@ -31,17 +31,74 @@ namespace
{
{
static const int invalidWeatherID = - 1 ;
static const int invalidWeatherID = - 1 ;
// linear interpolate between x and y based on factor.
float lerp ( float x , float y , float factor )
float lerp ( float x , float y , float factor )
{
{
return x * ( 1 - factor ) + y * factor ;
return x * ( 1 - factor ) + y * factor ;
}
}
// linear interpolate between x and y based on factor.
osg : : Vec4f lerp ( const osg : : Vec4f & x , const osg : : Vec4f & y , float factor )
osg : : Vec4f lerp ( const osg : : Vec4f & x , const osg : : Vec4f & y , float factor )
{
{
return x * ( 1 - factor ) + y * factor ;
return x * ( 1 - factor ) + y * factor ;
}
}
}
}
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 ( ) ;
}
template class TimeOfDayInterpolator < float > ;
template class TimeOfDayInterpolator < osg : : Vec4f > ;
Weather : : Weather ( const std : : string & name ,
Weather : : Weather ( const std : : string & name ,
const MWWorld : : Fallback & fallback ,
const MWWorld : : Fallback & fallback ,
float stormWindSpeed ,
float stormWindSpeed ,
@ -49,24 +106,26 @@ Weather::Weather(const std::string& name,
const std : : string & ambientLoopSoundID ,
const std : : string & ambientLoopSoundID ,
const std : : string & particleEffect )
const std : : string & particleEffect )
: mCloudTexture ( fallback . getFallbackString ( " Weather_ " + name + " _Cloud_Texture " ) )
: mCloudTexture ( fallback . getFallbackString ( " Weather_ " + name + " _Cloud_Texture " ) )
, mSkySunriseColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Sunrise_Color " ) )
, mSkyColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Sunrise_Color " ) ,
, mSkyDayColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Day_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Day_Color " ) ,
, mSkySunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Sunset_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Sunset_Color " ) ,
, mSkyNightColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Night_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Sky_Night_Color " ) )
, mFogSunriseColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Sunrise_Color " ) )
, mFogColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Sunrise_Color " ) ,
, mFogDayColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Day_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Day_Color " ) ,
, mFogSunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Sunset_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Sunset_Color " ) ,
, mFogNightColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Night_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Fog_Night_Color " ) )
, mAmbientSunriseColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Sunrise_Color " ) )
, mAmbientColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Sunrise_Color " ) ,
, mAmbientDayColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Day_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Day_Color " ) ,
, mAmbientSunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Sunset_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Sunset_Color " ) ,
, mAmbientNightColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Night_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Ambient_Night_Color " ) )
, mSunSunriseColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Sunrise_Color " ) )
, mSunColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Sunrise_Color " ) ,
, mSunDayColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Day_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Day_Color " ) ,
, mSunSunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Sunset_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Sunset_Color " ) ,
, mSunNightColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Night_Color " ) )
fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Night_Color " ) )
, mLandFogDayDepth ( fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Day_Depth " ) )
, mLandFogDepth ( fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Day_Depth " ) ,
, mLandFogNightDepth ( fallback . getFallbackFloat ( " Weather_ " + name + " _Land_Fog_Night_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 " ) )
, mSunDiscSunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Disc_Sunset_Color " ) )
, mSunDiscSunsetColor ( fallback . getFallbackColour ( " Weather_ " + name + " _Sun_Disc_Sunset_Color " ) )
, mWindSpeed ( fallback . getFallbackFloat ( " Weather_ " + name + " _Wind_Speed " ) )
, mWindSpeed ( fallback . getFallbackFloat ( " Weather_ " + name + " _Wind_Speed " ) )
, mCloudSpeed ( fallback . getFallbackFloat ( " Weather_ " + name + " _Cloud_Speed " ) )
, mCloudSpeed ( fallback . getFallbackFloat ( " Weather_ " + name + " _Cloud_Speed " ) )
@ -432,12 +491,13 @@ WeatherManager::WeatherManager(MWRender::RenderingManager& rendering, const MWWo
, mSunriseDuration ( fallback . getFallbackFloat ( " Weather_Sunrise_Duration " ) )
, mSunriseDuration ( fallback . getFallbackFloat ( " Weather_Sunrise_Duration " ) )
, mSunsetDuration ( fallback . getFallbackFloat ( " Weather_Sunset_Duration " ) )
, mSunsetDuration ( fallback . getFallbackFloat ( " Weather_Sunset_Duration " ) )
, mSunPreSunsetTime ( fallback . getFallbackFloat ( " Weather_Sun_Pre-Sunset_Time " ) )
, mSunPreSunsetTime ( fallback . getFallbackFloat ( " Weather_Sun_Pre-Sunset_Time " ) )
, mNightStart ( mSunsetTime + mSunsetDuration )
, mNightFade ( 0 , 0 , 0 , 1 )
, mNightEnd ( mSunriseTime - 0.5f )
, mDayStart ( mSunriseTime + mSunriseDuration )
, mDayEnd ( mSunsetTime )
, mHoursBetweenWeatherChanges ( fallback . getFallbackFloat ( " Weather_Hours_Between_Weather_Changes " ) )
, mHoursBetweenWeatherChanges ( fallback . getFallbackFloat ( " Weather_Hours_Between_Weather_Changes " ) )
, mRainSpeed ( fallback . getFallbackFloat ( " Weather_Precip_Gravity " ) )
, mRainSpeed ( fallback . getFallbackFloat ( " Weather_Precip_Gravity " ) )
, mUnderwaterFog ( fallback . getFallbackFloat ( " Water_UnderwaterSunriseFog " ) ,
fallback . getFallbackFloat ( " Water_UnderwaterDayFog " ) ,
fallback . getFallbackFloat ( " Water_UnderwaterSunsetFog " ) ,
fallback . getFallbackFloat ( " Water_UnderwaterNightFog " ) )
, mWeatherSettings ( )
, mWeatherSettings ( )
, mMasser ( " Masser " , fallback )
, mMasser ( " Masser " , fallback )
, mSecunda ( " Secunda " , fallback )
, mSecunda ( " Secunda " , fallback )
@ -457,6 +517,12 @@ WeatherManager::WeatherManager(MWRender::RenderingManager& rendering, const MWWo
, mAmbientSound ( )
, mAmbientSound ( )
, mPlayingSoundID ( )
, mPlayingSoundID ( )
{
{
mTimeSettings . mNightStart = mSunsetTime + mSunsetDuration ;
mTimeSettings . mNightEnd = mSunriseTime - 0.5f ;
mTimeSettings . mDayStart = mSunriseTime + mSunriseDuration ;
mTimeSettings . mDayEnd = mSunsetTime ;
mTimeSettings . mSunriseTime = mSunriseTime ;
mWeatherSettings . reserve ( 10 ) ;
mWeatherSettings . reserve ( 10 ) ;
addWeather ( " Clear " , fallback ) ; // 0
addWeather ( " Clear " , fallback ) ; // 0
addWeather ( " Cloudy " , fallback ) ; // 1
addWeather ( " Cloudy " , fallback ) ; // 1
@ -589,7 +655,7 @@ void WeatherManager::update(float duration, bool paused)
}
}
// disable sun during night
// disable sun during night
if ( time . getHour ( ) > = m NightStart | | time . getHour ( ) < = mSunriseTime )
if ( time . getHour ( ) > = m TimeSettings. m NightStart | | time . getHour ( ) < = mSunriseTime )
mRendering . getSkyManager ( ) - > sunDisable ( ) ;
mRendering . getSkyManager ( ) - > sunDisable ( ) ;
else
else
mRendering . getSkyManager ( ) - > sunEnable ( ) ;
mRendering . getSkyManager ( ) - > sunEnable ( ) ;
@ -600,10 +666,10 @@ void WeatherManager::update(float duration, bool paused)
{
{
// Shift times into a 24-hour window beginning at mSunriseTime...
// Shift times into a 24-hour window beginning at mSunriseTime...
float adjustedHour = time . getHour ( ) ;
float adjustedHour = time . getHour ( ) ;
float adjustedNightStart = m NightStart;
float adjustedNightStart = m TimeSettings. m NightStart;
if ( time . getHour ( ) < mSunriseTime )
if ( time . getHour ( ) < mSunriseTime )
adjustedHour + = 24.f ;
adjustedHour + = 24.f ;
if ( m NightStart < mSunriseTime )
if ( m TimeSettings. m NightStart < mSunriseTime )
adjustedNightStart + = 24.f ;
adjustedNightStart + = 24.f ;
const bool is_night = adjustedHour > = adjustedNightStart ;
const bool is_night = adjustedHour > = adjustedNightStart ;
@ -624,6 +690,8 @@ void WeatherManager::update(float duration, bool paused)
mRendering . setSunDirection ( final * - 1 ) ;
mRendering . setSunDirection ( final * - 1 ) ;
}
}
float underwaterFog = mUnderwaterFog . getValue ( time . getHour ( ) , mTimeSettings ) ;
float peakHour = mSunriseTime + ( mSunsetTime - mSunriseTime ) / 2 ;
float peakHour = mSunriseTime + ( mSunsetTime - mSunriseTime ) / 2 ;
if ( time . getHour ( ) < mSunriseTime | | time . getHour ( ) > mSunsetTime )
if ( time . getHour ( ) < mSunriseTime | | time . getHour ( ) > mSunsetTime )
mRendering . getSkyManager ( ) - > setGlareTimeOfDayFade ( 0 ) ;
mRendering . getSkyManager ( ) - > setGlareTimeOfDayFade ( 0 ) ;
@ -635,7 +703,7 @@ void WeatherManager::update(float duration, bool paused)
mRendering . getSkyManager ( ) - > setMasserState ( mMasser . calculateState ( time ) ) ;
mRendering . getSkyManager ( ) - > setMasserState ( mMasser . calculateState ( time ) ) ;
mRendering . getSkyManager ( ) - > setSecundaState ( mSecunda . calculateState ( time ) ) ;
mRendering . getSkyManager ( ) - > setSecundaState ( mSecunda . calculateState ( time ) ) ;
mRendering . configureFog ( mResult . mFogDepth , mResult. mFogColor ) ;
mRendering . configureFog ( mResult . mFogDepth , underwaterFog, mResult. mFogColor ) ;
mRendering . setAmbientColour ( mResult . mAmbientColor ) ;
mRendering . setAmbientColour ( mResult . mAmbientColor ) ;
mRendering . setSunColour ( mResult . mSunColor ) ;
mRendering . setSunColour ( mResult . mSunColor ) ;
@ -697,7 +765,7 @@ bool WeatherManager::isDark() const
TimeStamp time = MWBase : : Environment : : get ( ) . getWorld ( ) - > getTimeStamp ( ) ;
TimeStamp time = MWBase : : Environment : : get ( ) . getWorld ( ) - > getTimeStamp ( ) ;
bool exterior = ( MWBase : : Environment : : get ( ) . getWorld ( ) - > isCellExterior ( )
bool exterior = ( MWBase : : Environment : : get ( ) . getWorld ( ) - > isCellExterior ( )
| | MWBase : : Environment : : get ( ) . getWorld ( ) - > isCellQuasiExterior ( ) ) ;
| | MWBase : : Environment : : get ( ) . getWorld ( ) - > isCellQuasiExterior ( ) ) ;
return exterior & & ( time . getHour ( ) < mSunriseTime | | time . getHour ( ) > m NightStart - 1 ) ;
return exterior & & ( time . getHour ( ) < mSunriseTime | | time . getHour ( ) > m TimeSettings. m NightStart - 1 ) ;
}
}
void WeatherManager : : write ( ESM : : ESMWriter & writer , Loading : : Listener & progress )
void WeatherManager : : write ( ESM : : ESMWriter & writer , Loading : : Listener & progress )
@ -975,81 +1043,14 @@ inline void WeatherManager::calculateResult(const int weatherID, const float gam
mResult . mParticleEffect = current . mParticleEffect ;
mResult . mParticleEffect = current . mParticleEffect ;
mResult . mRainEffect = current . mRainEffect ;
mResult . mRainEffect = current . mRainEffect ;
mResult . mNight = ( gameHour < mSunriseTime | | gameHour > mNightStart - 1 ) ;
mResult . mNight = ( gameHour < mSunriseTime | | gameHour > mTimeSettings . mNightStart - 1 ) ;
mResult . mFogDepth = mResult . mNight ? current . mLandFogNightDepth : current . mLandFogDayDepth ;
// TODO: use pre/post sunset/sunrise time values in [Weather] section
mResult . mFogDepth = current . mLandFogDepth . getValue ( gameHour , mTimeSettings ) ;
// night
mResult . mFogColor = current . mFogColor . getValue ( gameHour , mTimeSettings ) ;
if ( gameHour < = mNightEnd | | gameHour > = mNightStart + 1 )
mResult . mAmbientColor = current . mAmbientColor . getValue ( gameHour , mTimeSettings ) ;
{
mResult . mSunColor = current . mSunColor . getValue ( gameHour , mTimeSettings ) ;
mResult . mFogColor = current . mFogNightColor ;
mResult . mSkyColor = current . mSkyColor . getValue ( gameHour , mTimeSettings ) ;
mResult . mAmbientColor = current . mAmbientNightColor ;
mResult . mNightFade = mNightFade . getValue ( gameHour , mTimeSettings ) ;
mResult . mSunColor = current . mSunNightColor ;
mResult . mSkyColor = current . mSkyNightColor ;
mResult . mNightFade = 1.f ;
}
// sunrise
else if ( gameHour > = mNightEnd & & gameHour < = mDayStart + 1 )
{
if ( gameHour < = mSunriseTime )
{
// fade in
float advance = mSunriseTime - gameHour ;
float factor = advance / 0.5f ;
mResult . mFogColor = lerp ( current . mFogSunriseColor , current . mFogNightColor , factor ) ;
mResult . mAmbientColor = lerp ( current . mAmbientSunriseColor , current . mAmbientNightColor , factor ) ;
mResult . mSunColor = lerp ( current . mSunSunriseColor , current . mSunNightColor , factor ) ;
mResult . mSkyColor = lerp ( current . mSkySunriseColor , current . mSkyNightColor , factor ) ;
mResult . mNightFade = factor ;
}
else //if (gameHour >= 6)
{
// fade out
float advance = gameHour - mSunriseTime ;
float factor = advance / 3.f ;
mResult . mFogColor = lerp ( current . mFogSunriseColor , current . mFogDayColor , factor ) ;
mResult . mAmbientColor = lerp ( current . mAmbientSunriseColor , current . mAmbientDayColor , factor ) ;
mResult . mSunColor = lerp ( current . mSunSunriseColor , current . mSunDayColor , factor ) ;
mResult . mSkyColor = lerp ( current . mSkySunriseColor , current . mSkyDayColor , factor ) ;
}
}
// day
else if ( gameHour > = mDayStart + 1 & & gameHour < = mDayEnd - 1 )
{
mResult . mFogColor = current . mFogDayColor ;
mResult . mAmbientColor = current . mAmbientDayColor ;
mResult . mSunColor = current . mSunDayColor ;
mResult . mSkyColor = current . mSkyDayColor ;
}
// sunset
else if ( gameHour > = mDayEnd - 1 & & gameHour < = mNightStart + 1 )
{
if ( gameHour < = mDayEnd + 1 )
{
// fade in
float advance = ( mDayEnd + 1 ) - gameHour ;
float factor = ( advance / 2 ) ;
mResult . mFogColor = lerp ( current . mFogSunsetColor , current . mFogDayColor , factor ) ;
mResult . mAmbientColor = lerp ( current . mAmbientSunsetColor , current . mAmbientDayColor , factor ) ;
mResult . mSunColor = lerp ( current . mSunSunsetColor , current . mSunDayColor , factor ) ;
mResult . mSkyColor = lerp ( current . mSkySunsetColor , current . mSkyDayColor , factor ) ;
}
else //if (gameHour >= 19)
{
// fade out
float advance = gameHour - ( mDayEnd + 1 ) ;
float factor = advance / 2.f ;
mResult . mFogColor = lerp ( current . mFogSunsetColor , current . mFogNightColor , factor ) ;
mResult . mAmbientColor = lerp ( current . mAmbientSunsetColor , current . mAmbientNightColor , factor ) ;
mResult . mSunColor = lerp ( current . mSunSunsetColor , current . mSunNightColor , factor ) ;
mResult . mSkyColor = lerp ( current . mSkySunsetColor , current . mSkyNightColor , factor ) ;
mResult . mNightFade = factor ;
}
}
if ( gameHour > = mSunsetTime - mSunPreSunsetTime )
if ( gameHour > = mSunsetTime - mSunPreSunsetTime )
{
{