mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-25 10:00:21 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			183 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| // Copyright (C) 2023 Jon Beniston, M7RCE <jon@beniston.com>                     //
 | |
| //                                                                               //
 | |
| // This program is free software; you can redistribute it and/or modify          //
 | |
| // it under the terms of the GNU General Public License as published by          //
 | |
| // the Free Software Foundation as version 3 of the License, or                  //
 | |
| // (at your option) any later version.                                           //
 | |
| //                                                                               //
 | |
| // This program is distributed in the hope that it will be useful,               //
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of                //
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
 | |
| // GNU General Public License V3 for more details.                               //
 | |
| //                                                                               //
 | |
| // You should have received a copy of the GNU General Public License             //
 | |
| // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | |
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| #include <cmath>
 | |
| 
 | |
| #include <QPainter>
 | |
| 
 | |
| #include "coursedeviationindicator.h"
 | |
| 
 | |
| CourseDeviationIndicator::CourseDeviationIndicator(QWidget *parent) :
 | |
|     QWidget(parent),
 | |
|     m_localizerDDM(0.0f),
 | |
|     m_glideSlopeDDM(0.0f)
 | |
| {
 | |
| }
 | |
| 
 | |
| void CourseDeviationIndicator::setMode(Mode mode)
 | |
| {
 | |
|     m_mode = mode;
 | |
|     update();
 | |
| }
 | |
| 
 | |
| void CourseDeviationIndicator::setLocalizerDDM(float ddm)
 | |
| {
 | |
|     m_localizerDDM = ddm;
 | |
|     update();
 | |
| }
 | |
| 
 | |
| void CourseDeviationIndicator::setGlideSlopeDDM(float ddm)
 | |
| {
 | |
|     m_glideSlopeDDM = ddm;
 | |
|     update();
 | |
| }
 | |
| 
 | |
| void CourseDeviationIndicator::paintEvent(QPaintEvent *event)
 | |
| {
 | |
|     (void) event;
 | |
| 
 | |
|     QPainter painter(this);
 | |
| 
 | |
|     QRect r = rect();
 | |
|     int midW = r.width() / 2;
 | |
|     int midH = r.height() / 2;
 | |
|     int spacing;
 | |
| 
 | |
|     // A320 like CDI
 | |
| 
 | |
|     // Black background
 | |
|     int bgw, bgh;
 | |
|     if (m_mode == LOC)
 | |
|     {
 | |
|         bgw = r.width();
 | |
|         bgh = 20;
 | |
|         painter.fillRect(0, midH - bgh, bgw, bgh*2, QColor(0, 0, 0));
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         bgw = 20;
 | |
|         bgh = r.height();
 | |
|         painter.fillRect(midW - bgw, 0, bgw*2, bgh, QColor(0, 0, 0));
 | |
|     }
 | |
| 
 | |
|     const int dots = 5;
 | |
| 
 | |
|     // Circles
 | |
|     painter.setPen(QColor(255, 255, 255));
 | |
|     const int radius = 4;
 | |
|     int x, y;
 | |
| 
 | |
|     if (m_mode == LOC)
 | |
|     {
 | |
|         spacing = r.width() / 5;
 | |
|         x = spacing / 2;
 | |
|         y = midH;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         spacing = r.height() / 5;
 | |
|         x = midW;
 | |
|         y = spacing / 2;
 | |
|     }
 | |
|     for (int i = 0; i < dots; i++)
 | |
|     {
 | |
|         if (i != 2) {
 | |
|             painter.drawEllipse(QPointF(x, y), radius, radius);
 | |
|         }
 | |
|         if (m_mode == LOC) {
 | |
|             x += spacing;
 | |
|         } else {
 | |
|             y += spacing;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Diamond (index) - only draw half of symbol if out of range
 | |
|     // Shouldn't draw the symbol if signal not vaiid
 | |
|     // Typically, LOC full scale deflection 0.155 DDM (Which is ~2.5deg, so 1 deg per dot, but can be 3 degrees. A320 is 0.8 deg per dot)
 | |
|     // For GS, full deflection is 0.0875 DDM (0.7deg, so 0.14 deg per dot)
 | |
|     painter.setPen(QColor(255, 150, 250));
 | |
|     float dev;
 | |
|     if (m_mode == LOC) {
 | |
|         dev = m_localizerDDM / 0.155;
 | |
|     } else {
 | |
|         dev = m_glideSlopeDDM / 0.0875;
 | |
|     }
 | |
|     dev = std::min(dev, 1.0f);
 | |
|     dev = std::max(dev, -1.0f);
 | |
|     if (m_mode == LOC)
 | |
|     {
 | |
|         x = midW + dev * r.width() / 2;  // Positive DDM means we're to left of course line
 | |
|         y = midH;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         x = midW;
 | |
|         y = midH + dev * r.height() / 2; // Positive DDM means we're above glide path
 | |
|     }
 | |
|     int dw = 10;
 | |
|     int dh = 8;
 | |
|     painter.drawLine(x, y + dh, x - dw, y);
 | |
|     painter.drawLine(x - dw, y, x, y - dh);
 | |
|     painter.drawLine(x + dw, y, x, y - dh);
 | |
|     painter.drawLine(x, y + dh, x + dw, y);
 | |
| 
 | |
|     // Centre line
 | |
|     painter.setPen(QColor(255, 255, 70));
 | |
|     if (m_mode == LOC)
 | |
|     {
 | |
|         int lh = 14;
 | |
|         painter.drawLine(midW, midH + lh, midW, midH - lh);
 | |
|         painter.drawLine(midW-1, midH + lh, midW-1, midH - lh);
 | |
|         painter.drawLine(midW+1, midH + lh, midW+1, midH - lh);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         int lw = 14;
 | |
|         painter.drawLine(midW + lw, midH, midW - lw, midH);
 | |
|         painter.drawLine(midW + lw, midH - 1, midW - lw, midH - 1);
 | |
|         painter.drawLine(midW + lw, midH + 1, midW - lw, midH + 1);
 | |
|     }
 | |
| 
 | |
|     if (m_mode == LOC)
 | |
|     {
 | |
|         // Indicate localizer capture
 | |
|         if (std::abs(m_localizerDDM) < 0.175) // See 3.1.3.7.4
 | |
|         {
 | |
|             QFontMetrics fm(painter.font());
 | |
|             QString text = "LOC";
 | |
|             int tw = fm.horizontalAdvance(text);
 | |
|             int th = fm.descent();
 | |
|             painter.setPen(QColor(0, 255, 0));
 | |
|             painter.drawText(midW - tw/2, midH - bgh - th, text);
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // Indicate glideslope capture
 | |
|         if (std::abs(m_glideSlopeDDM) < 0.175) // Can't see a spec for this
 | |
|         {
 | |
|             QFontMetrics fm(painter.font());
 | |
|             QString text = "G/S";
 | |
|             int th = fm.ascent() / 2;
 | |
|             painter.setPen(QColor(0, 255, 0));
 | |
|             painter.drawText(midW + bgw + 2, midH + th, text);
 | |
|         }
 | |
|     }
 | |
| 
 | |
| }
 | |
| 
 |