2015-02-10 14:32:54 +00:00
# include "weather.hpp"
2015-04-22 15:58:55 +00:00
# include <components/misc/rng.hpp>
2015-03-15 01:07:47 +00:00
2015-08-27 14:57:32 +00:00
# include <components/esm/esmreader.hpp>
2015-07-09 17:22:04 +00:00
# include <components/esm/esmwriter.hpp>
2015-08-27 14:57:32 +00:00
# include <components/esm/savedgame.hpp>
2014-03-21 06:19:40 +00:00
# include <components/esm/weatherstate.hpp>
2016-01-06 11:46:06 +00:00
# include <components/fallback/fallback.hpp>
2014-03-20 06:25:52 +00:00
2012-04-23 13:27:03 +00:00
# include "../mwbase/environment.hpp"
2012-07-03 10:30:50 +00:00
# include "../mwbase/world.hpp"
2012-08-09 12:33:21 +00:00
# include "../mwbase/soundmanager.hpp"
2012-07-03 10:30:50 +00:00
2015-08-21 09:12:39 +00:00
# include "../mwmechanics/actorutil.hpp"
2014-12-01 19:31:33 +00:00
# include "../mwsound/sound.hpp"
2015-04-14 13:55:56 +00:00
# include "../mwrender/renderingmanager.hpp"
# include "../mwrender/sky.hpp"
2012-07-03 10:30:50 +00:00
# include "player.hpp"
2012-10-01 15:17:04 +00:00
# include "esmstore.hpp"
2014-02-23 19:11:05 +00:00
# include "cellstore.hpp"
2012-02-26 10:51:02 +00:00
2015-09-24 13:21:42 +00:00
# include <cmath>
2012-02-21 21:56:34 +00:00
using namespace MWWorld ;
2013-03-20 00:20:56 +00:00
namespace
{
2015-08-27 03:59:21 +00:00
static const int invalidWeatherID = - 1 ;
2015-11-01 21:47:40 +00:00
// linear interpolate between x and y based on factor.
2013-03-20 00:20:56 +00:00
float lerp ( float x , float y , float factor )
{
return x * ( 1 - factor ) + y * factor ;
}
2015-11-01 21:47:40 +00:00
// linear interpolate between x and y based on factor.
2015-04-14 13:55:56 +00:00
osg : : Vec4f lerp ( const osg : : Vec4f & x , const osg : : Vec4f & y , float factor )
2013-03-20 00:20:56 +00:00
{
return x * ( 1 - factor ) + y * factor ;
}
}
2015-11-01 21:47:40 +00:00
template < typename T >
T TimeOfDayInterpolator < T > : : getValue ( const float gameHour , const TimeOfDaySettings & timeSettings ) const
{
// TODO: use pre/post sunset/sunrise time values in [Weather] section
// night
if ( gameHour < = timeSettings . mNightEnd | | gameHour > = timeSettings . mNightStart + 1 )
return mNightValue ;
// sunrise
else if ( gameHour > = timeSettings . mNightEnd & & gameHour < = timeSettings . mDayStart + 1 )
{
if ( gameHour < = timeSettings . mSunriseTime )
{
// fade in
float advance = timeSettings . mSunriseTime - gameHour ;
float factor = advance / 0.5f ;
return lerp ( mSunriseValue , mNightValue , factor ) ;
}
else
{
// fade out
float advance = gameHour - timeSettings . mSunriseTime ;
float factor = advance / 3.f ;
return lerp ( mSunriseValue , mDayValue , factor ) ;
}
}
// day
else if ( gameHour > = timeSettings . mDayStart + 1 & & gameHour < = timeSettings . mDayEnd - 1 )
return mDayValue ;
// sunset
else if ( gameHour > = timeSettings . mDayEnd - 1 & & gameHour < = timeSettings . mNightStart + 1 )
{
if ( gameHour < = timeSettings . mDayEnd + 1 )
{
// fade in
float advance = ( timeSettings . mDayEnd + 1 ) - gameHour ;
float factor = ( advance / 2 ) ;
return lerp ( mSunsetValue , mDayValue , factor ) ;
}
else
{
// fade out
float advance = gameHour - ( timeSettings . mDayEnd + 1 ) ;
float factor = advance / 2.f ;
return lerp ( mSunsetValue , mNightValue , factor ) ;
}
}
// shut up compiler
return T ( ) ;
}
2016-08-14 14:17:09 +00:00
template class MWWorld : : TimeOfDayInterpolator < float > ;
template class MWWorld : : TimeOfDayInterpolator < osg : : Vec4f > ;
2015-11-01 21:47:40 +00:00
2015-08-16 04:38:49 +00:00
Weather : : Weather ( const std : : string & name ,
2016-01-06 11:46:06 +00:00
const Fallback : : Map & fallback ,
2015-08-16 04:38:49 +00:00
float stormWindSpeed ,
float rainSpeed ,
2017-09-21 10:08:45 +00:00
float dlFactor ,
float dlOffset ,
2015-08-16 04:38:49 +00:00
const std : : string & particleEffect )
: mCloudTexture ( fallback . getFallbackString ( " Weather_ " + name + " _Cloud_Texture " ) )
2015-11-01 21:47:40 +00:00
, mSkyColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Sunrise_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Day_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Sunset_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Night_Color " ) )
, mFogColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Sunrise_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Day_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Sunset_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Night_Color " ) )
, mAmbientColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Sunrise_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Day_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Sunset_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Night_Color " ) )
, mSunColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Sunrise_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Day_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Sunset_Color " ) ,
fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Night_Color " ) )
2015-11-01 22:03:16 +00:00
, mLandFogDepth ( fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Day_Depth " ) ,
fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Day_Depth " ) ,
fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Day_Depth " ) ,
fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Night_Depth " ) )
2015-08-16 04:38:49 +00:00
, mSunDiscSunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Disc_Sunset_Color " ) )
, mWindSpeed ( fallback . getFallbackFloat ( " Weather_ " + name + " _Wind_Speed " ) )
, mCloudSpeed ( fallback . getFallbackFloat ( " Weather_ " + name + " _Cloud_Speed " ) )
, mGlareView ( fallback . getFallbackFloat ( " Weather_ " + name + " _Glare_View " ) )
2017-09-21 10:08:45 +00:00
, mDL { dlFactor , dlOffset }
2015-08-16 04:38:49 +00:00
, mIsStorm ( mWindSpeed > stormWindSpeed )
, mRainSpeed ( rainSpeed )
, mRainFrequency ( fallback . getFallbackFloat ( " Weather_ " + name + " _Rain_Entrance_Speed " ) )
, mParticleEffect ( particleEffect )
, mRainEffect ( fallback . getFallbackBool ( " Weather_ " + name + " _Using_Precip " ) ? " meshes \\ raindrop.nif " : " " )
, mTransitionDelta ( fallback . getFallbackFloat ( " Weather_ " + name + " _Transition_Delta " ) )
, mCloudsMaximumPercent ( fallback . getFallbackFloat ( " Weather_ " + name + " _Clouds_Maximum_Percent " ) )
2015-09-09 03:05:33 +00:00
, mThunderFrequency ( fallback . getFallbackFloat ( " Weather_ " + name + " _Thunder_Frequency " ) )
, mThunderThreshold ( fallback . getFallbackFloat ( " Weather_ " + name + " _Thunder_Threshold " ) )
, mThunderSoundID ( )
, mFlashDecrement ( fallback . getFallbackFloat ( " Weather_ " + name + " _Flash_Decrement " ) )
, mFlashBrightness ( 0.0f )
{
mThunderSoundID [ 0 ] = fallback . getFallbackString ( " Weather_ " + name + " _Thunder_Sound_ID_0 " ) ;
mThunderSoundID [ 1 ] = fallback . getFallbackString ( " Weather_ " + name + " _Thunder_Sound_ID_1 " ) ;
mThunderSoundID [ 2 ] = fallback . getFallbackString ( " Weather_ " + name + " _Thunder_Sound_ID_2 " ) ;
mThunderSoundID [ 3 ] = fallback . getFallbackString ( " Weather_ " + name + " _Thunder_Sound_ID_3 " ) ;
2015-11-12 15:29:59 +00:00
// TODO: support weathers that have both "Ambient Loop Sound ID" and "Rain Loop Sound ID", need to play both sounds at the same time.
if ( ! mRainEffect . empty ( ) ) // NOTE: in vanilla, the weathers with rain seem to be hardcoded; changing Using_Precip has no effect
{
mAmbientLoopSoundID = fallback . getFallbackString ( " Weather_ " + name + " _Rain_Loop_Sound_ID " ) ;
if ( mAmbientLoopSoundID . empty ( ) ) // default to "rain" if not set
mAmbientLoopSoundID = " rain " ;
}
else
mAmbientLoopSoundID = fallback . getFallbackString ( " Weather_ " + name + " _Ambient_Loop_Sound_ID " ) ;
2015-11-28 04:55:36 +00:00
if ( Misc : : StringUtils : : ciEqual ( mAmbientLoopSoundID , " None " ) )
mAmbientLoopSoundID . clear ( ) ;
2015-09-09 03:05:33 +00:00
/*
Unhandled :
Rain Diameter = 600 ?
Rain Height Min = 200 ?
Rain Height Max = 700 ?
Rain Threshold = 0.6 ?
Max Raindrops = 650 ?
*/
2015-08-16 04:38:49 +00:00
}
2015-08-27 03:59:21 +00:00
float Weather : : transitionDelta ( ) const
2015-08-16 04:38:49 +00:00
{
2015-08-27 03:59:21 +00:00
// Transition Delta describes how quickly transitioning to the weather in question will take, in Hz. Note that the
// measurement is in real time, not in-game time.
return mTransitionDelta ;
2015-08-16 04:38:49 +00:00
}
2015-09-09 03:05:33 +00:00
float Weather : : cloudBlendFactor ( const float transitionRatio ) const
2015-08-16 04:38:49 +00:00
{
// Clouds Maximum Percent affects how quickly the sky transitions from one sky texture to the next.
return transitionRatio / mCloudsMaximumPercent ;
}
2015-09-09 03:05:33 +00:00
float Weather : : calculateThunder ( const float transitionRatio , const float elapsedSeconds , const bool isPaused )
{
// When paused, the flash brightness remains the same and no new strikes can occur.
if ( ! isPaused )
{
// Morrowind doesn't appear to do any calculations unless the transition ratio is higher than the Thunder Threshold.
if ( transitionRatio > = mThunderThreshold & & mThunderFrequency > 0.0f )
{
flashDecrement ( elapsedSeconds ) ;
if ( Misc : : Rng : : rollProbability ( ) < = thunderChance ( transitionRatio , elapsedSeconds ) )
{
lightningAndThunder ( ) ;
}
}
else
{
mFlashBrightness = 0.0f ;
}
}
return mFlashBrightness ;
}
inline void Weather : : flashDecrement ( const float elapsedSeconds )
{
// The Flash Decrement is measured in whole units per second. This means that if the flash brightness was
// currently 1.0, then it should take approximately 0.25 seconds to decay to 0.0 (the minimum).
float decrement = mFlashDecrement * elapsedSeconds ;
mFlashBrightness = decrement > mFlashBrightness ? 0.0f : mFlashBrightness - decrement ;
}
inline float Weather : : thunderChance ( const float transitionRatio , const float elapsedSeconds ) const
{
// This formula is reversed from the observation that with Thunder Frequency set to 1, there are roughly 10 strikes
// per minute. It doesn't appear to be tied to in game time as Timescale doesn't affect it. Various values of
// Thunder Frequency seem to change the average number of strikes in a linear fashion.. During a transition, it appears to
// scaled based on how far past it is past the Thunder Threshold.
float scaleFactor = ( transitionRatio - mThunderThreshold ) / ( 1.0f - mThunderThreshold ) ;
return ( ( mThunderFrequency * 10.0f ) / 60.0f ) * elapsedSeconds * scaleFactor ;
}
inline void Weather : : lightningAndThunder ( void )
{
// Morrowind seems to vary the intensity of the brightness based on which of the four sound IDs it selects.
// They appear to go from 0 (brightest, closest) to 3 (faintest, farthest). The value of 0.25 per distance
// was derived by setting the Flash Decrement to 0.1 and measuring how long each value took to decay to 0.
// TODO: Determine the distribution of each distance to see if it's evenly weighted.
unsigned int distance = Misc : : Rng : : rollDice ( 4 ) ;
// Flash brightness appears additive, since if multiple strikes occur, it takes longer for it to decay to 0.
mFlashBrightness + = 1 - ( distance * 0.25f ) ;
MWBase : : Environment : : get ( ) . getSoundManager ( ) - > playSound ( mThunderSoundID [ distance ] , 1.0 , 1.0 ) ;
}
2015-08-27 03:59:21 +00:00
RegionWeather : : RegionWeather ( const ESM : : Region & region )
: mWeather ( invalidWeatherID )
, mChances ( )
{
mChances . reserve ( 10 ) ;
mChances . push_back ( region . mData . mClear ) ;
mChances . push_back ( region . mData . mCloudy ) ;
mChances . push_back ( region . mData . mFoggy ) ;
mChances . push_back ( region . mData . mOvercast ) ;
mChances . push_back ( region . mData . mRain ) ;
mChances . push_back ( region . mData . mThunder ) ;
mChances . push_back ( region . mData . mAsh ) ;
mChances . push_back ( region . mData . mBlight ) ;
mChances . push_back ( region . mData . mA ) ;
mChances . push_back ( region . mData . mB ) ;
}
RegionWeather : : RegionWeather ( const ESM : : RegionWeatherState & state )
: mWeather ( state . mWeather )
, mChances ( state . mChances )
{
}
RegionWeather : : operator ESM : : RegionWeatherState ( ) const
{
2015-08-27 04:34:15 +00:00
ESM : : RegionWeatherState state =
{
mWeather ,
mChances
} ;
return state ;
2015-08-27 03:59:21 +00:00
}
void RegionWeather : : setChances ( const std : : vector < char > & chances )
{
if ( mChances . size ( ) < chances . size ( ) )
{
mChances . reserve ( chances . size ( ) ) ;
}
std : : vector < char > : : const_iterator it = chances . begin ( ) ;
for ( size_t i = 0 ; it ! = chances . end ( ) ; + + it , + + i )
{
mChances [ i ] = * it ;
}
// Regional weather no longer supports the current type, select a new weather pattern.
if ( ( static_cast < size_t > ( mWeather ) > = mChances . size ( ) ) | | ( mChances [ mWeather ] = = 0 ) )
{
chooseNewWeather ( ) ;
}
}
void RegionWeather : : setWeather ( int weatherID )
{
mWeather = weatherID ;
}
int RegionWeather : : getWeather ( )
{
// If the region weather was already set (by ChangeWeather, or by a previous call) then just return that value.
// Note that the region weather will be expired periodically when the weather update timer expires.
if ( mWeather = = invalidWeatherID )
{
chooseNewWeather ( ) ;
}
return mWeather ;
}
void RegionWeather : : chooseNewWeather ( )
{
// All probabilities must add to 100 (responsibility of the user).
// If chances A and B has values 30 and 70 then by generating 100 numbers 1..100, 30% will be lesser or equal 30
// and 70% will be greater than 30 (in theory).
int chance = Misc : : Rng : : rollDice ( 100 ) + 1 ; // 1..100
int sum = 0 ;
int i = 0 ;
for ( ; static_cast < size_t > ( i ) < mChances . size ( ) ; + + i )
{
sum + = mChances [ i ] ;
if ( chance < = sum )
2016-01-14 16:41:39 +00:00
{
mWeather = i ;
return ;
}
2015-08-27 03:59:21 +00:00
}
2016-01-14 16:41:39 +00:00
// if we hit this path then the chances don't add to 100, choose a default weather instead
mWeather = 0 ;
2015-08-27 03:59:21 +00:00
}
2016-01-06 11:46:06 +00:00
MoonModel : : MoonModel ( const std : : string & name , const Fallback : : Map & fallback )
2015-07-30 04:57:45 +00:00
: mFadeInStart ( fallback . getFallbackFloat ( " Moons_ " + name + " _Fade_In_Start " ) )
, mFadeInFinish ( fallback . getFallbackFloat ( " Moons_ " + name + " _Fade_In_Finish " ) )
, mFadeOutStart ( fallback . getFallbackFloat ( " Moons_ " + name + " _Fade_Out_Start " ) )
, mFadeOutFinish ( fallback . getFallbackFloat ( " Moons_ " + name + " _Fade_Out_Finish " ) )
, mAxisOffset ( fallback . getFallbackFloat ( " Moons_ " + name + " _Axis_Offset " ) )
, mSpeed ( fallback . getFallbackFloat ( " Moons_ " + name + " _Speed " ) )
, mDailyIncrement ( fallback . getFallbackFloat ( " Moons_ " + name + " _Daily_Increment " ) )
, mFadeStartAngle ( fallback . getFallbackFloat ( " Moons_ " + name + " _Fade_Start_Angle " ) )
, mFadeEndAngle ( fallback . getFallbackFloat ( " Moons_ " + name + " _Fade_End_Angle " ) )
, mMoonShadowEarlyFadeAngle ( fallback . getFallbackFloat ( " Moons_ " + name + " _Moon_Shadow_Early_Fade_Angle " ) )
{
// Morrowind appears to have a minimum speed in order to avoid situations where the moon couldn't conceivably
// complete a rotation in a single 24 hour period. The value of 180/23 was deduced from reverse engineering.
mSpeed = std : : min ( mSpeed , 180.0f / 23.0f ) ;
}
2015-08-27 03:59:21 +00:00
MWRender : : MoonState MoonModel : : calculateState ( const TimeStamp & gameTime ) const
2015-07-30 04:57:45 +00:00
{
2015-08-27 03:59:21 +00:00
float rotationFromHorizon = angle ( gameTime ) ;
2015-07-30 04:57:45 +00:00
MWRender : : MoonState state =
2015-08-27 03:59:21 +00:00
{
rotationFromHorizon ,
mAxisOffset , // Reverse engineered from Morrowind's scene graph rotation matrices.
static_cast < MWRender : : MoonState : : Phase > ( phase ( gameTime ) ) ,
shadowBlend ( rotationFromHorizon ) ,
earlyMoonShadowAlpha ( rotationFromHorizon ) * hourlyAlpha ( gameTime . getHour ( ) )
} ;
2015-07-30 04:57:45 +00:00
return state ;
}
2015-08-27 03:59:21 +00:00
inline float MoonModel : : angle ( const TimeStamp & gameTime ) const
2015-07-30 04:57:45 +00:00
{
// Morrowind's moons start travel on one side of the horizon (let's call it H-rise) and travel 180 degrees to the
// opposite horizon (let's call it H-set). Upon reaching H-set, they reset to H-rise until the next moon rise.
// When calculating the angle of the moon, several cases have to be taken into account:
// 1. Moon rises and then sets in one day.
// 2. Moon sets and doesn't rise in one day (occurs when the moon rise hour is >= 24).
// 3. Moon sets and then rises in one day.
2015-08-27 03:59:21 +00:00
float moonRiseHourToday = moonRiseHour ( gameTime . getDay ( ) ) ;
2015-07-30 04:57:45 +00:00
float moonRiseAngleToday = 0 ;
2015-08-27 03:59:21 +00:00
if ( gameTime . getHour ( ) < moonRiseHourToday )
2015-07-30 04:57:45 +00:00
{
2015-08-27 03:59:21 +00:00
float moonRiseHourYesterday = moonRiseHour ( gameTime . getDay ( ) - 1 ) ;
2015-07-30 04:57:45 +00:00
if ( moonRiseHourYesterday < 24 )
{
float moonRiseAngleYesterday = rotation ( 24 - moonRiseHourYesterday ) ;
if ( moonRiseAngleYesterday < 180 )
{
// The moon rose but did not set yesterday, so accumulate yesterday's angle with how much we've travelled today.
2015-08-27 03:59:21 +00:00
moonRiseAngleToday = rotation ( gameTime . getHour ( ) ) + moonRiseAngleYesterday ;
2015-07-30 04:57:45 +00:00
}
}
}
else
{
2015-08-27 03:59:21 +00:00
moonRiseAngleToday = rotation ( gameTime . getHour ( ) - moonRiseHourToday ) ;
2015-07-30 04:57:45 +00:00
}
if ( moonRiseAngleToday > = 180 )
{
// The moon set today, reset the angle to the horizon.
moonRiseAngleToday = 0 ;
}
return moonRiseAngleToday ;
}
2015-07-30 19:00:08 +00:00
inline float MoonModel : : moonRiseHour ( unsigned int daysPassed ) const
2015-07-30 04:57:45 +00:00
{
// This arises from the start date of 16 Last Seed, 427
// TODO: Find an alternate formula that doesn't rely on this day being fixed.
static const unsigned int startDay = 16 ;
// This odd formula arises from the fact that on 16 Last Seed, 17 increments have occurred, meaning
// that upon starting a new game, it must only calculate the moon phase as far back as 1 Last Seed.
// Note that we don't modulo after adding the latest daily increment because other calculations need to
// know if doing so would cause the moon rise to be postponed until the next day (which happens when
// the moon rise hour is >= 24 in Morrowind).
return mDailyIncrement + std : : fmod ( ( daysPassed - 1 + startDay ) * mDailyIncrement , 24.0f ) ;
}
2015-07-30 19:00:08 +00:00
inline float MoonModel : : rotation ( float hours ) const
2015-07-30 04:57:45 +00:00
{
// 15 degrees per hour was reverse engineered from the rotation matrices of the Morrowind scene graph.
// Note that this correlates to 360 / 24, which is a full rotation every 24 hours, so speed is a measure
// of whole rotations that could be completed in a day.
return 15.0f * mSpeed * hours ;
}
2015-08-27 03:59:21 +00:00
inline unsigned int MoonModel : : phase ( const TimeStamp & gameTime ) const
2015-07-30 04:57:45 +00:00
{
// Morrowind starts with a full moon on 16 Last Seed and then begins to wane 17 Last Seed, working on 3 day phase cycle.
// Note: this is an internal helper, and as such we don't want to return MWRender::MoonState::Phase since we can't
// forward declare it (C++11 strongly typed enums solve this).
2015-08-05 02:07:42 +00:00
// If the moon didn't rise yet today, use yesterday's moon phase.
2015-08-27 03:59:21 +00:00
if ( gameTime . getHour ( ) < moonRiseHour ( gameTime . getDay ( ) ) )
return ( gameTime . getDay ( ) / 3 ) % 8 ;
2015-08-05 02:07:42 +00:00
else
2015-08-27 03:59:21 +00:00
return ( ( gameTime . getDay ( ) + 1 ) / 3 ) % 8 ;
2015-07-30 04:57:45 +00:00
}
2015-07-30 19:00:08 +00:00
inline float MoonModel : : shadowBlend ( float angle ) const
2015-07-30 04:57:45 +00:00
{
// The Fade End Angle and Fade Start Angle describe a region where the moon transitions from a solid disk
// that is roughly the color of the sky, to a textured surface.
// Depending on the current angle, the following values describe the ratio between the textured moon
// and the solid disk:
// 1. From Fade End Angle 1 to Fade Start Angle 1 (during moon rise): 0..1
// 2. From Fade Start Angle 1 to Fade Start Angle 2 (between moon rise and moon set): 1 (textured)
// 3. From Fade Start Angle 2 to Fade End Angle 2 (during moon set): 1..0
// 4. From Fade End Angle 2 to Fade End Angle 1 (between moon set and moon rise): 0 (solid disk)
float fadeAngle = mFadeStartAngle - mFadeEndAngle ;
float fadeEndAngle2 = 180.0f - mFadeEndAngle ;
float fadeStartAngle2 = 180.0f - mFadeStartAngle ;
if ( ( angle > = mFadeEndAngle ) & & ( angle < mFadeStartAngle ) )
return ( angle - mFadeEndAngle ) / fadeAngle ;
else if ( ( angle > = mFadeStartAngle ) & & ( angle < fadeStartAngle2 ) )
return 1.0f ;
else if ( ( angle > = fadeStartAngle2 ) & & ( angle < fadeEndAngle2 ) )
return ( fadeEndAngle2 - angle ) / fadeAngle ;
else
return 0.0f ;
}
2015-07-30 19:00:08 +00:00
inline float MoonModel : : hourlyAlpha ( float gameHour ) const
2015-07-30 04:57:45 +00:00
{
// The Fade Out Start / Finish and Fade In Start / Finish describe the hours at which the moon
// appears and disappears.
// Depending on the current hour, the following values describe how transparent the moon is.
// 1. From Fade Out Start to Fade Out Finish: 1..0
// 2. From Fade Out Finish to Fade In Start: 0 (transparent)
// 3. From Fade In Start to Fade In Finish: 0..1
// 4. From Fade In Finish to Fade Out Start: 1 (solid)
if ( ( gameHour > = mFadeOutStart ) & & ( gameHour < mFadeOutFinish ) )
return ( mFadeOutFinish - gameHour ) / ( mFadeOutFinish - mFadeOutStart ) ;
else if ( ( gameHour > = mFadeOutFinish ) & & ( gameHour < mFadeInStart ) )
return 0.0f ;
else if ( ( gameHour > = mFadeInStart ) & & ( gameHour < mFadeInFinish ) )
return ( gameHour - mFadeInStart ) / ( mFadeInFinish - mFadeInStart ) ;
else
return 1.0f ;
}
2015-07-30 19:00:08 +00:00
inline float MoonModel : : earlyMoonShadowAlpha ( float angle ) const
2015-07-30 04:57:45 +00:00
{
// The Moon Shadow Early Fade Angle describes an arc relative to Fade End Angle.
// Depending on the current angle, the following values describe how transparent the moon is.
// 1. From Moon Shadow Early Fade Angle 1 to Fade End Angle 1 (during moon rise): 0..1
// 2. From Fade End Angle 1 to Fade End Angle 2 (between moon rise and moon set): 1 (solid)
// 3. From Fade End Angle 2 to Moon Shadow Early Fade Angle 2 (during moon set): 1..0
// 4. From Moon Shadow Early Fade Angle 2 to Moon Shadow Early Fade Angle 1: 0 (transparent)
float moonShadowEarlyFadeAngle1 = mFadeEndAngle - mMoonShadowEarlyFadeAngle ;
float fadeEndAngle2 = 180.0f - mFadeEndAngle ;
float moonShadowEarlyFadeAngle2 = fadeEndAngle2 + mMoonShadowEarlyFadeAngle ;
if ( ( angle > = moonShadowEarlyFadeAngle1 ) & & ( angle < mFadeEndAngle ) )
return ( angle - moonShadowEarlyFadeAngle1 ) / mMoonShadowEarlyFadeAngle ;
else if ( ( angle > = mFadeEndAngle ) & & ( angle < fadeEndAngle2 ) )
return 1.0f ;
else if ( ( angle > = fadeEndAngle2 ) & & ( angle < moonShadowEarlyFadeAngle2 ) )
return ( moonShadowEarlyFadeAngle2 - angle ) / mMoonShadowEarlyFadeAngle ;
else
return 0.0f ;
}
2016-01-06 11:46:06 +00:00
WeatherManager : : WeatherManager ( MWRender : : RenderingManager & rendering , const Fallback : : Map & fallback , MWWorld : : ESMStore & store )
2015-08-27 03:59:21 +00:00
: mStore ( store )
, mRendering ( rendering )
, mSunriseTime ( fallback . getFallbackFloat ( " Weather_Sunrise_Time " ) )
, mSunsetTime ( fallback . getFallbackFloat ( " Weather_Sunset_Time " ) )
, mSunriseDuration ( fallback . getFallbackFloat ( " Weather_Sunrise_Duration " ) )
, mSunsetDuration ( fallback . getFallbackFloat ( " Weather_Sunset_Duration " ) )
2015-09-21 17:43:48 +00:00
, mSunPreSunsetTime ( fallback . getFallbackFloat ( " Weather_Sun_Pre-Sunset_Time " ) )
2015-11-01 21:47:40 +00:00
, mNightFade ( 0 , 0 , 0 , 1 )
2015-08-27 03:59:21 +00:00
, mHoursBetweenWeatherChanges ( fallback . getFallbackFloat ( " Weather_Hours_Between_Weather_Changes " ) )
, mRainSpeed ( fallback . getFallbackFloat ( " Weather_Precip_Gravity " ) )
2015-11-01 21:47:40 +00:00
, mUnderwaterFog ( fallback . getFallbackFloat ( " Water_UnderwaterSunriseFog " ) ,
fallback . getFallbackFloat ( " Water_UnderwaterDayFog " ) ,
fallback . getFallbackFloat ( " Water_UnderwaterSunsetFog " ) ,
fallback . getFallbackFloat ( " Water_UnderwaterNightFog " ) )
2015-08-27 03:59:21 +00:00
, mWeatherSettings ( )
, mMasser ( " Masser " , fallback )
, mSecunda ( " Secunda " , fallback )
, mWindSpeed ( 0.f )
, mIsStorm ( false )
, mStormDirection ( 0 , 1 , 0 )
, mCurrentRegion ( )
, mTimePassed ( 0 )
, mFastForward ( false )
, mWeatherUpdateTime ( mHoursBetweenWeatherChanges )
, mTransitionFactor ( 0 )
, mCurrentWeather ( 0 )
, mNextWeather ( 0 )
, mQueuedWeather ( 0 )
, mRegions ( )
, mResult ( )
2017-09-12 04:33:18 +00:00
, mAmbientSound ( nullptr )
2015-08-27 03:59:21 +00:00
, mPlayingSoundID ( )
{
2015-11-01 21:47:40 +00:00
mTimeSettings . mNightStart = mSunsetTime + mSunsetDuration ;
mTimeSettings . mNightEnd = mSunriseTime - 0.5f ;
mTimeSettings . mDayStart = mSunriseTime + mSunriseDuration ;
mTimeSettings . mDayEnd = mSunsetTime ;
mTimeSettings . mSunriseTime = mSunriseTime ;
2015-08-27 03:59:21 +00:00
mWeatherSettings . reserve ( 10 ) ;
2017-09-21 10:08:45 +00:00
// These distant land fog factor and offset values are the defaults MGE XE provides. Should be
// provided by settings somewhere?
addWeather ( " Clear " , fallback , 1.0f , 0.0f ) ; // 0
addWeather ( " Cloudy " , fallback , 0.9f , 0.0f ) ; // 1
addWeather ( " Foggy " , fallback , 0.2f , 30.0f ) ; // 2
addWeather ( " Overcast " , fallback , 0.7f , 0.0f ) ; // 3
addWeather ( " Rain " , fallback , 0.5f , 10.0f ) ; // 4
addWeather ( " Thunderstorm " , fallback , 0.5f , 20.0f ) ; // 5
addWeather ( " Ashstorm " , fallback , 0.2f , 50.0f , " meshes \\ ashcloud.nif " ) ; // 6
addWeather ( " Blight " , fallback , 0.2f , 60.0f , " meshes \\ blightcloud.nif " ) ; // 7
addWeather ( " Snow " , fallback , 0.5f , 40.0f , " meshes \\ snow.nif " ) ; // 8
addWeather ( " Blizzard " , fallback , 0.16f , 70.0f , " meshes \\ blizzard.nif " ) ; // 9
2015-08-27 03:59:21 +00:00
Store < ESM : : Region > : : iterator it = store . get < ESM : : Region > ( ) . begin ( ) ;
for ( ; it ! = store . get < ESM : : Region > ( ) . end ( ) ; + + it )
{
std : : string regionID = Misc : : StringUtils : : lowerCase ( it - > mId ) ;
mRegions . insert ( std : : make_pair ( regionID , RegionWeather ( * it ) ) ) ;
}
forceWeather ( 0 ) ;
2012-02-22 18:17:37 +00:00
}
2013-08-29 13:15:40 +00:00
WeatherManager : : ~ WeatherManager ( )
{
2014-12-01 19:31:33 +00:00
stopSounds ( ) ;
2013-08-29 13:15:40 +00:00
}
2015-08-27 03:59:21 +00:00
void WeatherManager : : changeWeather ( const std : : string & regionID , const unsigned int weatherID )
2012-02-22 18:17:37 +00:00
{
2015-08-27 03:59:21 +00:00
// In Morrowind, this seems to have the following behavior, when applied to the current region:
// - When there is no transition in progress, start transitioning to the new weather.
// - If there is a transition in progress, queue up the transition and process it when the current one completes.
// - If there is a transition in progress, and a queued transition, overwrite the queued transition.
// - If multiple calls to ChangeWeather are made while paused (console up), only the last call will be used,
// meaning that if there was no transition in progress, only the last ChangeWeather will be processed.
// If the region isn't current, Morrowind will store the new weather for the region in question.
if ( weatherID < mWeatherSettings . size ( ) )
2012-02-22 18:17:37 +00:00
{
2015-08-27 03:59:21 +00:00
std : : string lowerCaseRegionID = Misc : : StringUtils : : lowerCase ( regionID ) ;
std : : map < std : : string , RegionWeather > : : iterator it = mRegions . find ( lowerCaseRegionID ) ;
if ( it ! = mRegions . end ( ) )
2012-02-27 11:21:00 +00:00
{
2015-08-27 03:59:21 +00:00
it - > second . setWeather ( weatherID ) ;
regionalWeatherChanged ( it - > first , it - > second ) ;
2012-02-27 11:21:00 +00:00
}
2012-02-22 18:17:37 +00:00
}
}
2015-08-27 03:59:21 +00:00
void WeatherManager : : modRegion ( const std : : string & regionID , const std : : vector < char > & chances )
2012-02-22 18:17:37 +00:00
{
2015-08-27 03:59:21 +00:00
// Sets the region's probability for various weather patterns. Note that this appears to be saved permanently.
// In Morrowind, this seems to have the following behavior when applied to the current region:
// - If the region supports the current weather, no change in current weather occurs.
// - If the region no longer supports the current weather, and there is no transition in progress, begin to
// transition to a new supported weather type.
// - If the region no longer supports the current weather, and there is a transition in progress, queue a
// transition to a new supported weather type.
std : : string lowerCaseRegionID = Misc : : StringUtils : : lowerCase ( regionID ) ;
std : : map < std : : string , RegionWeather > : : iterator it = mRegions . find ( lowerCaseRegionID ) ;
if ( it ! = mRegions . end ( ) )
2012-02-23 18:49:56 +00:00
{
2015-08-27 03:59:21 +00:00
it - > second . setChances ( chances ) ;
regionalWeatherChanged ( it - > first , it - > second ) ;
2012-02-23 18:49:56 +00:00
}
2015-08-27 03:59:21 +00:00
}
2012-03-16 18:49:01 +00:00
2015-08-27 03:59:21 +00:00
void WeatherManager : : playerTeleported ( )
{
// If the player teleports to an outdoors cell in a new region (for instance, by travelling), the weather needs to
// be changed immediately, and any transitions for the previous region discarded.
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
if ( world - > isCellExterior ( ) | | world - > isCellQuasiExterior ( ) )
2012-02-23 18:49:56 +00:00
{
2015-08-27 03:59:21 +00:00
std : : string playerRegion = Misc : : StringUtils : : lowerCase ( world - > getPlayerPtr ( ) . getCell ( ) - > getCell ( ) - > mRegion ) ;
std : : map < std : : string , RegionWeather > : : iterator it = mRegions . find ( playerRegion ) ;
if ( it ! = mRegions . end ( ) & & playerRegion ! = mCurrentRegion )
2012-02-24 16:42:31 +00:00
{
2015-08-27 03:59:21 +00:00
mCurrentRegion = playerRegion ;
forceWeather ( it - > second . getWeather ( ) ) ;
2012-02-24 16:42:31 +00:00
}
2012-02-23 18:49:56 +00:00
}
2015-08-27 03:59:21 +00:00
}
2012-03-16 18:49:01 +00:00
2015-08-27 03:59:21 +00:00
void WeatherManager : : update ( float duration , bool paused )
{
2015-12-19 14:50:13 +00:00
MWWorld : : ConstPtr player = MWMechanics : : getPlayer ( ) ;
2015-08-27 03:59:21 +00:00
MWBase : : World & world = * MWBase : : Environment : : get ( ) . getWorld ( ) ;
TimeStamp time = world . getTimeStamp ( ) ;
2012-03-16 18:49:01 +00:00
2015-11-28 16:57:35 +00:00
if ( ! paused | | mFastForward )
2012-02-23 18:49:56 +00:00
{
2015-08-27 03:59:21 +00:00
// Add new transitions when either the player's current external region changes.
std : : string playerRegion = Misc : : StringUtils : : lowerCase ( player . getCell ( ) - > getCell ( ) - > mRegion ) ;
if ( updateWeatherTime ( ) | | updateWeatherRegion ( playerRegion ) )
2012-02-24 16:42:31 +00:00
{
2015-08-27 03:59:21 +00:00
std : : map < std : : string , RegionWeather > : : iterator it = mRegions . find ( mCurrentRegion ) ;
if ( it ! = mRegions . end ( ) )
{
addWeatherTransition ( it - > second . getWeather ( ) ) ;
}
2012-02-24 16:42:31 +00:00
}
2012-02-22 18:17:37 +00:00
2015-08-27 03:59:21 +00:00
updateWeatherTransitions ( duration ) ;
2014-12-01 19:31:33 +00:00
}
2012-09-25 19:28:25 +00:00
2015-08-27 03:59:21 +00:00
const bool exterior = ( world . isCellExterior ( ) | | world . isCellQuasiExterior ( ) ) ;
if ( ! exterior )
2014-01-01 15:45:39 +00:00
{
2015-08-27 03:59:21 +00:00
mRendering . setSkyEnabled ( false ) ;
2014-12-01 19:31:33 +00:00
stopSounds ( ) ;
2014-01-01 15:45:39 +00:00
return ;
}
2015-09-09 03:05:33 +00:00
calculateWeatherResult ( time . getHour ( ) , duration , paused ) ;
2012-03-17 12:14:31 +00:00
2013-06-19 14:18:43 +00:00
mWindSpeed = mResult . mWindSpeed ;
2014-06-25 16:20:21 +00:00
mIsStorm = mResult . mIsStorm ;
2013-05-01 09:42:24 +00:00
2014-06-26 17:01:49 +00:00
if ( mIsStorm )
{
2015-05-12 01:02:15 +00:00
osg : : Vec3f playerPos ( player . getRefData ( ) . getPosition ( ) . asVec3 ( ) ) ;
osg : : Vec3f redMountainPos ( 19950 , 72032 , 27831 ) ;
2014-06-26 17:01:49 +00:00
mStormDirection = ( playerPos - redMountainPos ) ;
2015-05-12 01:02:15 +00:00
mStormDirection . z ( ) = 0 ;
2015-05-31 23:57:15 +00:00
mStormDirection . normalize ( ) ;
2015-08-27 03:59:21 +00:00
mRendering . getSkyManager ( ) - > setStormDirection ( mStormDirection ) ;
2014-06-26 17:01:49 +00:00
}
2013-06-19 14:18:43 +00:00
// disable sun during night
2015-11-01 21:47:40 +00:00
if ( time . getHour ( ) > = mTimeSettings . mNightStart | | time . getHour ( ) < = mSunriseTime )
2015-08-27 03:59:21 +00:00
mRendering . getSkyManager ( ) - > sunDisable ( ) ;
2015-04-15 16:50:50 +00:00
else
2015-08-27 03:59:21 +00:00
mRendering . getSkyManager ( ) - > sunEnable ( ) ;
2012-03-17 11:57:52 +00:00
2015-02-08 20:26:38 +00:00
// Update the sun direction. Run it east to west at a fixed angle from overhead.
// The sun's speed at day and night may differ, since mSunriseTime and mNightStart
2015-02-07 18:36:27 +00:00
// mark when the sun is level with the horizon.
{
2015-02-08 20:26:38 +00:00
// Shift times into a 24-hour window beginning at mSunriseTime...
2015-08-27 03:59:21 +00:00
float adjustedHour = time . getHour ( ) ;
2015-11-01 21:47:40 +00:00
float adjustedNightStart = mTimeSettings . mNightStart ;
2015-08-27 03:59:21 +00:00
if ( time . getHour ( ) < mSunriseTime )
2015-02-07 18:36:27 +00:00
adjustedHour + = 24.f ;
2015-11-01 21:47:40 +00:00
if ( mTimeSettings . mNightStart < mSunriseTime )
2015-02-08 20:26:38 +00:00
adjustedNightStart + = 24.f ;
2012-03-17 12:14:31 +00:00
2015-02-08 20:26:38 +00:00
const bool is_night = adjustedHour > = adjustedNightStart ;
const float dayDuration = adjustedNightStart - mSunriseTime ;
2015-02-07 18:36:27 +00:00
const float nightDuration = 24.f - dayDuration ;
2012-03-17 12:14:31 +00:00
2015-02-07 18:36:27 +00:00
double theta ;
if ( ! is_night ) {
2016-12-21 15:49:37 +00:00
theta = static_cast < float > ( osg : : PI ) * ( adjustedHour - mSunriseTime ) / dayDuration ;
2015-02-07 18:36:27 +00:00
} else {
2016-12-21 15:49:37 +00:00
theta = static_cast < float > ( osg : : PI ) * ( 1.f - ( adjustedHour - adjustedNightStart ) / nightDuration ) ;
2015-02-07 18:36:27 +00:00
}
2014-06-16 21:08:02 +00:00
2015-04-14 13:55:56 +00:00
osg : : Vec3f final (
2015-03-08 00:07:29 +00:00
static_cast < float > ( cos ( theta ) ) ,
2015-02-08 20:26:38 +00:00
- 0.268f , // approx tan( -15 degrees )
2015-03-08 00:07:29 +00:00
static_cast < float > ( sin ( theta ) ) ) ;
2015-08-27 03:59:21 +00:00
mRendering . setSunDirection ( final * - 1 ) ;
2015-02-07 18:36:27 +00:00
}
2012-03-16 18:49:01 +00:00
2015-11-01 21:47:40 +00:00
float underwaterFog = mUnderwaterFog . getValue ( time . getHour ( ) , mTimeSettings ) ;
2015-11-01 21:09:02 +00:00
2015-09-21 14:03:30 +00:00
float peakHour = mSunriseTime + ( mSunsetTime - mSunriseTime ) / 2 ;
if ( time . getHour ( ) < mSunriseTime | | time . getHour ( ) > mSunsetTime )
mRendering . getSkyManager ( ) - > setGlareTimeOfDayFade ( 0 ) ;
else if ( time . getHour ( ) < peakHour )
mRendering . getSkyManager ( ) - > setGlareTimeOfDayFade ( 1 - ( peakHour - time . getHour ( ) ) / ( peakHour - mSunriseTime ) ) ;
else
mRendering . getSkyManager ( ) - > setGlareTimeOfDayFade ( 1 - ( time . getHour ( ) - peakHour ) / ( mSunsetTime - peakHour ) ) ;
2015-08-27 03:59:21 +00:00
mRendering . getSkyManager ( ) - > setMasserState ( mMasser . calculateState ( time ) ) ;
mRendering . getSkyManager ( ) - > setSecundaState ( mSecunda . calculateState ( time ) ) ;
2012-03-16 18:49:01 +00:00
2017-09-21 10:08:45 +00:00
mRendering . configureFog ( mResult . mFogDepth , underwaterFog , mResult . mDLFogFactor ,
mResult . mDLFogOffset / 100.0f , mResult . mFogColor ) ;
2015-08-27 03:59:21 +00:00
mRendering . setAmbientColour ( mResult . mAmbientColor ) ;
2016-05-10 15:39:57 +00:00
mRendering . setSunColour ( mResult . mSunColor , mResult . mSunColor * mResult . mGlareView ) ;
2013-06-19 14:18:43 +00:00
2015-08-27 03:59:21 +00:00
mRendering . getSkyManager ( ) - > setWeather ( mResult ) ;
2013-06-19 14:18:43 +00:00
// Play sounds
2014-12-01 19:31:33 +00:00
if ( mPlayingSoundID ! = mResult . mAmbientLoopSoundID )
2012-03-09 17:30:03 +00:00
{
2014-12-01 19:31:33 +00:00
stopSounds ( ) ;
if ( ! mResult . mAmbientLoopSoundID . empty ( ) )
2017-09-15 08:03:41 +00:00
mAmbientSound = MWBase : : Environment : : get ( ) . getSoundManager ( ) - > playSound (
mResult . mAmbientLoopSoundID , mResult . mAmbientSoundVolume , 1.0 ,
MWSound : : Type : : Sfx , MWSound : : PlayMode : : Loop
) ;
2014-12-01 19:31:33 +00:00
mPlayingSoundID = mResult . mAmbientLoopSoundID ;
2012-03-09 17:30:03 +00:00
}
2017-09-12 04:33:18 +00:00
else if ( mAmbientSound )
2014-12-01 19:31:33 +00:00
mAmbientSound - > setVolume ( mResult . mAmbientSoundVolume ) ;
2013-06-19 14:18:43 +00:00
}
2014-12-01 19:31:33 +00:00
void WeatherManager : : stopSounds ( )
2013-06-19 14:18:43 +00:00
{
2017-09-12 04:33:18 +00:00
if ( mAmbientSound )
2015-11-30 13:42:51 +00:00
MWBase : : Environment : : get ( ) . getSoundManager ( ) - > stopSound ( mAmbientSound ) ;
2017-09-12 04:33:18 +00:00
mAmbientSound = nullptr ;
2015-11-25 08:09:40 +00:00
mPlayingSoundID . clear ( ) ;
2013-06-19 14:18:43 +00:00
}
2015-08-27 03:59:21 +00:00
float WeatherManager : : getWindSpeed ( ) const
2013-06-19 14:18:43 +00:00
{
2015-08-27 03:59:21 +00:00
return mWindSpeed ;
2012-02-21 21:56:34 +00:00
}
2012-02-22 19:12:08 +00:00
2015-08-27 03:59:21 +00:00
bool WeatherManager : : isInStorm ( ) const
2012-02-22 19:12:08 +00:00
{
2015-08-27 03:59:21 +00:00
return mIsStorm ;
2012-02-22 19:12:08 +00:00
}
2015-08-27 03:59:21 +00:00
osg : : Vec3f WeatherManager : : getStormDirection ( ) const
2012-02-25 20:34:38 +00:00
{
2015-08-27 03:59:21 +00:00
return mStormDirection ;
}
2013-04-28 03:29:34 +00:00
2015-08-27 19:20:45 +00:00
void WeatherManager : : advanceTime ( double hours , bool incremental )
2015-08-27 03:59:21 +00:00
{
2015-08-27 19:20:45 +00:00
// In Morrowind, when the player sleeps/waits, serves jail time, travels, or trains, all weather transitions are
// immediately applied, regardless of whatever transition time might have been remaining.
2015-08-27 03:59:21 +00:00
mTimePassed + = hours ;
2015-08-28 23:04:22 +00:00
mFastForward = ! incremental ? true : mFastForward ;
2013-07-27 14:10:18 +00:00
}
2015-08-27 03:59:21 +00:00
unsigned int WeatherManager : : getWeatherID ( ) const
2013-05-01 09:42:24 +00:00
{
2015-08-27 03:59:21 +00:00
return mCurrentWeather ;
2013-05-01 09:42:24 +00:00
}
2013-12-10 22:48:49 +00:00
2013-12-30 16:53:02 +00:00
bool WeatherManager : : isDark ( ) const
2013-12-10 22:48:49 +00:00
{
2015-08-27 03:59:21 +00:00
TimeStamp time = MWBase : : Environment : : get ( ) . getWorld ( ) - > getTimeStamp ( ) ;
2013-12-30 16:53:02 +00:00
bool exterior = ( MWBase : : Environment : : get ( ) . getWorld ( ) - > isCellExterior ( )
| | MWBase : : Environment : : get ( ) . getWorld ( ) - > isCellQuasiExterior ( ) ) ;
2015-11-01 21:47:40 +00:00
return exterior & & ( time . getHour ( ) < mSunriseTime | | time . getHour ( ) > mTimeSettings . mNightStart - 1 ) ;
2013-12-10 22:48:49 +00:00
}
2014-01-01 13:57:14 +00:00
2014-04-28 09:29:57 +00:00
void WeatherManager : : write ( ESM : : ESMWriter & writer , Loading : : Listener & progress )
2014-03-20 06:25:52 +00:00
{
2014-03-21 06:19:40 +00:00
ESM : : WeatherState state ;
state . mCurrentRegion = mCurrentRegion ;
state . mTimePassed = mTimePassed ;
2015-08-27 03:59:21 +00:00
state . mFastForward = mFastForward ;
state . mWeatherUpdateTime = mWeatherUpdateTime ;
state . mTransitionFactor = mTransitionFactor ;
state . mCurrentWeather = mCurrentWeather ;
state . mNextWeather = mNextWeather ;
state . mQueuedWeather = mQueuedWeather ;
std : : map < std : : string , RegionWeather > : : iterator it = mRegions . begin ( ) ;
for ( ; it ! = mRegions . end ( ) ; + + it )
{
2015-08-27 04:34:15 +00:00
state . mRegions . insert ( std : : make_pair ( it - > first , it - > second ) ) ;
2015-08-27 03:59:21 +00:00
}
2014-03-21 06:19:40 +00:00
2014-03-20 06:25:52 +00:00
writer . startRecord ( ESM : : REC_WTHR ) ;
2014-03-21 06:19:40 +00:00
state . save ( writer ) ;
2014-03-20 06:25:52 +00:00
writer . endRecord ( ESM : : REC_WTHR ) ;
}
2015-01-22 18:04:59 +00:00
bool WeatherManager : : readRecord ( ESM : : ESMReader & reader , uint32_t type )
2014-03-20 06:25:52 +00:00
{
if ( ESM : : REC_WTHR = = type )
{
2015-08-27 15:34:35 +00:00
static const int oldestCompatibleSaveFormat = 2 ;
if ( reader . getFormat ( ) < oldestCompatibleSaveFormat )
2015-08-27 03:59:21 +00:00
{
2015-08-27 14:57:32 +00:00
// Weather state isn't really all that important, so to preserve older save games, we'll just discard the
// older weather records, rather than fail to handle the record.
reader . skipRecord ( ) ;
2015-08-27 03:59:21 +00:00
}
else
{
2015-08-27 14:57:32 +00:00
ESM : : WeatherState state ;
state . load ( reader ) ;
mCurrentRegion . swap ( state . mCurrentRegion ) ;
mTimePassed = state . mTimePassed ;
mFastForward = state . mFastForward ;
mWeatherUpdateTime = state . mWeatherUpdateTime ;
mTransitionFactor = state . mTransitionFactor ;
mCurrentWeather = state . mCurrentWeather ;
2015-08-29 04:12:39 +00:00
mNextWeather = state . mNextWeather ;
2015-08-27 14:57:32 +00:00
mQueuedWeather = state . mQueuedWeather ;
mRegions . clear ( ) ;
2016-10-15 15:34:03 +00:00
importRegions ( ) ;
for ( std : : map < std : : string , ESM : : RegionWeatherState > : : iterator it = state . mRegions . begin ( ) ; it ! = state . mRegions . end ( ) ; + + it )
2015-08-27 14:57:32 +00:00
{
2016-10-15 15:34:03 +00:00
std : : map < std : : string , RegionWeather > : : iterator found = mRegions . find ( it - > first ) ;
if ( found ! = mRegions . end ( ) )
2015-08-27 14:57:32 +00:00
{
2016-10-15 15:34:03 +00:00
found - > second = RegionWeather ( it - > second ) ;
2015-08-27 14:57:32 +00:00
}
2015-08-27 03:59:21 +00:00
}
}
2014-03-20 06:25:52 +00:00
return true ;
}
return false ;
}
2015-01-24 15:26:43 +00:00
void WeatherManager : : clear ( )
{
stopSounds ( ) ;
2015-08-27 03:59:21 +00:00
mCurrentRegion = " " ;
mTimePassed = 0.0f ;
mWeatherUpdateTime = 0.0f ;
forceWeather ( 0 ) ;
mRegions . clear ( ) ;
importRegions ( ) ;
2015-01-24 15:26:43 +00:00
}
2015-08-27 03:59:21 +00:00
inline void WeatherManager : : addWeather ( const std : : string & name ,
2016-01-06 11:46:06 +00:00
const Fallback : : Map & fallback ,
2017-09-21 10:08:45 +00:00
float dlFactor , float dlOffset ,
2015-08-27 03:59:21 +00:00
const std : : string & particleEffect )
2013-12-29 11:47:44 +00:00
{
2015-08-27 03:59:21 +00:00
static const float fStromWindSpeed = mStore . get < ESM : : GameSetting > ( ) . find ( " fStromWindSpeed " ) - > getFloat ( ) ;
2017-09-21 10:08:45 +00:00
Weather weather ( name , fallback , fStromWindSpeed , mRainSpeed , dlFactor , dlOffset , particleEffect ) ;
2015-08-27 03:59:21 +00:00
mWeatherSettings . push_back ( weather ) ;
}
inline void WeatherManager : : importRegions ( )
{
Store < ESM : : Region > : : iterator it = mStore . get < ESM : : Region > ( ) . begin ( ) ;
for ( ; it ! = mStore . get < ESM : : Region > ( ) . end ( ) ; + + it )
{
std : : string regionID = Misc : : StringUtils : : lowerCase ( it - > mId ) ;
mRegions . insert ( std : : make_pair ( regionID , RegionWeather ( * it ) ) ) ;
}
}
inline void WeatherManager : : regionalWeatherChanged ( const std : : string & regionID , RegionWeather & region )
{
// If the region is current, then add a weather transition for it.
2015-12-19 14:50:13 +00:00
MWWorld : : ConstPtr player = MWMechanics : : getPlayer ( ) ;
2015-08-27 03:59:21 +00:00
if ( player . isInCell ( ) )
2013-12-31 19:40:23 +00:00
{
2016-05-16 00:05:02 +00:00
if ( Misc : : StringUtils : : ciEqual ( regionID , mCurrentRegion ) )
2015-08-27 03:59:21 +00:00
{
addWeatherTransition ( region . getWeather ( ) ) ;
}
}
}
2013-12-31 19:40:23 +00:00
2015-08-27 03:59:21 +00:00
inline bool WeatherManager : : updateWeatherTime ( )
{
mWeatherUpdateTime - = mTimePassed ;
mTimePassed = 0.0f ;
if ( mWeatherUpdateTime < = 0.0f )
{
// Expire all regional weather, so that any call to getWeather() will return a new weather ID.
std : : map < std : : string , RegionWeather > : : iterator it = mRegions . begin ( ) ;
for ( ; it ! = mRegions . end ( ) ; + + it )
2013-12-31 19:40:23 +00:00
{
2015-08-27 03:59:21 +00:00
it - > second . setWeather ( invalidWeatherID ) ;
}
mWeatherUpdateTime + = mHoursBetweenWeatherChanges ;
return true ;
}
return false ;
}
inline bool WeatherManager : : updateWeatherRegion ( const std : : string & playerRegion )
{
if ( ! playerRegion . empty ( ) & & playerRegion ! = mCurrentRegion )
{
mCurrentRegion = playerRegion ;
return true ;
}
2014-01-01 15:45:39 +00:00
2015-08-27 03:59:21 +00:00
return false ;
}
inline void WeatherManager : : updateWeatherTransitions ( const float elapsedRealSeconds )
{
// When a player chooses to train, wait, or serves jail time, any transitions will be fast forwarded to the last
// weather type set, regardless of the remaining transition time.
if ( ! mFastForward & & inTransition ( ) )
{
const float delta = mWeatherSettings [ mNextWeather ] . transitionDelta ( ) ;
mTransitionFactor - = elapsedRealSeconds * delta ;
if ( mTransitionFactor < = 0.0f )
{
mCurrentWeather = mNextWeather ;
mNextWeather = mQueuedWeather ;
mQueuedWeather = invalidWeatherID ;
2013-12-31 19:40:23 +00:00
2015-08-27 03:59:21 +00:00
// We may have begun processing the queued transition, so we need to apply the remaining time towards it.
if ( inTransition ( ) )
2013-12-31 19:40:23 +00:00
{
2015-08-27 03:59:21 +00:00
const float newDelta = mWeatherSettings [ mNextWeather ] . transitionDelta ( ) ;
const float remainingSeconds = - ( mTransitionFactor / delta ) ;
mTransitionFactor = 1.0f - ( remainingSeconds * newDelta ) ;
2014-01-01 15:45:39 +00:00
}
else
{
2015-08-27 03:59:21 +00:00
mTransitionFactor = 0.0f ;
2013-12-31 19:40:23 +00:00
}
2014-01-01 15:45:39 +00:00
}
2013-12-31 19:40:23 +00:00
}
2015-08-27 03:59:21 +00:00
else
{
if ( mQueuedWeather ! = invalidWeatherID )
{
mCurrentWeather = mQueuedWeather ;
}
else if ( mNextWeather ! = invalidWeatherID )
{
mCurrentWeather = mNextWeather ;
}
2015-08-29 04:12:39 +00:00
mNextWeather = invalidWeatherID ;
mQueuedWeather = invalidWeatherID ;
2015-08-27 03:59:21 +00:00
mFastForward = false ;
}
2013-12-29 11:47:44 +00:00
}
2014-06-24 16:37:38 +00:00
2015-08-27 03:59:21 +00:00
inline void WeatherManager : : forceWeather ( const int weatherID )
2014-06-24 16:37:38 +00:00
{
2015-08-27 03:59:21 +00:00
mTransitionFactor = 0.0f ;
mCurrentWeather = weatherID ;
mNextWeather = invalidWeatherID ;
mQueuedWeather = invalidWeatherID ;
2014-06-24 16:37:38 +00:00
}
2015-08-27 03:59:21 +00:00
inline bool WeatherManager : : inTransition ( )
2014-06-24 16:37:38 +00:00
{
2015-08-27 03:59:21 +00:00
return mNextWeather ! = invalidWeatherID ;
2014-06-24 16:37:38 +00:00
}
2015-04-14 13:55:56 +00:00
2015-08-27 03:59:21 +00:00
inline void WeatherManager : : addWeatherTransition ( const int weatherID )
2015-04-14 13:55:56 +00:00
{
2015-08-27 03:59:21 +00:00
// In order to work like ChangeWeather expects, this method begins transitioning to the new weather immediately if
// no transition is in progress, otherwise it queues it to be transitioned.
assert ( weatherID > = 0 & & static_cast < size_t > ( weatherID ) < mWeatherSettings . size ( ) ) ;
if ( ! inTransition ( ) & & ( weatherID ! = mCurrentWeather ) )
{
mNextWeather = weatherID ;
mTransitionFactor = 1.0f ;
}
else if ( inTransition ( ) & & ( weatherID ! = mNextWeather ) )
{
mQueuedWeather = weatherID ;
}
2015-04-14 13:55:56 +00:00
}
2015-08-16 04:38:49 +00:00
2015-09-09 03:05:33 +00:00
inline void WeatherManager : : calculateWeatherResult ( const float gameHour ,
const float elapsedSeconds ,
const bool isPaused )
2015-08-16 04:38:49 +00:00
{
2015-09-09 03:05:33 +00:00
float flash = 0.0f ;
2015-08-27 03:59:21 +00:00
if ( ! inTransition ( ) )
{
calculateResult ( mCurrentWeather , gameHour ) ;
2015-09-09 03:05:33 +00:00
flash = mWeatherSettings [ mCurrentWeather ] . calculateThunder ( 1.0f , elapsedSeconds , isPaused ) ;
2015-08-27 03:59:21 +00:00
}
else
{
calculateTransitionResult ( 1 - mTransitionFactor , gameHour ) ;
2015-09-09 03:05:33 +00:00
float currentFlash = mWeatherSettings [ mCurrentWeather ] . calculateThunder ( mTransitionFactor ,
elapsedSeconds ,
isPaused ) ;
float nextFlash = mWeatherSettings [ mNextWeather ] . calculateThunder ( 1 - mTransitionFactor ,
elapsedSeconds ,
isPaused ) ;
flash = currentFlash + nextFlash ;
}
osg : : Vec4f flashColor ( flash , flash , flash , 0.0f ) ;
mResult . mFogColor + = flashColor ;
mResult . mAmbientColor + = flashColor ;
mResult . mSunColor + = flashColor ;
2015-08-27 03:59:21 +00:00
}
2015-08-16 04:38:49 +00:00
2015-08-27 03:59:21 +00:00
inline void WeatherManager : : calculateResult ( const int weatherID , const float gameHour )
{
const Weather & current = mWeatherSettings [ weatherID ] ;
mResult . mCloudTexture = current . mCloudTexture ;
mResult . mCloudBlendFactor = 0 ;
mResult . mWindSpeed = current . mWindSpeed ;
mResult . mCloudSpeed = current . mCloudSpeed ;
mResult . mGlareView = current . mGlareView ;
mResult . mAmbientLoopSoundID = current . mAmbientLoopSoundID ;
mResult . mAmbientSoundVolume = 1.f ;
mResult . mEffectFade = 1.f ;
mResult . mIsStorm = current . mIsStorm ;
2015-08-16 04:38:49 +00:00
2015-08-27 03:59:21 +00:00
mResult . mRainSpeed = current . mRainSpeed ;
mResult . mRainFrequency = current . mRainFrequency ;
mResult . mParticleEffect = current . mParticleEffect ;
mResult . mRainEffect = current . mRainEffect ;
2015-11-01 21:47:40 +00:00
mResult . mNight = ( gameHour < mSunriseTime | | gameHour > mTimeSettings . mNightStart - 1 ) ;
2015-08-27 03:59:21 +00:00
2015-11-01 22:03:16 +00:00
mResult . mFogDepth = current . mLandFogDepth . getValue ( gameHour , mTimeSettings ) ;
2017-09-21 10:08:45 +00:00
mResult . mDLFogFactor = current . mDL . FogFactor ;
mResult . mDLFogOffset = current . mDL . FogOffset ;
2015-11-01 21:47:40 +00:00
mResult . mFogColor = current . mFogColor . getValue ( gameHour , mTimeSettings ) ;
mResult . mAmbientColor = current . mAmbientColor . getValue ( gameHour , mTimeSettings ) ;
mResult . mSunColor = current . mSunColor . getValue ( gameHour , mTimeSettings ) ;
mResult . mSkyColor = current . mSkyColor . getValue ( gameHour , mTimeSettings ) ;
mResult . mNightFade = mNightFade . getValue ( gameHour , mTimeSettings ) ;
2015-09-21 17:43:48 +00:00
if ( gameHour > = mSunsetTime - mSunPreSunsetTime )
{
float factor = ( gameHour - ( mSunsetTime - mSunPreSunsetTime ) ) / mSunPreSunsetTime ;
factor = std : : min ( 1.f , factor ) ;
mResult . mSunDiscColor = lerp ( osg : : Vec4f ( 1 , 1 , 1 , 1 ) , current . mSunDiscSunsetColor , factor ) ;
// The SunDiscSunsetColor in the INI isn't exactly the resulting color on screen, most likely because
// MW applied the color to the ambient term as well. After the ambient and emissive terms are added together, the fixed pipeline
2016-12-14 15:39:33 +00:00
// would then clamp the total lighting to (1,1,1). A noticeable change in color tone can be observed when only one of the color components gets clamped.
2015-09-21 17:43:48 +00:00
// Unfortunately that means we can't use the INI color as is, have to replicate the above nonsense.
mResult . mSunDiscColor = mResult . mSunDiscColor + osg : : componentMultiply ( mResult . mSunDiscColor , mResult . mAmbientColor ) ;
for ( int i = 0 ; i < 3 ; + + i )
mResult . mSunDiscColor [ i ] = std : : min ( 1.f , mResult . mSunDiscColor [ i ] ) ;
}
else
mResult . mSunDiscColor = osg : : Vec4f ( 1 , 1 , 1 , 1 ) ;
if ( gameHour > = mSunsetTime )
{
float fade = std : : min ( 1.f , ( gameHour - mSunsetTime ) / 2.f ) ;
fade = fade * fade ;
mResult . mSunDiscColor . a ( ) = 1.f - fade ;
}
else if ( gameHour > = mSunriseTime & & gameHour < = mSunriseTime + 1 )
{
mResult . mSunDiscColor . a ( ) = gameHour - mSunriseTime ;
}
else
mResult . mSunDiscColor . a ( ) = 1 ;
2015-08-16 04:38:49 +00:00
}
2015-08-27 03:59:21 +00:00
inline void WeatherManager : : calculateTransitionResult ( const float factor , const float gameHour )
2015-08-16 04:38:49 +00:00
{
2015-08-27 03:59:21 +00:00
calculateResult ( mCurrentWeather , gameHour ) ;
const MWRender : : WeatherResult current = mResult ;
calculateResult ( mNextWeather , gameHour ) ;
const MWRender : : WeatherResult other = mResult ;
mResult . mCloudTexture = current . mCloudTexture ;
mResult . mNextCloudTexture = other . mCloudTexture ;
mResult . mCloudBlendFactor = mWeatherSettings [ mNextWeather ] . cloudBlendFactor ( factor ) ;
mResult . mFogColor = lerp ( current . mFogColor , other . mFogColor , factor ) ;
mResult . mSunColor = lerp ( current . mSunColor , other . mSunColor , factor ) ;
mResult . mSkyColor = lerp ( current . mSkyColor , other . mSkyColor , factor ) ;
mResult . mAmbientColor = lerp ( current . mAmbientColor , other . mAmbientColor , factor ) ;
mResult . mSunDiscColor = lerp ( current . mSunDiscColor , other . mSunDiscColor , factor ) ;
mResult . mFogDepth = lerp ( current . mFogDepth , other . mFogDepth , factor ) ;
2017-09-21 10:08:45 +00:00
mResult . mDLFogFactor = lerp ( current . mDLFogFactor , other . mDLFogFactor , factor ) ;
mResult . mDLFogOffset = lerp ( current . mDLFogOffset , other . mDLFogOffset , factor ) ;
2015-08-27 03:59:21 +00:00
mResult . mWindSpeed = lerp ( current . mWindSpeed , other . mWindSpeed , factor ) ;
mResult . mCloudSpeed = lerp ( current . mCloudSpeed , other . mCloudSpeed , factor ) ;
mResult . mGlareView = lerp ( current . mGlareView , other . mGlareView , factor ) ;
mResult . mNightFade = lerp ( current . mNightFade , other . mNightFade , factor ) ;
mResult . mNight = current . mNight ;
if ( factor < 0.5 )
{
mResult . mIsStorm = current . mIsStorm ;
mResult . mParticleEffect = current . mParticleEffect ;
mResult . mRainEffect = current . mRainEffect ;
mResult . mParticleEffect = current . mParticleEffect ;
mResult . mRainSpeed = current . mRainSpeed ;
mResult . mRainFrequency = current . mRainFrequency ;
mResult . mAmbientSoundVolume = 1 - ( factor * 2 ) ;
mResult . mEffectFade = mResult . mAmbientSoundVolume ;
mResult . mAmbientLoopSoundID = current . mAmbientLoopSoundID ;
}
else
{
mResult . mIsStorm = other . mIsStorm ;
mResult . mParticleEffect = other . mParticleEffect ;
mResult . mRainEffect = other . mRainEffect ;
mResult . mParticleEffect = other . mParticleEffect ;
mResult . mRainSpeed = other . mRainSpeed ;
mResult . mRainFrequency = other . mRainFrequency ;
mResult . mAmbientSoundVolume = 2 * ( factor - 0.5f ) ;
mResult . mEffectFade = mResult . mAmbientSoundVolume ;
mResult . mAmbientLoopSoundID = other . mAmbientLoopSoundID ;
}
2015-08-16 04:38:49 +00:00
}