2015-02-10 15:32:54 +01:00
|
|
|
#define _USE_MATH_DEFINES
|
2015-02-07 12:36:27 -06:00
|
|
|
#include <cmath>
|
|
|
|
|
2015-02-10 15:32:54 +01:00
|
|
|
#include "weather.hpp"
|
|
|
|
|
2015-04-22 17:58:55 +02:00
|
|
|
#include <components/misc/rng.hpp>
|
2015-03-15 14:07:47 +13:00
|
|
|
|
2015-07-09 19:22:04 +02:00
|
|
|
#include <components/esm/esmwriter.hpp>
|
2014-03-21 01:19:40 -05:00
|
|
|
#include <components/esm/weatherstate.hpp>
|
2014-03-20 01:25:52 -05:00
|
|
|
|
2012-04-23 15:27:03 +02:00
|
|
|
#include "../mwbase/environment.hpp"
|
2012-07-03 12:30:50 +02:00
|
|
|
#include "../mwbase/world.hpp"
|
2012-08-09 14:33:21 +02:00
|
|
|
#include "../mwbase/soundmanager.hpp"
|
2012-07-03 12:30:50 +02:00
|
|
|
|
2014-12-01 20:31:33 +01:00
|
|
|
#include "../mwsound/sound.hpp"
|
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
#include "../mwrender/renderingmanager.hpp"
|
|
|
|
#include "../mwrender/sky.hpp"
|
2012-07-03 12:30:50 +02:00
|
|
|
|
|
|
|
#include "player.hpp"
|
2012-10-01 19:17:04 +04:00
|
|
|
#include "esmstore.hpp"
|
2013-03-15 10:26:04 +01:00
|
|
|
#include "fallback.hpp"
|
2014-02-23 20:11:05 +01:00
|
|
|
#include "cellstore.hpp"
|
2012-02-26 11:51:02 +01:00
|
|
|
|
2012-02-21 22:56:34 +01:00
|
|
|
using namespace MWWorld;
|
|
|
|
|
2013-03-20 01:20:56 +01:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
float lerp (float x, float y, float factor)
|
|
|
|
{
|
|
|
|
return x * (1-factor) + y * factor;
|
|
|
|
}
|
2013-05-09 00:36:58 +02:00
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
osg::Vec4f lerp (const osg::Vec4f& x, const osg::Vec4f& y, float factor)
|
2013-03-20 01:20:56 +01:00
|
|
|
{
|
|
|
|
return x * (1-factor) + y * factor;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
void WeatherManager::setFallbackWeather(Weather& weather,const std::string& name)
|
2013-03-10 00:24:14 +01:00
|
|
|
{
|
2013-06-20 21:44:28 +02:00
|
|
|
std::string upper=name;
|
|
|
|
upper[0]=toupper(name[0]);
|
|
|
|
weather.mCloudsMaximumPercent = mFallback->getFallbackFloat("Weather_"+upper+"_Clouds_Maximum_Percent");
|
|
|
|
weather.mTransitionDelta = mFallback->getFallbackFloat("Weather_"+upper+"_Transition_Delta");
|
|
|
|
weather.mSkySunriseColor=mFallback->getFallbackColour("Weather_"+upper+"_Sky_Sunrise_Color");
|
|
|
|
weather.mSkyDayColor = mFallback->getFallbackColour("Weather_"+upper+"_Sky_Day_Color");
|
|
|
|
weather.mSkySunsetColor = mFallback->getFallbackColour("Weather_"+upper+"_Sky_Sunset_Color");
|
|
|
|
weather.mSkyNightColor = mFallback->getFallbackColour("Weather_"+upper+"_Sky_Night_Color");
|
|
|
|
weather.mFogSunriseColor = mFallback->getFallbackColour("Weather_"+upper+"_Fog_Sunrise_Color");
|
|
|
|
weather.mFogDayColor = mFallback->getFallbackColour("Weather_"+upper+"_Fog_Day_Color");
|
|
|
|
weather.mFogSunsetColor = mFallback->getFallbackColour("Weather_"+upper+"_Fog_Sunset_Color");
|
|
|
|
weather.mFogNightColor = mFallback->getFallbackColour("Weather_"+upper+"_Fog_Night_Color");
|
|
|
|
weather.mAmbientSunriseColor = mFallback->getFallbackColour("Weather_"+upper+"_Ambient_Sunrise_Color");
|
|
|
|
weather.mAmbientDayColor = mFallback->getFallbackColour("Weather_"+upper+"_Ambient_Day_Color");
|
|
|
|
weather.mAmbientSunsetColor = mFallback->getFallbackColour("Weather_"+upper+"_Ambient_Sunset_Color");
|
|
|
|
weather.mAmbientNightColor = mFallback->getFallbackColour("Weather_"+upper+"_Ambient_Night_Color");
|
|
|
|
weather.mSunSunriseColor = mFallback->getFallbackColour("Weather_"+upper+"_Sun_Sunrise_Color");
|
|
|
|
weather.mSunDayColor = mFallback->getFallbackColour("Weather_"+upper+"_Sun_Day_Color");
|
|
|
|
weather.mSunSunsetColor = mFallback->getFallbackColour("Weather_"+upper+"_Sun_Sunset_Color");
|
|
|
|
weather.mSunNightColor = mFallback->getFallbackColour("Weather_"+upper+"_Sun_Night_Color");
|
|
|
|
weather.mSunDiscSunsetColor = mFallback->getFallbackColour("Weather_"+upper+"_Sun_Disc_Sunset_Color");
|
|
|
|
weather.mLandFogDayDepth = mFallback->getFallbackFloat("Weather_"+upper+"_Land_Fog_Day_Depth");
|
|
|
|
weather.mLandFogNightDepth = mFallback->getFallbackFloat("Weather_"+upper+"_Land_Fog_Night_Depth");
|
|
|
|
weather.mWindSpeed = mFallback->getFallbackFloat("Weather_"+upper+"_Wind_Speed");
|
|
|
|
weather.mCloudSpeed = mFallback->getFallbackFloat("Weather_"+upper+"_Cloud_Speed");
|
|
|
|
weather.mGlareView = mFallback->getFallbackFloat("Weather_"+upper+"_Glare_View");
|
2014-06-24 15:00:15 +02:00
|
|
|
weather.mCloudTexture = mFallback->getFallbackString("Weather_"+upper+"_Cloud_Texture");
|
2014-06-25 18:20:21 +02:00
|
|
|
|
2015-07-10 02:34:00 +02:00
|
|
|
static const float fStromWindSpeed = mStore->get<ESM::GameSetting>().find("fStromWindSpeed")->getFloat();
|
|
|
|
|
|
|
|
weather.mIsStorm = weather.mWindSpeed > fStromWindSpeed;
|
|
|
|
|
2014-06-25 18:20:21 +02:00
|
|
|
bool usesPrecip = mFallback->getFallbackBool("Weather_"+upper+"_Using_Precip");
|
|
|
|
if (usesPrecip)
|
|
|
|
weather.mRainEffect = "meshes\\raindrop.nif";
|
|
|
|
weather.mRainSpeed = mRainSpeed;
|
|
|
|
weather.mRainFrequency = mFallback->getFallbackFloat("Weather_"+upper+"_Rain_Entrance_Speed");
|
|
|
|
/*
|
|
|
|
Unhandled:
|
|
|
|
Rain Diameter=600 ?
|
|
|
|
Rain Height Min=200 ?
|
|
|
|
Rain Height Max=700 ?
|
|
|
|
Rain Threshold=0.6 ?
|
|
|
|
Max Raindrops=650 ?
|
|
|
|
*/
|
2014-06-24 18:37:38 +02:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
mWeatherSettings[name] = weather;
|
2013-03-10 00:24:14 +01:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-05-09 00:36:58 +02:00
|
|
|
|
2013-05-09 13:56:13 +02:00
|
|
|
float WeatherManager::calculateHourFade (const std::string& moonName) const
|
2013-05-09 00:36:58 +02:00
|
|
|
{
|
|
|
|
float fadeOutStart=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_Out_Start");
|
2013-05-14 19:58:08 +02:00
|
|
|
float fadeOutFinish=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_Out_Finish");
|
2013-05-09 00:36:58 +02:00
|
|
|
float fadeInStart=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_In_Start");
|
|
|
|
float fadeInFinish=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_In_Finish");
|
|
|
|
|
2013-05-10 18:55:56 +02:00
|
|
|
if (mHour >= fadeOutStart && mHour <= fadeOutFinish)
|
2013-05-14 19:58:08 +02:00
|
|
|
return (1 - ((mHour - fadeOutStart) / (fadeOutFinish - fadeOutStart)));
|
|
|
|
if (mHour >= fadeInStart && mHour <= fadeInFinish)
|
|
|
|
return (1 - ((mHour - fadeInStart) / (fadeInFinish - fadeInStart)));
|
2013-05-09 00:36:58 +02:00
|
|
|
else
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-05-09 13:56:13 +02:00
|
|
|
float WeatherManager::calculateAngleFade (const std::string& moonName, float angle) const
|
2013-05-09 00:36:58 +02:00
|
|
|
{
|
|
|
|
float endAngle=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_End_Angle");
|
|
|
|
float startAngle=mFallback->getFallbackFloat("Moons_"+moonName+"_Fade_Start_Angle");
|
2013-05-14 19:58:08 +02:00
|
|
|
if (angle <= startAngle && angle >= endAngle)
|
|
|
|
return (1 - ((angle - endAngle)/(startAngle-endAngle)));
|
|
|
|
else if (angle > startAngle)
|
2013-05-09 00:36:58 +02:00
|
|
|
return 0.f;
|
|
|
|
else
|
|
|
|
return 1.f;
|
|
|
|
}
|
|
|
|
|
2015-07-10 02:34:00 +02:00
|
|
|
WeatherManager::WeatherManager(MWRender::RenderingManager* rendering, MWWorld::Fallback* fallback, MWWorld::ESMStore* store) :
|
|
|
|
mHour(14), mWindSpeed(0.f), mIsStorm(false), mStormDirection(0,1,0), mFallback(fallback), mStore(store),
|
2015-04-30 19:24:27 -05:00
|
|
|
mRendering(rendering), mCurrentWeather("clear"), mNextWeather(""), mFirstUpdate(true),
|
|
|
|
mRemainingTransitionTime(0), mThunderFlash(0), mThunderChance(0), mThunderChanceNeeded(50),
|
|
|
|
mTimePassed(0), mWeatherUpdateTime(0), mThunderSoundDelay(0)
|
2012-02-21 22:56:34 +01:00
|
|
|
{
|
2013-03-15 10:32:03 +01:00
|
|
|
//Globals
|
2013-03-15 10:28:39 +01: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");
|
2013-03-20 02:29:03 +01:00
|
|
|
mHoursBetweenWeatherChanges = mFallback->getFallbackFloat("Weather_Hours_Between_Weather_Changes");
|
2013-05-07 10:27:37 +02:00
|
|
|
mWeatherUpdateTime = mHoursBetweenWeatherChanges * 3600;
|
2013-03-15 10:28:39 +01:00
|
|
|
mThunderFrequency = mFallback->getFallbackFloat("Weather_Thunderstorm_Thunder_Frequency");
|
|
|
|
mThunderThreshold = mFallback->getFallbackFloat("Weather_Thunderstorm_Thunder_Threshold");
|
|
|
|
mThunderSoundDelay = 0.25;
|
2013-05-07 10:27:37 +02:00
|
|
|
|
2014-06-25 18:20:21 +02:00
|
|
|
mRainSpeed = mFallback->getFallbackFloat("Weather_Precip_Gravity");
|
|
|
|
|
2013-05-07 10:27:37 +02: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;
|
2015-03-08 13:07:29 +13:00
|
|
|
mNightEnd = mSunriseTime - 0.5f;
|
2013-05-07 10:27:37 +02:00
|
|
|
mDayStart = mSunriseTime + mSunriseDuration;
|
|
|
|
mDayEnd = mSunsetTime;
|
|
|
|
|
2013-03-15 10:28:39 +01:00
|
|
|
//Weather
|
2013-03-15 10:22:02 +01:00
|
|
|
Weather clear;
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(clear,"clear");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-22 19:17:37 +01:00
|
|
|
Weather cloudy;
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(cloudy,"cloudy");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-25 21:06:03 +01:00
|
|
|
Weather foggy;
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(foggy,"foggy");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-25 13:46:17 +01:00
|
|
|
Weather thunderstorm;
|
2014-12-01 20:31:33 +01:00
|
|
|
thunderstorm.mAmbientLoopSoundID = "rain heavy";
|
2014-06-25 18:20:21 +02:00
|
|
|
thunderstorm.mRainEffect = "meshes\\raindrop.nif";
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(thunderstorm,"thunderstorm");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-25 21:06:03 +01:00
|
|
|
Weather rain;
|
2014-12-01 20:31:33 +01:00
|
|
|
rain.mAmbientLoopSoundID = "rain";
|
2014-06-25 18:20:21 +02:00
|
|
|
rain.mRainEffect = "meshes\\raindrop.nif";
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(rain,"rain");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-22 19:17:37 +01:00
|
|
|
Weather overcast;
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(overcast,"overcast");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-25 21:06:03 +01:00
|
|
|
Weather ashstorm;
|
|
|
|
ashstorm.mAmbientLoopSoundID = "ashstorm";
|
2014-06-24 15:00:15 +02:00
|
|
|
ashstorm.mParticleEffect = "meshes\\ashcloud.nif";
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(ashstorm,"ashstorm");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-25 21:06:03 +01:00
|
|
|
Weather blight;
|
|
|
|
blight.mAmbientLoopSoundID = "blight";
|
2014-06-24 15:00:15 +02:00
|
|
|
blight.mParticleEffect = "meshes\\blightcloud.nif";
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(blight,"blight");
|
2012-03-19 22:54:04 +01:00
|
|
|
|
2012-02-25 21:06:03 +01:00
|
|
|
Weather snow;
|
2014-06-24 15:00:15 +02:00
|
|
|
snow.mParticleEffect = "meshes\\snow.nif";
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(snow, "snow");
|
2012-04-23 15:27:03 +02:00
|
|
|
|
2012-02-25 21:06:03 +01:00
|
|
|
Weather blizzard;
|
|
|
|
blizzard.mAmbientLoopSoundID = "BM Blizzard";
|
2014-06-24 15:00:15 +02:00
|
|
|
blizzard.mParticleEffect = "meshes\\blizzard.nif";
|
2013-06-20 21:44:28 +02:00
|
|
|
setFallbackWeather(blizzard,"blizzard");
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
|
|
|
|
2013-08-29 15:15:40 +02:00
|
|
|
WeatherManager::~WeatherManager()
|
|
|
|
{
|
2014-12-01 20:31:33 +01:00
|
|
|
stopSounds();
|
2013-08-29 15:15:40 +02:00
|
|
|
}
|
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
void WeatherManager::setWeather(const std::string& weather, bool instant)
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2013-06-20 21:44:28 +02:00
|
|
|
if (weather == mCurrentWeather && mNextWeather == "")
|
2012-04-07 22:55:25 +02:00
|
|
|
{
|
|
|
|
mFirstUpdate = false;
|
2012-03-17 13:14:31 +01:00
|
|
|
return;
|
2012-04-07 22:55:25 +02:00
|
|
|
}
|
2012-03-17 13:14:31 +01:00
|
|
|
|
2012-02-25 23:41:02 +01:00
|
|
|
if (instant || mFirstUpdate)
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2013-06-20 21:44:28 +02:00
|
|
|
mNextWeather = "";
|
|
|
|
mCurrentWeather = weather;
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
2012-02-25 23:41:02 +01:00
|
|
|
else
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2013-06-20 21:44:28 +02:00
|
|
|
if (mNextWeather != "")
|
2012-02-27 12:21:00 +01:00
|
|
|
{
|
|
|
|
// transition more than 50% finished?
|
2013-05-07 10:27:37 +02:00
|
|
|
if (mRemainingTransitionTime/(mWeatherSettings[mCurrentWeather].mTransitionDelta * 24.f * 3600) <= 0.5)
|
2012-02-27 12:21:00 +01:00
|
|
|
mCurrentWeather = mNextWeather;
|
|
|
|
}
|
2012-03-17 13:14:31 +01:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
mNextWeather = weather;
|
2013-12-29 12:47:44 +01:00
|
|
|
mRemainingTransitionTime = mWeatherSettings[mCurrentWeather].mTransitionDelta * 24.f * 3600.f;
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
2012-04-07 16:33:03 +02:00
|
|
|
mFirstUpdate = false;
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
void WeatherManager::setResult(const std::string& weatherType)
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2013-06-19 03:24:30 +02:00
|
|
|
const Weather& current = mWeatherSettings[weatherType];
|
2013-06-19 06:50:36 +02:00
|
|
|
|
|
|
|
mResult.mCloudTexture = current.mCloudTexture;
|
|
|
|
mResult.mCloudBlendFactor = 0;
|
|
|
|
mResult.mCloudOpacity = current.mCloudsMaximumPercent;
|
|
|
|
mResult.mWindSpeed = current.mWindSpeed;
|
|
|
|
mResult.mCloudSpeed = current.mCloudSpeed;
|
|
|
|
mResult.mGlareView = current.mGlareView;
|
|
|
|
mResult.mAmbientLoopSoundID = current.mAmbientLoopSoundID;
|
2014-12-01 20:31:33 +01:00
|
|
|
mResult.mAmbientSoundVolume = 1.f;
|
|
|
|
mResult.mEffectFade = 1.f;
|
2013-06-19 06:50:36 +02:00
|
|
|
mResult.mSunColor = current.mSunDiscSunsetColor;
|
2013-06-21 08:58:52 +02:00
|
|
|
|
2014-06-24 18:37:38 +02:00
|
|
|
mResult.mIsStorm = current.mIsStorm;
|
|
|
|
|
2014-06-25 18:20:21 +02:00
|
|
|
mResult.mRainSpeed = current.mRainSpeed;
|
|
|
|
mResult.mRainFrequency = current.mRainFrequency;
|
|
|
|
|
2014-06-24 15:00:15 +02:00
|
|
|
mResult.mParticleEffect = current.mParticleEffect;
|
2014-06-25 18:20:21 +02:00
|
|
|
mResult.mRainEffect = current.mRainEffect;
|
2014-06-24 15:00:15 +02:00
|
|
|
|
2013-06-19 06:50:36 +02:00
|
|
|
mResult.mNight = (mHour < mSunriseTime || mHour > mNightStart - 1);
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-19 06:50:36 +02:00
|
|
|
mResult.mFogDepth = mResult.mNight ? current.mLandFogNightDepth : current.mLandFogDayDepth;
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2012-02-23 19:49:56 +01:00
|
|
|
// night
|
2013-05-07 10:27:37 +02:00
|
|
|
if (mHour <= mNightEnd || mHour >= mNightStart + 1)
|
2012-02-23 19:49:56 +01:00
|
|
|
{
|
2013-06-19 06:50:36 +02:00
|
|
|
mResult.mFogColor = current.mFogNightColor;
|
|
|
|
mResult.mAmbientColor = current.mAmbientNightColor;
|
|
|
|
mResult.mSunColor = current.mSunNightColor;
|
|
|
|
mResult.mSkyColor = current.mSkyNightColor;
|
|
|
|
mResult.mNightFade = 1.f;
|
2012-02-23 19:49:56 +01:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2012-02-23 19:49:56 +01:00
|
|
|
// sunrise
|
2013-05-07 10:27:37 +02:00
|
|
|
else if (mHour >= mNightEnd && mHour <= mDayStart + 1)
|
2012-02-23 19:49:56 +01:00
|
|
|
{
|
2013-05-07 10:27:37 +02:00
|
|
|
if (mHour <= mSunriseTime)
|
2012-02-24 17:42:31 +01:00
|
|
|
{
|
|
|
|
// fade in
|
2013-05-07 10:27:37 +02:00
|
|
|
float advance = mSunriseTime - mHour;
|
2012-03-16 19:49:01 +01:00
|
|
|
float factor = advance / 0.5f;
|
2013-06-19 06:50:36 +02:00
|
|
|
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;
|
2012-02-24 17:42:31 +01:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
else //if (mHour >= 6)
|
2012-02-24 17:42:31 +01:00
|
|
|
{
|
|
|
|
// fade out
|
2013-05-07 10:27:37 +02:00
|
|
|
float advance = mHour - mSunriseTime;
|
2012-03-16 19:49:01 +01:00
|
|
|
float factor = advance / 3.f;
|
2013-06-19 06:50:36 +02:00
|
|
|
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);
|
2012-02-24 17:42:31 +01:00
|
|
|
}
|
2012-02-23 19:49:56 +01:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2012-02-23 19:49:56 +01:00
|
|
|
// day
|
2013-05-07 10:27:37 +02:00
|
|
|
else if (mHour >= mDayStart + 1 && mHour <= mDayEnd - 1)
|
2012-02-23 19:49:56 +01:00
|
|
|
{
|
2013-06-19 06:50:36 +02:00
|
|
|
mResult.mFogColor = current.mFogDayColor;
|
|
|
|
mResult.mAmbientColor = current.mAmbientDayColor;
|
|
|
|
mResult.mSunColor = current.mSunDayColor;
|
|
|
|
mResult.mSkyColor = current.mSkyDayColor;
|
2012-02-23 19:49:56 +01:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2012-02-23 19:49:56 +01:00
|
|
|
// sunset
|
2013-05-07 10:27:37 +02:00
|
|
|
else if (mHour >= mDayEnd - 1 && mHour <= mNightStart + 1)
|
2012-02-23 19:49:56 +01:00
|
|
|
{
|
2013-05-07 10:27:37 +02:00
|
|
|
if (mHour <= mDayEnd + 1)
|
2012-02-24 17:42:31 +01:00
|
|
|
{
|
|
|
|
// fade in
|
2013-05-07 10:27:37 +02:00
|
|
|
float advance = (mDayEnd + 1) - mHour;
|
2012-03-16 19:49:01 +01:00
|
|
|
float factor = (advance / 2);
|
2013-06-19 06:50:36 +02:00
|
|
|
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);
|
2012-02-24 17:42:31 +01:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
else //if (mHour >= 19)
|
2012-02-24 17:42:31 +01:00
|
|
|
{
|
|
|
|
// fade out
|
2013-05-07 10:27:37 +02:00
|
|
|
float advance = mHour - (mDayEnd + 1);
|
2012-03-16 19:49:01 +01:00
|
|
|
float factor = advance / 2.f;
|
2013-06-19 06:50:36 +02:00
|
|
|
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;
|
2012-02-24 17:42:31 +01:00
|
|
|
}
|
2012-02-23 19:49:56 +01:00
|
|
|
}
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
|
|
|
|
2013-06-19 06:50:36 +02:00
|
|
|
void WeatherManager::transition(float factor)
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2013-06-19 06:50:36 +02:00
|
|
|
setResult(mCurrentWeather);
|
|
|
|
const WeatherResult current = mResult;
|
|
|
|
setResult(mNextWeather);
|
|
|
|
const WeatherResult other = mResult;
|
|
|
|
|
|
|
|
mResult.mCloudTexture = current.mCloudTexture;
|
|
|
|
mResult.mNextCloudTexture = other.mCloudTexture;
|
|
|
|
mResult.mCloudBlendFactor = factor;
|
|
|
|
|
|
|
|
mResult.mCloudOpacity = lerp(current.mCloudOpacity, other.mCloudOpacity, 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);
|
|
|
|
mResult.mWindSpeed = lerp(current.mWindSpeed, other.mWindSpeed, factor);
|
|
|
|
mResult.mCloudSpeed = lerp(current.mCloudSpeed, other.mCloudSpeed, factor);
|
|
|
|
mResult.mCloudOpacity = lerp(current.mCloudOpacity, other.mCloudOpacity, factor);
|
|
|
|
mResult.mGlareView = lerp(current.mGlareView, other.mGlareView, factor);
|
|
|
|
mResult.mNightFade = lerp(current.mNightFade, other.mNightFade, factor);
|
|
|
|
|
|
|
|
mResult.mNight = current.mNight;
|
2014-06-24 15:00:15 +02:00
|
|
|
|
2014-12-01 20:31:33 +01:00
|
|
|
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;
|
2015-03-08 13:07:29 +13:00
|
|
|
mResult.mAmbientSoundVolume = 2*(factor-0.5f);
|
2014-12-01 20:31:33 +01:00
|
|
|
mResult.mEffectFade = mResult.mAmbientSoundVolume;
|
|
|
|
mResult.mAmbientLoopSoundID = other.mAmbientLoopSoundID;
|
|
|
|
}
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
|
|
|
|
2014-10-09 19:41:51 +13:00
|
|
|
void WeatherManager::update(float duration, bool paused)
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2015-03-08 13:07:29 +13:00
|
|
|
float timePassed = static_cast<float>(mTimePassed);
|
2012-09-25 21:28:25 +02:00
|
|
|
mTimePassed = 0;
|
|
|
|
|
|
|
|
mWeatherUpdateTime -= timePassed;
|
2012-03-13 17:25:07 +01:00
|
|
|
|
2014-01-01 16:45:39 +01:00
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
|
|
|
const bool exterior = (world->isCellExterior() || world->isCellQuasiExterior());
|
|
|
|
if (!exterior)
|
|
|
|
{
|
2015-04-14 15:55:56 +02:00
|
|
|
mRendering->setSkyEnabled(false);
|
|
|
|
//mRendering->getSkyManager()->setLightningStrength(0.f);
|
2014-12-01 20:31:33 +01:00
|
|
|
stopSounds();
|
2014-01-01 16:45:39 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-12-31 20:40:23 +01:00
|
|
|
switchToNextWeather(false);
|
2013-06-19 16:18:43 +02:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
if (mNextWeather != "")
|
2013-06-19 16:18:43 +02:00
|
|
|
{
|
|
|
|
mRemainingTransitionTime -= timePassed;
|
|
|
|
if (mRemainingTransitionTime < 0)
|
2012-02-22 19:17:37 +01:00
|
|
|
{
|
2013-06-19 16:18:43 +02:00
|
|
|
mCurrentWeather = mNextWeather;
|
2013-06-20 21:44:28 +02:00
|
|
|
mNextWeather = "";
|
2012-02-22 19:17:37 +01:00
|
|
|
}
|
2013-06-19 16:18:43 +02:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
if (mNextWeather != "")
|
2013-06-19 16:18:43 +02:00
|
|
|
transition(1 - (mRemainingTransitionTime / (mWeatherSettings[mCurrentWeather].mTransitionDelta * 24.f * 3600)));
|
|
|
|
else
|
|
|
|
setResult(mCurrentWeather);
|
2012-03-17 13:14:31 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
mWindSpeed = mResult.mWindSpeed;
|
2014-06-25 18:20:21 +02:00
|
|
|
mIsStorm = mResult.mIsStorm;
|
2013-05-01 11:42:24 +02:00
|
|
|
|
2014-06-26 19:01:49 +02:00
|
|
|
if (mIsStorm)
|
|
|
|
{
|
|
|
|
MWWorld::Ptr player = world->getPlayerPtr();
|
2015-05-12 03:02:15 +02:00
|
|
|
osg::Vec3f playerPos (player.getRefData().getPosition().asVec3());
|
|
|
|
osg::Vec3f redMountainPos (19950, 72032, 27831);
|
2014-06-26 19:01:49 +02:00
|
|
|
|
|
|
|
mStormDirection = (playerPos - redMountainPos);
|
2015-05-12 03:02:15 +02:00
|
|
|
mStormDirection.z() = 0;
|
2015-06-01 01:57:15 +02:00
|
|
|
mStormDirection.normalize();
|
2015-05-22 01:54:06 +02:00
|
|
|
mRendering->getSkyManager()->setStormDirection(mStormDirection);
|
2014-06-26 19:01:49 +02:00
|
|
|
}
|
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
mRendering->configureFog(mResult.mFogDepth, mResult.mFogColor);
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
// disable sun during night
|
2015-04-15 18:50:50 +02:00
|
|
|
if (mHour >= mNightStart || mHour <= mSunriseTime)
|
|
|
|
mRendering->getSkyManager()->sunDisable();
|
|
|
|
else
|
|
|
|
mRendering->getSkyManager()->sunEnable();
|
2012-03-17 12:57:52 +01:00
|
|
|
|
2015-02-08 14:26:38 -06: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 12:36:27 -06:00
|
|
|
// mark when the sun is level with the horizon.
|
|
|
|
{
|
2015-02-08 14:26:38 -06:00
|
|
|
// Shift times into a 24-hour window beginning at mSunriseTime...
|
2015-02-07 12:36:27 -06:00
|
|
|
float adjustedHour = mHour;
|
2015-02-08 14:26:38 -06:00
|
|
|
float adjustedNightStart = mNightStart;
|
2015-02-07 12:36:27 -06:00
|
|
|
if ( mHour < mSunriseTime )
|
|
|
|
adjustedHour += 24.f;
|
2015-02-08 14:26:38 -06:00
|
|
|
if ( mNightStart < mSunriseTime )
|
|
|
|
adjustedNightStart += 24.f;
|
2012-03-17 13:14:31 +01:00
|
|
|
|
2015-02-08 14:26:38 -06:00
|
|
|
const bool is_night = adjustedHour >= adjustedNightStart;
|
|
|
|
const float dayDuration = adjustedNightStart - mSunriseTime;
|
2015-02-07 12:36:27 -06:00
|
|
|
const float nightDuration = 24.f - dayDuration;
|
2012-03-17 13:14:31 +01:00
|
|
|
|
2015-02-07 12:36:27 -06:00
|
|
|
double theta;
|
|
|
|
if ( !is_night ) {
|
|
|
|
theta = M_PI * (adjustedHour - mSunriseTime) / dayDuration;
|
|
|
|
} else {
|
2015-02-08 14:26:38 -06:00
|
|
|
theta = M_PI * (adjustedHour - adjustedNightStart) / nightDuration;
|
2015-02-07 12:36:27 -06:00
|
|
|
}
|
2014-06-16 17:08:02 -04:00
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
osg::Vec3f final(
|
2015-03-08 13:07:29 +13:00
|
|
|
static_cast<float>(cos(theta)),
|
2015-02-08 14:26:38 -06:00
|
|
|
-0.268f, // approx tan( -15 degrees )
|
2015-03-08 13:07:29 +13:00
|
|
|
static_cast<float>(sin(theta)));
|
2015-04-14 15:55:56 +02:00
|
|
|
mRendering->setSunDirection( final * -1 );
|
2015-02-07 12:36:27 -06:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
/*
|
|
|
|
* TODO: import separated fadeInStart/Finish, fadeOutStart/Finish
|
|
|
|
* for masser and secunda
|
|
|
|
*/
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
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;
|
|
|
|
else
|
|
|
|
moonHeight = 0;
|
|
|
|
|
|
|
|
moonHeight /= (24.f - (fadeInStart - fadeOutFinish));
|
|
|
|
|
|
|
|
if (moonHeight != 0)
|
|
|
|
{
|
|
|
|
int facing = (moonHeight <= 1) ? 1 : -1;
|
2015-04-14 15:55:56 +02:00
|
|
|
osg::Vec3f masser(
|
2013-05-14 19:58:08 +02:00
|
|
|
(moonHeight - 1) * facing,
|
2013-05-07 10:27:37 +02:00
|
|
|
(1 - moonHeight) * facing,
|
2012-02-25 16:36:45 +01:00
|
|
|
moonHeight);
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
osg::Vec3f secunda(
|
2015-03-08 13:07:29 +13:00
|
|
|
(moonHeight - 1) * facing * 1.25f,
|
|
|
|
(1 - moonHeight) * facing * 0.8f,
|
2012-02-25 16:36:45 +01:00
|
|
|
moonHeight);
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2015-04-15 18:50:50 +02:00
|
|
|
mRendering->getSkyManager()->setMasserDirection(masser);
|
|
|
|
mRendering->getSkyManager()->setSecundaDirection(secunda);
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
float angle = (1-moonHeight) * 90.f * facing;
|
|
|
|
float masserHourFade = calculateHourFade("Masser");
|
|
|
|
float secundaHourFade = calculateHourFade("Secunda");
|
|
|
|
float masserAngleFade = calculateAngleFade("Masser", angle);
|
|
|
|
float secundaAngleFade = calculateAngleFade("Secunda", angle);
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
masserAngleFade *= masserHourFade;
|
|
|
|
secundaAngleFade *= secundaHourFade;
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2015-06-19 18:15:23 +02:00
|
|
|
if (masserAngleFade > 0)
|
|
|
|
{
|
|
|
|
mRendering->getSkyManager()->setMasserFade(masserAngleFade);
|
|
|
|
mRendering->getSkyManager()->masserEnable();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
mRendering->getSkyManager()->masserDisable();
|
|
|
|
|
|
|
|
if (secundaAngleFade > 0)
|
|
|
|
{
|
|
|
|
mRendering->getSkyManager()->setSecundaFade(secundaAngleFade);
|
|
|
|
mRendering->getSkyManager()->secundaEnable();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
mRendering->getSkyManager()->secundaDisable();
|
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2015-04-15 18:50:50 +02:00
|
|
|
mRendering->getSkyManager()->masserDisable();
|
|
|
|
mRendering->getSkyManager()->secundaDisable();
|
2013-06-19 16:18:43 +02:00
|
|
|
}
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2014-10-10 23:29:51 +13:00
|
|
|
if (!paused)
|
2013-06-19 16:18:43 +02:00
|
|
|
{
|
2014-10-10 23:29:51 +13:00
|
|
|
if (mCurrentWeather == "thunderstorm" && mNextWeather == "")
|
2012-02-25 13:46:17 +01:00
|
|
|
{
|
2013-06-19 16:18:43 +02:00
|
|
|
if (mThunderFlash > 0)
|
2012-02-25 13:46:17 +01:00
|
|
|
{
|
2014-10-10 23:29:51 +13:00
|
|
|
// play the sound after a delay
|
|
|
|
mThunderSoundDelay -= duration;
|
|
|
|
if (mThunderSoundDelay <= 0)
|
|
|
|
{
|
|
|
|
// pick a random sound
|
2015-04-22 17:58:55 +02:00
|
|
|
int sound = Misc::Rng::rollDice(4);
|
2014-10-10 23:29:51 +13:00
|
|
|
std::string* soundName = NULL;
|
|
|
|
if (sound == 0) soundName = &mThunderSoundID0;
|
|
|
|
else if (sound == 1) soundName = &mThunderSoundID1;
|
|
|
|
else if (sound == 2) soundName = &mThunderSoundID2;
|
|
|
|
else if (sound == 3) soundName = &mThunderSoundID3;
|
|
|
|
if (soundName)
|
|
|
|
MWBase::Environment::get().getSoundManager()->playSound(*soundName, 1.0, 1.0);
|
|
|
|
mThunderSoundDelay = 1000;
|
|
|
|
}
|
|
|
|
|
|
|
|
mThunderFlash -= duration;
|
2015-04-14 15:55:56 +02:00
|
|
|
//if (mThunderFlash > 0)
|
|
|
|
//mRendering->getSkyManager()->setLightningStrength( mThunderFlash / mThunderThreshold );
|
|
|
|
//else
|
2014-10-10 23:29:51 +13:00
|
|
|
{
|
2015-04-22 17:58:55 +02:00
|
|
|
mThunderChanceNeeded = static_cast<float>(Misc::Rng::rollDice(100));
|
2014-10-10 23:29:51 +13:00
|
|
|
mThunderChance = 0;
|
2015-04-14 15:55:56 +02:00
|
|
|
//mRendering->getSkyManager()->setLightningStrength( 0.f );
|
2014-10-10 23:29:51 +13:00
|
|
|
}
|
2012-02-25 13:46:17 +01:00
|
|
|
}
|
2014-10-10 23:29:51 +13:00
|
|
|
else
|
2013-06-19 16:18:43 +02:00
|
|
|
{
|
2014-10-10 23:29:51 +13:00
|
|
|
// no thunder active
|
|
|
|
mThunderChance += duration*4; // chance increases by 4 percent every second
|
|
|
|
if (mThunderChance >= mThunderChanceNeeded)
|
|
|
|
{
|
|
|
|
mThunderFlash = mThunderThreshold;
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2015-04-14 15:55:56 +02:00
|
|
|
//mRendering->getSkyManager()->setLightningStrength( mThunderFlash / mThunderThreshold );
|
2012-03-16 19:49:01 +01:00
|
|
|
|
2014-10-10 23:29:51 +13:00
|
|
|
mThunderSoundDelay = 0.25;
|
|
|
|
}
|
2013-06-19 16:18:43 +02:00
|
|
|
}
|
|
|
|
}
|
2015-04-14 15:55:56 +02:00
|
|
|
//else
|
|
|
|
//mRendering->getSkyManager()->setLightningStrength(0.f);
|
2012-02-23 21:12:06 +01:00
|
|
|
}
|
2014-10-10 23:29:51 +13:00
|
|
|
|
2012-03-09 18:30:03 +01:00
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
mRendering->setAmbientColour(mResult.mAmbientColor);
|
|
|
|
mRendering->setSunColour(mResult.mSunColor);
|
|
|
|
|
|
|
|
mRendering->getSkyManager()->setWeather(mResult);
|
|
|
|
|
|
|
|
// Play sounds
|
2014-12-01 20:31:33 +01:00
|
|
|
if (mPlayingSoundID != mResult.mAmbientLoopSoundID)
|
2012-03-09 18:30:03 +01:00
|
|
|
{
|
2014-12-01 20:31:33 +01:00
|
|
|
stopSounds();
|
|
|
|
if (!mResult.mAmbientLoopSoundID.empty())
|
|
|
|
mAmbientSound = MWBase::Environment::get().getSoundManager()->playSound(mResult.mAmbientLoopSoundID, 1.0, 1.0, MWBase::SoundManager::Play_TypeSfx, MWBase::SoundManager::Play_Loop);
|
2012-03-09 18:30:03 +01:00
|
|
|
|
2014-12-01 20:31:33 +01:00
|
|
|
mPlayingSoundID = mResult.mAmbientLoopSoundID;
|
2012-03-09 18:30:03 +01:00
|
|
|
}
|
2014-12-01 20:31:33 +01:00
|
|
|
if (mAmbientSound.get())
|
|
|
|
mAmbientSound->setVolume(mResult.mAmbientSoundVolume);
|
2013-06-19 16:18:43 +02:00
|
|
|
}
|
|
|
|
|
2014-12-01 20:31:33 +01:00
|
|
|
void WeatherManager::stopSounds()
|
2013-06-19 16:18:43 +02:00
|
|
|
{
|
2014-12-01 20:31:33 +01:00
|
|
|
if (mAmbientSound.get())
|
2012-03-09 18:30:03 +01:00
|
|
|
{
|
2014-12-01 20:31:33 +01:00
|
|
|
MWBase::Environment::get().getSoundManager()->stopSound(mAmbientSound);
|
|
|
|
mAmbientSound.reset();
|
|
|
|
mPlayingSoundID.clear();
|
2013-06-19 16:18:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-20 01:25:52 -05:00
|
|
|
std::string WeatherManager::nextWeather(const ESM::Region* region) const
|
2013-06-19 16:18:43 +02:00
|
|
|
{
|
2013-07-27 07:10:18 -07:00
|
|
|
std::vector<char> probability;
|
|
|
|
|
|
|
|
RegionModMap::const_iterator iter = mRegionMods.find(Misc::StringUtils::lowerCase(region->mId));
|
|
|
|
if(iter != mRegionMods.end())
|
|
|
|
probability = iter->second;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
probability.reserve(10);
|
|
|
|
probability.push_back(region->mData.mClear);
|
|
|
|
probability.push_back(region->mData.mCloudy);
|
|
|
|
probability.push_back(region->mData.mFoggy);
|
|
|
|
probability.push_back(region->mData.mOvercast);
|
|
|
|
probability.push_back(region->mData.mRain);
|
|
|
|
probability.push_back(region->mData.mThunder);
|
|
|
|
probability.push_back(region->mData.mAsh);
|
|
|
|
probability.push_back(region->mData.mBlight);
|
|
|
|
probability.push_back(region->mData.mA);
|
|
|
|
probability.push_back(region->mData.mB);
|
|
|
|
}
|
|
|
|
|
2013-06-19 16:18:43 +02:00
|
|
|
/*
|
|
|
|
* 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).
|
|
|
|
*/
|
|
|
|
|
2015-04-22 17:58:55 +02:00
|
|
|
int chance = Misc::Rng::rollDice(100) + 1; // 1..100
|
2013-06-19 16:18:43 +02:00
|
|
|
int sum = 0;
|
2013-07-29 02:32:08 +02:00
|
|
|
unsigned int i = 0;
|
2013-07-27 07:10:18 -07:00
|
|
|
for (; i < probability.size(); ++i)
|
2013-06-19 16:18:43 +02:00
|
|
|
{
|
|
|
|
sum += probability[i];
|
|
|
|
if (chance < sum)
|
2013-06-28 09:00:21 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (i)
|
|
|
|
{
|
|
|
|
case 1:
|
|
|
|
return "cloudy";
|
|
|
|
case 2:
|
|
|
|
return "foggy";
|
|
|
|
case 3:
|
|
|
|
return "overcast";
|
|
|
|
case 4:
|
|
|
|
return "rain";
|
|
|
|
case 5:
|
|
|
|
return "thunderstorm";
|
|
|
|
case 6:
|
|
|
|
return "ashstorm";
|
|
|
|
case 7:
|
|
|
|
return "blight";
|
|
|
|
case 8:
|
|
|
|
return "snow";
|
|
|
|
case 9:
|
|
|
|
return "blizzard";
|
|
|
|
default: // case 0
|
|
|
|
return "clear";
|
2012-03-09 18:30:03 +01:00
|
|
|
}
|
2012-02-21 22:56:34 +01:00
|
|
|
}
|
2012-02-22 20:12:08 +01:00
|
|
|
|
|
|
|
void WeatherManager::setHour(const float hour)
|
|
|
|
{
|
|
|
|
mHour = hour;
|
|
|
|
}
|
|
|
|
|
2012-02-25 21:34:38 +01:00
|
|
|
unsigned int WeatherManager::getWeatherID() const
|
|
|
|
{
|
|
|
|
// Source: http://www.uesp.net/wiki/Tes3Mod:GetCurrentWeather
|
2013-06-20 21:44:28 +02:00
|
|
|
|
|
|
|
if (mCurrentWeather == "clear")
|
|
|
|
return 0;
|
|
|
|
else if (mCurrentWeather == "cloudy")
|
|
|
|
return 1;
|
|
|
|
else if (mCurrentWeather == "foggy")
|
|
|
|
return 2;
|
|
|
|
else if (mCurrentWeather == "overcast")
|
|
|
|
return 3;
|
|
|
|
else if (mCurrentWeather == "rain")
|
|
|
|
return 4;
|
|
|
|
else if (mCurrentWeather == "thunderstorm")
|
|
|
|
return 5;
|
|
|
|
else if (mCurrentWeather == "ashstorm")
|
|
|
|
return 6;
|
|
|
|
else if (mCurrentWeather == "blight")
|
|
|
|
return 7;
|
|
|
|
else if (mCurrentWeather == "snow")
|
|
|
|
return 8;
|
|
|
|
else if (mCurrentWeather == "blizzard")
|
|
|
|
return 9;
|
|
|
|
|
|
|
|
else
|
|
|
|
return 0;
|
2012-02-25 21:34:38 +01:00
|
|
|
}
|
2012-02-26 11:51:02 +01:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
void WeatherManager::changeWeather(const std::string& region, const unsigned int id)
|
2012-02-26 11:51:02 +01:00
|
|
|
{
|
2013-04-28 05:29:34 +02:00
|
|
|
// make sure this region exists
|
2013-04-28 11:05:01 +02:00
|
|
|
MWBase::Environment::get().getWorld()->getStore().get<ESM::Region>().find(region);
|
2013-04-28 05:29:34 +02:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
std::string weather;
|
|
|
|
if (id==0)
|
|
|
|
weather = "clear";
|
|
|
|
else if (id==1)
|
|
|
|
weather = "cloudy";
|
|
|
|
else if (id==2)
|
|
|
|
weather = "foggy";
|
|
|
|
else if (id==3)
|
|
|
|
weather = "overcast";
|
|
|
|
else if (id==4)
|
|
|
|
weather = "rain";
|
|
|
|
else if (id==5)
|
|
|
|
weather = "thunderstorm";
|
|
|
|
else if (id==6)
|
|
|
|
weather = "ashstorm";
|
|
|
|
else if (id==7)
|
|
|
|
weather = "blight";
|
|
|
|
else if (id==8)
|
|
|
|
weather = "snow";
|
|
|
|
else if (id==9)
|
|
|
|
weather = "blizzard";
|
|
|
|
else
|
|
|
|
weather = "clear";
|
2012-02-26 11:51:02 +01:00
|
|
|
|
2013-06-20 21:44:28 +02:00
|
|
|
mRegionOverrides[Misc::StringUtils::lowerCase(region)] = weather;
|
2013-04-28 05:29:34 +02:00
|
|
|
|
2014-09-14 08:33:41 +02:00
|
|
|
MWWorld::Ptr player = MWBase::Environment::get().getWorld()->getPlayerPtr();
|
|
|
|
if (player.isInCell())
|
|
|
|
{
|
|
|
|
std::string playerRegion = player.getCell()->getCell()->mRegion;
|
|
|
|
if (Misc::StringUtils::ciEqual(region, playerRegion))
|
|
|
|
setWeather(weather);
|
|
|
|
}
|
2012-02-26 11:51:02 +01:00
|
|
|
}
|
2013-05-01 11:42:24 +02:00
|
|
|
|
2013-07-27 07:10:18 -07:00
|
|
|
void WeatherManager::modRegion(const std::string ®ionid, const std::vector<char> &chances)
|
|
|
|
{
|
|
|
|
mRegionMods[Misc::StringUtils::lowerCase(regionid)] = chances;
|
|
|
|
// Start transitioning right away if the region no longer supports the current weather type
|
|
|
|
unsigned int current = getWeatherID();
|
|
|
|
if(current >= chances.size() || chances[current] == 0)
|
|
|
|
mWeatherUpdateTime = 0.0f;
|
|
|
|
}
|
|
|
|
|
2013-05-01 11:42:24 +02:00
|
|
|
float WeatherManager::getWindSpeed() const
|
|
|
|
{
|
|
|
|
return mWindSpeed;
|
|
|
|
}
|
2013-12-10 23:48:49 +01:00
|
|
|
|
2013-12-30 17:53:02 +01:00
|
|
|
bool WeatherManager::isDark() const
|
2013-12-10 23:48:49 +01:00
|
|
|
{
|
2013-12-30 17:53:02 +01:00
|
|
|
bool exterior = (MWBase::Environment::get().getWorld()->isCellExterior()
|
|
|
|
|| MWBase::Environment::get().getWorld()->isCellQuasiExterior());
|
|
|
|
return exterior && (mHour < mSunriseTime || mHour > mNightStart - 1);
|
2013-12-10 23:48:49 +01:00
|
|
|
}
|
2014-01-01 14:57:14 +01:00
|
|
|
|
2014-04-28 11:29:57 +02:00
|
|
|
void WeatherManager::write(ESM::ESMWriter& writer, Loading::Listener& progress)
|
2014-03-20 01:25:52 -05:00
|
|
|
{
|
2014-03-21 01:19:40 -05:00
|
|
|
ESM::WeatherState state;
|
|
|
|
state.mHour = mHour;
|
|
|
|
state.mWindSpeed = mWindSpeed;
|
|
|
|
state.mCurrentWeather = mCurrentWeather;
|
|
|
|
state.mNextWeather = mNextWeather;
|
|
|
|
state.mCurrentRegion = mCurrentRegion;
|
|
|
|
state.mFirstUpdate = mFirstUpdate;
|
|
|
|
state.mRemainingTransitionTime = mRemainingTransitionTime;
|
|
|
|
state.mTimePassed = mTimePassed;
|
|
|
|
|
2014-03-20 01:25:52 -05:00
|
|
|
writer.startRecord(ESM::REC_WTHR);
|
2014-03-21 01:19:40 -05:00
|
|
|
state.save(writer);
|
2014-03-20 01:25:52 -05:00
|
|
|
writer.endRecord(ESM::REC_WTHR);
|
|
|
|
}
|
|
|
|
|
2015-01-22 19:04:59 +01:00
|
|
|
bool WeatherManager::readRecord(ESM::ESMReader& reader, uint32_t type)
|
2014-03-20 01:25:52 -05:00
|
|
|
{
|
|
|
|
if(ESM::REC_WTHR == type)
|
|
|
|
{
|
2014-03-21 01:19:40 -05:00
|
|
|
// load first so that if it fails, we haven't accidentally reset the state below
|
|
|
|
ESM::WeatherState state;
|
|
|
|
state.load(reader);
|
|
|
|
|
|
|
|
// swap in the loaded values now that we can't fail
|
|
|
|
mHour = state.mHour;
|
|
|
|
mWindSpeed = state.mWindSpeed;
|
|
|
|
mCurrentWeather.swap(state.mCurrentWeather);
|
|
|
|
mNextWeather.swap(state.mNextWeather);
|
|
|
|
mCurrentRegion.swap(state.mCurrentRegion);
|
|
|
|
mFirstUpdate = state.mFirstUpdate;
|
|
|
|
mRemainingTransitionTime = state.mRemainingTransitionTime;
|
|
|
|
mTimePassed = state.mTimePassed;
|
2014-03-20 01:25:52 -05:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-01-24 16:26:43 +01:00
|
|
|
void WeatherManager::clear()
|
|
|
|
{
|
|
|
|
stopSounds();
|
|
|
|
mRegionOverrides.clear();
|
|
|
|
mRegionMods.clear();
|
|
|
|
mThunderFlash = 0.0;
|
|
|
|
mThunderChance = 0.0;
|
|
|
|
mThunderChanceNeeded = 50.0;
|
|
|
|
}
|
|
|
|
|
2013-12-29 12:47:44 +01:00
|
|
|
void WeatherManager::switchToNextWeather(bool instantly)
|
|
|
|
{
|
2013-12-31 20:40:23 +01:00
|
|
|
MWBase::World* world = MWBase::Environment::get().getWorld();
|
2014-01-01 16:45:39 +01:00
|
|
|
if (world->isCellExterior() || world->isCellQuasiExterior())
|
2013-12-31 20:40:23 +01:00
|
|
|
{
|
2014-02-21 11:35:46 +01:00
|
|
|
std::string regionstr = Misc::StringUtils::lowerCase(world->getPlayerPtr().getCell()->getCell()->mRegion);
|
2013-12-31 20:40:23 +01:00
|
|
|
|
2014-01-01 16:45:39 +01:00
|
|
|
if (mWeatherUpdateTime <= 0 || regionstr != mCurrentRegion)
|
2013-12-31 20:40:23 +01:00
|
|
|
{
|
2014-01-01 16:45:39 +01:00
|
|
|
mCurrentRegion = regionstr;
|
|
|
|
mWeatherUpdateTime = mHoursBetweenWeatherChanges * 3600;
|
|
|
|
|
|
|
|
std::string weatherType = "clear";
|
2013-12-31 20:40:23 +01:00
|
|
|
|
2014-01-01 16:45:39 +01:00
|
|
|
if (mRegionOverrides.find(regionstr) != mRegionOverrides.end())
|
2013-12-31 20:40:23 +01:00
|
|
|
{
|
2014-01-01 16:45:39 +01:00
|
|
|
weatherType = mRegionOverrides[regionstr];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// get weather probabilities for the current region
|
|
|
|
const ESM::Region *region = world->getStore().get<ESM::Region>().search (regionstr);
|
|
|
|
|
|
|
|
if (region != 0)
|
|
|
|
{
|
|
|
|
weatherType = nextWeather(region);
|
|
|
|
}
|
2013-12-31 20:40:23 +01:00
|
|
|
}
|
|
|
|
|
2014-01-01 16:45:39 +01:00
|
|
|
setWeather(weatherType, instantly);
|
|
|
|
}
|
2013-12-31 20:40:23 +01:00
|
|
|
}
|
2013-12-29 12:47:44 +01:00
|
|
|
}
|
2014-06-24 18:37:38 +02:00
|
|
|
|
|
|
|
bool WeatherManager::isInStorm() const
|
|
|
|
{
|
2014-06-25 18:20:21 +02:00
|
|
|
return mIsStorm;
|
2014-06-24 18:37:38 +02:00
|
|
|
}
|
|
|
|
|
2015-05-12 03:02:15 +02:00
|
|
|
osg::Vec3f WeatherManager::getStormDirection() const
|
2014-06-24 18:37:38 +02:00
|
|
|
{
|
2014-06-26 19:01:49 +02:00
|
|
|
return mStormDirection;
|
2014-06-24 18:37:38 +02:00
|
|
|
}
|
2015-04-14 15:55:56 +02:00
|
|
|
|
|
|
|
void WeatherManager::advanceTime(double hours)
|
|
|
|
{
|
|
|
|
mTimePassed += hours*3600;
|
|
|
|
}
|