1 // kln89_page_*.[ch]xx - this file is one of the "pages" that
2 // are used in the KLN89 GPS unit simulation.
4 // Written by David Luff, started 2005.
6 // Copyright (C) 2005 - David C Luff - daveluff AT ntlworld.com
8 // This program is free software; you can redistribute it and/or
9 // modify it under the terms of the GNU General Public License as
10 // published by the Free Software Foundation; either version 2 of the
11 // License, or (at your option) any later version.
13 // This program is distributed in the hope that it will be useful, but
14 // WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 // General Public License for more details.
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the Free Software
20 // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
28 #include "kln89_page_apt.hxx"
30 #include <simgear/structure/exception.hxx>
33 #include <ATC/CommStation.hxx>
34 #include <Main/globals.hxx>
35 #include <Airports/runways.hxx>
36 #include <Airports/simple.hxx>
40 KLN89AptPage::KLN89AptPage(KLN89* parent)
46 // Make sure that _last_apt_id doesn't match at startup to force airport data to be fetched on first update.
47 _last_apt_id = "XXXX";
59 _replaceDialog = false;
64 KLN89AptPage::~KLN89AptPage() {
67 void KLN89AptPage::Update(double dt) {
68 bool actPage = (_kln89->_activePage->GetName() == "ACT" ? true : false);
69 bool multi; // Not set by FindFirst...
71 if(_apt_id.size() == 4) exact = true;
72 // TODO - move this search out to where the button is pressed, and cache the result!
73 if(_apt_id != _last_apt_id || ap == NULL) ap = _kln89->FindFirstAptById(_apt_id, multi, exact);
74 //if(np == NULL) cout << "NULL... ";
75 //if(b == false) cout << "false...\n";
78 cout << "VOR FOUND!\n";
85 //cout << "Valid airport found! id = " << ap->getId() << ", elev = " << ap->getElevation() << '\n';
86 if(_apt_id != _last_apt_id) {
87 UpdateAirport(ap->getId());
88 _last_apt_id = _apt_id;
92 _apt_id = ap->getId();
93 if(_kln89->GetActiveWaypoint()) {
94 if(_apt_id == _kln89->GetActiveWaypoint()->id) {
95 if(!(_kln89->_waypointAlert && _kln89->_blink)) {
96 // Active waypoint arrow
97 _kln89->DrawSpecialChar(4, 2, 0, 3);
101 if(_kln89->_mode != KLN89_MODE_CRSR) {
102 if(!(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) { // Don't draw the airport name when the IAP dialogs are active
105 _kln89->DrawText(ap->getId(), 2, 1, 3);
107 // If it's the ACT page, The ID is shifted slightly right to make space for the waypoint index.
108 _kln89->DrawText(ap->getId(), 2, 4, 3);
110 int n = snprintf(buf, 3, "%i", _kln89->GetActiveWaypointIndex() + 1);
111 _kln89->DrawText((string)buf, 2, 3 - n, 3);
114 if(!_kln89->_blink) {
115 _kln89->DrawText(ap->getId(), 2, 1, 3, false, 99);
123 _kln89->DrawText(ap->getName(), 2, 0, 2);
125 _kln89->DrawText(_kln89->_altUnits == GPS_ALT_UNITS_FT ? "ft" : "m", 2, 14, 3);
127 int n = snprintf(buf, 5, "%i", (_kln89->_altUnits == GPS_ALT_UNITS_FT ? (int)(ap->getElevation()) : (int)((double)ap->getElevation() * SG_FEET_TO_METER)));
128 _kln89->DrawText((string)buf, 2, 14 - n, 3);
130 airport_id_str_map_iterator itr = _kln89->_airportTowns.find(_apt_id);
131 if(itr != _kln89->_airportTowns.end()) {
132 _kln89->DrawText(itr->second, 2, 0, 1);
134 // State / Province / Country
135 itr = _kln89->_airportStates.find(_apt_id);
136 if(itr != _kln89->_airportStates.end()) {
137 _kln89->DrawText(itr->second, 2, 0, 0);
139 } else if(_subPage == 1) {
140 _kln89->DrawLatitude(ap->getLatitude(), 2, 3, 2);
141 _kln89->DrawLongitude(ap->getLongitude(), 2, 3, 1);
142 _kln89->DrawDirDistField(ap->getLatitude() * SG_DEGREES_TO_RADIANS, ap->getLongitude() * SG_DEGREES_TO_RADIANS,
143 2, 0, 0, _to_flag, (_kln89->_mode == KLN89_MODE_CRSR && _uLinePos == 5 ? true : false));
144 } else if(_subPage == 2) {
145 // Try and calculate a realistic difference from UTC based on longitude
146 // Since 0 longitude is the middle of UTC, the boundaries will be at 7.5, 22.5, 37.5 etc.
147 int hrDiff = ((int)((fabs(ap->getLongitude())) + 7.5)) / 15;
148 _kln89->DrawText("UTC", 2, 0, 2);
150 _kln89->DrawText(ap->getLongitude() >= 0.0 ? "+" : "-", 2, 3, 2);
152 snprintf(buf, 3, "%02i", hrDiff);
153 _kln89->DrawText((string)buf, 2, 4, 2);
154 _kln89->DrawText("( DT)", 2, 6, 2);
155 if(ap->getLongitude() >= 0.0) {
160 _kln89->DrawText(ap->getLongitude() >= 0.0 ? "+" : "-", 2, 7, 2);
161 snprintf(buf, 3, "%02i", hrDiff);
162 _kln89->DrawText((string)buf, 2, 8, 2);
164 // I guess we can make a heuristic guess as to fuel availability from the runway sizes
165 // For now assume that airports with asphalt or concrete runways will have at least 100L,
166 // and that runways over 4000ft will have JET.
167 if(_aptRwys[0]->surface() <= 2) {
168 if(_aptRwys[0]->lengthFt() >= 4000) {
169 _kln89->DrawText("JET 100L", 2, 0, 1);
171 _kln89->DrawText("100L", 2, 0, 1);
175 _kln89->DrawText("NO APR", 2, 0, 0);
177 // TODO - output proper differentiation of ILS and NP APR and NP APR type eg GPS(R)
178 _kln89->DrawText("NP APR", 2, 0, 0);
180 } else if(_subPage == 3) {
182 _kln89->DrawChar('+', 1, 3, 0);
184 unsigned int i = _curRwyPage * 2;
186 if(i < _aptRwys.size()) {
188 string s = _aptRwys[i]->ident();
189 _kln89->DrawText(s, 2, 9, 3);
190 _kln89->DrawText("/", 2, 12, 3);
191 string recipIdent = _aptRwys[i]->reciprocalRunway()->ident();
192 _kln89->DrawText(recipIdent, 2, 13, 3);
194 s = GPSitoa(int(float(_aptRwys[i]->lengthFt()) * (_kln89->_altUnits == GPS_ALT_UNITS_FT ? 1.0 : SG_FEET_TO_METER) + 0.5));
195 _kln89->DrawText(s, 2, 5 - s.size(), 2);
196 _kln89->DrawText((_kln89->_altUnits == GPS_ALT_UNITS_FT ? "ft" : "m"), 2, 5, 2);
198 // TODO - why not store these strings as an array?
199 switch(_aptRwys[i]->surface()) {
201 // Asphalt - fall through
204 _kln89->DrawText("HRD", 2, 9, 2);
208 // Turf / Turf helipad
209 _kln89->DrawText("TRF", 2, 9, 2);
213 // Dirt / Dirt helipad
214 _kln89->DrawText("DRT", 2, 9, 2);
218 _kln89->DrawText("GRV", 2, 9, 2);
221 // Asphalt helipad - fall through
224 _kln89->DrawText("HRD", 2, 9, 2);
228 _kln89->DrawText("CLY", 2, 9, 2);
231 _kln89->DrawText("MAT", 2, 9, 2);
235 if(i < _aptRwys.size()) {
237 string s = _aptRwys[i]->ident();
238 _kln89->DrawText(s, 2, 9, 1);
239 _kln89->DrawText("/", 2, 12, 1);
240 string recip = _aptRwys[i]->reciprocalRunway()->ident();
241 _kln89->DrawText(recip, 2, 13, 1);
243 s = GPSitoa(int(float(_aptRwys[i]->lengthFt()) * (_kln89->_altUnits == GPS_ALT_UNITS_FT ? 1.0 : SG_FEET_TO_METER) + 0.5));
244 _kln89->DrawText(s, 2, 5 - s.size(), 0);
245 _kln89->DrawText((_kln89->_altUnits == GPS_ALT_UNITS_FT ? "ft" : "m"), 2, 5, 0);
247 // TODO - why not store these strings as an array?
248 switch(_aptRwys[i]->surface()) {
250 // Asphalt - fall through
253 _kln89->DrawText("HRD", 2, 9, 0);
257 // Turf / Turf helipad
258 _kln89->DrawText("TRF", 2, 9, 0);
262 // Dirt / Dirt helipad
263 _kln89->DrawText("DRT", 2, 9, 0);
267 _kln89->DrawText("GRV", 2, 9, 0);
270 // Asphalt helipad - fall through
273 _kln89->DrawText("HRD", 2, 9, 0);
277 _kln89->DrawText("CLY", 2, 9, 0);
280 _kln89->DrawText("MAT", 2, 9, 0);
283 } else if(_subPage == 4) {
284 if(_nFreqPages > 1) {
285 _kln89->DrawChar('+', 1, 3, 0);
287 unsigned int i = _curFreqPage * 3;
288 if(i < _aptFreqs.size()) {
289 _kln89->DrawText(_aptFreqs[i].service, 2, 0, 2);
290 _kln89->DrawFreq(_aptFreqs[i].freq, 2, 7, 2);
293 if(i < _aptFreqs.size()) {
294 _kln89->DrawText(_aptFreqs[i].service, 2, 0, 1);
295 _kln89->DrawFreq(_aptFreqs[i].freq, 2, 7, 1);
298 if(i < _aptFreqs.size()) {
299 _kln89->DrawText(_aptFreqs[i].service, 2, 0, 0);
300 _kln89->DrawFreq(_aptFreqs[i].freq, 2, 7, 0);
302 } else if(_subPage == 5) {
303 // TODO - user ought to be allowed to leave persistent remarks
304 _kln89->DrawText("[Remarks]", 2, 2, 2);
305 } else if(_subPage == 6) {
306 // We don't have SID/STAR database yet
308 _kln89->DrawText("No SID/STAR", 2, 3, 2);
309 _kln89->DrawText("In Data Base", 2, 2, 1);
310 _kln89->DrawText("For This Airport", 2, 0, 0);
311 } else if(_subPage == 7) {
313 _kln89->DrawText("IAP", 2, 11, 3);
314 _kln89->DrawText("No Approach", 2, 3, 2);
315 _kln89->DrawText("In Data Base", 2, 2, 1);
316 _kln89->DrawText("For This Airport", 2, 0, 0);
319 _kln89->DrawText(_iaps[_curIap]->_ident, 2, 1, 3);
320 _kln89->DrawText(_iaps[_curIap]->_rwyStr, 2, 7, 3);
321 _kln89->DrawText(_iaps[_curIap]->_aptIdent, 2, 12, 3);
322 _kln89->DrawText("IAF", 2, 2, 2);
323 unsigned int line = 0;
324 for(unsigned int i=_iafStart; i<_approachRoutes.size(); ++i) {
326 i = _approachRoutes.size() - 1;
328 // Assume that the IAF number is always single digit!
329 _kln89->DrawText(GPSitoa(i+1), 2, 6, 2-line);
330 if(!(_kln89->_mode == KLN89_MODE_CRSR && _kln89->_blink && _uLinePos == (line + 1))) {
331 _kln89->DrawText(_approachRoutes[i]->waypoints[0]->id, 2, 8, 2-line);
333 if(_kln89->_mode == KLN89_MODE_CRSR && _uLinePos == (line + 1) && !(_kln89->_blink )) {
334 _kln89->Underline(2, 8, 2-line, 5);
338 if(_uLinePos > 0 && !(_kln89->_blink)) {
341 } else if(_addDialog) {
342 _kln89->DrawText(_iaps[_curIap]->_ident, 2, 1, 3);
343 _kln89->DrawText(_iaps[_curIap]->_rwyStr, 2, 7, 3);
344 _kln89->DrawText(_iaps[_curIap]->_aptIdent, 2, 12, 3);
345 string s = GPSitoa(_fStart + 1);
346 _kln89->DrawText(s, 2, 2-s.size(), 2);
347 s = GPSitoa(_kln89->_approachFP->waypoints.size());
348 _kln89->DrawText(s, 2, 2-s.size(), 1);
349 if(!(_uLinePos == _fStart+1 && _kln89->_blink)) {
350 _kln89->DrawText(_kln89->_approachFP->waypoints[_fStart]->id, 2, 4, 2);
351 if(_uLinePos == _fStart+1) _kln89->Underline(2, 4, 2, 6);
353 if(!(_uLinePos == _maxULinePos-1 && _kln89->_blink)) {
354 _kln89->DrawText(_kln89->_approachFP->waypoints[_kln89->_approachFP->waypoints.size()-1]->id, 2, 4, 1);
355 if(_uLinePos == _maxULinePos-1) _kln89->Underline(2, 4, 1, 6);
357 if(!(_uLinePos > _kln89->_approachFP->waypoints.size() && _kln89->_blink)) {
358 _kln89->DrawText("ADD TO FPL 0?", 2, 2, 0);
359 if(_uLinePos > _kln89->_approachFP->waypoints.size()) {
360 _kln89->Underline(2, 2, 0, 13);
364 } else if(_replaceDialog) {
365 _kln89->DrawText(_iaps[_curIap]->_ident, 2, 1, 3);
366 _kln89->DrawText(_iaps[_curIap]->_rwyStr, 2, 7, 3);
367 _kln89->DrawText(_iaps[_curIap]->_aptIdent, 2, 12, 3);
368 _kln89->DrawText("Replace Existing", 2, 0, 2);
369 _kln89->DrawText("Approach", 2, 4, 1);
370 if(_uLinePos > 0 && !(_kln89->_blink)) {
371 _kln89->DrawText("APPROVE?", 2, 4, 0);
372 _kln89->Underline(2, 4, 0, 8);
376 _kln89->DrawText("IAP", 2, 11, 3);
378 if(_kln89->_mode == KLN89_MODE_CRSR && _uLinePos > 4) {
380 if(!_kln89->_blink) _kln89->DrawEnt();
382 // _maxULine pos should be 4 + iaps.size() at this point.
383 // Draw a maximum of 3 IAPs.
384 // If there are more than 3 IAPs for this airport, then we need to offset the start
385 // of the list if _uLinePos is pointing at the 4th or later IAP.
386 unsigned int offset = 0;
389 offset = _uLinePos - 7;
391 for(unsigned int i=0; i<3; ++i) {
393 if(index < _iaps.size()) {
394 string s = GPSitoa(index+1);
395 _kln89->DrawText(s, 2, 2 - s.size(), 2-i);
396 if(!(selApp && _uLinePos == index+5 && _kln89->_blink)) {
397 _kln89->DrawText(_iaps[index]->_ident, 2, 3, 2-i);
398 _kln89->DrawText(_iaps[index]->_rwyStr, 2, 9, 2-i);
400 if(selApp && _uLinePos == index+5 && !_kln89->_blink) {
401 _kln89->Underline(2, 3, 2-i, 9);
411 if(_kln89->_mode != KLN89_MODE_CRSR) _kln89->DrawText(_apt_id, 2, 1, 3);
414 _kln89->DrawText("----.-", 2, 9, 3);
415 _kln89->DrawText("--------------", 2, 0, 2);
416 _kln89->DrawText("- -- --.--'", 2, 3, 1);
417 _kln89->DrawText("---- --.--'", 2, 3, 0);
418 _kln89->DrawSpecialChar(0, 2, 7, 1);
419 _kln89->DrawSpecialChar(0, 2, 7, 0);
424 if(_kln89->_mode == KLN89_MODE_CRSR) {
425 if(!(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) {
426 if(_uLinePos > 0 && _uLinePos < 5) {
427 // TODO - blink as well
428 _kln89->Underline(2, _uLinePos, 3, 1);
430 for(unsigned int i = 0; i < _apt_id.size(); ++i) {
431 if(_uLinePos != (i + 1)) {
432 _kln89->DrawChar(_apt_id[i], 2, i + 1, 3);
434 if(!_kln89->_blink) _kln89->DrawChar(_apt_id[i], 2, i + 1, 3);
442 KLN89Page::Update(dt);
445 void KLN89AptPage::SetId(const string& s) {
446 if(s != _apt_id || s != _last_apt_id) {
447 UpdateAirport(s); // If we don't do this here we break things if s is the same as the current ID since the update wouldn't get called then.
449 DCL: Hmmm - I wrote the comment above, but I don't quite understand it!
450 I'm not quite sure why I don't simply set _apt_id here (and NOT _last_apt_id)
451 and let the logic in Update(...) handle the airport details cache update.
454 _last_apt_id = _apt_id;
455 _save_apt_id = _apt_id;
459 // Update the cached airport details
460 void KLN89AptPage::UpdateAirport(const string& id) {
464 const FGAirport* apt = fgFindAirportID(id);
466 throw sg_exception("UpdateAirport: unknown airport id " + id);
469 for (unsigned int c=0; c<apt->commStations().size(); ++c) {
470 flightgear::CommStation* comm = apt->commStations()[c];
472 aq.freq = comm->freqKHz();
473 switch (comm->type()) {
474 case FGPositioned::FREQ_ATIS:
475 aq.service = "ATIS*"; break;
476 case FGPositioned::FREQ_GROUND:
477 aq.service = "GRND*"; break;
478 case FGPositioned::FREQ_TOWER:
479 aq.service = "TWR *"; break;
480 case FGPositioned::FREQ_APP_DEP:
481 aq.service = "APR *"; break;
487 _nFreqPages = (unsigned int)ceil((float(_aptFreqs.size())) / 3.0f);
492 // build local array, longest runway first
493 for (unsigned int r=0; r<apt->numRunways(); ++r) {
494 FGRunway* rwy(apt->getRunwayByIndex(r));
495 if ((r > 0) && (rwy->lengthFt() > _aptRwys.front()->lengthFt())) {
496 _aptRwys.insert(_aptRwys.begin(), rwy);
498 _aptRwys.push_back(rwy);
502 _nRwyPages = (_aptRwys.size() + 1) / 2; // 2 runways per page.
503 if(_nFreqPages < 1) _nFreqPages = 1;
504 if(_nRwyPages < 1) _nRwyPages = 1;
506 // Instrument approaches
507 // Only non-precision for now - TODO - handle precision approaches if necessary
509 iap_map_iterator itr = _kln89->_np_iap.find(id);
510 if(itr != _kln89->_np_iap.end()) {
514 if(_iafDialog || _addDialog || _replaceDialog) {
515 // Eek - major logic error if an airport details cache update occurs
516 // with one of these dialogs active.
517 // TODO - output a warning.
518 //cout << "HELP!!!!!!!!!!\n";
520 _maxULinePos = 4 + _iaps.size(); // We shouldn't need to check the crsr for out-of-bounds here since we only update the airport details when the airport code is changed - ie. _uLinePos <= 4!
525 void KLN89AptPage::CrsrPressed() {
526 if(_kln89->_mode == KLN89_MODE_DISP) {
528 // Pressing crsr jumps back to vanilla IAP page.
531 _replaceDialog = false;
535 if(_kln89->_obsMode) {
542 } else if(_subPage == 7) {
543 // Don't *think* we need some of this since some of it we can only get to by pressing ENT, not CRSR.
545 _maxULinePos = _approachRoutes.size();
547 } else if(_addDialog) {
550 } else if(_replaceDialog) {
554 _maxULinePos = 4 + _iaps.size();
566 void KLN89AptPage::ClrPressed() {
567 if(_subPage == 1 && _uLinePos == 5) {
568 _to_flag = !_to_flag;
569 } else if(_subPage == 7) {
570 // Clear backs out IAP selection one step at a time
573 _maxULinePos = 4 + _iaps.size();
579 } else if(_addDialog) {
581 if(_approachRoutes.size() > 1) {
584 // Don't reset _curIaf since it is remembed.
585 _uLinePos = 1 + _curIaf; // TODO - make this robust to more than 3 IAF
587 _maxULinePos = 4 + _iaps.size();
594 } else if(_replaceDialog) {
595 _replaceDialog = false;
603 void KLN89AptPage::EntPressed() {
606 if(_kln89->_dtoReview) {
607 _kln89->DtoInitiate(_apt_id);
609 _last_apt_id = _apt_id;
610 _apt_id = _save_apt_id;
612 } else if(_subPage == 7 && _kln89->_mode == KLN89_MODE_CRSR && _uLinePos > 0) {
613 // We are selecting an approach
616 // Record the IAF that was picked
618 _curIaf = _approachRoutes.size() - 1;
620 _curIaf = _uLinePos - 1 + _iafStart;
622 //cout << "_curIaf = " << _curIaf << '\n';
623 // TODO - delete the waypoints inside _approachFP before clearing them!!!!!!!
624 _kln89->_approachFP->waypoints.clear();
625 GPSWaypoint* wp = new GPSWaypoint;
626 *wp = *(_approachRoutes[_curIaf]->waypoints[0]); // Need to make copies here since we're going to alter ID and type sometimes
627 string iafid = wp->id;
628 _kln89->_approachFP->waypoints.push_back(wp);
629 for(unsigned int i=0; i<_IAP.size(); ++i) {
630 if(_IAP[i]->id != iafid) { // Don't duplicate waypoints that are part of the initial fix list and the approach procedure list.
631 // FIXME - allow the same waypoint to be both the IAF and the FAF in some
632 // approaches that have a procedure turn eg. KDLL
633 // Also allow MAF to be the same as IAF!
634 wp = new GPSWaypoint;
636 //cout << "Adding waypoint " << wp->id << ", type is " << wp->appType << '\n';
637 //if(wp->appType == GPS_FAF) wp->id += 'f';
638 //if(wp->appType == GPS_MAP) wp->id += 'm';
639 //cout << "New id = " << wp->id << '\n';
640 _kln89->_approachFP->waypoints.push_back(wp);
645 _maxULinePos = _kln89->_approachFP->waypoints.size() + 1;
646 _uLinePos = _maxULinePos;
648 } else if(_addDialog) {
649 if(_uLinePos == _maxULinePos) {
651 if(_kln89->ApproachLoaded()) {
652 _replaceDialog = true;
656 // Now load the approach into the active flightplan.
657 // As far as I can tell, the rules are this:
658 // If the airport of the approach is in the flightplan, insert it prior to this. (Not sure what happens if airport has already been passed).
659 // If the airport is not in the flightplan, append the approach to the flightplan, even if it is closer than the current active leg,
660 // in which case reorientate to flightplan might put us on the approach, but unable to activate it.
661 // However, it appears from the sim as if this can indeed happen if the user is not carefull.
663 for(unsigned int i=0; i<_kln89->_activeFP->waypoints.size(); ++i) {
664 if(_kln89->_activeFP->waypoints[i]->id == _apt_id) {
665 _kln89->_activeFP->waypoints.insert(_kln89->_activeFP->waypoints.begin()+i, _kln89->_approachFP->waypoints.begin(), _kln89->_approachFP->waypoints.end());
671 _kln89->_activeFP->waypoints.insert(_kln89->_activeFP->waypoints.end(), _kln89->_approachFP->waypoints.begin(), _kln89->_approachFP->waypoints.end());
673 _kln89->_approachID = _apt_id;
674 _kln89->_approachAbbrev = _iaps[_curIap]->_ident;
675 _kln89->_approachRwyStr = _iaps[_curIap]->_rwyStr;
676 _kln89->_approachLoaded = true;
677 //_kln89->_messageStack.push_back("*Press ALT To Set Baro");
678 // Actually - this message is only sent when we go into appraoch-arm mode.
679 // TODO - check the flightplan for consistency
680 _kln89->OrientateToActiveFlightPlan();
681 _kln89->_mode = KLN89_MODE_DISP;
682 _kln89->_curPage = 7;
683 _kln89->_activePage = _kln89->_pages[7]; // Do we need to clean up here at all before jumping?
686 } else if(_replaceDialog) {
687 // TODO - load the approach!
688 } else if(_uLinePos > 4) {
689 _approachRoutes.clear();
692 _approachRoutes = ((FGNPIAP*)(_iaps[_uLinePos-5]))->_approachRoutes;
693 _IAP = ((FGNPIAP*)(_iaps[_uLinePos-5]))->_IAP;
694 _curIap = _uLinePos - 5; // TODO - handle the start of list ! no. 1, and the end of list not sequential!
696 if(_approachRoutes.size() > 1) {
697 // More than 1 IAF - display the selection dialog
699 _maxULinePos = _approachRoutes.size();
701 // There is only 1 IAF, so load the waypoints into the approach flightplan here.
702 // TODO - there is nasty code duplication loading the approach FP between the case here where we have only one
703 // IAF and the case where we must choose the IAF from a list. Try to tidy this after it is all working properly.
704 _kln89->_approachFP->waypoints.clear();
705 GPSWaypoint* wp = new GPSWaypoint;
706 *wp = *(_approachRoutes[0]->waypoints[0]); // Need to make copies here since we're going to alter ID and type sometimes
707 string iafid = wp->id;
708 _kln89->_approachFP->waypoints.push_back(wp);
709 for(unsigned int i=0; i<_IAP.size(); ++i) {
710 if(_IAP[i]->id != iafid) { // Don't duplicate waypoints that are part of the initial fix list and the approach procedure list.
711 // FIXME - allow the same waypoint to be both the IAF and the FAF in some
712 // approaches that have a procedure turn eg. KDLL
713 // Also allow MAF to be the same as IAF!
714 wp = new GPSWaypoint;
716 _kln89->_approachFP->waypoints.push_back(wp);
726 void KLN89AptPage::Knob1Left1() {
727 if(_kln89->_mode == KLN89_MODE_CRSR && _subPage == 7 && _addDialog) {
728 if(_uLinePos == _maxULinePos) {
730 if(_kln89->_approachFP->waypoints.size() > 1) _fStart = _kln89->_approachFP->waypoints.size() - 2;
731 } else if(_uLinePos == _maxULinePos - 1) {
733 } else if(_uLinePos > 0) {
742 KLN89Page::Knob1Left1();
746 void KLN89AptPage::Knob1Right1() {
747 if(_kln89->_mode == KLN89_MODE_CRSR && _subPage == 7 && _addDialog) {
748 if(_uLinePos == _maxULinePos) {
750 } else if(_uLinePos == _maxULinePos - 1) {
753 } else if(_uLinePos > 0) {
754 if(_fStart >= _kln89->_approachFP->waypoints.size() - 2) {
760 } else if(_uLinePos == 0) {
765 KLN89Page::Knob1Right1();
769 void KLN89AptPage::Knob2Left1() {
770 if(_kln89->_mode != KLN89_MODE_CRSR || _uLinePos == 0) {
771 if(_uLinePos == 0 && _kln89->_mode == KLN89_MODE_CRSR && _kln89->_obsMode) {
772 KLN89Page::Knob2Left1();
773 } else if(_subPage == 5) {
775 _curFreqPage = _nFreqPages - 1;
776 } else if(_subPage == 4) {
778 if(_curFreqPage == 0) {
780 _curRwyPage = _nRwyPages - 1;
784 } else if(_subPage == 3) {
785 if(_curRwyPage == 0) {
786 KLN89Page::Knob2Left1();
790 } else if(_subPage == 0) {
792 // We have to set _uLinePos here even though the cursor isn't pressed, to
793 // ensure that the list displays properly.
800 KLN89Page::Knob2Left1();
803 if(_uLinePos < 5 && !(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) {
804 // Same logic for all pages - set the ID
805 _apt_id = _apt_id.substr(0, _uLinePos);
806 // ASSERT(_uLinePos > 0);
807 if(_uLinePos == (_apt_id.size() + 1)) {
810 _apt_id[_uLinePos - 1] = _kln89->DecChar(_apt_id[_uLinePos - 1], (_uLinePos == 1 ? false : true));
814 // TODO - set by name
816 // NO-OP - to/fr is cycled by clr button
822 void KLN89AptPage::Knob2Right1() {
823 if(_kln89->_mode != KLN89_MODE_CRSR || _uLinePos == 0) {
824 if(_uLinePos == 0 && _kln89->_mode == KLN89_MODE_CRSR && _kln89->_obsMode) {
825 KLN89Page::Knob2Right1();
826 } else if(_subPage == 2) {
829 } else if(_subPage == 3) {
830 if(_curRwyPage == _nRwyPages - 1) {
836 } else if(_subPage == 4) {
837 if(_curFreqPage == _nFreqPages - 1) {
842 } else if(_subPage == 6) {
844 // We have to set _uLinePos here even though the cursor isn't pressed, to
845 // ensure that the list displays properly.
852 KLN89Page::Knob2Right1();
855 if(_uLinePos < 5 && !(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) {
856 // Same logic for all pages - set the ID
857 _apt_id = _apt_id.substr(0, _uLinePos);
858 // ASSERT(_uLinePos > 0);
859 if(_uLinePos == (_apt_id.size() + 1)) {
862 _apt_id[_uLinePos - 1] = _kln89->IncChar(_apt_id[_uLinePos - 1], (_uLinePos == 1 ? false : true));
866 // TODO - set by name
868 // NO-OP - to/fr is cycled by clr button