1
+ #include < imgui.h>
2
+ #include < config.h>
3
+ #include < core.h>
4
+ #include < gui/style.h>
5
+ #include < gui/gui.h>
6
+ #include < signal_path/signal_path.h>
7
+ #include < module.h>
8
+ #include < filesystem>
9
+ #include < dsp/buffer/reshaper.h>
10
+ #include < dsp/sink/handler_sink.h>
11
+ #include < gui/widgets/constellation_diagram.h>
12
+ #include " vor_decoder.h"
13
+ #include < fstream>
14
+
15
+ #define CONCAT (a, b ) ((std::string(a) + b).c_str())
16
+
17
+ SDRPP_MOD_INFO{
18
+ /* Name: */ " vor_receiver" ,
19
+ /* Description: */ " VOR Receiver for SDR++" ,
20
+ /* Author: */ " Ryzerth" ,
21
+ /* Version: */ 0 , 1 , 0 ,
22
+ /* Max instances */ -1
23
+ };
24
+
25
+ ConfigManager config;
26
+
27
+ #define INPUT_SAMPLE_RATE VOR_IN_SR
28
+
29
+ class VORReceiverModule : public ModuleManager ::Instance {
30
+ public:
31
+ VORReceiverModule (std::string name) {
32
+ this ->name = name;
33
+
34
+ // Load config
35
+ config.acquire ();
36
+ // TODO: Load config
37
+ config.release ();
38
+
39
+ vfo = sigpath::vfoManager.createVFO (name, ImGui::WaterfallVFO::REF_CENTER, 0 , INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, true );
40
+ decoder = new vor::Decoder (vfo->output , 1 );
41
+ decoder->onBearing .bind (&VORReceiverModule::onBearing, this );
42
+
43
+ decoder->start ();
44
+
45
+ gui::menu.registerEntry (name, menuHandler, this , this );
46
+ }
47
+
48
+ ~VORReceiverModule () {
49
+ decoder->stop ();
50
+ sigpath::vfoManager.deleteVFO (vfo);
51
+ gui::menu.removeEntry (name);
52
+ delete decoder;
53
+ }
54
+
55
+ void postInit () {}
56
+
57
+ void enable () {
58
+ double bw = gui::waterfall.getBandwidth ();
59
+ vfo = sigpath::vfoManager.createVFO (name, ImGui::WaterfallVFO::REF_CENTER, 0 , INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, INPUT_SAMPLE_RATE, true );
60
+
61
+ decoder->setInput (vfo->output );
62
+
63
+ decoder->start ();
64
+
65
+ enabled = true ;
66
+ }
67
+
68
+ void disable () {
69
+ decoder->stop ();
70
+
71
+ sigpath::vfoManager.deleteVFO (vfo);
72
+ enabled = false ;
73
+ }
74
+
75
+ bool isEnabled () {
76
+ return enabled;
77
+ }
78
+
79
+ private:
80
+ static void menuHandler (void * ctx) {
81
+ VORReceiverModule* _this = (VORReceiverModule*)ctx;
82
+
83
+ float menuWidth = ImGui::GetContentRegionAvail ().x ;
84
+
85
+ if (!_this->enabled ) { style::beginDisabled (); }
86
+
87
+ ImGui::Text (" Bearing: %f°" , _this->bearing );
88
+ ImGui::Text (" Quality: %0.1f%%" , _this->quality );
89
+
90
+ if (!_this->enabled ) { style::endDisabled (); }
91
+ }
92
+
93
+ void onBearing (float nbearing, float nquality) {
94
+ bearing = (180 .0f * nbearing / FL_M_PI);
95
+ quality = nquality * 100 .0f ;
96
+ }
97
+
98
+ std::string name;
99
+ bool enabled = true ;
100
+
101
+ // DSP Chain
102
+ VFOManager::VFO* vfo;
103
+ vor::Decoder* decoder;
104
+
105
+ float bearing = 0 .0f , quality = 0 .0f ;
106
+ };
107
+
108
+ MOD_EXPORT void _INIT_ () {
109
+ // Create default recording directory
110
+ std::string root = (std::string)core::args[" root" ];
111
+ json def = json ({});
112
+ config.setPath (root + " /vor_receiver_config.json" );
113
+ config.load (def);
114
+ config.enableAutoSave ();
115
+ }
116
+
117
+ MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_ (std::string name) {
118
+ return new VORReceiverModule (name);
119
+ }
120
+
121
+ MOD_EXPORT void _DELETE_INSTANCE_ (void * instance) {
122
+ delete (VORReceiverModule*)instance;
123
+ }
124
+
125
+ MOD_EXPORT void _END_ () {
126
+ config.disableAutoSave ();
127
+ config.save ();
128
+ }
0 commit comments