diff --git a/changelog.md b/changelog.md index 1b6f2be6..5f388ec7 100644 --- a/changelog.md +++ b/changelog.md @@ -10,6 +10,7 @@ Saves from 5.x are not compatible with 6.0. * **[Mission Generation]** Reworked the ground object generation which now uses a new layout system * **[Mission Generation]** Added information about the modulation (AM/FM) of the assigned frequencies to the kneeboard and assign AM modulation instead of FM for JTAC. * **[Mission Generation]** Adjusted wind speeds. Wind speeds at high altitude are generally higher now. +* **[Mission Generation]** Added turbulance. Higher in Summer and Winter, also higher at day time than at night time. * **[Factions]** Updated the Faction file structure. Older custom faction files will not work correctly and have to be updated to the new structure. * **[Flight Planning]** Added preset formations for different flight types at hold, join, ingress, and split waypoints. Air to Air flights will tend toward line-abreast and spread-four formations. Air to ground flights will tend towards trail formation. * **[Flight Planning]** Added the ability to plan tankers for recovery on package flights. AI does not plan. diff --git a/game/missiongenerator/environmentgenerator.py b/game/missiongenerator/environmentgenerator.py index dec34969..c969f868 100644 --- a/game/missiongenerator/environmentgenerator.py +++ b/game/missiongenerator/environmentgenerator.py @@ -17,6 +17,7 @@ class EnvironmentGenerator: def set_atmospheric(self, atmospheric: AtmosphericConditions) -> None: self.mission.weather.qnh = atmospheric.qnh.mm_hg self.mission.weather.season_temperature = atmospheric.temperature_celsius + self.mission.weather.turbulence_at_ground = int(atmospheric.turbulance_per_10cm) def set_clouds(self, clouds: Optional[Clouds]) -> None: if clouds is None: diff --git a/game/missiongenerator/kneeboard.py b/game/missiongenerator/kneeboard.py index a1bc0c6f..a8892a5f 100644 --- a/game/missiongenerator/kneeboard.py +++ b/game/missiongenerator/kneeboard.py @@ -400,6 +400,9 @@ class BriefingPage(KneeboardPage): f"Temperature: {round(self.weather.atmospheric.temperature_celsius)} °C at sea level" ) writer.text(f"QNH: {qnh_in_hg} inHg / {qnh_mm_hg} mmHg / {qnh_hpa} hPa") + writer.text( + f"Turbulance: {round(self.weather.atmospheric.turbulance_per_10cm)} per 10cm at ground level." + ) fl = self.flight diff --git a/game/theater/seasonalconditions.py b/game/theater/seasonalconditions.py index 2280d15e..e478689c 100644 --- a/game/theater/seasonalconditions.py +++ b/game/theater/seasonalconditions.py @@ -45,4 +45,9 @@ class SeasonalConditions: winter_avg_temperature: float temperature_day_night_difference: float + high_avg_yearly_turbulance_per_10cm: float + low_avg_yearly_turbulance_per_10cm: float + solar_noon_turbulance_per_10cm: float + midnight_turbulance_per_10cm: float + weather_type_chances: dict[Season, WeatherTypeChances] diff --git a/game/theater/theaterloader.py b/game/theater/theaterloader.py index 3dad8c19..f3e94d51 100644 --- a/game/theater/theaterloader.py +++ b/game/theater/theaterloader.py @@ -57,6 +57,23 @@ class SeasonData: ) +@dataclass(frozen=True) +class TurbulanceData: + high_avg_yearly_turbulance_per_10cm: float | None + low_avg_yearly_turbulance_per_10cm: float | None + solar_noon_turbulance_per_10cm: float | None + midnight_turbulance_per_10cm: float | None + + @staticmethod + def from_yaml(data: dict[str, Any]) -> TurbulanceData: + return TurbulanceData( + data.get("high_avg_yearly_turbulance_per_10cm"), + data.get("low_avg_yearly_turbulance_per_10cm"), + data.get("solar_noon_turbulance_per_10cm"), + data.get("midnight_turbulance_per_10cm"), + ) + + class TheaterLoader: THEATER_RESOURCE_DIR = Path("resources/theaters") @@ -113,6 +130,7 @@ class TheaterLoader: spring = SeasonData.from_yaml(climate_data["seasons"]["spring"]) summer = SeasonData.from_yaml(climate_data["seasons"]["summer"]) fall = SeasonData.from_yaml(climate_data["seasons"]["fall"]) + turbulance = TurbulanceData.from_yaml(climate_data["turbulance"]) if summer.average_pressure is None: raise RuntimeError( f"{self.descriptor_path} does not define a summer average pressure" @@ -129,12 +147,32 @@ class TheaterLoader: raise RuntimeError( f"{self.descriptor_path} does not define a winter average temperature" ) + if turbulance.high_avg_yearly_turbulance_per_10cm is None: + raise RuntimeError( + f"{self.descriptor_path} does not define a yearly average high turbulance" + ) + if turbulance.low_avg_yearly_turbulance_per_10cm is None: + raise RuntimeError( + f"{self.descriptor_path} does not define a yearly average low turbulance" + ) + if turbulance.solar_noon_turbulance_per_10cm is None: + raise RuntimeError( + f"{self.descriptor_path} does not define a solar noon turbulance" + ) + if turbulance.midnight_turbulance_per_10cm is None: + raise RuntimeError( + f"{self.descriptor_path} does not define a midnight turbulance" + ) return SeasonalConditions( summer.average_pressure, winter.average_pressure, summer.average_temperature, winter.average_temperature, climate_data["day_night_temperature_difference"], + turbulance.high_avg_yearly_turbulance_per_10cm, + turbulance.low_avg_yearly_turbulance_per_10cm, + turbulance.solar_noon_turbulance_per_10cm, + turbulance.midnight_turbulance_per_10cm, { Season.Winter: winter.weather, Season.Spring: spring.weather, diff --git a/game/weather.py b/game/weather.py index a5fc203f..48f51184 100644 --- a/game/weather.py +++ b/game/weather.py @@ -2,6 +2,7 @@ from __future__ import annotations import datetime import logging +import math import random from dataclasses import dataclass, field from typing import Optional, TYPE_CHECKING @@ -36,6 +37,9 @@ class AtmosphericConditions: #: Temperature at sea level in Celcius. temperature_celsius: float + #: Turbulance per 10 cm. + turbulance_per_10cm: float + @dataclass(frozen=True) class WindConditions: @@ -108,18 +112,38 @@ class Weather: day, ) + seasonal_turbulance = self.interpolate_seasonal_turbulance( + seasonal_conditions.high_avg_yearly_turbulance_per_10cm, + seasonal_conditions.low_avg_yearly_turbulance_per_10cm, + day, + ) + + day_turbulance = seasonal_conditions.solar_noon_turbulance_per_10cm + night_turbulance = seasonal_conditions.midnight_turbulance_per_10cm + time_of_day_turbulance = self.interpolate_solar_activity( + time_of_day, day_turbulance, night_turbulance + ) + + random_turbulance = random.normalvariate(mu=0, sigma=0.5) + + turbulance = abs( + seasonal_turbulance + time_of_day_turbulance + random_turbulance + ) + if time_of_day == TimeOfDay.Day: temperature += seasonal_conditions.temperature_day_night_difference / 2 if time_of_day == TimeOfDay.Night: temperature -= seasonal_conditions.temperature_day_night_difference / 2 pressure += self.pressure_adjustment temperature += self.temperature_adjustment + turbulance += self.turbulance_adjustment logging.debug( "Weather: Before random: temp {} press {}".format(temperature, pressure) ) conditions = AtmosphericConditions( qnh=self.random_pressure(pressure), temperature_celsius=self.random_temperature(temperature), + turbulance_per_10cm=turbulance, ) logging.debug( "Weather: After random: temp {} press {}".format( @@ -136,6 +160,10 @@ class Weather: def temperature_adjustment(self) -> float: raise NotImplementedError + @property + def turbulance_adjustment(self) -> float: + raise NotImplementedError + def generate_clouds(self) -> Optional[Clouds]: raise NotImplementedError @@ -243,6 +271,42 @@ class Weather: winter_factor = distance_from_peak_summer / day_of_year_peak_summer return interpolate(summer_value, winter_value, winter_factor, clamp=True) + @staticmethod + def interpolate_seasonal_turbulance( + high_value: float, low_value: float, day: datetime.date + ) -> float: + day_of_year = day.timetuple().tm_yday + day_of_year_peak_summer = 183 + distance_from_peak_summer = -day_of_year_peak_summer + day_of_year + + amplitude = 0.5 * (high_value - low_value) + offset = amplitude + low_value + + # A high peak in summer and winter, between high_value and low_value. + return ( + amplitude * math.cos(4 * math.pi * distance_from_peak_summer / 365.25) + + offset + ) + + @staticmethod + def interpolate_solar_activity( + time_of_day: TimeOfDay, high: float, low: float + ) -> float: + + scale: float = 0 + + match time_of_day: + case TimeOfDay.Dawn: + scale = 0.4 + case TimeOfDay.Day: + scale = 1 + case TimeOfDay.Dusk: + scale = 0.6 + case TimeOfDay.Night: + scale = 0 + + return interpolate(value1=low, value2=high, factor=scale, clamp=True) + class ClearSkies(Weather): @property @@ -253,6 +317,10 @@ class ClearSkies(Weather): def temperature_adjustment(self) -> float: return 3.0 + @property + def turbulance_adjustment(self) -> float: + return 0.3 + def generate_clouds(self) -> Optional[Clouds]: return None @@ -272,6 +340,10 @@ class Cloudy(Weather): def temperature_adjustment(self) -> float: return 0.0 + @property + def turbulance_adjustment(self) -> float: + return 0.6 + def generate_clouds(self) -> Optional[Clouds]: return Clouds.random_preset(rain=False) @@ -292,6 +364,10 @@ class Raining(Weather): def temperature_adjustment(self) -> float: return -3.0 + @property + def turbulance_adjustment(self) -> float: + return 0.9 + def generate_clouds(self) -> Optional[Clouds]: return Clouds.random_preset(rain=True) @@ -312,6 +388,10 @@ class Thunderstorm(Weather): def temperature_adjustment(self) -> float: return -3.0 + @property + def turbulance_adjustment(self) -> float: + return 1.2 + def generate_clouds(self) -> Optional[Clouds]: return Clouds( base=self.random_cloud_base(), diff --git a/resources/theaters/caucasus/info.yaml b/resources/theaters/caucasus/info.yaml index 54b232d8..701af30c 100644 --- a/resources/theaters/caucasus/info.yaml +++ b/resources/theaters/caucasus/info.yaml @@ -37,3 +37,8 @@ climate: raining: 30 cloudy: 50 clear: 20 + turbulance: + high_avg_yearly_turbulance_per_10cm: 4 + low_avg_yearly_turbulance_per_10cm: 1 + solar_noon_turbulance_per_10cm: 1 + midnight_turbulance_per_10cm: 0 \ No newline at end of file diff --git a/resources/theaters/falklands/info.yaml b/resources/theaters/falklands/info.yaml index 1d4be26d..04a5fc6a 100644 --- a/resources/theaters/falklands/info.yaml +++ b/resources/theaters/falklands/info.yaml @@ -46,3 +46,8 @@ climate: raining: 30 cloudy: 45 clear: 25 + turbulance: + high_avg_yearly_turbulance_per_10cm: 9 + low_avg_yearly_turbulance_per_10cm: 3 + solar_noon_turbulance_per_10cm: 2 + midnight_turbulance_per_10cm: 0 \ No newline at end of file diff --git a/resources/theaters/marianaislands/info.yaml b/resources/theaters/marianaislands/info.yaml index 6cb3340c..734ddc22 100644 --- a/resources/theaters/marianaislands/info.yaml +++ b/resources/theaters/marianaislands/info.yaml @@ -38,3 +38,8 @@ climate: raining: 45 cloudy: 30 clear: 20 + turbulance: + high_avg_yearly_turbulance_per_10cm: 3 + low_avg_yearly_turbulance_per_10cm: 1 + solar_noon_turbulance_per_10cm: 2 + midnight_turbulance_per_10cm: 1 \ No newline at end of file diff --git a/resources/theaters/nevada/info.yaml b/resources/theaters/nevada/info.yaml index 401071fc..28467512 100644 --- a/resources/theaters/nevada/info.yaml +++ b/resources/theaters/nevada/info.yaml @@ -37,3 +37,8 @@ climate: raining: 10 cloudy: 45 clear: 45 + turbulance: + high_avg_yearly_turbulance_per_10cm: 9 + low_avg_yearly_turbulance_per_10cm: 1 + solar_noon_turbulance_per_10cm: 6 + midnight_turbulance_per_10cm: 0 \ No newline at end of file diff --git a/resources/theaters/normandy/info.yaml b/resources/theaters/normandy/info.yaml index 109039cf..e2510988 100644 --- a/resources/theaters/normandy/info.yaml +++ b/resources/theaters/normandy/info.yaml @@ -37,3 +37,8 @@ climate: raining: 30 cloudy: 50 clear: 20 + turbulance: + high_avg_yearly_turbulance_per_10cm: 5 + low_avg_yearly_turbulance_per_10cm: 2 + solar_noon_turbulance_per_10cm: 3 + midnight_turbulance_per_10cm: 1 \ No newline at end of file diff --git a/resources/theaters/persian gulf/info.yaml b/resources/theaters/persian gulf/info.yaml index 0f9d01ce..2d161aca 100644 --- a/resources/theaters/persian gulf/info.yaml +++ b/resources/theaters/persian gulf/info.yaml @@ -38,3 +38,8 @@ climate: raining: 2 cloudy: 28 clear: 70 + turbulance: + high_avg_yearly_turbulance_per_10cm: 8 + low_avg_yearly_turbulance_per_10cm: 1 + solar_noon_turbulance_per_10cm: 5 + midnight_turbulance_per_10cm: 0 \ No newline at end of file diff --git a/resources/theaters/syria/info.yaml b/resources/theaters/syria/info.yaml index 2b3eec9e..2801c6bf 100644 --- a/resources/theaters/syria/info.yaml +++ b/resources/theaters/syria/info.yaml @@ -37,3 +37,8 @@ climate: raining: 15 cloudy: 35 clear: 50 + turbulance: + high_avg_yearly_turbulance_per_10cm: 7 + low_avg_yearly_turbulance_per_10cm: 2 + solar_noon_turbulance_per_10cm: 4 + midnight_turbulance_per_10cm: 1 \ No newline at end of file diff --git a/resources/theaters/the channel/info.yaml b/resources/theaters/the channel/info.yaml index f6ed1f7b..b49d14d7 100644 --- a/resources/theaters/the channel/info.yaml +++ b/resources/theaters/the channel/info.yaml @@ -38,3 +38,8 @@ climate: raining: 30 cloudy: 50 clear: 20 + turbulance: + high_avg_yearly_turbulance_per_10cm: 5 + low_avg_yearly_turbulance_per_10cm: 2 + solar_noon_turbulance_per_10cm: 3 + midnight_turbulance_per_10cm: 1 \ No newline at end of file