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 - david.luff@nottingham.ac.uk
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., 675 Mass Ave, Cambridge, MA 02139, USA.
24 #include "kln89_page_apt.hxx"
25 #include <ATC/commlist.hxx>
27 // This function is copied from Airports/runways.cxx
28 // TODO - Make the original properly available and remove this instance!!!!
29 // Return reverse rwy number
32 static string GetReverseRunwayNo(string rwyno) {
33 // cout << "Original rwyno = " << rwyNo << '\n';
35 // standardize input number
36 string tmp = rwyno.substr(1, 1);
37 if (( tmp == "L" || tmp == "R" || tmp == "C" ) || (rwyno.size() == 1)) {
40 SG_LOG( SG_GENERAL, SG_INFO,
41 "Standardising rwy number from " << tmp << " to " << rwyno );
45 int rn = atoi(rwyno.substr(0,2).c_str());
50 sprintf(buf, "%02i", rn);
51 if(rwyno.size() == 3) {
52 if(rwyno.substr(2,1) == "L") {
55 } else if (rwyno.substr(2,1) == "R") {
58 } else if (rwyno.substr(2,1) == "C") {
61 } else if (rwyno.substr(2,1) == "T") {
65 SG_LOG(SG_GENERAL, SG_ALERT, "Unknown runway code "
66 << rwyno << " passed to GetReverseRunwayNo(...)");
72 KLN89AptPage::KLN89AptPage(KLN89* parent)
78 // Make sure that _last_apt_id doesn't match at startup to force airport data to be fetched on first update.
79 _last_apt_id = "XXXX";
91 _replaceDialog = false;
96 KLN89AptPage::~KLN89AptPage() {
99 void KLN89AptPage::Update(double dt) {
100 bool actPage = (_kln89->_activePage->GetName() == "ACT" ? true : false);
101 bool multi; // Not set by FindFirst...
103 if(_apt_id.size() == 4) exact = true;
104 // TODO - move this search out to where the button is pressed, and cache the result!
105 if(_apt_id != _last_apt_id || ap == NULL) ap = _kln89->FindFirstAptById(_apt_id, multi, exact);
106 //if(np == NULL) cout << "NULL... ";
107 //if(b == false) cout << "false...\n";
110 cout << "VOR FOUND!\n";
117 //cout << "Valid airport found! id = " << ap->getId() << ", elev = " << ap->getElevation() << '\n';
118 if(_apt_id != _last_apt_id) {
119 UpdateAirport(ap->getId());
120 _last_apt_id = _apt_id;
124 _apt_id = ap->getId();
125 if(_kln89->GetActiveWaypoint()) {
126 if(_apt_id == _kln89->GetActiveWaypoint()->id) {
127 if(!(_kln89->_waypointAlert && _kln89->_blink)) {
128 // Active waypoint arrow
129 _kln89->DrawSpecialChar(4, 2, 0, 3);
133 if(_kln89->_mode != KLN89_MODE_CRSR) {
134 if(!(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) { // Don't draw the airport name when the IAP dialogs are active
137 _kln89->DrawText(ap->getId(), 2, 1, 3);
139 // If it's the ACT page, The ID is shifted slightly right to make space for the waypoint index.
140 _kln89->DrawText(ap->getId(), 2, 4, 3);
142 int n = snprintf(buf, 3, "%i", _kln89->GetActiveWaypointIndex() + 1);
143 _kln89->DrawText((string)buf, 2, 3 - n, 3);
146 if(!_kln89->_blink) {
147 _kln89->DrawText(ap->getId(), 2, 1, 3, false, 99);
155 _kln89->DrawText(ap->getName(), 2, 0, 2);
157 _kln89->DrawText(_kln89->_altUnits == GPS_ALT_UNITS_FT ? "ft" : "m", 2, 14, 3);
159 int n = snprintf(buf, 5, "%i", (_kln89->_altUnits == GPS_ALT_UNITS_FT ? (int)(ap->getElevation()) : (int)((double)ap->getElevation() * SG_FEET_TO_METER)));
160 _kln89->DrawText((string)buf, 2, 14 - n, 3);
162 airport_id_str_map_iterator itr = _kln89->_airportTowns.find(_apt_id);
163 if(itr != _kln89->_airportTowns.end()) {
164 _kln89->DrawText(itr->second, 2, 0, 1);
166 // State / Province / Country
167 itr = _kln89->_airportStates.find(_apt_id);
168 if(itr != _kln89->_airportStates.end()) {
169 _kln89->DrawText(itr->second, 2, 0, 0);
171 } else if(_subPage == 1) {
172 _kln89->DrawLatitude(ap->getLatitude(), 2, 3, 2);
173 _kln89->DrawLongitude(ap->getLongitude(), 2, 3, 1);
174 _kln89->DrawDirDistField(ap->getLatitude() * SG_DEGREES_TO_RADIANS, ap->getLongitude() * SG_DEGREES_TO_RADIANS,
175 2, 0, 0, _to_flag, (_kln89->_mode == KLN89_MODE_CRSR && _uLinePos == 5 ? true : false));
176 } else if(_subPage == 2) {
177 // Try and calculate a realistic difference from UTC based on longitude
178 float degLonPerHr = 360.0 / 24.0; // 15 degrees per hour difference.
179 // Since 0 longitude is the middle of UTC, the boundaries will be at 7.5, 22.5, 37.5 etc.
180 int hrDiff = ((int)((fabs(ap->getLongitude())) + 7.5)) / 15;
181 _kln89->DrawText("UTC", 2, 0, 2);
183 _kln89->DrawText(ap->getLongitude() >= 0.0 ? "+" : "-", 2, 3, 2);
185 snprintf(buf, 3, "%02i", hrDiff);
186 _kln89->DrawText((string)buf, 2, 4, 2);
187 _kln89->DrawText("( DT)", 2, 6, 2);
188 if(ap->getLongitude() >= 0.0) {
193 _kln89->DrawText(ap->getLongitude() >= 0.0 ? "+" : "-", 2, 7, 2);
194 snprintf(buf, 3, "%02i", hrDiff);
195 _kln89->DrawText((string)buf, 2, 8, 2);
197 // I guess we can make a heuristic guess as to fuel availability from the runway sizes
198 // For now assume that airports with asphalt or concrete runways will have at least 100L,
199 // and that runways over 4000ft will have JET.
200 if(_aptRwys[0]._surface_code <= 2) {
201 if(_aptRwys[0]._length >= 4000) {
202 _kln89->DrawText("JET 100L", 2, 0, 1);
204 _kln89->DrawText("100L", 2, 0, 1);
208 _kln89->DrawText("NO APR", 2, 0, 0);
210 // TODO - output proper differentiation of ILS and NP APR and NP APR type eg GPS(R)
211 _kln89->DrawText("NP APR", 2, 0, 0);
213 } else if(_subPage == 3) {
215 _kln89->DrawChar('+', 1, 3, 0);
217 unsigned int i = _curRwyPage * 2;
219 if(i < _aptRwys.size()) {
221 string s = _aptRwys[i]._rwy_no;
222 _kln89->DrawText(s, 2, 9, 3);
223 _kln89->DrawText("/", 2, 12, 3);
224 _kln89->DrawText(GetReverseRunwayNo(s), 2, 13, 3);
226 s = GPSitoa(int(float(_aptRwys[i]._length) * (_kln89->_altUnits == GPS_ALT_UNITS_FT ? 1.0 : SG_FEET_TO_METER) + 0.5));
227 _kln89->DrawText(s, 2, 5 - s.size(), 2);
228 _kln89->DrawText((_kln89->_altUnits == GPS_ALT_UNITS_FT ? "ft" : "m"), 2, 5, 2);
230 // TODO - why not store these strings as an array?
231 switch(_aptRwys[i]._surface_code) {
233 // Asphalt - fall through
236 _kln89->DrawText("HRD", 2, 9, 2);
240 // Turf / Turf helipad
241 _kln89->DrawText("TRF", 2, 9, 2);
245 // Dirt / Dirt helipad
246 _kln89->DrawText("DRT", 2, 9, 2);
250 _kln89->DrawText("GRV", 2, 9, 2);
253 // Asphalt helipad - fall through
256 _kln89->DrawText("HRD", 2, 9, 2);
260 _kln89->DrawText("CLY", 2, 9, 2);
263 _kln89->DrawText("MAT", 2, 9, 2);
267 if(i < _aptRwys.size()) {
269 string s = _aptRwys[i]._rwy_no;
270 _kln89->DrawText(s, 2, 9, 1);
271 _kln89->DrawText("/", 2, 12, 1);
272 _kln89->DrawText(GetReverseRunwayNo(s), 2, 13, 1);
274 s = GPSitoa(int(float(_aptRwys[i]._length) * (_kln89->_altUnits == GPS_ALT_UNITS_FT ? 1.0 : SG_FEET_TO_METER) + 0.5));
275 _kln89->DrawText(s, 2, 5 - s.size(), 0);
276 _kln89->DrawText((_kln89->_altUnits == GPS_ALT_UNITS_FT ? "ft" : "m"), 2, 5, 0);
278 // TODO - why not store these strings as an array?
279 switch(_aptRwys[i]._surface_code) {
281 // Asphalt - fall through
284 _kln89->DrawText("HRD", 2, 9, 0);
288 // Turf / Turf helipad
289 _kln89->DrawText("TRF", 2, 9, 0);
293 // Dirt / Dirt helipad
294 _kln89->DrawText("DRT", 2, 9, 0);
298 _kln89->DrawText("GRV", 2, 9, 0);
301 // Asphalt helipad - fall through
304 _kln89->DrawText("HRD", 2, 9, 0);
308 _kln89->DrawText("CLY", 2, 9, 0);
311 _kln89->DrawText("MAT", 2, 9, 0);
314 } else if(_subPage == 4) {
315 if(_nFreqPages > 1) {
316 _kln89->DrawChar('+', 1, 3, 0);
318 unsigned int i = _curFreqPage * 3;
319 if(i < _aptFreqs.size()) {
320 _kln89->DrawText(_aptFreqs[i].service, 2, 0, 2);
321 _kln89->DrawFreq(_aptFreqs[i].freq, 2, 7, 2);
324 if(i < _aptFreqs.size()) {
325 _kln89->DrawText(_aptFreqs[i].service, 2, 0, 1);
326 _kln89->DrawFreq(_aptFreqs[i].freq, 2, 7, 1);
329 if(i < _aptFreqs.size()) {
330 _kln89->DrawText(_aptFreqs[i].service, 2, 0, 0);
331 _kln89->DrawFreq(_aptFreqs[i].freq, 2, 7, 0);
333 } else if(_subPage == 5) {
334 // TODO - user ought to be allowed to leave persistent remarks
335 _kln89->DrawText("[Remarks]", 2, 2, 2);
336 } else if(_subPage == 6) {
337 // We don't have SID/STAR database yet
339 _kln89->DrawText("No SID/STAR", 2, 3, 2);
340 _kln89->DrawText("In Data Base", 2, 2, 1);
341 _kln89->DrawText("For This Airport", 2, 0, 0);
342 } else if(_subPage == 7) {
344 _kln89->DrawText("IAP", 2, 11, 3);
345 _kln89->DrawText("No Approach", 2, 3, 2);
346 _kln89->DrawText("In Data Base", 2, 2, 1);
347 _kln89->DrawText("For This Airport", 2, 0, 0);
350 _kln89->DrawText(_iaps[_curIap]->_abbrev, 2, 1, 3);
351 _kln89->DrawText(_iaps[_curIap]->_rwyStr, 2, 7, 3);
352 _kln89->DrawText(_iaps[_curIap]->_id, 2, 12, 3);
353 _kln89->DrawText("IAF", 2, 2, 2);
355 for(unsigned int i=_iafStart; i<_IAF.size(); ++i) {
359 // Assume that the IAF number is always single digit!
360 _kln89->DrawText(GPSitoa(i+1), 2, 6, 2-line);
361 if(!(_kln89->_mode == KLN89_MODE_CRSR && _kln89->_blink && _uLinePos == (line + 1))) {
362 _kln89->DrawText(_IAF[i]->id, 2, 8, 2-line);
364 if(_kln89->_mode == KLN89_MODE_CRSR && _uLinePos == (line + 1) && !(_kln89->_blink )) {
365 _kln89->Underline(2, 8, 2-line, 5);
369 if(_uLinePos > 0 && !(_kln89->_blink)) {
372 } else if(_addDialog) {
373 _kln89->DrawText(_iaps[_curIap]->_abbrev, 2, 1, 3);
374 _kln89->DrawText(_iaps[_curIap]->_rwyStr, 2, 7, 3);
375 _kln89->DrawText(_iaps[_curIap]->_id, 2, 12, 3);
376 string s = GPSitoa(_fStart + 1);
377 _kln89->DrawText(s, 2, 2-s.size(), 2);
378 s = GPSitoa(_kln89->_approachFP->waypoints.size());
379 _kln89->DrawText(s, 2, 2-s.size(), 1);
380 if(!(_uLinePos == _fStart+1 && _kln89->_blink)) {
381 _kln89->DrawText(_kln89->_approachFP->waypoints[_fStart]->id, 2, 4, 2);
382 if(_uLinePos == _fStart+1) _kln89->Underline(2, 4, 2, 6);
384 if(!(_uLinePos == _maxULinePos-1 && _kln89->_blink)) {
385 _kln89->DrawText(_kln89->_approachFP->waypoints[_kln89->_approachFP->waypoints.size()-1]->id, 2, 4, 1);
386 if(_uLinePos == _maxULinePos-1) _kln89->Underline(2, 4, 1, 6);
388 if(!(_uLinePos > _kln89->_approachFP->waypoints.size() && _kln89->_blink)) {
389 _kln89->DrawText("ADD TO FPL 0?", 2, 2, 0);
390 if(_uLinePos > _kln89->_approachFP->waypoints.size()) {
391 _kln89->Underline(2, 2, 0, 13);
395 } else if(_replaceDialog) {
396 _kln89->DrawText(_iaps[_curIap]->_abbrev, 2, 1, 3);
397 _kln89->DrawText(_iaps[_curIap]->_rwyStr, 2, 7, 3);
398 _kln89->DrawText(_iaps[_curIap]->_id, 2, 12, 3);
399 _kln89->DrawText("Replace Existing", 2, 0, 2);
400 _kln89->DrawText("Approach", 2, 4, 1);
401 if(_uLinePos > 0 && !(_kln89->_blink)) {
402 _kln89->DrawText("APPROVE?", 2, 4, 0);
403 _kln89->Underline(2, 4, 0, 8);
407 _kln89->DrawText("IAP", 2, 11, 3);
410 if(_kln89->_mode == KLN89_MODE_CRSR && _uLinePos > 4) {
412 if(!_kln89->_blink) _kln89->DrawEnt();
414 for(unsigned int i=0; i<_iaps.size(); ++i) { // TODO - do this properly when > 3 IAPs
415 string s = GPSitoa(i+1);
416 _kln89->DrawText(s, 2, 2 - s.size(), 2-i);
417 if(!(selApp && _uLinePos == 5+i && _kln89->_blink)) {
418 _kln89->DrawText(_iaps[i]->_abbrev, 2, 3, 2-i);
419 _kln89->DrawText(_iaps[i]->_rwyStr, 2, 9, 2-i);
421 if(selApp && _uLinePos == 5+i && !_kln89->_blink) {
422 _kln89->Underline(2, 3, 2-i, 9);
431 if(_kln89->_mode != KLN89_MODE_CRSR) _kln89->DrawText(_apt_id, 2, 1, 3);
434 _kln89->DrawText("----.-", 2, 9, 3);
435 _kln89->DrawText("--------------", 2, 0, 2);
436 _kln89->DrawText("- -- --.--'", 2, 3, 1);
437 _kln89->DrawText("---- --.--'", 2, 3, 0);
438 _kln89->DrawSpecialChar(0, 2, 7, 1);
439 _kln89->DrawSpecialChar(0, 2, 7, 0);
444 if(_kln89->_mode == KLN89_MODE_CRSR) {
445 if(!(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) {
446 if(_uLinePos > 0 && _uLinePos < 5) {
447 // TODO - blink as well
448 _kln89->Underline(2, _uLinePos, 3, 1);
450 for(unsigned int i = 0; i < _apt_id.size(); ++i) {
451 if(_uLinePos != (i + 1)) {
452 _kln89->DrawChar(_apt_id[i], 2, i + 1, 3);
454 if(!_kln89->_blink) _kln89->DrawChar(_apt_id[i], 2, i + 1, 3);
462 KLN89Page::Update(dt);
465 void KLN89AptPage::SetId(string s) {
466 _last_apt_id = _apt_id;
467 _save_apt_id = _apt_id;
469 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.
472 // Update the cached airport details
473 void KLN89AptPage::UpdateAirport(const string& id) {
478 //cout << "UpdateAirport called, id = " << id << '\n';
479 // TODO - the logic below only returns one service per type per airport - they can be on more than one freq though.
480 if(current_commlist->FindByCode(id, ad, ATIS)) {
481 //cout << "Found ATIS\n";
482 aq.service = "ATIS*";
484 _aptFreqs.push_back(aq);
486 if(current_commlist->FindByCode(id, ad, GROUND)) {
487 aq.service = "GRND*";
489 _aptFreqs.push_back(aq);
491 if(current_commlist->FindByCode(id, ad, TOWER)) {
492 aq.service = "TWR *";
494 _aptFreqs.push_back(aq);
496 if(current_commlist->FindByCode(id, ad, APPROACH)) {
499 _aptFreqs.push_back(aq);
501 _nFreqPages = (unsigned int)ceil((float(_aptFreqs.size())) / 3.0f);
506 bool haveRwy = globals->get_runways()->search(id, &r);
507 while(haveRwy && r._id == id) {
508 // Insert the runway with longest at the start of the array
509 for(unsigned int i = 0; i <= _aptRwys.size(); ++i) {
510 if(i == _aptRwys.size()) {
511 _aptRwys.push_back(r);
514 if(r._length > _aptRwys[i]._length) {
515 _aptRwys.insert(_aptRwys.begin() + i, r);
520 haveRwy = globals->get_runways()->next(&r);
522 _nRwyPages = (_aptRwys.size() + 1) / 2; // 2 runways per page.
523 if(_nFreqPages < 1) _nFreqPages = 1;
524 if(_nRwyPages < 1) _nRwyPages = 1;
526 // Instrument approaches
527 // Only non-precision for now - TODO - handle precision approaches if necessary
529 iap_map_iterator itr = _kln89->_np_iap.find(id);
530 if(itr != _kln89->_np_iap.end()) {
533 if(_subPage == 7) _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!
536 void KLN89AptPage::CrsrPressed() {
537 if(_kln89->_mode == KLN89_MODE_DISP) {
539 // Pressing crsr jumps back to vanilla IAP page.
542 _replaceDialog = false;
546 if(_kln89->_obsMode) {
553 } else if(_subPage == 7) {
554 // Don't *think* we need some of this since some of it we can only get to by pressing ENT, not CRSR.
556 _maxULinePos = _IAF.size();
558 } else if(_addDialog) {
561 } else if(_replaceDialog) {
565 _maxULinePos = 4 + _iaps.size();
577 void KLN89AptPage::ClrPressed() {
578 if(_subPage == 1 && _uLinePos == 5) {
579 _to_flag = !_to_flag;
580 } else if(_subPage == 7) {
581 // Clear backs out IAP selection one step at a time
584 _maxULinePos = 4 + _iaps.size();
590 } else if(_addDialog) {
592 if(_IAF.size() > 1) {
595 // Don't reset _curIaf since it is remembed.
596 _uLinePos = 1 + _curIaf; // TODO - make this robust to more than 3 IAF
598 _maxULinePos = 4 + _iaps.size();
605 } else if(_replaceDialog) {
606 _replaceDialog = false;
614 void KLN89AptPage::EntPressed() {
618 _last_apt_id = _apt_id;
619 _apt_id = _save_apt_id;
620 } else if(_subPage == 7 && _kln89->_mode == KLN89_MODE_CRSR && _uLinePos > 0) {
622 // We are selecting an approach
626 // Record the IAF that was picked
628 _curIaf = _IAF.size() - 1;
630 _curIaf = _uLinePos - 1 + _iafStart;
632 //cout << "_curIaf = " << _curIaf << '\n';
633 // TODO - delete the waypoints inside _approachFP before clearing them!!!!!!!
634 _kln89->_approachFP->waypoints.clear();
635 GPSWaypoint* wp = new GPSWaypoint;
636 *wp = *_IAF[_curIaf]; // Need to make copies here since we're going to alter ID and type sometimes
637 string iafid = wp->id;
639 _kln89->_approachFP->waypoints.push_back(wp);
640 for(unsigned int i=0; i<_IAP.size(); ++i) {
641 if(_IAP[i]->id != iafid) { // Don't duplicate waypoints that are part of the initial fix list and the approach procedure list.
642 // FIXME - allow the same waypoint to be both the IAF and the FAF in some
643 // approaches that have a procedure turn eg. KDLL
644 // Also allow MAF to be the same as IAF!
645 wp = new GPSWaypoint;
647 //cout << "Adding waypoint " << wp->id << ", type is " << wp->appType << '\n';
648 //if(wp->appType == GPS_FAF) wp->id += 'f';
649 //if(wp->appType == GPS_MAP) wp->id += 'm';
650 //cout << "New id = " << wp->id << '\n';
651 _kln89->_approachFP->waypoints.push_back(wp);
654 // Only add 1 missed approach procedure waypoint for now. I think this might be standard always anyway.
655 wp = new GPSWaypoint;
658 _kln89->_approachFP->waypoints.push_back(wp);
661 _maxULinePos = _kln89->_approachFP->waypoints.size() + 1;
662 _uLinePos = _maxULinePos;
664 } else if(_addDialog) {
665 if(_uLinePos == _maxULinePos) {
667 if(_kln89->ApproachLoaded()) {
668 _replaceDialog = true;
672 // Now load the approach into the active flightplan.
673 // As far as I can tell, the rules are this:
674 // 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).
675 // If the airport is not in the flightplan, append the approach to the flightplan, even if it is closer than the current active leg,
676 // in which case reorientate to flightplan might put us on the approach, but unable to activate it.
677 // However, it appears from the sim as if this can indeed happen if the user is not carefull.
679 for(unsigned int i=0; i<_kln89->_activeFP->waypoints.size(); ++i) {
680 if(_kln89->_activeFP->waypoints[i]->id == _apt_id) {
681 _kln89->_activeFP->waypoints.insert(_kln89->_activeFP->waypoints.begin()+i, _kln89->_approachFP->waypoints.begin(), _kln89->_approachFP->waypoints.end());
687 _kln89->_activeFP->waypoints.insert(_kln89->_activeFP->waypoints.end(), _kln89->_approachFP->waypoints.begin(), _kln89->_approachFP->waypoints.end());
689 _kln89->_approachID = _apt_id;
690 _kln89->_approachAbbrev = _iaps[_curIap]->_abbrev;
691 _kln89->_approachRwyStr = _iaps[_curIap]->_rwyStr;
692 _kln89->_approachLoaded = true;
693 //_kln89->_messageStack.push_back("*Press ALT To Set Baro");
694 // Actually - this message is only sent when we go into appraoch-arm mode.
695 // TODO - check the flightplan for consistency
696 _kln89->OrientateToActiveFlightPlan();
697 _kln89->_mode = KLN89_MODE_DISP;
698 _kln89->_curPage = 7;
699 _kln89->_activePage = _kln89->_pages[7]; // Do we need to clean up here at all before jumping?
702 } else if(_replaceDialog) {
703 // TODO - load the approach!
704 } else if(_uLinePos > 4) {
709 _IAF = ((FGNPIAP*)(_iaps[_uLinePos-5]))->_IAF;
710 _IAP = ((FGNPIAP*)(_iaps[_uLinePos-5]))->_IAP;
711 _MAP = ((FGNPIAP*)(_iaps[_uLinePos-5]))->_MAP;
712 _curIap = _uLinePos - 5; // TODO - handle the start of list ! no. 1, and the end of list not sequential!
714 if(_IAF.size() > 1) {
715 // More than 1 IAF - display the selection dialog
717 _maxULinePos = _IAF.size();
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();
791 KLN89Page::Knob2Left1();
794 if(_uLinePos < 5 && !(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) {
795 // Same logic for all pages - set the ID
796 _apt_id = _apt_id.substr(0, _uLinePos);
797 // ASSERT(_uLinePos > 0);
798 if(_uLinePos == (_apt_id.size() + 1)) {
801 _apt_id[_uLinePos - 1] = _kln89->DecChar(_apt_id[_uLinePos - 1], (_uLinePos == 1 ? false : true));
805 // TODO - set by name
807 // NO-OP - to/fr is cycled by clr button
813 void KLN89AptPage::Knob2Right1() {
814 if(_kln89->_mode != KLN89_MODE_CRSR || _uLinePos == 0) {
815 if(_uLinePos == 0 && _kln89->_mode == KLN89_MODE_CRSR && _kln89->_obsMode) {
816 KLN89Page::Knob2Right1();
817 } else if(_subPage == 2) {
820 } else if(_subPage == 3) {
821 if(_curRwyPage == _nRwyPages - 1) {
827 } else if(_subPage == 4) {
828 if(_curFreqPage == _nFreqPages - 1) {
834 KLN89Page::Knob2Right1();
837 if(_uLinePos < 5 && !(_subPage == 7 && (_iafDialog || _addDialog || _replaceDialog))) {
838 // Same logic for all pages - set the ID
839 _apt_id = _apt_id.substr(0, _uLinePos);
840 // ASSERT(_uLinePos > 0);
841 if(_uLinePos == (_apt_id.size() + 1)) {
844 _apt_id[_uLinePos - 1] = _kln89->IncChar(_apt_id[_uLinePos - 1], (_uLinePos == 1 ? false : true));
848 // TODO - set by name
850 // NO-OP - to/fr is cycled by clr button