Improve thunderstorm support.

Reversed settings for thunderstorms. Added thunder support to all
weather types. Implemented a simple lightning flash effect similar to
MW.
sceneinput
slothlife 9 years ago
parent 7a96a04b75
commit 29d74f0249

@ -1181,20 +1181,6 @@ void SkyManager::setSecundaState(const MoonState& state)
mSecunda->setState(state); mSecunda->setState(state);
} }
void SkyManager::setLightningStrength(const float factor)
{
if (!mCreated) return;
/*
if (factor > 0.f)
{
mLightning->setDiffuseColour (ColourValue(2*factor, 2*factor, 2*factor));
mLightning->setVisible(true);
}
else
mLightning->setVisible(false);
*/
}
void SkyManager::setDate(int day, int month) void SkyManager::setDate(int day, int month)
{ {
mDay = day; mDay = day;

@ -140,8 +140,6 @@ namespace MWRender
void setMasserState(const MoonState& state); void setMasserState(const MoonState& state);
void setSecundaState(const MoonState& state); void setSecundaState(const MoonState& state);
void setLightningStrength(const float factor);
void setGlare(const float glare); void setGlare(const float glare);
void setGlareEnabled(bool enabled); void setGlareEnabled(bool enabled);

@ -80,7 +80,16 @@ Weather::Weather(const std::string& name,
, mRainEffect(fallback.getFallbackBool("Weather_" + name + "_Using_Precip") ? "meshes\\raindrop.nif" : "") , mRainEffect(fallback.getFallbackBool("Weather_" + name + "_Using_Precip") ? "meshes\\raindrop.nif" : "")
, mTransitionDelta(fallback.getFallbackFloat("Weather_" + name + "_Transition_Delta")) , mTransitionDelta(fallback.getFallbackFloat("Weather_" + name + "_Transition_Delta"))
, mCloudsMaximumPercent(fallback.getFallbackFloat("Weather_" + name + "_Clouds_Maximum_Percent")) , mCloudsMaximumPercent(fallback.getFallbackFloat("Weather_" + name + "_Clouds_Maximum_Percent"))
{ , 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");
/* /*
Unhandled: Unhandled:
Rain Diameter=600 ? Rain Diameter=600 ?
@ -98,12 +107,66 @@ float Weather::transitionDelta() const
return mTransitionDelta; return mTransitionDelta;
} }
float Weather::cloudBlendFactor(float transitionRatio) const float Weather::cloudBlendFactor(const float transitionRatio) const
{ {
// Clouds Maximum Percent affects how quickly the sky transitions from one sky texture to the next. // Clouds Maximum Percent affects how quickly the sky transitions from one sky texture to the next.
return transitionRatio / mCloudsMaximumPercent; return transitionRatio / mCloudsMaximumPercent;
} }
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);
}
RegionWeather::RegionWeather(const ESM::Region& region) RegionWeather::RegionWeather(const ESM::Region& region)
: mWeather(invalidWeatherID) : mWeather(invalidWeatherID)
, mChances() , mChances()
@ -378,19 +441,9 @@ WeatherManager::WeatherManager(MWRender::RenderingManager& rendering, const MWWo
, mWeatherSettings() , mWeatherSettings()
, mMasser("Masser", fallback) , mMasser("Masser", fallback)
, mSecunda("Secunda", fallback) , mSecunda("Secunda", fallback)
, mThunderFrequency(fallback.getFallbackFloat("Weather_Thunderstorm_Thunder_Frequency"))
, mThunderThreshold(fallback.getFallbackFloat("Weather_Thunderstorm_Thunder_Threshold"))
, mThunderSoundID0(fallback.getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_0"))
, mThunderSoundID1(fallback.getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_1"))
, mThunderSoundID2(fallback.getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_2"))
, mThunderSoundID3(fallback.getFallbackString("Weather_Thunderstorm_Thunder_Sound_ID_3"))
, mWindSpeed(0.f) , mWindSpeed(0.f)
, mIsStorm(false) , mIsStorm(false)
, mStormDirection(0,1,0) , mStormDirection(0,1,0)
, mThunderSoundDelay(0.25)
, mThunderFlash(0)
, mThunderChance(0)
, mThunderChanceNeeded(50)
, mCurrentRegion() , mCurrentRegion()
, mTimePassed(0) , mTimePassed(0)
, mFastForward(false) , mFastForward(false)
@ -515,12 +568,11 @@ void WeatherManager::update(float duration, bool paused)
if(!exterior) if(!exterior)
{ {
mRendering.setSkyEnabled(false); mRendering.setSkyEnabled(false);
//mRendering->getSkyManager()->setLightningStrength(0.f);
stopSounds(); stopSounds();
return; return;
} }
calculateWeatherResult(time.getHour()); calculateWeatherResult(time.getHour(), duration, paused);
mWindSpeed = mResult.mWindSpeed; mWindSpeed = mResult.mWindSpeed;
mIsStorm = mResult.mIsStorm; mIsStorm = mResult.mIsStorm;
@ -536,8 +588,6 @@ void WeatherManager::update(float duration, bool paused)
mRendering.getSkyManager()->setStormDirection(mStormDirection); mRendering.getSkyManager()->setStormDirection(mStormDirection);
} }
mRendering.configureFog(mResult.mFogDepth, mResult.mFogColor);
// disable sun during night // disable sun during night
if (time.getHour() >= mNightStart || time.getHour() <= mSunriseTime) if (time.getHour() >= mNightStart || time.getHour() <= mSunriseTime)
mRendering.getSkyManager()->sunDisable(); mRendering.getSkyManager()->sunDisable();
@ -577,56 +627,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));
if (!paused) mRendering.configureFog(mResult.mFogDepth, mResult.mFogColor);
{
if(mCurrentWeather == 5 && !inTransition())
{
if (mThunderFlash > 0)
{
// play the sound after a delay
mThunderSoundDelay -= duration;
if (mThunderSoundDelay <= 0)
{
// pick a random sound
int sound = Misc::Rng::rollDice(4);
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;
//if (mThunderFlash > 0)
//mRendering->getSkyManager()->setLightningStrength( mThunderFlash / mThunderThreshold );
//else
{
mThunderChanceNeeded = static_cast<float>(Misc::Rng::rollDice(100));
mThunderChance = 0;
//mRendering->getSkyManager()->setLightningStrength( 0.f );
}
}
else
{
// no thunder active
mThunderChance += duration*4; // chance increases by 4 percent every second
if (mThunderChance >= mThunderChanceNeeded)
{
mThunderFlash = mThunderThreshold;
//mRendering->getSkyManager()->setLightningStrength( mThunderFlash / mThunderThreshold );
mThunderSoundDelay = 0.25;
}
}
}
//else
//mRendering->getSkyManager()->setLightningStrength(0.f);
}
mRendering.setAmbientColour(mResult.mAmbientColor); mRendering.setAmbientColour(mResult.mAmbientColor);
mRendering.setSunColour(mResult.mSunColor); mRendering.setSunColour(mResult.mSunColor);
@ -765,10 +766,6 @@ void WeatherManager::clear()
{ {
stopSounds(); stopSounds();
mThunderFlash = 0.0;
mThunderChance = 0.0;
mThunderChanceNeeded = 50.0;
mCurrentRegion = ""; mCurrentRegion = "";
mTimePassed = 0.0f; mTimePassed = 0.0f;
mWeatherUpdateTime = 0.0f; mWeatherUpdateTime = 0.0f;
@ -921,16 +918,32 @@ inline void WeatherManager::addWeatherTransition(const int weatherID)
} }
} }
inline void WeatherManager::calculateWeatherResult(const float gameHour) inline void WeatherManager::calculateWeatherResult(const float gameHour,
const float elapsedSeconds,
const bool isPaused)
{ {
float flash = 0.0f;
if(!inTransition()) if(!inTransition())
{ {
calculateResult(mCurrentWeather, gameHour); calculateResult(mCurrentWeather, gameHour);
flash = mWeatherSettings[mCurrentWeather].calculateThunder(1.0f, elapsedSeconds, isPaused);
} }
else else
{ {
calculateTransitionResult(1 - mTransitionFactor, gameHour); calculateTransitionResult(1 - mTransitionFactor, gameHour);
} 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;
} }
inline void WeatherManager::calculateResult(const int weatherID, const float gameHour) inline void WeatherManager::calculateResult(const int weatherID, const float gameHour)

@ -113,11 +113,27 @@ namespace MWWorld
// is broken in the vanilla game and was disabled. // is broken in the vanilla game and was disabled.
float transitionDelta() const; float transitionDelta() const;
float cloudBlendFactor(float transitionRatio) const; float cloudBlendFactor(const float transitionRatio) const;
float calculateThunder(const float transitionRatio, const float elapsedSeconds, const bool isPaused);
private: private:
float mTransitionDelta; float mTransitionDelta;
float mCloudsMaximumPercent; float mCloudsMaximumPercent;
// Note: In MW, only thunderstorms support these attributes, but in the interest of making weather more
// flexible, these settings are imported for all weather types. Only thunderstorms will normally have any
// non-zero values.
float mThunderFrequency;
float mThunderThreshold;
std::string mThunderSoundID[4];
float mFlashDecrement;
float mFlashBrightness;
void flashDecrement(const float elapsedSeconds);
float thunderChance(const float transitionRatio, const float elapsedSeconds) const;
void lightningAndThunder(void);
}; };
/// A class for storing a region's weather. /// A class for storing a region's weather.
@ -242,22 +258,10 @@ namespace MWWorld
MoonModel mMasser; MoonModel mMasser;
MoonModel mSecunda; MoonModel mSecunda;
float mThunderFrequency;
float mThunderThreshold;
std::string mThunderSoundID0;
std::string mThunderSoundID1;
std::string mThunderSoundID2;
std::string mThunderSoundID3;
float mWindSpeed; float mWindSpeed;
bool mIsStorm; bool mIsStorm;
osg::Vec3f mStormDirection; osg::Vec3f mStormDirection;
float mThunderSoundDelay;
float mThunderFlash;
float mThunderChance;
float mThunderChanceNeeded;
std::string mCurrentRegion; std::string mCurrentRegion;
float mTimePassed; float mTimePassed;
bool mFastForward; bool mFastForward;
@ -288,7 +292,7 @@ namespace MWWorld
bool inTransition(); bool inTransition();
void addWeatherTransition(const int weatherID); void addWeatherTransition(const int weatherID);
void calculateWeatherResult(const float gameHour); void calculateWeatherResult(const float gameHour, const float elapsedSeconds, const bool isPaused);
void calculateResult(const int weatherID, const float gameHour); void calculateResult(const int weatherID, const float gameHour);
void calculateTransitionResult(const float factor, const float gameHour); void calculateTransitionResult(const float factor, const float gameHour);
}; };

Loading…
Cancel
Save