openmw-tes3coop/apps/openmw/mwworld/weather.cpp

652 lines
25 KiB
C++
Raw Normal View History

2012-02-21 21:56:34 +00:00
#include "weather.hpp"
2012-02-25 12:46:17 +00:00
#include <ctime>
#include <cstdlib>
2012-02-21 21:56:34 +00:00
#include <boost/algorithm/string.hpp>
#include "../mwbase/environment.hpp"
#include "../mwbase/world.hpp"
#include "../mwbase/soundmanager.hpp"
#include "../mwrender/renderingmanager.hpp"
#include "player.hpp"
2012-10-01 15:17:04 +00:00
#include "esmstore.hpp"
2013-03-15 09:26:04 +00:00
#include "fallback.hpp"
using namespace Ogre;
2012-02-21 21:56:34 +00:00
using namespace MWWorld;
2012-02-25 12:46:17 +00:00
using namespace MWSound;
2012-02-21 21:56:34 +00:00
2013-03-20 00:20:56 +00:00
namespace
{
float lerp (float x, float y, float factor)
{
return x * (1-factor) + y * factor;
}
2013-05-08 22:36:58 +00:00
2013-03-20 00:20:56 +00:00
Ogre::ColourValue lerp (const Ogre::ColourValue& x, const Ogre::ColourValue& y, float factor)
{
return x * (1-factor) + y * factor;
}
}
std::string Weather::weatherTypeToStr(Weather::Type type)
2013-03-09 23:24:14 +00:00
{
switch (type) {
case Type_Clear:
return "Clear";
case Type_Cloudy:
return "Cloudy";
case Type_Foggy:
return "Foggy";
case Type_Overcast:
return "Overcast";
case Type_Rain:
return "Rain";
case Type_Thunderstorm:
return "Thunderstorm";
case Type_Ashstorm:
return "Ashstorm";
case Type_Blight:
return "Blight";
case Type_Snow:
return "Snow";
case Type_Blizzard:
return "Blizzard";
default: // Type_Unknown
return "";
}
}
void WeatherManager::setFallbackWeather(Weather& weather, Weather::Type type)
{
const std::string weatherName = Weather::weatherTypeToStr(type);
weather.mCloudsMaximumPercent = mFallback->getFallbackFloat("Weather_"+weatherName+"_Clouds_Maximum_Percent");
weather.mTransitionDelta = mFallback->getFallbackFloat("Weather_"+weatherName+"_Transition_Delta");
weather.mSkySunriseColor= mFallback->getFallbackColour("Weather_"+weatherName+"_Sky_Sunrise_Color");
weather.mSkyDayColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sky_Day_Color");
weather.mSkySunsetColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sky_Sunset_Color");
weather.mSkyNightColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sky_Night_Color");
weather.mFogSunriseColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Fog_Sunrise_Color");
weather.mFogDayColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Fog_Day_Color");
weather.mFogSunsetColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Fog_Sunset_Color");
weather.mFogNightColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Fog_Night_Color");
weather.mAmbientSunriseColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Ambient_Sunrise_Color");
weather.mAmbientDayColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Ambient_Day_Color");
weather.mAmbientSunsetColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Ambient_Sunset_Color");
weather.mAmbientNightColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Ambient_Night_Color");
weather.mSunSunriseColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sun_Sunrise_Color");
weather.mSunDayColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sun_Day_Color");
weather.mSunSunsetColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sun_Sunset_Color");
weather.mSunNightColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sun_Night_Color");
weather.mSunDiscSunsetColor = mFallback->getFallbackColour("Weather_"+weatherName+"_Sun_Disc_Sunset_Color");
weather.mLandFogDayDepth = mFallback->getFallbackFloat("Weather_"+weatherName+"_Land_Fog_Day_Depth");
weather.mLandFogNightDepth = mFallback->getFallbackFloat("Weather_"+weatherName+"_Land_Fog_Night_Depth");
weather.mWindSpeed = mFallback->getFallbackFloat("Weather_"+weatherName+"_Wind_Speed");
weather.mCloudSpeed = mFallback->getFallbackFloat("Weather_"+weatherName+"_Cloud_Speed");
weather.mGlareView = mFallback->getFallbackFloat("Weather_"+weatherName+"_Glare_View");
mWeatherSettings[type] = weather;
2013-03-09 23:24:14 +00:00
}
2012-03-16 18:49:01 +00:00
2013-05-08 22:36:58 +00:00
2013-05-09 11:56:13 +00:00
float WeatherManager::calculateHourFade (const std::string& moonName) const
2013-05-08 22:36:58 +00:00
{
float fadeOutStart=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_Out_Start");
2013-05-14 17:58:08 +00:00
float fadeOutFinish=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_Out_Finish");
2013-05-08 22:36:58 +00:00
float fadeInStart=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_In_Start");
float fadeInFinish=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_In_Finish");
2013-05-10 16:55:56 +00:00
if (mHour >= fadeOutStart && mHour <= fadeOutFinish)
2013-05-14 17:58:08 +00:00
return (1 - ((mHour - fadeOutStart) / (fadeOutFinish - fadeOutStart)));
if (mHour >= fadeInStart && mHour <= fadeInFinish)
return (1 - ((mHour - fadeInStart) / (fadeInFinish - fadeInStart)));
2013-05-08 22:36:58 +00:00
else
return 1;
}
2013-05-09 11:56:13 +00:00
float WeatherManager::calculateAngleFade (const std::string& moonName, float angle) const
2013-05-08 22:36:58 +00:00
{
float endAngle=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_End_Angle");
float startAngle=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_Start_Angle");
2013-05-14 17:58:08 +00:00
if (angle <= startAngle && angle >= endAngle)
return (1 - ((angle - endAngle)/(startAngle-endAngle)));
else if (angle > startAngle)
2013-05-08 22:36:58 +00:00
return 0.f;
else
return 1.f;
}
WeatherManager::WeatherManager(MWRender::RenderingManager* rendering,MWWorld::Fallback* fallback) :
mHour(14), mCurrentWeather(Weather::Type_Clear), mFirstUpdate(true), mWeatherUpdateTime(0),
2012-06-06 18:29:30 +00:00
mThunderFlash(0), mThunderChance(0), mThunderChanceNeeded(50), mThunderSoundDelay(0),
mRemainingTransitionTime(0), mMonth(0), mDay(0),
2013-05-07 08:27:37 +00:00
mTimePassed(0), mFallback(fallback), mWindSpeed(0.f), mRendering(rendering)
2012-02-21 21:56:34 +00:00
{
2013-03-15 09:32:03 +00:00
//Globals
2013-03-15 09:28:39 +00:00
mThunderSoundID0 = mFallback->getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_0");
mThunderSoundID1 = mFallback->getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_1");
mThunderSoundID2 = mFallback->getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_2");
mThunderSoundID3 = mFallback->getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_3");
mSunriseTime = mFallback->getFallbackFloat("Weather_Sunrise_Time");
mSunsetTime = mFallback->getFallbackFloat("Weather_Sunset_Time");
mSunriseDuration = mFallback->getFallbackFloat("Weather_Sunrise_Duration");
mSunsetDuration = mFallback->getFallbackFloat("Weather_Sunset_Duration");
mHoursBetweenWeatherChanges = mFallback->getFallbackFloat("Weather_Hours_Between_Weather_Changes");
2013-05-07 08:27:37 +00:00
mWeatherUpdateTime = mHoursBetweenWeatherChanges * 3600;
2013-03-15 09:28:39 +00:00
mThunderFrequency = mFallback->getFallbackFloat("Weather_Thunderstorm_Thunder_Frequency");
mThunderThreshold = mFallback->getFallbackFloat("Weather_Thunderstorm_Thunder_Threshold");
mThunderSoundDelay = 0.25;
2013-05-07 08:27:37 +00:00
//Some useful values
/* TODO: Use pre-sunrise_time, pre-sunset_time,
* post-sunrise_time, and post-sunset_time to better
* describe sunrise/sunset time.
* These values are fallbacks attached to weather.
*/
mNightStart = mSunsetTime + mSunsetDuration;
mNightEnd = mSunriseTime - 0.5;
mDayStart = mSunriseTime + mSunriseDuration;
mDayEnd = mSunsetTime;
2013-03-15 09:28:39 +00:00
//Weather
2013-03-15 09:22:02 +00:00
Weather clear;
clear.mCloudTexture = "tx_sky_clear.dds";
setFallbackWeather(clear, Weather::Type_Clear);
Weather cloudy;
2013-03-15 09:22:02 +00:00
cloudy.mCloudTexture = "tx_sky_cloudy.dds";
setFallbackWeather(cloudy, Weather::Type_Cloudy);
Weather foggy;
2013-03-15 09:22:02 +00:00
foggy.mCloudTexture = "tx_sky_foggy.dds";
setFallbackWeather(foggy, Weather::Type_Foggy);
2012-02-25 12:46:17 +00:00
Weather thunderstorm;
thunderstorm.mCloudTexture = "tx_sky_thunder.dds";
thunderstorm.mRainLoopSoundID = "rain heavy";
setFallbackWeather(thunderstorm, Weather::Type_Thunderstorm);
Weather rain;
rain.mCloudTexture = "tx_sky_rainy.dds";
rain.mRainLoopSoundID = "rain";
setFallbackWeather(rain, Weather::Type_Rain);
Weather overcast;
overcast.mCloudTexture = "tx_sky_overcast.dds";
setFallbackWeather(overcast, Weather::Type_Overcast);
Weather ashstorm;
ashstorm.mCloudTexture = "tx_sky_ashstorm.dds";
ashstorm.mAmbientLoopSoundID = "ashstorm";
setFallbackWeather(ashstorm, Weather::Type_Ashstorm);
Weather blight;
blight.mCloudTexture = "tx_sky_blight.dds";
blight.mAmbientLoopSoundID = "blight";
setFallbackWeather(blight, Weather::Type_Blight);
Weather snow;
snow.mCloudTexture = "tx_bm_sky_snow.dds";
setFallbackWeather(snow, Weather::Type_Snow);
Weather blizzard;
blizzard.mCloudTexture = "tx_bm_sky_blizzard.dds";
blizzard.mAmbientLoopSoundID = "BM Blizzard";
setFallbackWeather(blizzard, Weather::Type_Blizzard);
}
void WeatherManager::setWeather(Weather::Type weatherType, bool instant)
{
if (weatherType == mCurrentWeather && mNextWeather == Weather::Type_Unknown)
2012-04-07 20:55:25 +00:00
{
mFirstUpdate = false;
return;
2012-04-07 20:55:25 +00:00
}
if (instant || mFirstUpdate)
{
mNextWeather = Weather::Type_Unknown;
mCurrentWeather = weatherType;
}
else
{
if (mNextWeather != Weather::Type_Unknown)
2012-02-27 11:21:00 +00:00
{
// transition more than 50% finished?
2013-05-07 08:27:37 +00:00
if (mRemainingTransitionTime/(mWeatherSettings[mCurrentWeather].mTransitionDelta * 24.f * 3600) <= 0.5)
2012-02-27 11:21:00 +00:00
mCurrentWeather = mNextWeather;
}
mNextWeather = weatherType;
2013-05-07 08:27:37 +00:00
mRemainingTransitionTime = mWeatherSettings[mCurrentWeather].mTransitionDelta * 24.f * 3600;
}
mFirstUpdate = false;
}
WeatherResult WeatherManager::getResult(Weather::Type weatherType)
{
const Weather& current = mWeatherSettings[weatherType];
WeatherResult result;
2012-03-16 18:49:01 +00:00
result.mCloudTexture = current.mCloudTexture;
result.mCloudBlendFactor = 0;
result.mCloudOpacity = current.mCloudsMaximumPercent;
result.mWindSpeed = current.mWindSpeed;
result.mCloudSpeed = current.mCloudSpeed;
result.mGlareView = current.mGlareView;
result.mAmbientLoopSoundID = current.mAmbientLoopSoundID;
2012-02-26 19:46:09 +00:00
result.mSunColor = current.mSunDiscSunsetColor;
2013-05-07 08:27:37 +00:00
result.mNight = (mHour < mSunriseTime || mHour > mNightStart - 1);
2012-03-16 18:49:01 +00:00
2012-02-26 12:13:29 +00:00
result.mFogDepth = result.mNight ? current.mLandFogNightDepth : current.mLandFogDayDepth;
2012-03-16 18:49:01 +00:00
// night
2013-05-07 08:27:37 +00:00
if (mHour <= mNightEnd || mHour >= mNightStart + 1)
{
result.mFogColor = current.mFogNightColor;
result.mAmbientColor = current.mAmbientNightColor;
result.mSunColor = current.mSunNightColor;
result.mSkyColor = current.mSkyNightColor;
result.mNightFade = 1.f;
}
2012-03-16 18:49:01 +00:00
// sunrise
2013-05-07 08:27:37 +00:00
else if (mHour >= mNightEnd && mHour <= mDayStart + 1)
{
2013-05-07 08:27:37 +00:00
if (mHour <= mSunriseTime)
2012-02-24 16:42:31 +00:00
{
// fade in
2013-05-07 08:27:37 +00:00
float advance = mSunriseTime - mHour;
2012-03-16 18:49:01 +00:00
float factor = advance / 0.5f;
2013-03-20 00:20:56 +00:00
result.mFogColor = lerp(current.mFogSunriseColor, current.mFogNightColor, factor);
result.mAmbientColor = lerp(current.mAmbientSunriseColor, current.mAmbientNightColor, factor);
result.mSunColor = lerp(current.mSunSunriseColor, current.mSunNightColor, factor);
result.mSkyColor = lerp(current.mSkySunriseColor, current.mSkyNightColor, factor);
result.mNightFade = factor;
2012-02-24 16:42:31 +00:00
}
2012-03-16 18:49:01 +00:00
else //if (mHour >= 6)
2012-02-24 16:42:31 +00:00
{
// fade out
2013-05-07 08:27:37 +00:00
float advance = mHour - mSunriseTime;
2012-03-16 18:49:01 +00:00
float factor = advance / 3.f;
2013-03-20 00:20:56 +00:00
result.mFogColor = lerp(current.mFogSunriseColor, current.mFogDayColor, factor);
result.mAmbientColor = lerp(current.mAmbientSunriseColor, current.mAmbientDayColor, factor);
result.mSunColor = lerp(current.mSunSunriseColor, current.mSunDayColor, factor);
result.mSkyColor = lerp(current.mSkySunriseColor, current.mSkyDayColor, factor);
2012-02-24 16:42:31 +00:00
}
}
2012-03-16 18:49:01 +00:00
// day
2013-05-07 08:27:37 +00:00
else if (mHour >= mDayStart + 1 && mHour <= mDayEnd - 1)
{
result.mFogColor = current.mFogDayColor;
result.mAmbientColor = current.mAmbientDayColor;
result.mSunColor = current.mSunDayColor;
result.mSkyColor = current.mSkyDayColor;
}
2012-03-16 18:49:01 +00:00
// sunset
2013-05-07 08:27:37 +00:00
else if (mHour >= mDayEnd - 1 && mHour <= mNightStart + 1)
{
2013-05-07 08:27:37 +00:00
if (mHour <= mDayEnd + 1)
2012-02-24 16:42:31 +00:00
{
// fade in
2013-05-07 08:27:37 +00:00
float advance = (mDayEnd + 1) - mHour;
2012-03-16 18:49:01 +00:00
float factor = (advance / 2);
2013-03-20 00:20:56 +00:00
result.mFogColor = lerp(current.mFogSunsetColor, current.mFogDayColor, factor);
result.mAmbientColor = lerp(current.mAmbientSunsetColor, current.mAmbientDayColor, factor);
result.mSunColor = lerp(current.mSunSunsetColor, current.mSunDayColor, factor);
result.mSkyColor = lerp(current.mSkySunsetColor, current.mSkyDayColor, factor);
2012-02-24 16:42:31 +00:00
}
2012-03-16 18:49:01 +00:00
else //if (mHour >= 19)
2012-02-24 16:42:31 +00:00
{
// fade out
2013-05-07 08:27:37 +00:00
float advance = mHour - (mDayEnd + 1);
2012-03-16 18:49:01 +00:00
float factor = advance / 2.f;
2013-03-20 00:20:56 +00:00
result.mFogColor = lerp(current.mFogSunsetColor, current.mFogNightColor, factor);
result.mAmbientColor = lerp(current.mAmbientSunsetColor, current.mAmbientNightColor, factor);
result.mSunColor = lerp(current.mSunSunsetColor, current.mSunNightColor, factor);
result.mSkyColor = lerp(current.mSkySunsetColor, current.mSkyNightColor, factor);
result.mNightFade = factor;
2012-02-24 16:42:31 +00:00
}
}
2012-03-16 18:49:01 +00:00
return result;
}
2012-02-22 19:12:08 +00:00
WeatherResult WeatherManager::transition(float factor)
{
2012-02-22 19:12:08 +00:00
const WeatherResult& current = getResult(mCurrentWeather);
const WeatherResult& other = getResult(mNextWeather);
WeatherResult result;
2012-03-16 18:49:01 +00:00
result.mCloudTexture = current.mCloudTexture;
result.mNextCloudTexture = other.mCloudTexture;
result.mCloudBlendFactor = factor;
2012-03-16 18:49:01 +00:00
2013-03-20 00:20:56 +00:00
result.mCloudOpacity = lerp(current.mCloudOpacity, other.mCloudOpacity, factor);
result.mFogColor = lerp(current.mFogColor, other.mFogColor, factor);
result.mSunColor = lerp(current.mSunColor, other.mSunColor, factor);
result.mSkyColor = lerp(current.mSkyColor, other.mSkyColor, factor);
2012-03-16 18:49:01 +00:00
2013-03-20 00:20:56 +00:00
result.mAmbientColor = lerp(current.mAmbientColor, other.mAmbientColor, factor);
result.mSunDiscColor = lerp(current.mSunDiscColor, other.mSunDiscColor, factor);
result.mFogDepth = lerp(current.mFogDepth, other.mFogDepth, factor);
result.mWindSpeed = lerp(current.mWindSpeed, other.mWindSpeed, factor);
result.mCloudSpeed = lerp(current.mCloudSpeed, other.mCloudSpeed, factor);
result.mCloudOpacity = lerp(current.mCloudOpacity, other.mCloudOpacity, factor);
result.mGlareView = lerp(current.mGlareView, other.mGlareView, factor);
result.mNightFade = lerp(current.mNightFade, other.mNightFade, factor);
2012-03-16 18:49:01 +00:00
result.mNight = current.mNight;
2012-03-16 18:49:01 +00:00
return result;
}
void WeatherManager::update(float duration)
{
float timePassed = mTimePassed;
mTimePassed = 0;
mWeatherUpdateTime -= timePassed;
bool exterior = (MWBase::Environment::get().getWorld()->isCellExterior() || MWBase::Environment::get().getWorld()->isCellQuasiExterior());
2012-03-16 18:49:01 +00:00
if (exterior)
{
2013-04-28 03:29:34 +00:00
std::string regionstr = Misc::StringUtils::lowerCase(MWBase::Environment::get().getWorld()->getPlayer().getPlayer().getCell()->mCell->mRegion);
if (mWeatherUpdateTime <= 0 || regionstr != mCurrentRegion)
{
mCurrentRegion = regionstr;
2013-05-07 08:27:37 +00:00
mWeatherUpdateTime = mHoursBetweenWeatherChanges * 3600;
2012-03-16 18:49:01 +00:00
Weather::Type weatherType = Weather::Type_Clear;
2012-03-16 18:49:01 +00:00
if (mRegionOverrides.find(regionstr) != mRegionOverrides.end())
weatherType = mRegionOverrides[regionstr];
else
{
// get weather probabilities for the current region
const ESM::Region *region =
MWBase::Environment::get().getWorld()->getStore().get<ESM::Region>().search (regionstr);
if (region != 0)
{
/*
* 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).
*/
const int probability[] = {
region->mData.mClear,
region->mData.mCloudy,
region->mData.mFoggy,
region->mData.mOvercast,
region->mData.mRain,
region->mData.mThunder,
region->mData.mAsh,
region->mData.mBlight,
region->mData.mA,
region->mData.mB
}; // 10 elements
int chance = (rand() % 100) + 1; // 1..100
int sum = 0;
for (int i = 0; i < 10; ++i)
{
sum += probability[i];
if (chance < sum)
{
weatherType = (Weather::Type)i;
break;
}
}
}
}
2012-03-16 18:49:01 +00:00
setWeather(weatherType, false);
}
2012-03-16 18:49:01 +00:00
2012-02-25 12:46:17 +00:00
WeatherResult result;
2012-03-16 18:49:01 +00:00
if (mNextWeather != Weather::Type_Unknown)
{
mRemainingTransitionTime -= timePassed;
2012-02-25 12:46:17 +00:00
if (mRemainingTransitionTime < 0)
{
mCurrentWeather = mNextWeather;
mNextWeather = Weather::Type_Unknown;
2012-02-25 12:46:17 +00:00
}
}
2012-03-16 18:49:01 +00:00
if (mNextWeather != Weather::Type_Unknown)
2013-05-07 08:27:37 +00:00
result = transition(1 - (mRemainingTransitionTime / (mWeatherSettings[mCurrentWeather].mTransitionDelta * 24.f * 3600)));
2012-02-25 12:46:17 +00:00
else
result = getResult(mCurrentWeather);
2013-05-01 09:42:24 +00:00
mWindSpeed = result.mWindSpeed;
2012-02-26 12:13:29 +00:00
mRendering->configureFog(result.mFogDepth, result.mFogColor);
2012-03-16 18:49:01 +00:00
2012-02-25 12:46:17 +00:00
// disable sun during night
2013-05-07 08:27:37 +00:00
if (mHour >= mNightStart || mHour <= mSunriseTime)
2012-02-25 12:46:17 +00:00
mRendering->getSkyManager()->sunDisable();
else
mRendering->getSkyManager()->sunEnable();
2012-03-17 11:57:52 +00:00
// sun angle
float height;
2013-05-07 08:27:37 +00:00
//Day duration
float dayDuration = (mNightStart - 1) - mSunriseTime;
2012-03-17 11:57:52 +00:00
// rise at 6, set at 20
2013-05-07 08:27:37 +00:00
if (mHour >= mSunriseTime && mHour <= mNightStart)
height = 1 - std::abs(((mHour - dayDuration) / 7.f));
else if (mHour > mNightStart)
height = (mHour - mNightStart) / 4.f;
2012-03-17 11:57:52 +00:00
else //if (mHour > 0 && mHour < 6)
2013-05-07 08:27:37 +00:00
height = 1 - (mHour / mSunriseTime);
2013-05-10 16:55:56 +00:00
int facing = (mHour > 13.f) ? 1 : -1;
2012-03-17 11:57:52 +00:00
Vector3 final(
(height - 1) * facing,
(height - 1) * facing,
2012-03-17 11:57:52 +00:00
height);
mRendering->setSunDirection(final);
2012-03-16 18:49:01 +00:00
2013-05-14 17:58:08 +00:00
/*
* TODO: import separated fadeInStart/Finish, fadeOutStart/Finish
* for masser and secunda
*/
float fadeOutFinish=mFallback->getFallbackFloat("Moons_Masser_Fade_Out_Finish");
float fadeInStart=mFallback->getFallbackFloat("Moons_Masser_Fade_In_Start");
//moon calculations
float moonHeight;
if (mHour >= fadeInStart)
moonHeight = mHour - fadeInStart;
else if (mHour <= fadeOutFinish)
moonHeight = mHour + fadeOutFinish;
2012-02-25 15:36:45 +00:00
else
2013-05-14 17:58:08 +00:00
moonHeight = 0;
2012-03-16 18:49:01 +00:00
2013-05-14 17:58:08 +00:00
moonHeight /= (24.f - (fadeInStart - fadeOutFinish));
2012-03-16 18:49:01 +00:00
2013-05-14 17:58:08 +00:00
if (moonHeight != 0)
2012-02-25 15:36:45 +00:00
{
2013-05-14 17:58:08 +00:00
int facing = (moonHeight <= 1) ? 1 : -1;
2012-02-25 15:36:45 +00:00
Vector3 masser(
2013-05-14 17:58:08 +00:00
(moonHeight - 1) * facing,
2013-05-07 08:27:37 +00:00
(1 - moonHeight) * facing,
2012-02-25 15:36:45 +00:00
moonHeight);
2012-03-16 18:49:01 +00:00
2012-02-25 15:36:45 +00:00
Vector3 secunda(
2013-05-14 17:58:08 +00:00
(moonHeight - 1) * facing * 1.25,
2013-05-07 08:27:37 +00:00
(1 - moonHeight) * facing * 0.8,
2012-02-25 15:36:45 +00:00
moonHeight);
2012-03-16 18:49:01 +00:00
2012-02-25 15:36:45 +00:00
mRendering->getSkyManager()->setMasserDirection(masser);
mRendering->getSkyManager()->setSecundaDirection(secunda);
mRendering->getSkyManager()->masserEnable();
mRendering->getSkyManager()->secundaEnable();
2012-03-16 18:49:01 +00:00
2013-05-14 17:58:08 +00:00
float angle = (1-moonHeight) * 90.f * facing;
2013-05-08 22:36:58 +00:00
float masserHourFade = calculateHourFade("Masser");
float secundaHourFade = calculateHourFade("Secunda");
float masserAngleFade = calculateAngleFade("Masser", angle);
float secundaAngleFade = calculateAngleFade("Secunda", angle);
2012-03-16 18:49:01 +00:00
2013-05-07 08:27:37 +00:00
masserAngleFade *= masserHourFade;
secundaAngleFade *= secundaHourFade;
2012-03-16 18:49:01 +00:00
2013-05-07 08:27:37 +00:00
mRendering->getSkyManager()->setMasserFade(masserAngleFade);
mRendering->getSkyManager()->setSecundaFade(secundaAngleFade);
2012-02-25 15:36:45 +00:00
}
else
{
mRendering->getSkyManager()->masserDisable();
mRendering->getSkyManager()->secundaDisable();
}
2012-03-16 18:49:01 +00:00
if (mCurrentWeather == Weather::Type_Thunderstorm && mNextWeather == Weather::Type_Unknown && exterior)
2012-02-25 12:46:17 +00:00
{
if (mThunderFlash > 0)
{
// play the sound after a delay
mThunderSoundDelay -= duration;
if (mThunderSoundDelay <= 0)
{
// pick a random sound
int sound = rand() % 4;
std::string soundname;
2013-03-09 23:24:14 +00:00
if (sound == 0) soundname = mThunderSoundID0;
else if (sound == 1) soundname = mThunderSoundID1;
else if (sound == 2) soundname = mThunderSoundID2;
else if (sound == 3) soundname = mThunderSoundID3;
MWBase::Environment::get().getSoundManager()->playSound(soundname, 1.0, 1.0);
2012-02-25 12:46:17 +00:00
mThunderSoundDelay = 1000;
}
2012-03-16 18:49:01 +00:00
2012-02-25 12:46:17 +00:00
mThunderFlash -= duration;
if (mThunderFlash > 0)
2013-03-09 23:24:14 +00:00
mRendering->getSkyManager()->setLightningStrength( mThunderFlash / mThunderThreshold );
2012-02-25 12:46:17 +00:00
else
{
srand(time(NULL));
mThunderChanceNeeded = rand() % 100;
mThunderChance = 0;
2012-07-13 07:13:12 +00:00
mRendering->getSkyManager()->setLightningStrength( 0.f );
2012-02-25 12:46:17 +00:00
}
}
else
{
// no thunder active
mThunderChance += duration*4; // chance increases by 4 percent every second
if (mThunderChance >= mThunderChanceNeeded)
{
2013-03-09 23:24:14 +00:00
mThunderFlash = mThunderThreshold;
2012-03-16 18:49:01 +00:00
2013-03-09 23:24:14 +00:00
mRendering->getSkyManager()->setLightningStrength( mThunderFlash / mThunderThreshold );
2012-03-16 18:49:01 +00:00
2013-03-09 23:24:14 +00:00
mThunderSoundDelay = 0.25;
2012-02-25 12:46:17 +00:00
}
}
}
2012-02-26 12:13:29 +00:00
else
2012-07-13 07:13:12 +00:00
mRendering->getSkyManager()->setLightningStrength(0.f);
2012-03-16 18:49:01 +00:00
mRendering->setAmbientColour(result.mAmbientColor);
mRendering->sunEnable(false);
mRendering->setSunColour(result.mSunColor);
2012-03-16 18:49:01 +00:00
mRendering->getSkyManager()->setWeather(result);
}
else
{
mRendering->sunDisable(false);
mRendering->skyDisable();
2012-07-13 07:13:12 +00:00
mRendering->getSkyManager()->setLightningStrength(0.f);
}
2012-03-09 17:30:03 +00:00
// play sounds
std::string ambientSnd = (mNextWeather == Weather::Type_Unknown ? mWeatherSettings[mCurrentWeather].mAmbientLoopSoundID : "");
if (!exterior) ambientSnd = "";
2012-03-09 17:30:03 +00:00
if (ambientSnd != "")
{
if (std::find(mSoundsPlaying.begin(), mSoundsPlaying.end(), ambientSnd) == mSoundsPlaying.end())
{
mSoundsPlaying.push_back(ambientSnd);
MWBase::Environment::get().getSoundManager()->playSound(ambientSnd, 1.0, 1.0, MWBase::SoundManager::Play_Loop);
2012-03-09 17:30:03 +00:00
}
}
std::string rainSnd = (mNextWeather == Weather::Type_Unknown ? mWeatherSettings[mCurrentWeather].mRainLoopSoundID : "");
if (!exterior) rainSnd = "";
2012-03-09 17:30:03 +00:00
if (rainSnd != "")
{
if (std::find(mSoundsPlaying.begin(), mSoundsPlaying.end(), rainSnd) == mSoundsPlaying.end())
{
mSoundsPlaying.push_back(rainSnd);
MWBase::Environment::get().getSoundManager()->playSound(rainSnd, 1.0, 1.0, MWBase::SoundManager::Play_Loop);
2012-03-09 17:30:03 +00:00
}
}
// stop sounds
std::vector<std::string>::iterator it=mSoundsPlaying.begin();
while (it!=mSoundsPlaying.end())
{
if ( *it != ambientSnd && *it != rainSnd)
{
MWBase::Environment::get().getSoundManager()->stopSound(*it);
2012-03-09 17:30:03 +00:00
it = mSoundsPlaying.erase(it);
}
else
++it;
}
2012-02-21 21:56:34 +00:00
}
2012-02-22 19:12:08 +00:00
void WeatherManager::setHour(const float hour)
{
mHour = hour;
}
void WeatherManager::setDate(const int day, const int month)
{
mDay = day;
mMonth = month;
}
unsigned int WeatherManager::getWeatherID() const
{
// Source: http://www.uesp.net/wiki/Tes3Mod:GetCurrentWeather
return mCurrentWeather;
}
void WeatherManager::changeWeather(const std::string& region, const int id)
{
2013-04-28 03:29:34 +00:00
// make sure this region exists
2013-04-28 09:05:01 +00:00
MWBase::Environment::get().getWorld()->getStore().get<ESM::Region>().find(region);
2013-04-28 03:29:34 +00:00
Weather::Type weatherType = Weather::Type_Clear;
if (id >= Weather::Type_Clear && id < Weather::Type_Unknown)
weatherType = (Weather::Type)id;
mRegionOverrides[Misc::StringUtils::lowerCase(region)] = weatherType;
2013-04-28 03:29:34 +00:00
std::string playerRegion = MWBase::Environment::get().getWorld()->getPlayer().getPlayer().getCell()->mCell->mRegion;
if (Misc::StringUtils::ciEqual(region, playerRegion))
setWeather(weatherType);
}
2013-05-01 09:42:24 +00:00
float WeatherManager::getWindSpeed() const
{
return mWindSpeed;
}