JSBSim Flight Dynamics Model 1.2.2 (22 Mar 2025)
An Open Source Flight Dynamics and Control Software Library in C++
Loading...
Searching...
No Matches
FGScript Class Reference

Detailed Description

Encapsulates the JSBSim scripting capability.

Scripting support provided via FGScript.

There is support for scripting provided in the FGScript class. Commands are specified using the Scripting Directives for JSBSim. The script file is in XML format. A test condition (or conditions) can be set up in an event in a script and when the condition evaluates to true, the specified action[s] is/are taken. An event can be persistent, meaning that at every time the test condition first evaluates to true (toggling from false to true) then the specified set actions take place. An event can also be defined to execute or evaluate continuously while the condition is true. When the set of tests evaluates to true for a given condition, an item may be set to another value. This value may be a value, or a delta value, and the change from the current value to the new value can be either via a step action, a ramp, or an exponential approach. The speed of a ramp or exponential approach is specified via the time constant. Here is an example illustrating the format of the script file:

<?xml version="1.0"?>
<runscript name="C172-01A takeoff run">
<!--
This run is for testing the C172 altitude hold autopilot
-->
<use aircraft="c172x"/>
<use initialize="reset00"/>
<run start="0.0" end="3000" dt="0.0083333">
<event name="engine start">
<notify/>
<condition>
sim-time-sec >= 0.25
</condition>
<set name="fcs/throttle-cmd-norm" value="1.0" action="FG_RAMP" tc ="0.5"/>
<set name="fcs/mixture-cmd-norm" value="0.87" action="FG_RAMP" tc ="0.5"/>
<set name="propulsion/magneto_cmd" value="3"/>
<set name="propulsion/starter_cmd" value="1"/>
</event>
<event name="set heading hold">
<!-- Set Heading when reach 5 ft -->
<notify/>
<condition>
position/h-agl-ft >= 5
</condition>
<set name="ap/heading_setpoint" value="200"/>
<set name="ap/attitude_hold" value="0"/>
<set name="ap/heading_hold" value="1"/>
</event>
<event name="set autopilot">
<!-- Set Autopilot for 20 ft -->
<notify/>
<condition>
aero/qbar-psf >= 4
</condition>
<set name="ap/altitude_setpoint" value="100.0" action="FG_EXP" tc ="2.0"/>
<set name="ap/altitude_hold" value="1"/>
<set name="fcs/flap-cmd-norm" value=".33"/>
</event>
<event name="set autopilot 2" persistent="true">
<!-- Set Autopilot for 6000 ft -->
<notify/>
<condition>
aero/qbar-psf > 5
</condition>
<set name="ap/altitude_setpoint" value="6000.0"/>
</event>
<event name="Time Notify">
<notify/>
<condition> sim-time-sec >= 500 </condition>
</event>
<event name="Time Notify">
<notify/>
<condition> sim-time-sec >= 1000 </condition>
</event>
</run>
</runscript>

The first line must always be present - it identifies the file as an XML format file. The second line identifies this file as a script file, and gives a descriptive name to the script file. Comments are next, delineated by the <!– and –> symbols. The aircraft and initialization files to be used are specified in the "use" lines. Next, comes the "run" section, where the conditions are described in "event" clauses.

Author
Jon S. Berndt

Definition at line 167 of file FGScript.h.

#include <FGScript.h>

+ Inheritance diagram for FGScript:
+ Collaboration diagram for FGScript:

Public Member Functions

 FGScript (FGFDMExec *exec)
 Default constructor.
 
 ~FGScript ()
 Default destructor.
 
bool LoadScript (const SGPath &script, double default_dT, const SGPath &initfile)
 Loads a script to drive JSBSim (usually in standalone mode).
 
void ResetEvents (void)
 
bool RunScript (void)
 This function is called each pass through the executive Run() method IF scripting is enabled.
 
- Public Member Functions inherited from FGJSBBase
 FGJSBBase ()
 Constructor for FGJSBBase.
 
virtual ~FGJSBBase ()
 Destructor for FGJSBBase.
 
void disableHighLighting (void)
 Disables highlighting in the console output.
 

Additional Inherited Members

- Public Types inherited from FGJSBBase
enum  { eL = 1 , eM , eN }
 Moments L, M, N. More...
 
enum  { eP = 1 , eQ , eR }
 Rates P, Q, R. More...
 
enum  { eU = 1 , eV , eW }
 Velocities U, V, W. More...
 
enum  { eX = 1 , eY , eZ }
 Positions X, Y, Z. More...
 
enum  { ePhi = 1 , eTht , ePsi }
 Euler angles Phi, Theta, Psi. More...
 
enum  { eDrag = 1 , eSide , eLift }
 Stability axis forces, Drag, Side force, Lift. More...
 
enum  { eRoll = 1 , ePitch , eYaw }
 Local frame orientation Roll, Pitch, Yaw. More...
 
enum  { eNorth = 1 , eEast , eDown }
 Local frame position North, East, Down. More...
 
enum  { eLat = 1 , eLong , eRad }
 Locations Radius, Latitude, Longitude. More...
 
enum  {
  inNone = 0 , inDegrees , inRadians , inMeters ,
  inFeet
}
 Conversion specifiers. More...
 
- Static Public Member Functions inherited from FGJSBBase
static const std::string & GetVersion (void)
 Returns the version number of JSBSim.
 
static constexpr double KelvinToFahrenheit (double kelvin)
 Converts from degrees Kelvin to degrees Fahrenheit.
 
static constexpr double CelsiusToRankine (double celsius)
 Converts from degrees Celsius to degrees Rankine.
 
static constexpr double RankineToCelsius (double rankine)
 Converts from degrees Rankine to degrees Celsius.
 
static constexpr double KelvinToRankine (double kelvin)
 Converts from degrees Kelvin to degrees Rankine.
 
static constexpr double RankineToKelvin (double rankine)
 Converts from degrees Rankine to degrees Kelvin.
 
static constexpr double FahrenheitToCelsius (double fahrenheit)
 Converts from degrees Fahrenheit to degrees Celsius.
 
static constexpr double CelsiusToFahrenheit (double celsius)
 Converts from degrees Celsius to degrees Fahrenheit.
 
static constexpr double CelsiusToKelvin (double celsius)
 Converts from degrees Celsius to degrees Kelvin.
 
static constexpr double KelvinToCelsius (double kelvin)
 Converts from degrees Kelvin to degrees Celsius.
 
static constexpr double FeetToMeters (double measure)
 Converts from feet to meters.
 
static bool EqualToRoundoff (double a, double b)
 Finite precision comparison.
 
static bool EqualToRoundoff (float a, float b)
 Finite precision comparison.
 
static bool EqualToRoundoff (float a, double b)
 Finite precision comparison.
 
static bool EqualToRoundoff (double a, float b)
 Finite precision comparison.
 
static constexpr double Constrain (double min, double value, double max)
 Constrain a value between a minimum and a maximum value.
 
static constexpr double sign (double num)
 
- Static Public Attributes inherited from FGJSBBase
static char highint [5] = {27, '[', '1', 'm', '\0' }
 highlights text
 
static char halfint [5] = {27, '[', '2', 'm', '\0' }
 low intensity text
 
static char normint [6] = {27, '[', '2', '2', 'm', '\0' }
 normal intensity text
 
static char reset [5] = {27, '[', '0', 'm', '\0' }
 resets text properties
 
static char underon [5] = {27, '[', '4', 'm', '\0' }
 underlines text
 
static char underoff [6] = {27, '[', '2', '4', 'm', '\0' }
 underline off
 
static char fgblue [6] = {27, '[', '3', '4', 'm', '\0' }
 blue text
 
static char fgcyan [6] = {27, '[', '3', '6', 'm', '\0' }
 cyan text
 
static char fgred [6] = {27, '[', '3', '1', 'm', '\0' }
 red text
 
static char fggreen [6] = {27, '[', '3', '2', 'm', '\0' }
 green text
 
static char fgdef [6] = {27, '[', '3', '9', 'm', '\0' }
 default text
 
static short debug_lvl = 1
 
- Static Protected Member Functions inherited from FGJSBBase
static std::string CreateIndexedPropertyName (const std::string &Property, int index)
 
- Static Protected Attributes inherited from FGJSBBase
static constexpr double radtodeg = 180. / M_PI
 
static constexpr double degtorad = M_PI / 180.
 
static constexpr double hptoftlbssec = 550.0
 
static constexpr double psftoinhg = 0.014138
 
static constexpr double psftopa = 47.88
 
static constexpr double fttom = 0.3048
 
static constexpr double ktstofps = 1852./(3600*fttom)
 
static constexpr double fpstokts = 1.0 / ktstofps
 
static constexpr double inchtoft = 1.0/12.0
 
static constexpr double m3toft3 = 1.0/(fttom*fttom*fttom)
 
static constexpr double in3tom3 = inchtoft*inchtoft*inchtoft/m3toft3
 
static constexpr double inhgtopa = 3386.38
 
static constexpr double slugtolb = 32.174049
 Note that definition of lbtoslug by the inverse of slugtolb and not to a different constant you can also get from some tables will make lbtoslug*slugtolb == 1 up to the magnitude of roundoff.
 
static constexpr double lbtoslug = 1.0/slugtolb
 
static constexpr double kgtolb = 2.20462
 
static constexpr double kgtoslug = 0.06852168
 
static const std::string needed_cfg_version = "2.0"
 
static const std::string JSBSim_version = JSBSIM_VERSION " " __DATE__ " " __TIME__
 

Constructor & Destructor Documentation

◆ FGScript()

FGScript ( FGFDMExec exec)

Default constructor.

Definition at line 65 of file FGScript.cpp.

65 : FDMExec(fgex)
66{
67 PropertyManager=FDMExec->GetPropertyManager();
68
69 Debug(0);
70}
std::shared_ptr< FGPropertyManager > GetPropertyManager(void) const
Returns a pointer to the property manager object.
Definition FGFDMExec.h:421
+ Here is the call graph for this function:

◆ ~FGScript()

~FGScript ( )

Default destructor.

Definition at line 74 of file FGScript.cpp.

75{
76 unsigned int i, j;
77
78 for (i=0; i<Events.size(); i++) {
79 delete Events[i].Condition;
80 for (j=0; j<Events[i].Functions.size(); j++)
81 delete Events[i].Functions[j];
82 for (j=0; j<Events[i].NotifyProperties.size(); j++)
83 delete Events[i].NotifyProperties[j];
84 }
85 Events.clear();
86
87 Debug(1);
88}

Member Function Documentation

◆ LoadScript()

bool LoadScript ( const SGPath &  script,
double  default_dT,
const SGPath &  initfile 
)

Loads a script to drive JSBSim (usually in standalone mode).

The language is the Script Directives for JSBSim. If a simulation step size has been supplied on the command line, it will override the script specified simulation step size.

Parameters
scriptthe filename (including path name, if any) for the script.
default_dTthe default simulation step size if no value is specified in the script
initfileAn optional initialization file name passed in, empty by default. If a file name is passed in, it will override the one present in the script.
Returns
true if successful

Definition at line 92 of file FGScript.cpp.

94{
95 SGPath initialize;
96 string aircraft="", prop_name="";
97 string notifyPropertyName="";
98 Element *element=0, *run_element=0, *event_element=0;
99 Element *set_element=0;
100 Element *notify_element = 0L, *notify_property_element = 0L;
101 double dt = 0.0, value = 0.0;
102 FGCondition *newCondition;
103
104 FGXMLFileRead XMLFileRead;
105 Element* document = XMLFileRead.LoadXMLDocument(script);
106
107 if (!document) {
108 cerr << "File: " << script << " could not be loaded." << endl;
109 return false;
110 }
111
112 if (document->GetName() != string("runscript")) {
113 cerr << "File: " << script << " is not a script file" << endl;
114 return false;
115 }
116
117 ScriptName = document->GetAttributeValue("name");
118
119 // First, find "run" element and set delta T
120
121 run_element = document->FindElement("run");
122
123 if (!run_element) {
124 cerr << "No \"run\" element found in script." << endl;
125 return false;
126 }
127
128 // Set sim timing
129
130 if (run_element->HasAttribute("start"))
131 StartTime = run_element->GetAttributeValueAsNumber("start");
132 else
133 StartTime = 0.0;
134 FDMExec->Setsim_time(StartTime);
135 if (run_element->HasAttribute("end")) {
136 EndTime = run_element->GetAttributeValueAsNumber("end");
137 } else {
138 cerr << "An end time (duration) for the script must be specified in the script <run> element." << endl;
139 return false;
140 }
141
142 if (default_dT == 0.0)
143 dt = run_element->GetAttributeValueAsNumber("dt");
144 else {
145 dt = default_dT;
146 cout << endl << "Overriding simulation step size from the command line. New step size is: "
147 << default_dT << " seconds (" << 1/default_dT << " Hz)" << endl << endl;
148 }
149
150 FDMExec->Setdt(dt);
151
152 // Make sure that the desired time is reached and executed.
153 EndTime += 0.99*FDMExec->GetDeltaT();
154
155 // read aircraft and initialization files
156
157 element = document->FindElement("use");
158 if (element) {
159 aircraft = element->GetAttributeValue("aircraft");
160 if (!aircraft.empty()) {
161 if (!FDMExec->LoadModel(aircraft))
162 return false;
163 } else {
164 cerr << "Aircraft must be specified in use element." << endl;
165 return false;
166 }
167
168 initialize = SGPath::fromLocal8Bit(element->GetAttributeValue("initialize").c_str());
169 if (initfile.isNull()) {
170 if (initialize.isNull()) {
171 cerr << "Initialization file must be specified in use element." << endl;
172 return false;
173 }
174 } else {
175 cout << endl << "The initialization file specified in the script file ("
176 << initialize << ") has been overridden with a specified file ("
177 << initfile << ")." << endl;
178 initialize = initfile;
179 }
180
181 } else {
182 cerr << "No \"use\" directives in the script file." << endl;
183 return false;
184 }
185
186 auto IC = FDMExec->GetIC();
187 if ( ! IC->Load( initialize )) {
188 cerr << "Initialization unsuccessful" << endl;
189 return false;
190 }
191
192 // Now, read input spec if given.
193 element = document->FindElement("input");
194 while (element) {
195 if (!FDMExec->GetInput()->Load(element))
196 return false;
197
198 element = document->FindNextElement("input");
199 }
200
201 // Now, read output spec if given.
202 element = document->FindElement("output");
203 SGPath scriptDir = SGPath(script.dir());
204 if (scriptDir.isNull())
205 scriptDir = SGPath(".");
206
207 while (element) {
208 if (!FDMExec->GetOutput()->Load(element, scriptDir))
209 return false;
210
211 element = document->FindNextElement("output");
212 }
213
214 // Read local property/value declarations
215 int saved_debug_lvl = debug_lvl;
216 debug_lvl = 0; // Disable messages
217 LocalProperties.Load(run_element, PropertyManager.get(), true);
218 debug_lvl = saved_debug_lvl;
219
220 // Read "events" from script
221
222 event_element = run_element->FindElement("event");
223 while (event_element) { // event processing
224
225 // Create the event structure
226 struct event *newEvent = new struct event();
227
228 // Retrieve the event name if given
229 newEvent->Name = event_element->GetAttributeValue("name");
230
231 // Is this event persistent? That is, does it execute every time the
232 // condition triggers to true, or does it execute as a one-shot event, only?
233 if (event_element->GetAttributeValue("persistent") == string("true")) {
234 newEvent->Persistent = true;
235 }
236
237 // Does this event execute continuously when triggered to true?
238 if (event_element->GetAttributeValue("continuous") == string("true")) {
239 newEvent->Continuous = true;
240 }
241
242 // Process the conditions
243 Element* condition_element = event_element->FindElement("condition");
244 if (condition_element) {
245 try {
246 newCondition = new FGCondition(condition_element, PropertyManager);
247 } catch(BaseException& e) {
248 cerr << condition_element->ReadFrom()
249 << fgred << e.what() << reset << endl << endl;
250 delete newEvent;
251 return false;
252 }
253 newEvent->Condition = newCondition;
254 } else {
255 cerr << "No condition specified in script event " << newEvent->Name
256 << endl;
257 delete newEvent;
258 return false;
259 }
260
261 // Is there a delay between the time this event is triggered, and when the
262 // event actions are executed?
263
264 Element* delay_element = event_element->FindElement("delay");
265 if (delay_element)
266 newEvent->Delay = event_element->FindElementValueAsNumber("delay");
267 else
268 newEvent->Delay = 0.0;
269
270 // Notify about when this event is triggered?
271 if ((notify_element = event_element->FindElement("notify")) != 0) {
272 if (notify_element->HasAttribute("format")) {
273 if (notify_element->GetAttributeValue("format") == "kml") newEvent->NotifyKML = true;
274 }
275 newEvent->Notify = true;
276 // Check here for new <description> tag that gets echoed
277 string notify_description = notify_element->FindElementValue("description");
278 if (!notify_description.empty()) {
279 newEvent->Description = notify_description;
280 }
281 notify_property_element = notify_element->FindElement("property");
282 while (notify_property_element) {
283 notifyPropertyName = notify_property_element->GetDataLine();
284
285 if (notify_property_element->HasAttribute("apply")) {
286 string function_str = notify_property_element->GetAttributeValue("apply");
287 auto f = FDMExec->GetTemplateFunc(function_str);
288 if (f)
289 newEvent->NotifyProperties.push_back(new FGFunctionValue(notifyPropertyName, PropertyManager, f,
290 notify_property_element));
291 else {
292 cerr << notify_property_element->ReadFrom()
293 << fgred << highint << " No function by the name "
294 << function_str << " has been defined. This property will "
295 << "not be logged. You should check your configuration file."
296 << reset << endl;
297 }
298 }
299 else
300 newEvent->NotifyProperties.push_back(new FGPropertyValue(notifyPropertyName, PropertyManager,
301 notify_property_element));
302
303 string caption_attribute = notify_property_element->GetAttributeValue("caption");
304 if (caption_attribute.empty()) {
305 newEvent->DisplayString.push_back(notifyPropertyName);
306 } else {
307 newEvent->DisplayString.push_back(caption_attribute);
308 }
309
310 notify_property_element = notify_element->FindNextElement("property");
311 }
312 }
313
314 // Read set definitions (these define the actions to be taken when the event
315 // is triggered).
316 set_element = event_element->FindElement("set");
317 while (set_element) {
318 prop_name = set_element->GetAttributeValue("name");
319 if (PropertyManager->HasNode(prop_name)) {
320 newEvent->SetParam.push_back( PropertyManager->GetNode(prop_name) );
321 } else {
322 newEvent->SetParam.push_back( 0L );
323 }
324 newEvent->SetParamName.push_back( prop_name );
325
326 // Todo - should probably do some safety checking here to make sure one or
327 // the other of value or function is specified.
328 if (!set_element->GetAttributeValue("value").empty()) {
329 value = set_element->GetAttributeValueAsNumber("value");
330 newEvent->Functions.push_back(nullptr);
331 } else if (set_element->FindElement("function")) {
332 value = 0.0;
333 newEvent->Functions.push_back(new FGFunction(FDMExec, set_element->FindElement("function")));
334 }
335 newEvent->SetValue.push_back(value);
336 newEvent->OriginalValue.push_back(0.0);
337 newEvent->newValue.push_back(0.0);
338 newEvent->ValueSpan.push_back(0.0);
339 string tempCompare = set_element->GetAttributeValue("type");
340 if (to_lower(tempCompare).find("delta") != string::npos) newEvent->Type.push_back(FG_DELTA);
341 else if (to_lower(tempCompare).find("bool") != string::npos) newEvent->Type.push_back(FG_BOOL);
342 else if (to_lower(tempCompare).find("value") != string::npos) newEvent->Type.push_back(FG_VALUE);
343 else newEvent->Type.push_back(FG_VALUE); // DEFAULT
344 tempCompare = set_element->GetAttributeValue("action");
345 if (to_lower(tempCompare).find("ramp") != string::npos) newEvent->Action.push_back(FG_RAMP);
346 else if (to_lower(tempCompare).find("step") != string::npos) newEvent->Action.push_back(FG_STEP);
347 else if (to_lower(tempCompare).find("exp") != string::npos) newEvent->Action.push_back(FG_EXP);
348 else newEvent->Action.push_back(FG_STEP); // DEFAULT
349
350 if (!set_element->GetAttributeValue("tc").empty())
351 newEvent->TC.push_back(set_element->GetAttributeValueAsNumber("tc"));
352 else
353 newEvent->TC.push_back(1.0); // DEFAULT
354
355 newEvent->Transiting.push_back(false);
356
357 set_element = event_element->FindNextElement("set");
358 }
359 Events.push_back(*newEvent);
360 delete newEvent;
361
362 event_element = run_element->FindNextElement("event");
363 }
364
365 Debug(4);
366
367 return true;
368}
std::shared_ptr< FGInitialCondition > GetIC(void) const
Returns a pointer to the FGInitialCondition object.
Definition FGFDMExec.h:389
std::shared_ptr< FGOutput > GetOutput(void) const
Returns the FGOutput pointer.
double GetDeltaT(void) const
Returns the simulation delta T.
Definition FGFDMExec.h:552
double Setsim_time(double cur_time)
Sets the current sim time.
bool LoadModel(const SGPath &AircraftPath, const SGPath &EnginePath, const SGPath &SystemsPath, const std::string &model, bool addModelToPath=true)
Loads an aircraft model.
void Setdt(double delta_t)
Sets the integration time step for the simulation executive.
Definition FGFDMExec.h:571
std::shared_ptr< FGInput > GetInput(void) const
Returns the FGInput pointer.
static char fgred[6]
red text
Definition FGJSBBase.h:166
static char reset[5]
resets text properties
Definition FGJSBBase.h:156
static char highint[5]
highlights text
Definition FGJSBBase.h:150
+ Here is the call graph for this function:

◆ ResetEvents()

void ResetEvents ( void  )

Definition at line 372 of file FGScript.cpp.

373{
374 LocalProperties.ResetToIC();
375 FDMExec->Setsim_time(StartTime);
376
377 for (unsigned int i=0; i<Events.size(); i++)
378 Events[i].reset();
379}

◆ RunScript()

bool RunScript ( void  )

This function is called each pass through the executive Run() method IF scripting is enabled.

Returns
false if script should exit (i.e. if time limits are violated

Definition at line 383 of file FGScript.cpp.

384{
385 unsigned i, j;
386 unsigned event_ctr = 0;
387
388 double currentTime = FDMExec->GetSimTime();
389 double newSetValue = 0;
390
391 if (currentTime > EndTime) return false;
392
393 // Iterate over all events.
394 for (unsigned int ev_ctr=0; ev_ctr < Events.size(); ev_ctr++) {
395
396 struct event &thisEvent = Events[ev_ctr];
397
398 // Determine whether the set of conditional tests for this condition equate
399 // to true and should cause the event to execute. If the conditions evaluate
400 // to true, then the event is triggered. If the event is not persistent,
401 // then this trigger will remain set true. If the event is persistent, the
402 // trigger will reset to false when the condition evaluates to false.
403 if (thisEvent.Condition->Evaluate()) {
404 if (!thisEvent.Triggered) {
405
406 // The conditions are true, do the setting of the desired Event
407 // parameters
408 for (i=0; i<thisEvent.SetValue.size(); i++) {
409 if (thisEvent.SetParam[i] == 0L) { // Late bind property if necessary
410 if (PropertyManager->HasNode(thisEvent.SetParamName[i])) {
411 thisEvent.SetParam[i] = PropertyManager->GetNode(thisEvent.SetParamName[i]);
412 } else {
413 throw("No property, \""+thisEvent.SetParamName[i]+"\" is defined.");
414 }
415 }
416 thisEvent.OriginalValue[i] = thisEvent.SetParam[i]->getDoubleValue();
417 if (thisEvent.Functions[i] != 0) { // Parameter should be set to a function value
418 try {
419 thisEvent.SetValue[i] = thisEvent.Functions[i]->GetValue();
420 } catch (string& msg) {
421 std::cerr << std::endl << "A problem occurred in the execution of the script. " << msg << endl;
422 throw;
423 }
424 }
425 switch (thisEvent.Type[i]) {
426 case FG_VALUE:
427 case FG_BOOL:
428 thisEvent.newValue[i] = thisEvent.SetValue[i];
429 break;
430 case FG_DELTA:
431 thisEvent.newValue[i] = thisEvent.OriginalValue[i] + thisEvent.SetValue[i];
432 break;
433 default:
434 cerr << "Invalid Type specified" << endl;
435 break;
436 }
437 thisEvent.StartTime = currentTime + thisEvent.Delay;
438 thisEvent.ValueSpan[i] = thisEvent.newValue[i] - thisEvent.OriginalValue[i];
439 thisEvent.Transiting[i] = true;
440 }
441 }
442 thisEvent.Triggered = true;
443
444 } else if (thisEvent.Persistent) { // If the event is persistent, reset the trigger.
445 thisEvent.Triggered = false; // Reset the trigger for persistent events
446 thisEvent.Notified = false; // Also reset the notification flag
447 } else if (thisEvent.Continuous) { // If the event is continuous, reset the trigger.
448 thisEvent.Triggered = false; // Reset the trigger for persistent events
449 thisEvent.Notified = false; // Also reset the notification flag
450 }
451
452 if ((currentTime >= thisEvent.StartTime) && thisEvent.Triggered) {
453
454 for (i=0; i<thisEvent.SetValue.size(); i++) {
455 if (thisEvent.Transiting[i]) {
456 thisEvent.TimeSpan = currentTime - thisEvent.StartTime;
457 switch (thisEvent.Action[i]) {
458 case FG_RAMP:
459 if (thisEvent.TimeSpan <= thisEvent.TC[i]) {
460 newSetValue = thisEvent.TimeSpan/thisEvent.TC[i] * thisEvent.ValueSpan[i] + thisEvent.OriginalValue[i];
461 } else {
462 newSetValue = thisEvent.newValue[i];
463 if (thisEvent.Continuous != true) thisEvent.Transiting[i] = false;
464 }
465 break;
466 case FG_STEP:
467 newSetValue = thisEvent.newValue[i];
468
469 // If this is not a continuous event, reset the transiting flag.
470 // Otherwise, it is known that the event is a continuous event.
471 // Furthermore, if the event is to be determined by a function,
472 // then the function will be continuously calculated.
473 if (thisEvent.Continuous != true)
474 thisEvent.Transiting[i] = false;
475 else if (thisEvent.Functions[i] != 0)
476 newSetValue = thisEvent.Functions[i]->GetValue();
477
478 break;
479 case FG_EXP:
480 newSetValue = (1 - exp( -thisEvent.TimeSpan/thisEvent.TC[i] )) * thisEvent.ValueSpan[i] + thisEvent.OriginalValue[i];
481 break;
482 default:
483 cerr << "Invalid Action specified" << endl;
484 break;
485 }
486 thisEvent.SetParam[i]->setDoubleValue(newSetValue);
487 }
488 }
489
490 // Print notification values after setting them
491 if (thisEvent.Notify && !thisEvent.Notified) {
492 if (thisEvent.NotifyKML) {
493 cout << endl << "<Placemark>" << endl;
494 cout << " <name> " << currentTime << " seconds" << " </name>"
495 << endl;
496 cout << " <description>" << endl;
497 cout << " <![CDATA[" << endl;
498 cout << " <b>" << thisEvent.Name << " (Event " << event_ctr << ")"
499 << " executed at time: " << currentTime << "</b><br/>" << endl;
500 } else {
501 cout << endl << underon
502 << highint << thisEvent.Name << normint << underoff
503 << " (Event " << event_ctr << ")"
504 << " executed at time: " << highint << currentTime << normint
505 << endl;
506 }
507 if (!thisEvent.Description.empty()) {
508 cout << " " << thisEvent.Description << endl;
509 }
510 for (j=0; j<thisEvent.NotifyProperties.size();j++) {
511 cout << " " << thisEvent.DisplayString[j] << " = "
512 << thisEvent.NotifyProperties[j]->getDoubleValue();
513 if (thisEvent.NotifyKML) cout << " <br/>";
514 cout << endl;
515 }
516 if (thisEvent.NotifyKML) {
517 cout << " ]]>" << endl;
518 cout << " </description>" << endl;
519 cout << " <Point>" << endl;
520 cout << " <altitudeMode> absolute </altitudeMode>" << endl;
521 cout << " <extrude> 1 </extrude>" << endl;
522 cout << " <coordinates>"
523 << FDMExec->GetPropagate()->GetLongitudeDeg() << ","
524 << FDMExec->GetPropagate()->GetGeodLatitudeDeg() << ","
525 << FDMExec->GetPropagate()->GetAltitudeASLmeters()
526 << "</coordinates>" << endl;
527 cout << " </Point>" << endl;
528 cout << "</Placemark>" << endl;
529 }
530 cout << endl;
531 thisEvent.Notified = true;
532 }
533
534 }
535
536 event_ctr++;
537 }
538 return true;
539}
std::shared_ptr< FGPropagate > GetPropagate(void) const
Returns the FGPropagate pointer.
double GetSimTime(void) const
Returns the cumulative simulation time in seconds.
Definition FGFDMExec.h:549
static char normint[6]
normal intensity text
Definition FGJSBBase.h:154
static char underon[5]
underlines text
Definition FGJSBBase.h:158
static char underoff[6]
underline off
Definition FGJSBBase.h:160
+ Here is the call graph for this function:

The documentation for this class was generated from the following files: