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
FGWaypoint.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: FGWaypoint.cpp
4 Author: Jon S. Berndt
5 Date started: 6/2013
6
7 ------------- Copyright (C) 2013 Jon S. Berndt (jon@jsbsim.org) -------------
8
9 This program is free software; you can redistribute it and/or modify it under
10 the terms of the GNU Lesser General Public License as published by the Free
11 Software Foundation; either version 2 of the License, or (at your option) any
12 later version.
13
14 This program is distributed in the hope that it will be useful, but WITHOUT
15 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
16 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
17 details.
18
19 You should have received a copy of the GNU Lesser General Public License along
20 with this program; if not, write to the Free Software Foundation, Inc., 59
21 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
22
23 Further information about the GNU Lesser General Public License can also be
24 found on the world wide web at http://www.gnu.org.
25
26FUNCTIONAL DESCRIPTION
27--------------------------------------------------------------------------------
28
29HISTORY
30--------------------------------------------------------------------------------
31
32%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
33COMMENTS, REFERENCES, and NOTES
34%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
35
36%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
37INCLUDES
38%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
39
40#include "FGWaypoint.h"
41#include "input_output/FGXMLElement.h"
42#include "math/FGLocation.h"
43#include "models/FGFCS.h"
44#include "models/FGInertial.h"
45#include "initialization/FGInitialCondition.h"
46
47using namespace std;
48
49namespace JSBSim {
50
51/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
52CLASS IMPLEMENTATION
53%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
54
55//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
56
57FGWaypoint::FGWaypoint(FGFCS* fcs, Element* element)
58 : FGFCSComponent(fcs, element)
59{
60 if (Type == "WAYPOINT_HEADING") WaypointType = eHeading;
61 else if (Type == "WAYPOINT_DISTANCE") WaypointType = eDistance;
62
63 target_latitude_unit = 1.0;
64 target_longitude_unit = 1.0;
65 source_latitude_unit = 1.0;
66 source_longitude_unit = 1.0;
67 source = fcs->GetExec()->GetIC()->GetPosition();
68
69 auto PropertyManager = fcs->GetPropertyManager();
70
71 if (element->FindElement("target_latitude") ) {
72 target_latitude = std::make_unique<FGPropertyValue>(element->FindElementValue("target_latitude"),
73 PropertyManager, element);
74 if (element->FindElement("target_latitude")->HasAttribute("unit")) {
75 if (element->FindElement("target_latitude")->GetAttributeValue("unit") == "DEG") {
76 target_latitude_unit = 0.017453293;
77 }
78 }
79 } else {
80 cerr << element->ReadFrom() << endl
81 << "Target latitude is required for waypoint component: " << Name
82 << endl;
83 throw("Malformed waypoint definition");
84 }
85
86 if (element->FindElement("target_longitude") ) {
87 target_longitude = std::make_unique<FGPropertyValue>(element->FindElementValue("target_longitude"),
88 PropertyManager, element);
89 if (element->FindElement("target_longitude")->HasAttribute("unit")) {
90 if (element->FindElement("target_longitude")->GetAttributeValue("unit") == "DEG") {
91 target_longitude_unit = 0.017453293;
92 }
93 }
94 } else {
95 cerr << element->ReadFrom() << endl
96 << "Target longitude is required for waypoint component: " << Name
97 << endl;
98 throw("Malformed waypoint definition");
99 }
100
101 if (element->FindElement("source_latitude") ) {
102 source_latitude = std::make_unique<FGPropertyValue>(element->FindElementValue("source_latitude"),
103 PropertyManager, element);
104 if (element->FindElement("source_latitude")->HasAttribute("unit")) {
105 if (element->FindElement("source_latitude")->GetAttributeValue("unit") == "DEG") {
106 source_latitude_unit = 0.017453293;
107 }
108 }
109 } else {
110 cerr << element->ReadFrom() << endl
111 << "Source latitude is required for waypoint component: " << Name
112 << endl;
113 throw("Malformed waypoint definition");
114 }
115
116 if (element->FindElement("source_longitude") ) {
117 source_longitude = std::make_unique<FGPropertyValue>(element->FindElementValue("source_longitude"),
118 PropertyManager, element);
119 if (element->FindElement("source_longitude")->HasAttribute("unit")) {
120 if (element->FindElement("source_longitude")->GetAttributeValue("unit") == "DEG") {
121 source_longitude_unit = 0.017453293;
122 }
123 }
124 } else {
125 cerr << element->ReadFrom() << endl
126 << "Source longitude is required for waypoint component: " << Name
127 << endl;
128 throw("Malformed waypoint definition");
129 }
130
131 unit = element->GetAttributeValue("unit");
132 if (WaypointType == eHeading) {
133 if (!unit.empty()) {
134 if (unit == "DEG") eUnit = eDeg;
135 else if (unit == "RAD") eUnit = eRad;
136 else {
137 cerr << element->ReadFrom() << endl
138 << "Unknown unit " << unit << " in HEADING waypoint component, "
139 << Name << endl;
140 throw("Malformed waypoint definition");
141 }
142 } else {
143 eUnit = eRad; // Default is radians if unspecified
144 }
145 } else {
146 if (!unit.empty()) {
147 if (unit == "FT") eUnit = eFeet;
148 else if (unit == "M") eUnit = eMeters;
149 else {
150 cerr << element->ReadFrom() << endl
151 << "Unknown unit " << unit << " in DISTANCE waypoint component, "
152 << Name << endl;
153 throw("Malformed waypoint definition");
154 }
155 } else {
156 eUnit = eFeet; // Default is feet if unspecified
157 }
158 }
159
160 bind(element, PropertyManager.get());
161 Debug(0);
162}
163
164//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
165
166FGWaypoint::~FGWaypoint()
167{
168 Debug(1);
169}
170
171//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
172
173bool FGWaypoint::Run(void )
174{
175 double source_latitude_rad = source_latitude->GetValue() * source_latitude_unit;
176 double source_longitude_rad = source_longitude->GetValue() * source_longitude_unit;
177 double target_latitude_rad = target_latitude->GetValue() * target_latitude_unit;
178 double target_longitude_rad = target_longitude->GetValue() * target_longitude_unit;
179 source.SetPositionGeodetic(source_longitude_rad, source_latitude_rad, 0.0);
180
181 if (fabs(target_latitude_rad) > M_PI/2.0) {
182 cerr << endl;
183 cerr << "Target latitude in waypoint \"" << Name << "\" must be less than or equal to 90 degrees." << endl;
184 cerr << "(is longitude being mistakenly supplied?)" << endl;
185 cerr << endl;
186 throw("Waypoint target latitude exceeded 90 degrees.");
187 }
188
189 if (fabs(source_latitude_rad) > M_PI/2.0) {
190 cerr << endl;
191 cerr << "Source latitude in waypoint \"" << Name << "\" must be less than or equal to 90 degrees." << endl;
192 cerr << "(is longitude being mistakenly supplied?)" << endl;
193 cerr << endl;
194 throw("Source latitude exceeded 90 degrees.");
195 }
196
197 if (WaypointType == eHeading) { // Calculate Heading
198
199 double heading_to_waypoint_rad = source.GetHeadingTo(target_longitude_rad,
200 target_latitude_rad);
201
202 if (eUnit == eDeg) Output = heading_to_waypoint_rad * radtodeg;
203 else Output = heading_to_waypoint_rad;
204
205 } else { // Calculate Distance
206
207 double wp_distance = source.GetDistanceTo(target_longitude_rad,
208 target_latitude_rad);
209 if (eUnit == eMeters) Output = FeetToMeters(wp_distance);
210 else Output = wp_distance;
211
212 }
213
214 Clip();
215 SetOutput();
216
217 return true;
218}
219
220//%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
221// The bitmasked value choices are as follows:
222// unset: In this case (the default) JSBSim would only print
223// out the normally expected messages, essentially echoing
224// the config files as they are read. If the environment
225// variable is not set, debug_lvl is set to 1 internally
226// 0: This requests JSBSim not to output any messages
227// whatsoever.
228// 1: This value explicity requests the normal JSBSim
229// startup messages
230// 2: This value asks for a message to be printed out when
231// a class is instantiated
232// 4: When this value is set, a message is displayed when a
233// FGModel object executes its Run() method
234// 8: When this value is set, various runtime state variables
235// are printed out periodically
236// 16: When set various parameters are sanity checked and
237// a message is printed out when they go out of bounds
238
239void FGWaypoint::Debug(int from)
240{
241 if (debug_lvl <= 0) return;
242
243 if (debug_lvl & 1) { // Standard console startup message output
244 if (from == 0) { // Constructor
245 }
246 }
247 if (debug_lvl & 2 ) { // Instantiation/Destruction notification
248 if (from == 0) cout << "Instantiated: FGWaypoint" << endl;
249 if (from == 1) cout << "Destroyed: FGWaypoint" << endl;
250 }
251 if (debug_lvl & 4 ) { // Run() method entry print for FGModel-derived objects
252 }
253 if (debug_lvl & 8 ) { // Runtime state variables
254 }
255 if (debug_lvl & 16) { // Sanity checking
256 }
257 if (debug_lvl & 64) {
258 if (from == 0) { // Constructor
259 }
260 }
261}
262}