// Ryzom - MMORPG Framework <http://dev.ryzom.com/projects/ryzom/>
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
# include "stdpch.h"
# include "light_cycle_manager.h"
# include "weather_manager_client.h"
# include "game_share/light_cycle.h"
# include "game_share/time_weather_season/weather_predict.h"
# include "precipitation.h"
# include "sheet_manager.h"
# include "client_cfg.h"
# include "continent.h"
# include "time_client.h"
# include "interface_v3/interface_manager.h"
//
# include "nel/misc/sheet_id.h"
# include "nel/3d/u_particle_system_instance.h"
//
# include "client_sheets/weather_function_params_sheet.h"
//
# include "r2/editor.h"
# ifdef DEBUG_NEW
# define new DEBUG_NEW
# endif
H_AUTO_DECL ( RZ_Weather )
CLightCycle WorldLightCycle ;
CLightCycleManager LightCycleManager ;
CWeatherManagerClient WeatherManager ;
CWeatherFunctionParamsSheet * WeatherFunctionParams = NULL ;
float ManualWeatherValue = 0.f ;
EGSPD : : CSeason : : TSeason ManualSeasonValue = EGSPD : : CSeason : : Spring ;
bool ForceTrueWeatherValue = false ;
float DayNightCycleHour = 12.f ;
float ForcedDayNightCycleHour = - 1.0f ;
EGSPD : : CSeason : : TSeason CurrSeason = EGSPD : : CSeason : : Invalid ;
EGSPD : : CSeason : : TSeason StartupSeason = EGSPD : : CSeason : : Invalid ;
extern NLPACS : : UGlobalRetriever * GR ;
using namespace NLMISC ;
void releaseWeather ( )
{
if ( WeatherFunctionParams )
delete WeatherFunctionParams ;
WeatherFunctionParams = NULL ;
}
void buildLightCycleDesc ( CLightCycleDesc & dest , EGSPD : : CSeason : : TSeason season )
{
if ( season > = EGSPD : : CSeason : : Invalid )
{
nlwarning ( " Invalid season " ) ;
return ;
}
dest . NumHours = WorldLightCycle . NumHours ;
dest . RealDayLength = WorldLightCycle . RealDayLength ;
dest . MaxNumColorSteps = WorldLightCycle . MaxNumColorSteps ;
const CSeasonLightCycle & slc = WorldLightCycle . SeasonLightCycle [ season ] ;
dest . NightTransitionStartHour = slc . DayToDuskHour ;
dest . NightTransitionEndHour = slc . NightHour ;
dest . DawnTransitionStartHour = slc . NightToDayHour ;
dest . DawnTransitionEndHour = slc . DayHour ;
float intervalDayNight = slc . NightHour > = slc . DayToDuskHour ? slc . NightHour - slc . DayToDuskHour : WorldLightCycle . NumHours - slc . DayToDuskHour + slc . NightHour ;
float intervalDayDusk = slc . DuskToNightHour > = slc . DayToDuskHour ? slc . DuskToNightHour - slc . DayToDuskHour : WorldLightCycle . NumHours - slc . DayToDuskHour + slc . DuskToNightHour ;
if ( intervalDayDusk ! = 0.f ) dest . DuskRatio = intervalDayDusk / intervalDayNight ;
else dest . DuskRatio = 0.5f ;
}
void loadWorldLightCycle ( )
{
CEntitySheet * sheet = SheetMngr . get ( CSheetId ( " ryzom.light_cycle " ) ) ;
nlassert ( sheet ) ;
if ( sheet - > type ( ) ! = CEntitySheet : : LIGHT_CYCLE )
{
nlwarning ( " Bad type for light cycle sheet " ) ;
return ;
}
CLightCycleSheet * lightCycleSheet = ( CLightCycleSheet * ) sheet ;
WorldLightCycle = lightCycleSheet - > LightCycle ;
}
void loadWeatherFunctionParams ( )
{
if ( WeatherFunctionParams = = NULL ) WeatherFunctionParams = new CWeatherFunctionParamsSheet ;
if ( ClientCfg . ManualWeatherSetup )
{
WeatherFunctionParams - > CWeatherFunctionParamsSheetBase : : build ( " ryzom.weather_function_params " ) ;
}
else
{
// Now, just get it from the sheet manager
std : : vector < CSheetId > result ;
CSheetId : : buildIdVector ( result , " weather_function_params " ) ;
if ( result . empty ( ) )
{
nlwarning ( " Couldn't load weather_function_params " ) ;
return ;
}
if ( result . size ( ) ! = 1 )
{
nlwarning ( " Multiple .weather_function_params sheets available, taking first " ) ;
}
CEntitySheet * sheet = SheetMngr . get ( result [ 0 ] ) ;
if ( sheet - > type ( ) ! = CEntitySheet : : WEATHER_FUNCTION_PARAMS )
{
nlwarning ( " Bad type for weather_function_params sheet " ) ;
return ;
}
WeatherFunctionParams = ( CWeatherFunctionParamsSheet * ) sheet ;
}
}
// ***************************************************************************
void updateWeatherManager ( CContinent * continent )
{
H_AUTO_USE ( RZ_Weather )
// build a weather context
CWeatherContext wc ;
wc . GR = NULL ;
wc . LC = & WorldLightCycle ;
wc . WF = continent - > WeatherFunction ;
// Update the weather manager
// NB nico : even when light hour is forced, we use the real time to compute the weather (so that the weather will reflect the real one, even if hour & light is fixed)
if ( ClientCfg . ManualWeatherSetup & & ! ForceTrueWeatherValue )
{
WeatherManager . manualUpdate ( RT . getRyzomDay ( ) , ( float ) RT . getRyzomTime ( ) , wc , ManualWeatherValue , CurrSeason ) ;
}
else
{
WeatherManager . update ( RT . getRyzomDay ( ) , ( float ) RT . getRyzomTime ( ) , wc ) ;
}
}
// ***************************************************************************
void updateWeatherManager ( const NLMISC : : CMatrix & camMatrix , CContinent * continent )
{
H_AUTO_USE ( RZ_Weather )
// build a weather context
CWeatherContext wc ;
wc . GR = GR ;
if ( continent )
wc . WF = continent - > WeatherFunction ;
else
wc . WF = NULL ;
if ( continent )
{
if ( ClientCfg . ManualWeatherSetup & & ! ForceTrueWeatherValue )
{
WeatherManager . manualUpdate ( RT . getRyzomDay ( ) , ( float ) RT . getRyzomTime ( ) , wc , ManualWeatherValue , CurrSeason , camMatrix , * continent ) ;
}
else
{
// NB nico : even when light hour is forced, we use the real time to compute the weather (so that the weather will reflect the real one, even if hour & light is fixed)
WeatherManager . update ( RT . getRyzomDay ( ) , RT . getRyzomTime ( ) , wc , camMatrix , * continent ) ;
}
}
}
///////////////////
// WEATHER VALUE //
///////////////////
float LocalServerWeather = 0.f ; // 'local' server driven weather, trying to reach the weather value given in the database as time passes (for smooth transition)
float ServerWeatherBlendFactor = 0.f ; // blend factor between server driven weather & local computed pseudo random weather (based on the current date, gives same result for all clients)
bool ServerDrivenWeather = false ;
const float WEATHER_BLEND_SPEED = 1.f / 8.f ; // number of seconds to blend betwen weather states
static NLMISC : : CRefPtr < NLMISC : : CCDBNodeLeaf > s_ServerWeatherValueDB ;
// ***************************************************************************
static uint16 getServerWeather ( )
{
CCDBNodeLeaf * node = s_ServerWeatherValueDB ? & * s_ServerWeatherValueDB
: & * ( s_ServerWeatherValueDB = NLGUI : : CDBManager : : getInstance ( ) - > getDbProp ( " SERVER:WEATHER:VALUE " ) ) ;
if ( ! node ) return 0 ;
return ( uint16 ) node - > getValue16 ( ) ;
}
// ***************************************************************************
void updateDBDrivenWeatherValue ( )
{
uint16 dbWeather = getServerWeather ( ) ;
if ( dbWeather = = 0 )
{
// not server driven
ServerDrivenWeather = false ;
NLMISC : : incrementalBlend ( ServerWeatherBlendFactor , 0.f , WEATHER_BLEND_SPEED * DT ) ;
}
else
{
float targetWeather = ( dbWeather - 1 ) / 1022.f ;
if ( ! ServerDrivenWeather )
{
// just switched to server driven weather ?
LocalServerWeather = targetWeather ;
ServerDrivenWeather = true ;
}
// slowly blend from predicted weather to server driven weather
NLMISC : : incrementalBlend ( ServerWeatherBlendFactor , 1.f , WEATHER_BLEND_SPEED * DT ) ;
// slowly blend between server driven weathers as values in the db changes
NLMISC : : incrementalBlend ( LocalServerWeather , targetWeather , WEATHER_BLEND_SPEED * DT ) ;
}
}
// ***************************************************************************
float getBlendedWeather ( uint64 day ,
float hour ,
const CWeatherFunctionParamsSheetBase & wfp ,
const CWeatherFunction wf [ EGSPD : : CSeason : : Invalid ]
)
{
return NLMISC : : blend ( CPredictWeather : : predictWeather ( day , hour , wfp , wf ) , LocalServerWeather , ServerWeatherBlendFactor ) ;
}
uint8 getSeasonDBValue ( )
{
CCDBNodeLeaf * serverSeason = NLGUI : : CDBManager : : getInstance ( ) - > getDbProp ( " SERVER:SEASON:VALUE " ) ;
return serverSeason ? serverSeason - > getValue8 ( ) : 0 ;
}
////////////
// SEASON //
////////////
uint8 ServerSeasonValue = 0 ;
bool ServerSeasonReceived = false ;
static bool WaitingServerSeason = false ;
static R2 : : CEditor : : TMode OldR2EDMode = R2 : : CEditor : : NotInitialized ;
static EGSPD : : CSeason : : TSeason LastSeason = EGSPD : : CSeason : : Unknown ;
// ***************************************************************************
EGSPD : : CSeason : : TSeason computeCurrSeason ( )
{
// Note : when season isn't imposed by the server or by the editor, we use
// the season found in StartupSeason
// This mean that when the season change due to time, it won't be updated on the client
// This is a deliberate choice to avoid possibly disadvantaging freeze on the client while in combat
// To have this done correctly, the server should be aware of the change, and freeze action accordingly.
// When entering a new area, the 'StartupSeason' variable is updated however, because the player is safe at that time.
// (see Continent::select)
if ( ! ClientCfg . R2EDEnabled | | ! ClientCfg . Local ) // ... Standard client
{
// standard case
if ( ServerSeasonValue ! = 0 )
{
ServerSeasonReceived = false ;
// server driven season
LastSeason = ( EGSPD : : CSeason : : TSeason ) ( ( ServerSeasonValue - 1 ) & 3 ) ;
}
else
{
LastSeason = ClientCfg . ManualWeatherSetup ? ManualSeasonValue : StartupSeason ;
}
return LastSeason ;
}
// When we change act( different location) the Server TP the player at the good location
// So the manual setting of season is useless (R2::getEditor().getSeason()) because
// there is a huge risk to (Change to 2 times season)
// eg Act1 Season1, Island1, Act2 Season2, Island2
// If server tp occurs before, client received change of Season: there will be a unecessary TP
// When in edition mode, the editor tell what is the season (it is local and given
// by each act of the current scenario, so ignore server value here...)
if ( R2 : : getEditor ( ) . getMode ( ) = = R2 : : CEditor : : EditionMode )
{
OldR2EDMode = R2 : : getEditor ( ) . getMode ( ) ;
switch ( R2 : : getEditor ( ) . getSeason ( ) )
{
case R2 : : CEditor : : Automatic : LastSeason = StartupSeason ; break ;
case R2 : : CEditor : : Spring : LastSeason = EGSPD : : CSeason : : Spring ; break ;
case R2 : : CEditor : : Summer : LastSeason = EGSPD : : CSeason : : Summer ; break ;
case R2 : : CEditor : : Autumn : LastSeason = EGSPD : : CSeason : : Autumn ; break ;
case R2 : : CEditor : : Winter : LastSeason = EGSPD : : CSeason : : Winter ; break ;
case R2 : : CEditor : : UnknownSeason :
if ( LastSeason = = EGSPD : : CSeason : : Unknown )
{
// don't change the season unless this is the first call
LastSeason = StartupSeason ; // default
}
break ;
default :
nlassert ( 0 ) ;
break ;
}
}
// Leaving edit -> Test 'mode' or GoingToTest 'mode' :
if ( OldR2EDMode = = R2 : : CEditor : : EditionMode )
{
// a transition from edition mode to test mode has been detected :
// At some point the server will impulse the current season, be it the same than the current one
// Until this occur, we just retain the last season that was set in the editor, because
// season change are VERY costly (several seconds freeze) and we don't want the season to change twice in a row just because we don't
// know its real value at this point.
WaitingServerSeason = true ;
}
OldR2EDMode = R2 : : getEditor ( ) . getMode ( ) ;
if ( WaitingServerSeason & & ! ServerSeasonReceived )
{
return LastSeason ;
}
ServerSeasonReceived = false ;
WaitingServerSeason = false ;
// back to standard case
// At this point we ignore the 'ClientCfg.ManualWeatherSetup' flag, however, because it may be set
// when the "change weather" dialog is shown, but we only want to preview weather change, not the season change.
if ( ServerSeasonValue ! = 0 )
{
// server driven season
LastSeason = ( EGSPD : : CSeason : : TSeason ) ( ( ServerSeasonValue - 1 ) & 3 ) ;
}
else
{
LastSeason = StartupSeason ;
}
return LastSeason ;
}