1
0
mirror of https://github.com/f4exb/sdrangel.git synced 2025-10-26 02:20:26 -04:00
sdrangel/sdrgui/gui/glscope.h

267 lines
9.5 KiB
C
Raw Normal View History

2017-01-29 19:51:45 +01:00
///////////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2017 F4EXB //
// written by Edouard Griffiths //
// //
// 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. //
2017-01-29 19:51:45 +01:00
// //
// 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/>. //
///////////////////////////////////////////////////////////////////////////////////
#ifndef SDRBASE_GUI_GLSCOPENG_H_
#define SDRBASE_GUI_GLSCOPENG_H_
#include <QGLWidget>
#include <QPen>
#include <QTimer>
#include <QMutex>
#include <QFont>
#include <QMatrix4x4>
#include <QAtomicInt>
2017-01-29 19:51:45 +01:00
#include "dsp/dsptypes.h"
2018-08-12 17:22:39 +02:00
#include "dsp/scopevis.h"
2017-01-29 19:51:45 +01:00
#include "gui/scaleengine.h"
#include "gui/glshadercolors.h"
2017-01-29 19:51:45 +01:00
#include "gui/glshadersimple.h"
#include "gui/glshadertextured.h"
#include "export.h"
2017-01-29 19:51:45 +01:00
#include "util/bitfieldindex.h"
2018-03-01 02:33:18 +01:00
#include "util/incrementalarray.h"
2017-01-29 19:51:45 +01:00
class QPainter;
2018-08-12 17:01:56 +02:00
class SDRGUI_API GLScope: public QGLWidget {
2017-01-29 19:51:45 +01:00
Q_OBJECT
public:
enum DisplayMode {
DisplayXYH,
DisplayXYV,
DisplayX,
DisplayY,
DisplayPol
};
2018-08-12 17:01:56 +02:00
GLScope(QWidget* parent = 0);
virtual ~GLScope();
2017-01-29 19:51:45 +01:00
void connectTimer(const QTimer& timer);
void disconnectTimer();
2017-01-29 19:51:45 +01:00
2018-08-12 17:18:58 +02:00
void setTraces(std::vector<ScopeVis::TraceData>* tracesData, std::vector<float *>* traces);
void newTraces(std::vector<float *>* traces, int traceIndex, std::vector<Projector::ProjectionType>* projectionTypes);
2017-01-29 19:51:45 +01:00
2017-01-29 22:52:38 +01:00
int getSampleRate() const { return m_sampleRate; }
2017-02-05 04:41:32 +01:00
int getTraceSize() const { return m_traceSize; }
2017-01-29 22:52:38 +01:00
void setTriggerPre(uint32_t triggerPre, bool emitSignal = false); //!< number of samples
2017-01-29 19:51:45 +01:00
void setTimeOfsProMill(int timeOfsProMill);
void setSampleRate(int sampleRate);
void setTimeBase(int timeBase);
void setFocusedTraceIndex(uint32_t traceIndex);
2017-01-29 19:51:45 +01:00
void setDisplayMode(DisplayMode displayMode);
void setTraceSize(int trceSize, bool emitSignal = false);
2017-02-05 13:26:07 +01:00
void updateDisplay();
2017-02-05 04:41:32 +01:00
void setDisplayGridIntensity(int intensity);
void setDisplayTraceIntensity(int intensity);
2018-08-12 17:18:58 +02:00
void setFocusedTriggerData(ScopeVis::TriggerData& triggerData) { m_focusedTriggerData = triggerData; }
void setConfigChanged() { m_configChanged = true; }
//void incrementTraceCounter() { m_traceCounter++; }
bool getDataChanged() const { return m_dataChanged; }
DisplayMode getDisplayMode() const { return m_displayMode; }
void setDisplayXYPoints(bool value) { m_displayXYPoints = value; }
void setDisplayXYPolarGrid(bool value) { m_displayPolGrid = value; }
const QAtomicInt& getProcessingTraceIndex() const { return m_processingTraceIndex; }
void setTraceModulo(int modulo) { m_traceModulo = modulo; }
2017-01-29 19:51:45 +01:00
2017-01-29 22:52:38 +01:00
signals:
void sampleRateChanged(int);
void traceSizeChanged(uint32_t);
void preTriggerChanged(uint32_t); //!< number of samples
2017-01-29 22:52:38 +01:00
2017-01-29 19:51:45 +01:00
private:
struct ScopeMarker {
QPointF m_point;
float m_time;
float m_value;
QString m_timeStr;
QString m_valueStr;
QString m_timeDeltaStr;
QString m_valueDeltaStr;
ScopeMarker() :
m_point(0, 0),
m_time(0),
m_value(0),
m_timeStr(),
m_valueStr(),
m_timeDeltaStr(),
m_valueDeltaStr()
{}
ScopeMarker(
const QPointF& point,
float time,
float value,
const QString timeStr,
const QString& valueStr,
const QString& timeDeltaStr,
const QString& valueDeltaStr
) :
m_point(point),
m_time(time),
m_value(value),
m_timeStr(timeStr),
m_valueStr(valueStr),
m_timeDeltaStr(timeDeltaStr),
m_valueDeltaStr(valueDeltaStr)
{}
ScopeMarker(const ScopeMarker& other) :
m_point(other.m_point),
m_time(other.m_time),
m_timeStr(other.m_timeStr),
m_valueStr(other.m_valueStr),
m_timeDeltaStr(other.m_timeDeltaStr),
m_valueDeltaStr(other.m_valueDeltaStr)
{}
};
QList<ScopeMarker> m_markers1;
QList<ScopeMarker> m_markers2;
2018-08-12 17:18:58 +02:00
std::vector<ScopeVis::TraceData> *m_tracesData;
2017-02-06 02:40:31 +01:00
std::vector<float *> *m_traces;
std::vector<Projector::ProjectionType> *m_projectionTypes;
QAtomicInt m_processingTraceIndex;
2018-08-12 17:18:58 +02:00
ScopeVis::TriggerData m_focusedTriggerData;
//int m_traceCounter;
2017-02-06 02:40:31 +01:00
uint32_t m_bufferIndex;
2017-01-29 19:51:45 +01:00
DisplayMode m_displayMode;
bool m_displayPolGrid;
2017-01-29 19:51:45 +01:00
QTimer m_timer;
const QTimer *m_masterTimer;
2017-01-29 19:51:45 +01:00
QMutex m_mutex;
QAtomicInt m_dataChanged;
2017-01-29 19:51:45 +01:00
bool m_configChanged;
int m_sampleRate;
int m_timeOfsProMill;
2017-02-07 18:50:08 +01:00
uint32_t m_triggerPre;
2017-01-29 19:51:45 +01:00
int m_traceSize;
int m_traceModulo; //!< ineffective if <2
2017-01-29 19:51:45 +01:00
int m_timeBase;
2017-02-05 04:41:32 +01:00
int m_timeOffset;
uint32_t m_focusedTraceIndex;
2017-01-29 19:51:45 +01:00
// graphics stuff
QRectF m_glScopeRect1;
QRectF m_glScopeRect2;
QMatrix4x4 m_glScopeMatrix1;
QMatrix4x4 m_glScopeMatrix2;
QMatrix4x4 m_glLeft1ScaleMatrix;
QMatrix4x4 m_glRight1ScaleMatrix;
QMatrix4x4 m_glLeft2ScaleMatrix;
QMatrix4x4 m_glBot1ScaleMatrix;
QMatrix4x4 m_glBot2ScaleMatrix;
QPixmap m_left1ScalePixmap;
QPixmap m_left2ScalePixmap;
QPixmap m_bot1ScalePixmap;
QPixmap m_bot2ScalePixmap;
QPixmap m_channelOverlayPixmap1;
QPixmap m_channelOverlayPixmap2;
2017-01-29 19:51:45 +01:00
int m_displayGridIntensity;
int m_displayTraceIntensity;
bool m_displayXYPoints;
2017-01-29 19:51:45 +01:00
ScaleEngine m_x1Scale; //!< Display #1 X scale. Time scale
ScaleEngine m_x2Scale; //!< Display #2 X scale. Time scale
ScaleEngine m_y1Scale; //!< Display #1 Y scale. Always connected to trace #0 (X trace)
ScaleEngine m_y2Scale; //!< Display #2 Y scale. Connected to highlighted Y trace (#1..n)
QFont m_channelOverlayFont;
QFont m_textOverlayFont;
2017-01-29 19:51:45 +01:00
GLShaderSimple m_glShaderSimple;
GLShaderColors m_glShaderColors;
2017-01-29 19:51:45 +01:00
GLShaderTextured m_glShaderLeft1Scale;
GLShaderTextured m_glShaderBottom1Scale;
GLShaderTextured m_glShaderLeft2Scale;
GLShaderTextured m_glShaderBottom2Scale;
GLShaderTextured m_glShaderPowerOverlay;
GLShaderTextured m_glShaderTextOverlay;
2017-01-29 19:51:45 +01:00
2018-03-01 02:33:18 +01:00
IncrementalArray<GLfloat> m_q3Polar;
IncrementalArray<GLfloat> m_q3TickY1;
IncrementalArray<GLfloat> m_q3TickY2;
IncrementalArray<GLfloat> m_q3TickX1;
IncrementalArray<GLfloat> m_q3TickX2;
2019-10-14 18:50:40 +02:00
IncrementalArray<GLfloat> m_q3Radii; //!< Polar grid radii
IncrementalArray<GLfloat> m_q3Circle; //!< Polar grid unit circle
IncrementalArray<GLfloat> m_q3Colors; //!< Colors for trace rainbow palette
2018-03-01 02:33:18 +01:00
2017-01-29 19:51:45 +01:00
static const int m_topMargin = 5;
static const int m_botMargin = 20;
static const int m_leftMargin = 35;
static const int m_rightMargin = 5;
2019-10-14 18:50:40 +02:00
static const GLfloat m_q3RadiiConst[];
2017-01-29 19:51:45 +01:00
void initializeGL();
void resizeGL(int width, int height);
void paintGL();
void drawMarkers();
2017-01-29 19:51:45 +01:00
void applyConfig();
void setYScale(ScaleEngine& scale, uint32_t highlightedTraceIndex);
void setUniqueDisplays(); //!< Arrange displays when X and Y are unique on screen
void setVerticalDisplays(); //!< Arrange displays when X and Y are stacked vertically
void setHorizontalDisplays(); //!< Arrange displays when X and Y are stacked horizontally
void setPolarDisplays(); //!< Arrange displays when X and Y are stacked over on the left and polar display is on the right
2017-01-29 19:51:45 +01:00
void mousePressEvent(QMouseEvent* event);
void drawChannelOverlay( //!< Draws a text overlay
const QString& text,
const QColor& color,
QPixmap& channelOverlayPixmap,
const QRectF& glScopeRect);
void drawTextOverlay( //!< Draws a text overlay
const QString& text,
const QColor& color,
const QFont& font,
float shiftX,
float shiftY,
bool leftHalf,
bool topHalf,
const QRectF& glRect);
static bool isPositiveProjection(Projector::ProjectionType& projectionType)
{
return (projectionType == Projector::ProjectionMagLin)
|| (projectionType == Projector::ProjectionMagDB)
|| (projectionType == Projector::ProjectionMagSq);
}
2019-10-14 18:50:40 +02:00
void drawRectGrid2();
void drawPolarGrid2();
QString displayScaled(float value, char type, int precision);
2019-10-14 18:50:40 +02:00
static void drawCircle(float cx, float cy, float r, int num_segments, bool dotted, GLfloat *vertices);
static void setColorPalette(int nbVertices, int modulo, GLfloat *colors);
2019-10-14 18:50:40 +02:00
2017-01-29 19:51:45 +01:00
protected slots:
void cleanup();
void tick();
};
#endif /* SDRBASE_GUI_GLSCOPENG_H_ */