1 // ATCDialog.cxx - Functions and classes to handle the pop-up ATC dialog
3 // Written by Alexander Kappes and David Luff, started February 2003.
5 // Copyright (C) 2003 Alexander Kappes and David Luff
7 // This program is free software; you can redistribute it and/or
8 // modify it under the terms of the GNU General Public License as
9 // published by the Free Software Foundation; either version 2 of the
10 // License, or (at your option) any later version.
12 // This program is distributed in the hope that it will be useful, but
13 // WITHOUT ANY WARRANTY; without even the implied warranty of
14 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 // General Public License for more details.
17 // You should have received a copy of the GNU General Public License
18 // along with this program; if not, write to the Free Software
19 // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
25 #include <simgear/compiler.h>
27 #include <simgear/structure/commands.hxx>
28 #include <simgear/props/props_io.hxx>
30 #include <Main/globals.hxx>
31 #include <GUI/gui.h> // mkDialog
32 #include <GUI/new_gui.hxx>
34 #include "ATCDialog.hxx"
37 #include "commlist.hxx"
38 #include "ATCutils.hxx"
39 #include <Airports/simple.hxx>
43 using std::ostringstream;
45 FGATCDialog *current_atcdialog;
47 // For the command manager - maybe eventually this should go in the built in command list
48 static bool do_ATC_dialog(const SGPropertyNode* arg) {
49 current_atcdialog->PopupDialog();
53 static bool do_ATC_freq_search(const SGPropertyNode* arg) {
54 current_atcdialog->FreqDialog();
58 ATCMenuEntry::ATCMenuEntry() {
66 ATCMenuEntry::~ATCMenuEntry() {
69 static void atcUppercase(string &s) {
70 for(unsigned int i=0; i<s.size(); ++i) {
75 // find child whose <name>...</name> entry matches 'name'
76 static SGPropertyNode *getNamedNode(SGPropertyNode *prop, const char *name) {
79 for (int i = 0; i < prop->nChildren(); i++)
80 if ((p = getNamedNode(prop->getChild(i), name)))
83 if (!strcmp(prop->getStringValue("name"), name))
90 FGATCDialog::FGATCDialog() {
91 _callbackPending = false;
96 _gui = (NewGUI *)globals->get_subsystem("gui");
99 FGATCDialog::~FGATCDialog() {
102 void FGATCDialog::Init() {
103 // Add ATC-dialog to the command list
104 globals->get_commands()->addCommand("ATC-dialog", do_ATC_dialog);
105 // Add ATC-freq-search to the command list
106 globals->get_commands()->addCommand("ATC-freq-search", do_ATC_freq_search);
108 // initialize properties polled in Update()
109 globals->get_props()->setStringValue("/sim/atc/freq-airport", "");
110 globals->get_props()->setIntValue("/sim/atc/transmission-num", -1);
113 void FGATCDialog::Update(double dt) {
114 static SGPropertyNode_ptr airport = globals->get_props()->getNode("/sim/atc/freq-airport", true);
115 string s = airport->getStringValue();
117 airport->setStringValue("");
121 static SGPropertyNode_ptr trans_num = globals->get_props()->getNode("/sim/atc/transmission-num", true);
122 int n = trans_num->getIntValue();
124 trans_num->setIntValue(-1);
128 if(_callbackPending) {
129 if(_callbackTimer > _callbackWait) {
130 _callbackPtr->ReceiveUserCallback(_callbackCode);
131 _callbackPtr->NotifyTransmissionFinished(fgGetString("/sim/user/callsign"));
132 _callbackPending = false;
134 _callbackTimer += dt;
140 void FGATCDialog::add_entry(const string& station, const string& transmission, const string& menutext, atc_type type, int code) {
144 a.stationid = station;
145 a.transmission = transmission;
146 a.menuentry = menutext;
147 a.callback_code = code;
149 (available_dialog[type])[station.c_str()].push_back(a);
153 void FGATCDialog::remove_entry( const string &station, const string &trans, atc_type type ) {
154 atcmentry_vec_type* p = &((available_dialog[type])[station]);
155 atcmentry_vec_iterator current = p->begin();
156 while(current != p->end()) {
157 if(current->transmission == trans) current = p->erase(current);
162 void FGATCDialog::remove_entry( const string &station, int code, atc_type type ) {
163 atcmentry_vec_type* p = &((available_dialog[type])[station]);
164 atcmentry_vec_iterator current = p->begin();
165 while(current != p->end()) {
166 if(current->callback_code == code) current = p->erase(current);
171 // query the database whether the transmission is already registered;
172 bool FGATCDialog::trans_reg( const string &station, const string &trans, atc_type type ) {
173 atcmentry_vec_type* p = &((available_dialog[type])[station]);
174 atcmentry_vec_iterator current = p->begin();
175 for ( ; current != p->end() ; ++current ) {
176 if ( current->transmission == trans ) return true;
181 // query the database whether the transmission is already registered;
182 bool FGATCDialog::trans_reg( const string &station, int code, atc_type type ) {
183 atcmentry_vec_type* p = &((available_dialog[type])[station]);
184 atcmentry_vec_iterator current = p->begin();
185 for ( ; current != p->end() ; ++current ) {
186 if ( current->callback_code == code ) return true;
191 // Display the ATC popup dialog box with options relevant to the users current situation.
192 void FGATCDialog::PopupDialog() {
193 const char *dialog_name = "atc-dialog";
194 SGPropertyNode_ptr dlg = _gui->getDialogProperties(dialog_name);
198 _gui->closeDialog(dialog_name);
200 SGPropertyNode_ptr button_group = getNamedNode(dlg, "transmission-choice");
201 // remove all transmission buttons
202 button_group->removeChildren("button", false);
205 FGATC* atcptr = globals->get_ATC_mgr()->GetComm1ATCPointer(); // Hardwired to comm1 at the moment
208 label = "Not currently tuned to any ATC service";
209 mkDialog(label.c_str());
213 if(atcptr->GetType() == ATIS) {
214 label = "Tuned to ATIS - no communication possible";
215 mkDialog(label.c_str());
219 atcmentry_vec_type atcmlist = (available_dialog[atcptr->GetType()])[atcptr->get_ident()];
220 atcmentry_vec_iterator current = atcmlist.begin();
221 atcmentry_vec_iterator last = atcmlist.end();
223 if(!atcmlist.size()) {
224 label = "No transmission available";
225 mkDialog(label.c_str());
229 const int bufsize = 32;
231 // loop over all entries in atcmentrylist
232 for (int n = 0; n < 10; ++n) {
233 snprintf(buf, bufsize, "/sim/atc/opt[%d]", n);
234 fgSetBool(buf, false);
239 // add transmission button (modified copy of <button-template>)
240 SGPropertyNode *entry = button_group->getNode("button", n, true);
241 copyProperties(button_group->getNode("button-template", true), entry);
242 entry->removeChildren("enabled", true);
243 entry->setStringValue("property", buf);
244 entry->setIntValue("keynum", '1' + n);
246 entry->setBoolValue("default", true);
248 snprintf(buf, bufsize, "%d", n + 1);
249 string legend = string(buf) + ". " + current->menuentry;
250 entry->setStringValue("legend", legend.c_str());
251 entry->setIntValue("binding/value", n);
255 _gui->showDialog(dialog_name);
259 void FGATCDialog::PopupCallback(int num) {
260 FGATC* atcptr = globals->get_ATC_mgr()->GetComm1ATCPointer(); // FIXME - Hardwired to comm1 at the moment
265 if (atcptr->GetType() == TOWER) {
266 //cout << "TOWER " << endl;
267 //cout << "ident is " << atcptr->get_ident() << endl;
268 atcmentry_vec_type atcmlist = (available_dialog[TOWER])[atcptr->get_ident()];
269 int size = atcmlist.size();
270 if(size && num < size) {
271 //cout << "Doing callback...\n";
272 ATCMenuEntry a = atcmlist[num];
273 atcptr->SetFreqInUse();
274 string pilot = atcptr->GenText(a.transmission, a.callback_code);
275 fgSetString("/sim/messages/pilot", pilot.c_str());
276 // This is the user's speech getting displayed.
277 _callbackPending = true;
278 _callbackTimer = 0.0;
280 _callbackPtr = atcptr;
281 _callbackCode = a.callback_code;
283 //cout << "No options available...\n";
285 //cout << "Donded" << endl;
289 // map() key data type (removes duplicates and sorts by distance)
292 atcdata(const string i, const string n, const double d) {
293 id = i, name = n, distance = d;
295 bool operator<(const atcdata& a) const {
296 return id != a.id && distance < a.distance;
298 bool operator==(const atcdata& a) const {
299 return id == a.id && distance == a.distance;
306 void FGATCDialog::FreqDialog() {
307 const char *dialog_name = "atc-freq-search";
308 SGPropertyNode_ptr dlg = _gui->getDialogProperties(dialog_name);
312 _gui->closeDialog(dialog_name);
314 SGPropertyNode_ptr button_group = getNamedNode(dlg, "quick-buttons");
315 // remove all dynamic airport/ATC buttons
316 button_group->removeChildren("button", false);
318 // Find the ATC stations within a reasonable range
319 comm_list_type atc_stations;
320 comm_list_iterator atc_stat_itr;
322 double lon = fgGetDouble("/position/longitude-deg");
323 double lat = fgGetDouble("/position/latitude-deg");
324 double elev = fgGetDouble("/position/altitude-ft");
325 SGVec3d aircraft = SGVec3d::fromGeod(SGGeod::fromDegM(lon, lat, elev));
327 // search stations in range
328 int num_stat = current_commlist->FindByPos(lon, lat, elev, 50.0, &atc_stations);
330 map<atcdata, bool> uniq;
331 // fill map (sorts by distance and removes duplicates)
332 comm_list_iterator itr = atc_stations.begin();
333 for (; itr != atc_stations.end(); ++itr) {
334 SGVec3d station(itr->x, itr->y, itr->z);
335 double distance = distSqr(aircraft, station);
336 uniq[atcdata(itr->ident, itr->name, distance)] = true;
338 // create button per map entry (modified copy of <button-template>)
339 map<atcdata, bool>::iterator uit = uniq.begin();
340 for (int n = 0; uit != uniq.end() && n < 6; ++uit, ++n) { // max 6 buttons
341 SGPropertyNode *entry = button_group->getNode("button", n, true);
342 copyProperties(button_group->getNode("button-template", true), entry);
343 entry->removeChildren("enabled", true);
344 entry->setStringValue("legend", uit->first.id.c_str());
345 entry->setStringValue("binding[0]/value", uit->first.id.c_str());
349 // (un)hide message saying no things in range
350 SGPropertyNode_ptr range_error = getNamedNode(dlg, "no-atc-in-range");
351 range_error->setBoolValue("enabled", !num_stat);
353 _gui->showDialog(dialog_name);
356 void FGATCDialog::FreqDisplay(string& ident) {
357 const char *dialog_name = "atc-freq-display";
358 SGPropertyNode_ptr dlg = _gui->getDialogProperties(dialog_name);
362 _gui->closeDialog(dialog_name);
364 SGPropertyNode_ptr freq_group = getNamedNode(dlg, "frequency-list");
365 // remove all frequency entries
366 freq_group->removeChildren("group", false);
371 const FGAirport *a = fgFindAirportID(ident);
373 label = "Airport " + ident + " not found in database.";
374 mkDialog(label.c_str());
379 label = ident + " Frequencies";
380 dlg->setStringValue("text/label", label.c_str());
382 int n = 0; // Number of ATC frequencies at this airport
384 comm_list_type stations;
385 int found = current_commlist->FindByPos(a->getLongitude(), a->getLatitude(), a->getElevation(), 20.0, &stations);
388 comm_list_iterator itr = stations.begin();
389 for (n = 0; itr != stations.end(); ++itr) {
390 if(itr->ident != ident)
393 if(itr->type == INVALID)
396 // add frequency line (modified copy of <group-template>)
397 SGPropertyNode *entry = freq_group->getNode("group", n, true);
398 copyProperties(freq_group->getNode("group-template", true), entry);
399 entry->removeChildren("enabled", true);
402 entry->setStringValue("text[0]/label", ostr.str().c_str());
405 snprintf(buf, 8, "%.2f", (itr->freq / 100.0)); // Convert from KHz to MHz
406 if(buf[5] == '3') buf[5] = '2';
407 if(buf[5] == '8') buf[5] = '7';
410 entry->setStringValue("text[1]/label", buf);
417 label = "No frequencies found for airport " + ident;
418 mkDialog(label.c_str());
422 _gui->showDialog(dialog_name);