1
0
Fork 0

Clean up the timeofday setting code a bit more, consolodate the solver

functions (note to Norman: I looked at the web page you listed and that
looks like a good idea, but I don't have time right now to go through and
debug an entirely new routine.  What we have works well enough for now I hope!)
This commit is contained in:
curt 2003-09-17 15:49:15 +00:00
parent 15f45ad9a5
commit a217c563ba
4 changed files with 128 additions and 193 deletions

View file

@ -501,30 +501,57 @@ do_timeofday (const SGPropertyNode * arg)
// cout << "orig_warp = " << orig_warp << endl; // cout << "orig_warp = " << orig_warp << endl;
int warp = 0; int warp = 0;
if ( offset_type == "noon" ) { if ( offset_type == "real" ) {
warp = fgTimeSecondsUntilNoon( cur_time, warp = 0;
longitude->getDoubleValue() } else if ( offset_type == "dawn" ) {
* SGD_DEGREES_TO_RADIANS, warp = fgTimeSecondsUntilSunAngle( cur_time,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS );
} else if ( offset_type == "midnight" ) {
warp = fgTimeSecondsUntilMidnight( cur_time,
longitude->getDoubleValue() longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS, * SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue() latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS ); * SGD_DEGREES_TO_RADIANS,
} else if ( offset_type == "dawn" ) { 90.0, true );
warp = fgTimeSecondsUntilDawn( cur_time, } else if ( offset_type == "morning" ) {
longitude->getDoubleValue() warp = fgTimeSecondsUntilSunAngle( cur_time,
* SGD_DEGREES_TO_RADIANS, longitude->getDoubleValue()
latitude->getDoubleValue() * SGD_DEGREES_TO_RADIANS,
* SGD_DEGREES_TO_RADIANS ); latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
75.0, true );
} else if ( offset_type == "noon" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
0.0, true );
} else if ( offset_type == "afternoon" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
60.0, false );
} else if ( offset_type == "dusk" ) { } else if ( offset_type == "dusk" ) {
warp = fgTimeSecondsUntilDusk( cur_time, warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue() longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS, * SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue() latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS ); * SGD_DEGREES_TO_RADIANS,
90.0, false );
} else if ( offset_type == "evening" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
100.0, false );
} else if ( offset_type == "midnight" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
180.0, false );
} }
// cout << "warp = " << warp << endl; // cout << "warp = " << warp << endl;
globals->set_warp( orig_warp + warp ); globals->set_warp( orig_warp + warp );

View file

@ -1286,32 +1286,60 @@ void fgInitTimeOffset() {
// Okay, we now have several possible scenarios // Okay, we now have several possible scenarios
int offset = fgGetInt("/sim/startup/time-offset"); int offset = fgGetInt("/sim/startup/time-offset");
int warp = 0;
const string &offset_type = fgGetString("/sim/startup/time-offset-type"); const string &offset_type = fgGetString("/sim/startup/time-offset-type");
if ( offset_type == "noon" ) {
warp = fgTimeSecondsUntilNoon( cur_time, int warp = 0;
longitude->getDoubleValue() if ( offset_type == "real" ) {
* SGD_DEGREES_TO_RADIANS, warp = 0;
latitude->getDoubleValue() } else if ( offset_type == "dawn" ) {
* SGD_DEGREES_TO_RADIANS ); warp = fgTimeSecondsUntilSunAngle( cur_time,
} else if ( offset_type == "midnight" ) {
warp = fgTimeSecondsUntilMidnight( cur_time,
longitude->getDoubleValue() longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS, * SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue() latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS ); * SGD_DEGREES_TO_RADIANS,
} else if ( offset_type == "dawn" ) { 90.0, true );
warp = fgTimeSecondsUntilDawn( cur_time, } else if ( offset_type == "morning" ) {
longitude->getDoubleValue() warp = fgTimeSecondsUntilSunAngle( cur_time,
* SGD_DEGREES_TO_RADIANS, longitude->getDoubleValue()
latitude->getDoubleValue() * SGD_DEGREES_TO_RADIANS,
* SGD_DEGREES_TO_RADIANS ); latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
75.0, true );
} else if ( offset_type == "noon" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
0.0, true );
} else if ( offset_type == "afternoon" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
60.0, false );
} else if ( offset_type == "dusk" ) { } else if ( offset_type == "dusk" ) {
warp = fgTimeSecondsUntilDusk( cur_time, warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue() longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS, * SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue() latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS ); * SGD_DEGREES_TO_RADIANS,
90.0, false );
} else if ( offset_type == "evening" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
100.0, false );
} else if ( offset_type == "midnight" ) {
warp = fgTimeSecondsUntilSunAngle( cur_time,
longitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
latitude->getDoubleValue()
* SGD_DEGREES_TO_RADIANS,
180.0, false );
} else if ( offset_type == "system-offset" ) { } else if ( offset_type == "system-offset" ) {
warp = offset; warp = offset;
} else if ( offset_type == "gmt-offset" ) { } else if ( offset_type == "gmt-offset" ) {

View file

@ -81,47 +81,18 @@ static double sun_angle( const SGTime &t, sgVec3 world_up,
/** /**
* Given the current unix time in seconds, calculate seconds to noon * Given the current unix time in seconds, calculate seconds to the
* specified sun angle (relative to straight up.) Also specify if we
* want the angle while the sun is ascending or descending. For
* instance noon is when the sun angle is 0 (or the closest it can
* get.) Dusk is when the sun angle is 90 and descending. Dawn is
* when the sun angle is 90 and ascending.
*/ */
time_t fgTimeSecondsUntilNoon( time_t cur_time, time_t fgTimeSecondsUntilSunAngle( time_t cur_time,
double lon_rad,
double lat_rad )
{
// cout << "location = " << lon_rad * SG_RADIANS_TO_DEGREES << ", "
// << lat_rad * SG_RADIANS_TO_DEGREES << endl;
Point3D geod( lon_rad, lat_rad, 0 );
Point3D tmp = sgGeodToCart( geod );
sgVec3 world_up;
sgSetVec3( world_up, tmp.x(), tmp.y(), tmp.z() );
SGTime t = SGTime( lon_rad, lat_rad, "", 0 );
double best_angle = 180.0;
time_t best_time = cur_time;
for ( time_t secs = cur_time - half_day_secs;
secs < cur_time + half_day_secs;
secs += step_secs )
{
t.update( lon_rad, lat_rad, secs, 0 );
double angle = sun_angle( t, world_up, lon_rad, lat_rad );
if ( angle < best_angle ) {
// cout << "best angle = " << angle << " offset = "
// << secs - cur_time << endl;
best_angle = angle;
best_time = secs;
}
}
return best_time - cur_time;
}
/**
* Given the current unix time in seconds, calculate seconds to midnight
*/
time_t fgTimeSecondsUntilMidnight( time_t cur_time,
double lon_rad, double lon_rad,
double lat_rad ) double lat_rad,
double target_angle_deg,
bool ascending )
{ {
// cout << "location = " << lon_rad * SG_RADIANS_TO_DEGREES << ", " // cout << "location = " << lon_rad * SG_RADIANS_TO_DEGREES << ", "
// << lat_rad * SG_RADIANS_TO_DEGREES << endl; // << lat_rad * SG_RADIANS_TO_DEGREES << endl;
@ -131,43 +102,7 @@ time_t fgTimeSecondsUntilMidnight( time_t cur_time,
sgSetVec3( world_up, tmp.x(), tmp.y(), tmp.z() ); sgSetVec3( world_up, tmp.x(), tmp.y(), tmp.z() );
SGTime t = SGTime( lon_rad, lat_rad, "", 0 ); SGTime t = SGTime( lon_rad, lat_rad, "", 0 );
double best_angle = 0.0; double best_diff = 180.0;
time_t best_time = cur_time;
for ( time_t secs = cur_time - half_day_secs;
secs < cur_time + half_day_secs;
secs += step_secs )
{
t.update( lon_rad, lat_rad, secs, 0 );
double angle = sun_angle( t, world_up, lon_rad, lat_rad );
if ( angle > best_angle ) {
// cout << "best angle = " << angle << " offset = "
// << secs - cur_time << endl;
best_angle = angle;
best_time = secs;
}
}
return best_time - cur_time;
}
/**
* Given the current unix time in seconds, calculate seconds to dusk
*/
time_t fgTimeSecondsUntilDusk( time_t cur_time,
double lon_rad,
double lat_rad )
{
// cout << "location = " << lon_rad * SG_RADIANS_TO_DEGREES << ", "
// << lat_rad * SG_RADIANS_TO_DEGREES << endl;
Point3D geod( lon_rad, lat_rad, 0 );
Point3D tmp = sgGeodToCart( geod );
sgVec3 world_up;
sgSetVec3( world_up, tmp.x(), tmp.y(), tmp.z() );
SGTime t = SGTime( lon_rad, lat_rad, "", 0 );
double best_diff = 90.0;
double last_angle = -99999.0; double last_angle = -99999.0;
time_t best_time = cur_time; time_t best_time = cur_time;
@ -176,10 +111,17 @@ time_t fgTimeSecondsUntilDusk( time_t cur_time,
secs += step_secs ) secs += step_secs )
{ {
t.update( lon_rad, lat_rad, secs, 0 ); t.update( lon_rad, lat_rad, secs, 0 );
double angle = sun_angle( t, world_up, lon_rad, lat_rad ); double angle_deg = sun_angle( t, world_up, lon_rad, lat_rad );
double diff = fabs( angle - 90.0 ); double diff = fabs( angle_deg - target_angle_deg );
if ( diff < best_diff ) { if ( diff < best_diff ) {
if ( last_angle <= 180.0 && ( last_angle < angle ) ) { if ( last_angle <= 180.0 && ascending
&& ( last_angle > angle_deg ) ) {
// cout << "best angle = " << angle << " offset = "
// << secs - cur_time << endl;
best_diff = diff;
best_time = secs;
} else if ( last_angle <= 180.0 && !ascending
&& ( last_angle < angle_deg ) ) {
// cout << "best angle = " << angle << " offset = " // cout << "best angle = " << angle << " offset = "
// << secs - cur_time << endl; // << secs - cur_time << endl;
best_diff = diff; best_diff = diff;
@ -187,49 +129,7 @@ time_t fgTimeSecondsUntilDusk( time_t cur_time,
} }
} }
last_angle = angle; last_angle = angle_deg;
}
return best_time - cur_time;
}
/**
* Given the current unix time in seconds, calculate seconds to dawn
*/
time_t fgTimeSecondsUntilDawn( time_t cur_time,
double lon_rad,
double lat_rad )
{
// cout << "location = " << lon_rad * SG_RADIANS_TO_DEGREES << ", "
// << lat_rad * SG_RADIANS_TO_DEGREES << endl;
Point3D geod( lon_rad, lat_rad, 0 );
Point3D tmp = sgGeodToCart( geod );
sgVec3 world_up;
sgSetVec3( world_up, tmp.x(), tmp.y(), tmp.z() );
SGTime t = SGTime( lon_rad, lat_rad, "", 0 );
double best_diff = 90.0;
double last_angle = -99999.0;
time_t best_time = cur_time;
for ( time_t secs = cur_time - half_day_secs;
secs < cur_time + half_day_secs;
secs += step_secs )
{
t.update( lon_rad, lat_rad, secs, 0 );
double angle = sun_angle( t, world_up, lon_rad, lat_rad );
double diff = fabs( angle - 90.0 );
if ( diff < best_diff ) {
if ( last_angle <= 180.0 && ( last_angle > angle ) ) {
// cout << "best angle = " << angle << " offset = "
// << secs - cur_time << endl;
best_diff = diff;
best_time = secs;
}
}
last_angle = angle;
} }
return best_time - cur_time; return best_time - cur_time;

View file

@ -41,37 +41,17 @@
#endif #endif
/** /**
* Given the current unix time in seconds, calculate seconds to * Given the current unix time in seconds, calculate seconds to the
* highest sun angle. * specified sun angle (relative to straight up.) Also specify if we
* want the angle while the sun is ascending or descending. For
* instance noon is when the sun angle is 0 (or the closest it can
* get.) Dusk is when the sun angle is 90 and descending. Dawn is
* when the sun angle is 90 and ascending.
*/ */
time_t fgTimeSecondsUntilNoon( time_t cur_time, time_t fgTimeSecondsUntilSunAngle( time_t cur_time,
double lon_rad,
double lat_rad );
/**
* Given the current unix time in seconds, calculate seconds to lowest
* sun angle.
*/
time_t fgTimeSecondsUntilMidnight( time_t cur_time,
double lon_rad, double lon_rad,
double lat_rad ); double lat_rad,
double target_angle_deg,
/** bool ascending );
* Given the current unix time in seconds, calculate seconds to dusk
*/
time_t fgTimeSecondsUntilDusk( time_t cur_time,
double lon_rad,
double lat_rad );
/**
* Given the current unix time in seconds, calculate seconds to dawn
*/
time_t fgTimeSecondsUntilDawn( time_t cur_time,
double lon_rad,
double lat_rad );
#endif /* _SUNSOLVER_HXX */ #endif /* _SUNSOLVER_HXX */