source: ntrip/trunk/BNC/bncwindow.cpp@ 1358

Last change on this file since 1358 was 1352, checked in by mervart, 15 years ago

* empty log message *

File size: 55.7 KB
Line 
1// Part of BNC, a utility for retrieving decoding and
2// converting GNSS data streams from NTRIP broadcasters.
3//
4// Copyright (C) 2007
5// German Federal Agency for Cartography and Geodesy (BKG)
6// http://www.bkg.bund.de
7// Czech Technical University Prague, Department of Geodesy
8// http://www.fsv.cvut.cz
9//
10// Email: euref-ip@bkg.bund.de
11//
12// This program is free software; you can redistribute it and/or
13// modify it under the terms of the GNU General Public License
14// as published by the Free Software Foundation, version 2.
15//
16// This program is distributed in the hope that it will be useful,
17// but WITHOUT ANY WARRANTY; without even the implied warranty of
18// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19// GNU General Public License for more details.
20//
21// You should have received a copy of the GNU General Public License
22// along with this program; if not, write to the Free Software
23// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25/* -------------------------------------------------------------------------
26 * BKG NTRIP Client
27 * -------------------------------------------------------------------------
28 *
29 * Class: bncWindow
30 *
31 * Purpose: This class implements the main application window.
32 *
33 * Author: L. Mervart
34 *
35 * Created: 24-Dec-2005
36 *
37 * Changes:
38 *
39 * -----------------------------------------------------------------------*/
40
41#include <iostream>
42
43#include <unistd.h>
44#include "bncwindow.h"
45#include "bncapp.h"
46#include "bncgetthread.h"
47#include "bnctabledlg.h"
48#include "bnchlpdlg.h"
49#include "bnchtml.h"
50#include "bnctableitem.h"
51
52using namespace std;
53
54// Constructor
55////////////////////////////////////////////////////////////////////////////
56bncWindow::bncWindow() {
57
58 _caster = 0;
59
60 int ww = QFontMetrics(this->font()).width('w');
61
62 static const QStringList labels = QString("account,broadcaster:port / mountpoint,decoder,lat,long,nmea,ntrip,bytes").split(",");
63
64 setMinimumSize(80*ww, 65*ww);
65
66 setWindowTitle(tr("BKG Ntrip Client (BNC) Version 1.7"));
67
68 connect((bncApp*)qApp, SIGNAL(newMessage(QByteArray,bool)),
69 this, SLOT(slotWindowMessage(QByteArray,bool)));
70
71 // Create Actions
72 // --------------
73 _actHelp = new QAction(tr("&Help Contents"),this);
74 connect(_actHelp, SIGNAL(triggered()), SLOT(slotHelp()));
75
76 _actAbout = new QAction(tr("&About BNC"),this);
77 connect(_actAbout, SIGNAL(triggered()), SLOT(slotAbout()));
78
79 _actFlowchart = new QAction(tr("&Flow Chart"),this);
80 connect(_actFlowchart, SIGNAL(triggered()), SLOT(slotFlowchart()));
81
82 _actFontSel = new QAction(tr("Select &Font"),this);
83 connect(_actFontSel, SIGNAL(triggered()), SLOT(slotFontSel()));
84
85 _actSaveOpt = new QAction(tr("&Save && Activate Options"),this);
86 connect(_actSaveOpt, SIGNAL(triggered()), SLOT(slotSaveOptions()));
87
88 _actQuit = new QAction(tr("&Quit"),this);
89 connect(_actQuit, SIGNAL(triggered()), SLOT(close()));
90
91 _actAddMountPoints = new QAction(tr("Add &Mountpoints"),this);
92 connect(_actAddMountPoints, SIGNAL(triggered()), SLOT(slotAddMountPoints()));
93
94 _actDeleteMountPoints = new QAction(tr("&Delete Mountpoints"),this);
95 connect(_actDeleteMountPoints, SIGNAL(triggered()), SLOT(slotDeleteMountPoints()));
96 _actDeleteMountPoints->setEnabled(false);
97
98 _actGetData = new QAction(tr("Sta&rt"),this);
99 connect(_actGetData, SIGNAL(triggered()), SLOT(slotGetData()));
100
101 _actStop = new QAction(tr("Sto&p"),this);
102 connect(_actStop, SIGNAL(triggered()), SLOT(slotStop()));
103 _actStop->setEnabled(false);
104
105 _actwhatsthis= new QAction(tr("Help=Shift+F1"),this);
106 connect(_actwhatsthis, SIGNAL(triggered()), SLOT(slotWhatsThis()));
107
108 CreateMenu();
109 AddToolbar();
110
111 QSettings settings;
112 _proxyHostLineEdit = new QLineEdit(settings.value("proxyHost").toString());
113 _proxyPortLineEdit = new QLineEdit(settings.value("proxyPort").toString());
114 _proxyPortLineEdit->setMaximumWidth(9*ww);
115 _miscMountLineEdit = new QLineEdit(settings.value("miscMount").toString());
116 _miscMountLineEdit->setMaximumWidth(20*ww);
117 _scanRTCMCheckBox = new QCheckBox();
118 _scanRTCMCheckBox->setCheckState(Qt::CheckState(
119 settings.value("scanRTCM").toInt()));
120 _waitTimeSpinBox = new QSpinBox();
121 _waitTimeSpinBox->setMinimum(1);
122 _waitTimeSpinBox->setMaximum(30);
123 _waitTimeSpinBox->setSingleStep(1);
124 _waitTimeSpinBox->setSuffix(" sec");
125 _waitTimeSpinBox->setMaximumWidth(9*ww);
126 _waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
127 _outFileLineEdit = new QLineEdit(settings.value("outFile").toString());
128 _outPortLineEdit = new QLineEdit(settings.value("outPort").toString());
129 _outPortLineEdit->setMaximumWidth(9*ww);
130 _outUPortLineEdit = new QLineEdit(settings.value("outUPort").toString());
131 _outUPortLineEdit->setMaximumWidth(9*ww);
132 _outEphPortLineEdit = new QLineEdit(settings.value("outEphPort").toString());
133 _outEphPortLineEdit->setMaximumWidth(9*ww);
134 _corrPortLineEdit = new QLineEdit(settings.value("corrPort").toString());
135 _corrPortLineEdit->setMaximumWidth(9*ww);
136 _rnxPathLineEdit = new QLineEdit(settings.value("rnxPath").toString());
137 _ephPathLineEdit = new QLineEdit(settings.value("ephPath").toString());
138 _corrPathLineEdit = new QLineEdit(settings.value("corrPath").toString());
139
140 _rnxV3CheckBox = new QCheckBox();
141 _rnxV3CheckBox->setCheckState(Qt::CheckState(settings.value("rnxV3").toInt()));
142 _ephV3CheckBox = new QCheckBox();
143 _ephV3CheckBox->setCheckState(Qt::CheckState(settings.value("ephV3").toInt()));
144 _rnxScrpLineEdit = new QLineEdit(settings.value("rnxScript").toString());
145 _rnxSkelLineEdit = new QLineEdit(settings.value("rnxSkel").toString());
146 _rnxSkelLineEdit->setMaximumWidth(5*ww);
147 _rnxAppendCheckBox = new QCheckBox();
148 _rnxAppendCheckBox->setCheckState(Qt::CheckState(
149 settings.value("rnxAppend").toInt()));
150 _autoStartCheckBox = new QCheckBox();
151 _autoStartCheckBox->setCheckState(Qt::CheckState(
152 settings.value("autoStart").toInt()));
153 _rnxIntrComboBox = new QComboBox();
154 _rnxIntrComboBox->setMaximumWidth(9*ww);
155 _rnxIntrComboBox->setEditable(false);
156 _rnxIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
157 int ii = _rnxIntrComboBox->findText(settings.value("rnxIntr").toString());
158 if (ii != -1) {
159 _rnxIntrComboBox->setCurrentIndex(ii);
160 }
161 _onTheFlyComboBox = new QComboBox();
162 _onTheFlyComboBox->setMaximumWidth(9*ww);
163 _onTheFlyComboBox->setEditable(false);
164 _onTheFlyComboBox->addItems(QString("1 day,1 hour,1 min").split(","));
165 ii = _onTheFlyComboBox->findText(settings.value("onTheFlyInterval").toString());
166 if (ii != -1) {
167 _onTheFlyComboBox->setCurrentIndex(ii);
168 }
169 _ephIntrComboBox = new QComboBox();
170 _ephIntrComboBox->setMaximumWidth(9*ww);
171 _ephIntrComboBox->setEditable(false);
172 _ephIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
173 int jj = _ephIntrComboBox->findText(settings.value("ephIntr").toString());
174 if (jj != -1) {
175 _ephIntrComboBox->setCurrentIndex(jj);
176 }
177 _corrIntrComboBox = new QComboBox();
178 _corrIntrComboBox->setMaximumWidth(9*ww);
179 _corrIntrComboBox->setEditable(false);
180 _corrIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
181 int mm = _corrIntrComboBox->findText(settings.value("corrIntr").toString());
182 if (mm != -1) {
183 _corrIntrComboBox->setCurrentIndex(mm);
184 }
185 _corrTimeSpinBox = new QSpinBox();
186 _corrTimeSpinBox->setMinimum(1);
187 _corrTimeSpinBox->setMaximum(30);
188 _corrTimeSpinBox->setSingleStep(1);
189 _corrTimeSpinBox->setSuffix(" sec");
190 _corrTimeSpinBox->setMaximumWidth(9*ww);
191 _corrTimeSpinBox->setValue(settings.value("corrTime").toInt());
192 _rnxSamplSpinBox = new QSpinBox();
193 _rnxSamplSpinBox->setMinimum(0);
194 _rnxSamplSpinBox->setMaximum(60);
195 _rnxSamplSpinBox->setSingleStep(5);
196 _rnxSamplSpinBox->setMaximumWidth(9*ww);
197 _rnxSamplSpinBox->setValue(settings.value("rnxSampl").toInt());
198 _rnxSamplSpinBox->setSuffix(" sec");
199
200 _binSamplSpinBox = new QSpinBox();
201 _binSamplSpinBox->setMinimum(0);
202 _binSamplSpinBox->setMaximum(60);
203 _binSamplSpinBox->setSingleStep(5);
204 _binSamplSpinBox->setMaximumWidth(9*ww);
205 _binSamplSpinBox->setValue(settings.value("binSampl").toInt());
206 _binSamplSpinBox->setSuffix(" sec");
207
208 _obsRateComboBox = new QComboBox();
209 _obsRateComboBox->setMaximumWidth(9*ww);
210 _obsRateComboBox->setEditable(false);
211 _obsRateComboBox->addItems(QString(",0.1 Hz,0.2 Hz,0.5 Hz,1 Hz,5 Hz").split(","));
212 int kk = _obsRateComboBox->findText(settings.value("obsRate").toString());
213 if (kk != -1) {
214 _obsRateComboBox->setCurrentIndex(kk);
215 }
216 _makePauseCheckBox = new QCheckBox();
217 _makePauseCheckBox->setCheckState(Qt::CheckState(
218 settings.value("makePause").toInt()));
219 _adviseRecoSpinBox = new QSpinBox();
220 _adviseRecoSpinBox->setMinimum(0);
221 _adviseRecoSpinBox->setMaximum(60);
222 _adviseRecoSpinBox->setSingleStep(1);
223 _adviseRecoSpinBox->setSuffix(" min");
224 _adviseRecoSpinBox->setMaximumWidth(9*ww);
225 _adviseRecoSpinBox->setValue(settings.value("adviseReco").toInt());
226 _adviseFailSpinBox = new QSpinBox();
227 _adviseFailSpinBox->setMinimum(0);
228 _adviseFailSpinBox->setMaximum(60);
229 _adviseFailSpinBox->setSingleStep(1);
230 _adviseFailSpinBox->setSuffix(" min");
231 _adviseFailSpinBox->setMaximumWidth(9*ww);
232 _adviseFailSpinBox->setValue(settings.value("adviseFail").toInt());
233 _logFileLineEdit = new QLineEdit(settings.value("logFile").toString());
234 _adviseScriptLineEdit = new QLineEdit(settings.value("adviseScript").toString());
235
236 _serialPortNameLineEdit = new QLineEdit(settings.value("serialPortName").toString());
237 _serialMountPointLineEdit = new QLineEdit(settings.value("serialMountPoint").toString());
238
239 _serialBaudRateComboBox = new QComboBox();
240 _serialBaudRateComboBox->setMaximumWidth(9*ww);
241 _serialBaudRateComboBox->addItems(QString("110,300,600,"
242 "1200,2400,4800,9600,19200,38400,57600,115200").split(","));
243 kk = _serialBaudRateComboBox->findText(settings.value("serialBaudRate").toString());
244 if (kk != -1) {
245 _serialBaudRateComboBox->setCurrentIndex(kk);
246 }
247 _serialParityComboBox = new QComboBox();
248 _serialParityComboBox->setMaximumWidth(9*ww);
249 _serialParityComboBox->addItems(QString("NONE,ODD,EVEN,SPACE").split(","));
250 kk = _serialParityComboBox->findText(settings.value("serialParity").toString());
251 if (kk != -1) {
252 _serialParityComboBox->setCurrentIndex(kk);
253 }
254 _serialDataBitsComboBox = new QComboBox();
255 _serialDataBitsComboBox->setMaximumWidth(5*ww);
256 _serialDataBitsComboBox->addItems(QString("5,6,7,8").split(","));
257 kk = _serialDataBitsComboBox->findText(settings.value("serialDataBits").toString());
258 if (kk != -1) {
259 _serialDataBitsComboBox->setCurrentIndex(kk);
260 }
261 _serialStopBitsComboBox = new QComboBox();
262 _serialStopBitsComboBox->setMaximumWidth(5*ww);
263 _serialStopBitsComboBox->addItems(QString("1,2").split(","));
264 kk = _serialStopBitsComboBox->findText(settings.value("serialStopBits").toString());
265 if (kk != -1) {
266 _serialStopBitsComboBox->setCurrentIndex(kk);
267 }
268 _serialAutoNMEACheckBox = new QCheckBox();
269 _serialAutoNMEACheckBox->setCheckState(Qt::CheckState(
270 settings.value("serialAutoNMEA").toInt()));
271
272 _perfIntrComboBox = new QComboBox();
273 _perfIntrComboBox->setMaximumWidth(9*ww);
274 _perfIntrComboBox->setEditable(false);
275 _perfIntrComboBox->addItems(QString(",1 min,5 min,15 min,1 hour,6 hours,1 day").split(","));
276 int ll = _perfIntrComboBox->findText(settings.value("perfIntr").toString());
277 if (ll != -1) {
278 _perfIntrComboBox->setCurrentIndex(ll);
279 }
280
281 _mountPointsTable = new QTableWidget(0,8);
282
283 _mountPointsTable->horizontalHeader()->resizeSection(1,34*ww);
284 _mountPointsTable->horizontalHeader()->resizeSection(2,9*ww);
285 _mountPointsTable->horizontalHeader()->resizeSection(3,7*ww);
286 _mountPointsTable->horizontalHeader()->resizeSection(4,7*ww);
287 _mountPointsTable->horizontalHeader()->resizeSection(5,5*ww);
288 _mountPointsTable->horizontalHeader()->resizeSection(6,4*ww);
289 _mountPointsTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
290 _mountPointsTable->horizontalHeader()->setStretchLastSection(true);
291 _mountPointsTable->setHorizontalHeaderLabels(labels);
292 _mountPointsTable->setGridStyle(Qt::NoPen);
293 _mountPointsTable->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
294 _mountPointsTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
295 _mountPointsTable->setSelectionBehavior(QAbstractItemView::SelectRows);
296 _mountPointsTable->hideColumn(0);
297 connect(_mountPointsTable, SIGNAL(itemSelectionChanged()),
298 SLOT(slotSelectionChanged()));
299 populateMountPointsTable();
300
301 _log = new QTextBrowser();
302 _log->setReadOnly(true);
303
304 // WhatsThis
305 // ---------
306 _proxyHostLineEdit->setWhatsThis(tr("<p>If you are running BNC within a protected Local Area Network (LAN), you might need to use a proxy server to access the Internet. Enter your proxy server IP and port number in case one is operated in front of BNC. If you do not know the IP and port of your proxy server, check the proxy server settings in your Internet browser or ask your network administrator.</p><p>Note that IP streaming is sometimes not allowed in a LAN. In this case you need to ask your network administrator for an appropriate modification of the local security policy or for the installation of a TCP relay to the NTRIP broadcasters. If these are not possible, you might need to run BNC outside your LAN on a network that has unobstructed connection to the Internet.</p>"));
307 _proxyPortLineEdit->setWhatsThis(tr("<p>If you are running BNC within a protected Local Area Network (LAN), you might need to use a proxy server to access the Internet. Enter your proxy server IP and port number in case one is operated in front of BNC. If you do not know the IP and port of your proxy server, check the proxy server settings in your Internet browser or ask your network administrator.</p><p>Note that IP streaming is sometimes not allowed in a LAN. In this case you need to ask your network administrator for an appropriate modification of the local security policy or for the installation of a TCP relay to the NTRIP broadcasters. If these are not possible, you might need to run BNC outside your LAN on a network that has unobstructed connection to the Internet.</p>"));
308 _waitTimeSpinBox->setWhatsThis(tr("<p>When feeding a real-time GNSS engine waiting for synchronized input epoch by epoch, BNC drops whatever is received later than 'Wait for full epoch' seconds. A value of 3 to 5 seconds is recommended, depending on the latency of the incoming streams and the delay acceptable to your real-time GNSS engine or products.</p><p>Note that 'Wait for full epoch' does not effect the RINEX Observation file content. Observations received later than 'Wait for full epoch' seconds will still be included in the RINEX Observation files.</p>"));
309 _outFileLineEdit->setWhatsThis(tr("Specify the full path to a file where synchronized observations are saved in plain ASCII format. Beware that the size of this file can rapidly increase depending on the number of incoming streams."));
310 _outPortLineEdit->setWhatsThis(tr("BNC can produce synchronized observations in binary format on your local host through an IP port. Specify a port number here to activate this function."));
311 _outUPortLineEdit->setWhatsThis(tr("BNC can produce unsynchronized observations in binary format on your local host through an IP port. Specify a port number here to activate this function."));
312 _outEphPortLineEdit->setWhatsThis(tr("BNC can produce ephemeris data in RINEX ASCII format on your local host through an IP port. Specify a port number here to activate this function."));
313 _corrPortLineEdit->setWhatsThis(tr("BNC can produce Broadcast Ephemeris Corrections on your local host through an IP port. Specify a port number here to activate this function."));
314 _corrTimeSpinBox->setWhatsThis(tr("Concerning output through IP port, BNC drops Broadcast Ephemeris Corrections received later than 'Wait for full epoch' seconds. A value of 2 to 5 seconds is recommended, depending on the latency of the incoming correction stream(s) and the delay acceptable to your real-time application."));
315 _rnxPathLineEdit->setWhatsThis(tr("Here you specify the path to where the RINEX Observation files will be stored. If the specified directory does not exist, BNC will not create RINEX Observation files."));
316 _ephPathLineEdit->setWhatsThis(tr("Specify the path for saving Broadcast Ephemeris data as RINEX Navigation files. If the specified directory does not exist, BNC will not create RINEX Navigation files."));
317 _corrPathLineEdit->setWhatsThis(tr("Specify a directory for saving Broadcast Ephemeris Correction files. If the specified directory does not exist, BNC will not create the files."));
318 _rnxScrpLineEdit->setWhatsThis(tr("<p>Whenever a RINEX Observation file is saved, you might want to compress, copy or upload it immediately via FTP. BNC allows you to execute a script/batch file to carry out these operations. To do that specify the full path of the script/batch file here. BNC will pass the full RINEX Observation file path to the script as a command line parameter (%1 on Windows systems, $1 onUnix/Linux systems).</p><p>The triggering event for calling the script or batch file is the end of a RINEX Observation file 'Interval'. If that is overridden by a stream outage, the triggering event is the stream reconnection.</p>"));
319 _rnxSkelLineEdit->setWhatsThis(tr("<p>Whenever BNC starts generating RINEX Observation files (and then once every day at midnight), it first tries to retrieve information needed for RINEX headers from so-called public RINEX header skeleton files which are derived from sitelogs. However, sometimes public RINEX header skeleton files are not available, its contents is not up to date, or you need to put additional/optional records in the RINEX header.</p><p>For that BNC allows using personal skeleton files that contain the header records you would like to include. You can derive a personal RINEX header skeleton file from the information given in an up to date sitelog. A file in the RINEX 'Directory' with the RINEX 'Skeleton extension' is interpreted by BNC as a personal RINEX header skeleton file for the corresponding stream.</p>"));
320 _rnxAppendCheckBox->setWhatsThis(tr("<p>When BNC is started, new files are created by default and any existing files with the same name will be overwritten. However, users might want to append already existing files following a restart of BNC, a system crash or when BNC crashed. Tick 'Append files' to continue with existing files and keep what has been recorded so far.</p>"));
321 _autoStartCheckBox->setWhatsThis(tr("<p>Tick 'Auto start' for auto-start of BNC at startup time in window mode with preassigned processing options.</p>"));
322 _onTheFlyComboBox->setWhatsThis(tr("<p>When operating BNC online in 'no window' mode, some configuration parameters can be changed on-the-fly without interrupting the running process. For that BNC rereads parts of its configuration in pre-defined intervals.<p></p>Select '1 min', '1 hour', or '1 day' to force BNC to reread its configuration every full minute, hour, or day and let in between edited configuration options become effective on-the-fly without terminating uninvolved threads.</p><p>Note that when operating BNC in window mode, on-the-fly changeable configuration options become effective immediately through 'Save & Activate Options'.</p>"));
323 _rnxIntrComboBox->setWhatsThis(tr("<p>Select the length of the RINEX Observation file.</p>"));
324 _ephIntrComboBox->setWhatsThis(tr("<p>Select the length of the RINEX Navigation file.</p>"));
325 _corrIntrComboBox->setWhatsThis(tr("<p>Select the length of the Broadcast Ephemeris Correction files.</p>"));
326 _rnxSamplSpinBox->setWhatsThis(tr("<p>Select the RINEX Observation sampling interval in seconds. A value of zero '0' tells BNC to store all received epochs into RINEX.</p>"));
327 _binSamplSpinBox->setWhatsThis(tr("<p>Select the Observation sampling interval in seconds. A value of zero '0' tells BNC to send/store all received epochs.</p>"));
328 _obsRateComboBox->setWhatsThis(tr("<p>BNC can collect all returns (success or failure) coming from a decoder within a certain short time span to then decide whether a stream has an outage or its content is corrupted. The procedure needs a rough estimate of the expected 'Observation rate' of the incoming streams. When a continuous problem is detected, BNC can inform its operator about this event through an advisory note.</p><p>An empty option field (default) means that you don't want an explicit information from BNC about stream outages and incoming streams that can not be decoded and that the special procedure for handling of corrupted streams is bypassed.</p>"));
329 _adviseRecoSpinBox->setWhatsThis(tr("<p>Following a stream outage or a longer series of bad observations, an advisory note is generated when valid observations are received again throughout the 'Recovery threshold' time span. A value of about 5min (default) is recommended.</p><p>A value of zero '0' means that for any stream recovery, however short, BNC immediately generates an advisory note.</p><p>Note that for using this function you need to specify the 'Observation rate'.</p>"));
330 _adviseFailSpinBox->setWhatsThis(tr("<p>An advisory note is generated when no (or only corrupted) observations are seen throughout the 'Failure threshold' time span. A value of 15 min (default) is recommended.</p><p>A value of zero '0' means that for any stream failure, however short, BNC immediately generates an advisory note.</p><p>Note that for using this function you need to specify the 'Observation rate'.</p>"));
331 _makePauseCheckBox->setWhatsThis(tr("<p>In case of a continuously corrupted stream, the decoding process can be paused and decodings are then attempted again at decreasing rate till the stream hopefully recovers. Tick 'Pause' to activate this function.</p><p>Do not tick 'Pause' (default) in order to prevent BNC from making any decoding pause. Be aware that this may incur an unnecessary workload.</p><p>Note that this function is only effective if an 'Observation rate' is specified.</p>"));
332 _logFileLineEdit->setWhatsThis(tr("Records of BNC's activities are shown in the Log section on the bottom of this window. They can be saved into a file when a valid path is specified in the 'Logfile (full path)' field."));
333 _adviseScriptLineEdit->setWhatsThis(tr("<p>Specify the full path to a script or batch file to handle advisory notes generated in the event of corrupted streams or stream outages. The affected mountpoint and one of the comments 'Begin_Outage', 'End_Outage', 'Begin_Corrupted', or 'End_Corrupted' are passed on to the script as command line parameters.</p><p>The script can be configured to send an email to BNC's operator and/or to the affected stream provider. An empty option field (default) or invalid path means that you don't want to use this option.</p><p> Note that for using this function you need to specify the 'Observation rate'.</p>"));
334 _perfIntrComboBox->setWhatsThis(tr("<p>BNC can average latencies per stream over a certain period of GPS time. The resulting mean latencies are recorded in the Log file/section at the end of each 'Log latency' interval together with results of a statistical evaluation (approximate number of covered epochs, data gaps).</p><p>Select a 'Log latency' interval or select the empty option field if you do not want BNC to log latencies and statistical information.</p>"));
335 _mountPointsTable->setWhatsThis(tr("<p>Streams selected for retrieval are listed in the 'Mountpoints' section. Clicking on 'Add Mountpoints' button will open a window that allows the user to select data streams from an NTRIP broadcaster according to their mountpoints. To remove a stream from the 'Mountpoints' list, highlight it by clicking on it and hit the 'Delete Mountpoints' button. You can also remove multiple mountpoints by highlighting them using +Shift and +Ctrl.</p><p>BNC automatically allocates one of its internal decoders to a stream based on the stream's 'format' and 'format-details' as given in the sourcetable. However, there might be cases where you need to override the automatic selection due to incorrect sourcetable for example. BNC allows users to manually select the required decoder by editing the decoder string. Double click on the 'decoder' field, enter your preferred decoder and then hit Enter. The accepted decoder strings are 'RTCM_2.x', 'RTCM_3.x', and 'RTIGS'.</p><p>In case you need to log the raw data as is, BNC allows users to by-pass its decoders and and directly save the input in daily log files. To do this specify the decoder string as 'ZERO'.</p><p>BNC can also retrieve streams from virtual reference stations (VRS). To initiate these streams, an approximate rover position needs to be sent in NMEA GGA message to the NTRIP broadcaster. In return, a user-specific data stream is generated, typically by a Network-RTK software. This stream is customized to the exact latitude and longitude as shown in the 'lat' and 'long' columns under 'Mountpoints'. These VRS streams are indicated by a 'yes' in the 'nmea' column under 'Mountpoints' as well as in the sourcetable. The default 'lat' and 'long' values are taken from the sourcetable. However, in most cases you would probably want to change this according to your requirement. Double click on 'lat' and 'long' fields, enter the values you wish to send and then hit Enter. The format is in positive north latitude degrees (e.g. for northern hemisphere: 52.436, for southern hemisphere: -24.567) and eastern longitude degrees (e.g.: 358.872 or -1.128). Only mountpoints with a 'yes' in its 'nmea' column can be edited. The position should preferably be a point within the coverage of the network.</p>"));
336 _log->setWhatsThis(tr("Records of BNC's activities are shown in the Log section. The message log covers the communication status between BNC and the NTRIP broadcaster as well as any problems that occur in the communication link, stream availability, stream delay, stream conversion etc."));
337 _ephV3CheckBox->setWhatsThis(tr("The default format for RINEX Navigation files containing Broadcast Ephemeris is RINEX Version 2.11. Select 'Version 3' if you want to save the ephemeris in RINEX Version 3 format."));
338 _rnxV3CheckBox->setWhatsThis(tr("The default format for RINEX Observation files is RINEX Version 2.11. Select 'Version 3' if you want to save the observations in RINEX Version 3 format."));
339 _miscMountLineEdit->setWhatsThis(tr("<p>Specify a mountpoint to apply any of the options shown below. Enter 'ALL' if you want to apply these options to all configured streams.</p><p>An empty option field (default) means that you don't want BNC to apply any of these options.</p>"));
340 _scanRTCMCheckBox->setWhatsThis(tr("<p>Tick 'Scan RTCM' to log the numbers of incomming message types as well as contained antenna coordinates, antenna heigt, and antenna descriptor.</p><p>An empty option field (default) means that you don't want BNC to log such information.</p>"));
341 _serialMountPointLineEdit->setWhatsThis(tr("<p>Enter a 'Mountpoint' to forward the corresponding stream to a serial connected device.</p>"));
342 _serialPortNameLineEdit->setWhatsThis(tr("<p>Enter the serial 'Port name' selected for communication with your serial connected device. Valid port names are</p><pre>Windows: COM1, COM2<br>Linux: /dev/ttyS0, /dev/ttyS1<br>FreeBSD: /dev/ttyd0, /dev/ttyd1<br>Digital Unix: /dev/tty01, /dev/tty02<br>HP-UX: /dev/tty1p0, /dev/tty2p0<br>SGI/IRIX: /dev/ttyf1, /dev/ttyf2<br>SunOS/Solaris: /dev/ttya, /dev/ttyb</pre>"));
343 _serialBaudRateComboBox->setWhatsThis(tr("<p>Select a 'Baud rate' for the serial link.</p><p>Note that your selection must equal the baud rate configured to the serial connected device. Note further that using a high baud rate is recommended.</p>"));
344 _serialParityComboBox->setWhatsThis(tr("<p>Select the 'Parity' for the serial link.</p><p>Note that your selection must equal the parity selection configured to the serial connected device. Note further that parity is often set to 'NONE'.</p>"));
345 _serialDataBitsComboBox->setWhatsThis(tr("<p>Select the number of 'Data bits' for the serial link.</p><p>Note that your selection must equal the number of data bits configured to the serial connected device. Note further that often 8 data bits are used.</p>"));
346 _serialStopBitsComboBox->setWhatsThis(tr("<p>Select the number of 'Stop bits' for the serial link.</p><p>Note that your selection must equal the number of stop bits configured to the serial connected device. Note further that often 1 stop bit is used.</p>"));
347 _serialAutoNMEACheckBox->setWhatsThis(tr("<p>Tick 'Auto NMEA' to forward NMEA GGA messages coming from your serial connected device to the NTRIP brodacaster.</p><p>Note that this replaces the simulation of an initial NMEA GGA message based on latitude/longitude from the broadcaster's sourcetable.</p>"));
348
349 // Canvas with Editable Fields
350 // ---------------------------
351 _canvas = new QWidget;
352 setCentralWidget(_canvas);
353
354 QTabWidget* aogroup = new QTabWidget();
355 QWidget* pgroup = new QWidget();
356 QWidget* ggroup = new QWidget();
357 QWidget* sgroup = new QWidget();
358 QWidget* egroup = new QWidget();
359 QWidget* agroup = new QWidget();
360 QWidget* cgroup = new QWidget();
361 QWidget* ogroup = new QWidget();
362 QWidget* rgroup = new QWidget();
363 QWidget* sergroup = new QWidget();
364 aogroup->addTab(pgroup,tr("Proxy"));
365 aogroup->addTab(ggroup,tr("General"));
366 aogroup->addTab(ogroup,tr("RINEX Observations"));
367 aogroup->addTab(egroup,tr("RINEX Ephemeris"));
368 aogroup->addTab(cgroup,tr("Ephemeris Corrections"));
369 aogroup->addTab(sgroup,tr("Feed Engine"));
370 aogroup->addTab(sergroup,tr("Serial Link"));
371 aogroup->addTab(agroup,tr("Outages"));
372 aogroup->addTab(rgroup,tr("Miscellaneous"));
373
374 QGridLayout* pLayout = new QGridLayout;
375 pLayout->setColumnMinimumWidth(0,14*ww);
376 pLayout->addWidget(new QLabel("Proxy host"),0,0, Qt::AlignLeft);
377 pLayout->addWidget(_proxyHostLineEdit,0, 1);
378 pLayout->addWidget(new QLabel("Proxy port"),1,0, Qt::AlignLeft);
379 pLayout->addWidget(_proxyPortLineEdit,1,1);
380 pLayout->addWidget(new QLabel("Settings for the proxy in protected networks, leave the boxes blank if none."),2, 0, 1, 2, Qt::AlignLeft);
381 pLayout->addWidget(new QLabel(" "),3,0);
382 pLayout->addWidget(new QLabel(" "),4,0);
383 pLayout->addWidget(new QLabel(" "),5,0);
384 pgroup->setLayout(pLayout);
385
386 QGridLayout* gLayout = new QGridLayout;
387 gLayout->setColumnMinimumWidth(0,14*ww);
388 gLayout->addWidget(new QLabel("Logfile (full path)"), 0, 0);
389 gLayout->addWidget(_logFileLineEdit, 0, 1);
390 gLayout->addWidget(new QLabel("Append files"), 1, 0);
391 gLayout->addWidget(_rnxAppendCheckBox, 1, 1);
392 gLayout->addWidget(new QLabel("Reread configuration"), 2, 0);
393 gLayout->addWidget(_onTheFlyComboBox, 2, 1);
394 gLayout->addWidget(new QLabel("Auto start"), 3, 0);
395 gLayout->addWidget(_autoStartCheckBox, 3, 1);
396 gLayout->addWidget(new QLabel("General settings for logfile, file handling, configuration on-the-fly, and auto-start."),4, 0, 1, 2, Qt::AlignLeft);
397 gLayout->addWidget(new QLabel(" "),5,0);
398 ggroup->setLayout(gLayout);
399
400 QGridLayout* sLayout = new QGridLayout;
401 sLayout->setColumnMinimumWidth(0,14*ww);
402 sLayout->addWidget(new QLabel("Port (synchronized)"), 0, 0);
403 sLayout->addWidget(_outPortLineEdit, 0, 1);
404 sLayout->addWidget(new QLabel("Wait for full epoch"), 0, 2, Qt::AlignRight);
405 sLayout->addWidget(_waitTimeSpinBox, 0, 3, Qt::AlignLeft);
406 sLayout->addWidget(new QLabel("Port (unsynchronized)"), 1, 0);
407 sLayout->addWidget(_outUPortLineEdit, 1, 1);
408 sLayout->addWidget(new QLabel("File (full path)"), 2, 0);
409 sLayout->addWidget(_outFileLineEdit, 2, 1, 1, 30);
410 sLayout->addWidget(new QLabel("Sampling"), 3, 0);
411 sLayout->addWidget(_binSamplSpinBox, 3, 1, Qt::AlignLeft);
412 sLayout->addWidget(new QLabel("Output decoded observations in a binary format to feed a real-time GNSS engine."),4,0,1,30);
413 sLayout->addWidget(new QLabel(" "),5,0);
414 sgroup->setLayout(sLayout);
415
416 QGridLayout* eLayout = new QGridLayout;
417 eLayout->setColumnMinimumWidth(0,14*ww);
418 eLayout->addWidget(new QLabel("Directory"), 0, 0);
419 eLayout->addWidget(_ephPathLineEdit, 0, 1);
420 eLayout->addWidget(new QLabel("Interval"), 1, 0);
421 eLayout->addWidget(_ephIntrComboBox, 1, 1);
422 eLayout->addWidget(new QLabel("Port"), 2, 0);
423 eLayout->addWidget(_outEphPortLineEdit, 2, 1);
424 eLayout->addWidget(new QLabel("Version 3"), 3, 0);
425 eLayout->addWidget(_ephV3CheckBox, 3, 1);
426 eLayout->addWidget(new QLabel("Saving RINEX ephemeris files and ephemeris output through IP port."),4,0,1,2,Qt::AlignLeft);
427 eLayout->addWidget(new QLabel(" "),5,0);
428 egroup->setLayout(eLayout);
429
430 QGridLayout* aLayout = new QGridLayout;
431 aLayout->setColumnMinimumWidth(0,14*ww);
432 aLayout->addWidget(new QLabel("Observation rate"), 0, 0);
433 aLayout->addWidget(_obsRateComboBox, 0, 1);
434 aLayout->addWidget(new QLabel("Failure threshold"), 1, 0);
435 aLayout->addWidget(_adviseFailSpinBox, 1, 1);
436 aLayout->addWidget(new QLabel("Recovery threshold"), 2, 0);
437 aLayout->addWidget(_adviseRecoSpinBox, 2, 1);
438 aLayout->addWidget(new QLabel("Pause"), 2, 2, Qt::AlignRight);
439 aLayout->addWidget(_makePauseCheckBox, 2, 3, Qt::AlignLeft);
440 aLayout->addWidget(new QLabel("Script (full path)"), 3, 0);
441 aLayout->addWidget(_adviseScriptLineEdit, 3, 1,1,10);
442 aLayout->addWidget(new QLabel("Outage report, handling of corrupted streams."),5,0,1,10,Qt::AlignLeft);
443 agroup->setLayout(aLayout);
444
445 QGridLayout* rLayout = new QGridLayout;
446 rLayout->setColumnMinimumWidth(0,14*ww);
447 rLayout->addWidget(new QLabel("Mountpoint"), 0, 0, Qt::AlignLeft);
448 rLayout->addWidget(_miscMountLineEdit, 0, 1,1,15,Qt::AlignLeft);
449 rLayout->addWidget(new QLabel("Log latency"), 1, 0);
450 rLayout->addWidget(_perfIntrComboBox, 1, 1);
451 rLayout->addWidget(new QLabel("Scan RTCM"), 2, 0);
452 rLayout->addWidget(_scanRTCMCheckBox, 2, 1);
453 rLayout->addWidget(new QLabel("Log latencies or scan RTCM streams for numbers of message types and antenna information."),3, 0, 1, 4, Qt::AlignLeft);
454 rLayout->addWidget(new QLabel(" "), 4, 0);
455 rLayout->addWidget(new QLabel(" "), 5, 0);
456 rgroup->setLayout(rLayout);
457
458 QGridLayout* oLayout = new QGridLayout;
459 oLayout->setColumnMinimumWidth(0,14*ww);
460 oLayout->addWidget(new QLabel("Directory"), 0, 0);
461 oLayout->addWidget(_rnxPathLineEdit, 0, 1,1,12);
462 oLayout->addWidget(new QLabel("Interval"), 1, 0);
463 oLayout->addWidget(_rnxIntrComboBox, 1, 1);
464 oLayout->addWidget(new QLabel("Sampling"), 1, 2, Qt::AlignRight);
465 oLayout->addWidget(_rnxSamplSpinBox, 1, 3, Qt::AlignLeft);
466 oLayout->addWidget(new QLabel("Skeleton extension"), 2, 0);
467 oLayout->addWidget(_rnxSkelLineEdit, 2, 1,1,1, Qt::AlignLeft);
468 oLayout->addWidget(new QLabel("Script (full path)"), 3, 0);
469 oLayout->addWidget(_rnxScrpLineEdit, 3, 1,1,12);
470 oLayout->addWidget(new QLabel("Version 3"), 4, 0);
471 oLayout->addWidget(_rnxV3CheckBox, 4, 1);
472 oLayout->addWidget(new QLabel("Saving RINEX observation files."),5,0,1,12, Qt::AlignLeft);
473 ogroup->setLayout(oLayout);
474
475 QGridLayout* cLayout = new QGridLayout;
476 cLayout->setColumnMinimumWidth(0,14*ww);
477 cLayout->addWidget(new QLabel("Directory"), 0, 0);
478 cLayout->addWidget(_corrPathLineEdit, 0, 1,1,30);
479 cLayout->addWidget(new QLabel("Interval"), 1, 0);
480 cLayout->addWidget(_corrIntrComboBox, 1, 1);
481 cLayout->addWidget(new QLabel("Port"), 2, 0);
482 cLayout->addWidget(_corrPortLineEdit, 2, 1);
483 cLayout->addWidget(new QLabel("Wait for full epoch"), 2, 2, Qt::AlignRight);
484 cLayout->addWidget(_corrTimeSpinBox, 2, 3, Qt::AlignLeft);
485 cLayout->addWidget(new QLabel("Saving Broadcast Ephemeris correction files and correction output through IP port."),3,0,1,30);
486 cLayout->addWidget(new QLabel(" "),4,0);
487 cLayout->addWidget(new QLabel(" "),5,0);
488 cgroup->setLayout(cLayout);
489
490 QGridLayout* serLayout = new QGridLayout;
491 serLayout->setColumnMinimumWidth(0,14*ww);
492 serLayout->addWidget(new QLabel("Mountpoint"), 0,0, Qt::AlignLeft);
493 serLayout->addWidget(_serialMountPointLineEdit, 0,1,1,2);
494 serLayout->addWidget(new QLabel("Port name"), 1,0, Qt::AlignLeft);
495 serLayout->addWidget(_serialPortNameLineEdit, 1,1,1,2);
496 serLayout->addWidget(new QLabel("Baud rate"), 2,0, Qt::AlignLeft);
497 serLayout->addWidget(_serialBaudRateComboBox, 2,1);
498 serLayout->addWidget(new QLabel(" Parity "), 2,2, Qt::AlignRight);
499 serLayout->addWidget(_serialParityComboBox, 2,3);
500 serLayout->addWidget(new QLabel("Data bits"), 3,0, Qt::AlignLeft);
501 serLayout->addWidget(_serialDataBitsComboBox, 3,1);
502 serLayout->addWidget(new QLabel(" Stop bits "), 3,2, Qt::AlignRight);
503 serLayout->addWidget(_serialStopBitsComboBox, 3,3);
504 serLayout->addWidget(new QLabel("Auto NMEA"), 4, 0);
505 serLayout->addWidget(_serialAutoNMEACheckBox, 4, 1);
506 serLayout->addWidget(new QLabel("Serial port settings to feed a serial connected device."),5,0,1,30);
507
508 sergroup->setLayout(serLayout);
509
510 QVBoxLayout* mLayout = new QVBoxLayout;
511 mLayout->addWidget(aogroup);
512 mLayout->addWidget(_mountPointsTable);
513 mLayout->addWidget(_log);
514
515 _canvas->setLayout(mLayout);
516
517 // Auto start
518 // ----------
519 if ( Qt::CheckState(settings.value("autoStart").toInt()) == Qt::Checked) {
520 slotGetData();
521 }
522}
523
524// Destructor
525////////////////////////////////////////////////////////////////////////////
526bncWindow::~bncWindow() {
527 delete _caster;
528}
529
530//
531////////////////////////////////////////////////////////////////////////////
532void bncWindow::populateMountPointsTable() {
533
534 for (int iRow = _mountPointsTable->rowCount()-1; iRow >=0; iRow--) {
535 _mountPointsTable->removeRow(iRow);
536 }
537
538 QSettings settings;
539
540 QListIterator<QString> it(settings.value("mountPoints").toStringList());
541 if (!it.hasNext()) {
542 _actGetData->setEnabled(false);
543 }
544 int iRow = 0;
545 while (it.hasNext()) {
546 QStringList hlp = it.next().split(" ");
547 if (hlp.size() < 5) continue;
548 _mountPointsTable->insertRow(iRow);
549
550 QUrl url(hlp[0]);
551
552 QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
553 QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
554 QString nmea(hlp[4]);
555 QString ntripVersion = "1";
556 if (hlp.size() >= 6) {
557 ntripVersion = (hlp[5]);
558 }
559
560 QTableWidgetItem* it;
561 it = new QTableWidgetItem(url.userInfo());
562 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
563 _mountPointsTable->setItem(iRow, 0, it);
564
565 it = new QTableWidgetItem(fullPath);
566 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
567 _mountPointsTable->setItem(iRow, 1, it);
568
569 it = new QTableWidgetItem(format);
570 _mountPointsTable->setItem(iRow, 2, it);
571
572 if (nmea == "yes") {
573 it = new QTableWidgetItem(latitude);
574 _mountPointsTable->setItem(iRow, 3, it);
575 it = new QTableWidgetItem(longitude);
576 _mountPointsTable->setItem(iRow, 4, it);
577 } else {
578 it = new QTableWidgetItem(latitude);
579 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
580 _mountPointsTable->setItem(iRow, 3, it);
581 it = new QTableWidgetItem(longitude);
582 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
583 _mountPointsTable->setItem(iRow, 4, it);
584 }
585
586 it = new QTableWidgetItem(nmea);
587 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
588 _mountPointsTable->setItem(iRow, 5, it);
589
590 it = new QTableWidgetItem(ntripVersion);
591 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
592 _mountPointsTable->setItem(iRow, 6, it);
593
594 bncTableItem* bncIt = new bncTableItem();
595 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
596 _mountPointsTable->setItem(iRow, 7, bncIt);
597
598 iRow++;
599 }
600
601 _mountPointsTable->sortItems(1);
602}
603
604// Retrieve Table
605////////////////////////////////////////////////////////////////////////////
606void bncWindow::slotAddMountPoints() {
607
608 QSettings settings;
609 QString proxyHost = settings.value("proxyHost").toString();
610 int proxyPort = settings.value("proxyPort").toInt();
611 if (proxyHost != _proxyHostLineEdit->text() ||
612 proxyPort != _proxyPortLineEdit->text().toInt()) {
613 int iRet = QMessageBox::question(this, "Question", "Proxy options "
614 "changed. Use the new ones?",
615 QMessageBox::Yes, QMessageBox::No,
616 QMessageBox::NoButton);
617 if (iRet == QMessageBox::Yes) {
618 settings.setValue("proxyHost", _proxyHostLineEdit->text());
619 settings.setValue("proxyPort", _proxyPortLineEdit->text());
620 }
621 }
622
623 bncTableDlg* dlg = new bncTableDlg(this);
624 dlg->move(this->pos().x()+50, this->pos().y()+50);
625 connect(dlg, SIGNAL(newMountPoints(QStringList*)),
626 this, SLOT(slotNewMountPoints(QStringList*)));
627 dlg->exec();
628 delete dlg;
629
630}
631
632// Delete Selected Mount Points
633////////////////////////////////////////////////////////////////////////////
634void bncWindow::slotDeleteMountPoints() {
635
636 int nRows = _mountPointsTable->rowCount();
637 bool flg[nRows];
638 for (int iRow = 0; iRow < nRows; iRow++) {
639 if (_mountPointsTable->isItemSelected(_mountPointsTable->item(iRow,1))) {
640 flg[iRow] = true;
641 }
642 else {
643 flg[iRow] = false;
644 }
645 }
646 for (int iRow = nRows-1; iRow >= 0; iRow--) {
647 if (flg[iRow]) {
648 _mountPointsTable->removeRow(iRow);
649 }
650 }
651 _actDeleteMountPoints->setEnabled(false);
652
653 if (_mountPointsTable->rowCount() == 0) {
654 _actGetData->setEnabled(false);
655 }
656}
657
658// New Mount Points Selected
659////////////////////////////////////////////////////////////////////////////
660void bncWindow::slotNewMountPoints(QStringList* mountPoints) {
661 int iRow = 0;
662 QListIterator<QString> it(*mountPoints);
663 while (it.hasNext()) {
664 QStringList hlp = it.next().split(" ");
665 QUrl url(hlp[0]);
666 QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
667 QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
668 QString nmea(hlp[4]);
669 QString ntripVersion = "1";
670 if (hlp.size() >= 6) {
671 ntripVersion = (hlp[5]);
672 }
673
674 _mountPointsTable->insertRow(iRow);
675
676 QTableWidgetItem* it;
677 it = new QTableWidgetItem(url.userInfo());
678 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
679 _mountPointsTable->setItem(iRow, 0, it);
680
681 it = new QTableWidgetItem(fullPath);
682 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
683 _mountPointsTable->setItem(iRow, 1, it);
684
685 it = new QTableWidgetItem(format);
686 _mountPointsTable->setItem(iRow, 2, it);
687
688 if (nmea == "yes") {
689 it = new QTableWidgetItem(latitude);
690 _mountPointsTable->setItem(iRow, 3, it);
691 it = new QTableWidgetItem(longitude);
692 _mountPointsTable->setItem(iRow, 4, it);
693 } else {
694 it = new QTableWidgetItem(latitude);
695 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
696 _mountPointsTable->setItem(iRow, 3, it);
697 it = new QTableWidgetItem(longitude);
698 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
699 _mountPointsTable->setItem(iRow, 4, it);
700 }
701
702 it = new QTableWidgetItem(nmea);
703 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
704 _mountPointsTable->setItem(iRow, 5, it);
705
706 it = new QTableWidgetItem(ntripVersion);
707 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
708 _mountPointsTable->setItem(iRow, 6, it);
709
710 bncTableItem* bncIt = new bncTableItem();
711 _mountPointsTable->setItem(iRow, 7, bncIt);
712
713 iRow++;
714 }
715 _mountPointsTable->hideColumn(0);
716 _mountPointsTable->sortItems(1);
717 if (mountPoints->count() > 0 && !_actStop->isEnabled()) {
718 _actGetData->setEnabled(true);
719 }
720 delete mountPoints;
721}
722
723// Save Options
724////////////////////////////////////////////////////////////////////////////
725void bncWindow::slotSaveOptions() {
726 QSettings settings;
727 settings.setValue("proxyHost", _proxyHostLineEdit->text());
728 settings.setValue("proxyPort", _proxyPortLineEdit->text());
729 settings.setValue("waitTime", _waitTimeSpinBox->value());
730 settings.setValue("obsRate", _obsRateComboBox->currentText());
731 settings.setValue("adviseFail", _adviseFailSpinBox->value());
732 settings.setValue("adviseReco", _adviseRecoSpinBox->value());
733 settings.setValue("makePause", _makePauseCheckBox->checkState());
734 settings.setValue("outFile", _outFileLineEdit->text());
735 settings.setValue("perfIntr", _perfIntrComboBox->currentText());
736 settings.setValue("outPort", _outPortLineEdit->text());
737 settings.setValue("outUPort", _outUPortLineEdit->text());
738 settings.setValue("outEphPort", _outEphPortLineEdit->text());
739 settings.setValue("corrPort", _corrPortLineEdit->text());
740 settings.setValue("corrTime", _corrTimeSpinBox->value());
741 settings.setValue("rnxPath", _rnxPathLineEdit->text());
742 settings.setValue("ephPath", _ephPathLineEdit->text());
743 settings.setValue("corrPath", _corrPathLineEdit->text());
744 settings.setValue("rnxScript", _rnxScrpLineEdit->text());
745 settings.setValue("rnxIntr", _rnxIntrComboBox->currentText());
746 settings.setValue("onTheFlyInterval", _onTheFlyComboBox->currentText());
747 settings.setValue("ephIntr", _ephIntrComboBox->currentText());
748 settings.setValue("corrIntr", _corrIntrComboBox->currentText());
749 settings.setValue("rnxSampl", _rnxSamplSpinBox->value());
750 settings.setValue("binSampl", _binSamplSpinBox->value());
751 settings.setValue("rnxSkel", _rnxSkelLineEdit->text());
752 settings.setValue("rnxAppend", _rnxAppendCheckBox->checkState());
753 settings.setValue("autoStart", _autoStartCheckBox->checkState());
754 settings.setValue("rnxV3", _rnxV3CheckBox->checkState());
755 settings.setValue("ephV3", _ephV3CheckBox->checkState());
756 settings.setValue("logFile", _logFileLineEdit->text());
757 settings.setValue("adviseScript",_adviseScriptLineEdit->text());
758 settings.setValue("miscMount", _miscMountLineEdit->text());
759 settings.setValue("scanRTCM", _scanRTCMCheckBox->checkState());
760 settings.setValue("serialPortName", _serialPortNameLineEdit->text());
761 settings.setValue("serialMountPoint", _serialMountPointLineEdit->text());
762 settings.setValue("serialBaudRate", _serialBaudRateComboBox->currentText());
763 settings.setValue("serialParity", _serialParityComboBox->currentText());
764 settings.setValue("serialDataBits", _serialDataBitsComboBox->currentText());
765 settings.setValue("serialStopBits", _serialStopBitsComboBox->currentText());
766 settings.setValue("serialAutoNMEA", _serialAutoNMEACheckBox->checkState());
767
768 QStringList mountPoints;
769
770 for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
771 QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
772 "@" + _mountPointsTable->item(iRow, 1)->text() );
773
774 mountPoints.append(url.toString() + " " +
775 _mountPointsTable->item(iRow, 2)->text()
776 + " " + _mountPointsTable->item(iRow, 3)->text()
777 + " " + _mountPointsTable->item(iRow, 4)->text()
778 + " " + _mountPointsTable->item(iRow, 5)->text()
779 + " " + _mountPointsTable->item(iRow, 6)->text());
780 }
781 settings.setValue("mountPoints", mountPoints);
782 if (_caster) {
783 _caster->slotReadMountPoints();
784 }
785}
786
787// All get slots terminated
788////////////////////////////////////////////////////////////////////////////
789void bncWindow::slotGetThreadErrors() {
790 ((bncApp*)qApp)->slotMessage("All Get Threads Terminated", true);
791 if (!_actStop->isEnabled()) {
792 _actGetData->setEnabled(true);
793 }
794}
795
796// Retrieve Data
797////////////////////////////////////////////////////////////////////////////
798void bncWindow::slotGetData() {
799 slotSaveOptions();
800
801 _actDeleteMountPoints->setEnabled(false);
802 _actGetData->setEnabled(false);
803 _actStop->setEnabled(true);
804
805 _caster = new bncCaster(_outFileLineEdit->text(),
806 _outPortLineEdit->text().toInt());
807
808 ((bncApp*)qApp)->setPort(_outEphPortLineEdit->text().toInt());
809 ((bncApp*)qApp)->setPortCorr(_corrPortLineEdit->text().toInt());
810
811 connect(_caster, SIGNAL(getThreadErrors()),
812 this, SLOT(slotGetThreadErrors()));
813
814 connect (_caster, SIGNAL(mountPointsRead(QList<bncGetThread*>)),
815 this, SLOT(slotMountPointsRead(QList<bncGetThread*>)));
816
817 ((bncApp*)qApp)->slotMessage("============ Start BNC ============", true);
818
819 _caster->slotReadMountPoints();
820}
821
822// Retrieve Data
823////////////////////////////////////////////////////////////////////////////
824void bncWindow::slotStop() {
825 int iRet = QMessageBox::question(this, "Stop", "Stop retrieving data?",
826 QMessageBox::Yes, QMessageBox::No,
827 QMessageBox::NoButton);
828 if (iRet == QMessageBox::Yes) {
829 delete _caster; _caster = 0;
830 _actGetData->setEnabled(true);
831 _actStop->setEnabled(false);
832 }
833}
834
835// Close Application gracefully
836////////////////////////////////////////////////////////////////////////////
837void bncWindow::closeEvent(QCloseEvent* event) {
838
839 int iRet = QMessageBox::question(this, "Close", "Save Options?",
840 QMessageBox::Yes, QMessageBox::No,
841 QMessageBox::Cancel);
842
843 if (iRet == QMessageBox::Cancel) {
844 event->ignore();
845 return;
846 }
847 else if (iRet == QMessageBox::Yes) {
848 slotSaveOptions();
849 }
850
851 QMainWindow::closeEvent(event);
852}
853
854// User changed the selection of mountPoints
855////////////////////////////////////////////////////////////////////////////
856void bncWindow::slotSelectionChanged() {
857 if (_mountPointsTable->selectedItems().isEmpty()) {
858 _actDeleteMountPoints->setEnabled(false);
859 }
860 else {
861 _actDeleteMountPoints->setEnabled(true);
862 }
863}
864
865// Display Program Messages
866////////////////////////////////////////////////////////////////////////////
867void bncWindow::slotWindowMessage(const QByteArray msg, bool showOnScreen) {
868
869#ifdef DEBUG_RTCM2_2021
870 const int maxBufferSize = 1000;
871#else
872 const int maxBufferSize = 10000;
873#endif
874
875 if (! showOnScreen ) {
876 return;
877 }
878
879 QString txt = _log->toPlainText() + "\n" +
880 QDateTime::currentDateTime().toUTC().toString("yy-MM-dd hh:mm:ss ") + msg;
881 _log->clear();
882 _log->append(txt.right(maxBufferSize));
883}
884
885// About Message
886////////////////////////////////////////////////////////////////////////////
887void bncWindow::slotAbout() {
888 new bncAboutDlg(0);
889}
890
891//Flowchart
892////////////////////////////////////////////////////////////////////////////
893void bncWindow::slotFlowchart() {
894 new bncFlowchartDlg(0);
895}
896
897// Help Window
898////////////////////////////////////////////////////////////////////////////
899void bncWindow::slotHelp() {
900 QUrl url;
901 url.setPath(":bnchelp.html");
902 new bncHlpDlg(0, url);
903}
904
905// Select Fonts
906////////////////////////////////////////////////////////////////////////////
907void bncWindow::slotFontSel() {
908 bool ok;
909 QFont newFont = QFontDialog::getFont(&ok, this->font(), this);
910 if (ok) {
911 QSettings settings;
912 settings.setValue("font", newFont.toString());
913 QApplication::setFont(newFont);
914 int ww = QFontMetrics(newFont).width('w');
915 setMinimumSize(60*ww, 80*ww);
916 resize(60*ww, 80*ww);
917 }
918}
919
920// Whats This Help
921void bncWindow::slotWhatsThis() {
922 QWhatsThis::enterWhatsThisMode();
923}
924
925//
926////////////////////////////////////////////////////////////////////////////
927void bncWindow::slotMountPointsRead(QList<bncGetThread*> threads) {
928 populateMountPointsTable();
929 QSettings settings;
930 _binSamplSpinBox->setValue(settings.value("binSampl").toInt());
931 _waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
932 QListIterator<bncGetThread*> iTh(threads);
933 while (iTh.hasNext()) {
934 bncGetThread* thread = iTh.next();
935 for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
936 QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
937 "@" + _mountPointsTable->item(iRow, 1)->text() );
938 if (url == thread->mountPoint() &&
939 _mountPointsTable->item(iRow, 3)->text() == thread->latitude() &&
940 _mountPointsTable->item(iRow, 4)->text() == thread->longitude() ) {
941 ((bncTableItem*) _mountPointsTable->item(iRow, 7))->setGetThread(thread);
942 break;
943 }
944 }
945 }
946}
947
948//
949////////////////////////////////////////////////////////////////////////////
950void bncWindow::CreateMenu() {
951 // Create Menus
952 // ------------
953 _menuFile = menuBar()->addMenu(tr("&File"));
954 _menuFile->addAction(_actFontSel);
955 _menuFile->addSeparator();
956 _menuFile->addAction(_actSaveOpt);
957 _menuFile->addSeparator();
958 _menuFile->addAction(_actQuit);
959
960 _menuHlp = menuBar()->addMenu(tr("&Help"));
961 _menuHlp->addAction(_actHelp);
962 _menuHlp->addAction(_actFlowchart);
963 _menuHlp->addAction(_actAbout);
964}
965
966//
967////////////////////////////////////////////////////////////////////////////
968void bncWindow::AddToolbar() {
969 // Tool (Command) Bar
970 // ------------------
971 QToolBar* toolBar = new QToolBar;
972 addToolBar(Qt::BottomToolBarArea, toolBar);
973 toolBar->setMovable(false);
974 toolBar->addAction(_actAddMountPoints);
975 toolBar->addAction(_actDeleteMountPoints);
976 toolBar->addAction(_actGetData);
977 toolBar->addAction(_actStop);
978 toolBar->addWidget(new QLabel(" "));
979 toolBar->addAction(_actwhatsthis);
980}
981
982//
983////////////////////////////////////////////////////////////////////////////
984bncAboutDlg::bncAboutDlg(QWidget* parent) :
985 QDialog(parent) {
986
987 QTextBrowser* tb = new QTextBrowser;
988 QUrl url; url.setPath(":bncabout.html");
989 tb->setSource(url);
990 tb->setReadOnly(true);
991
992 int ww = QFontMetrics(font()).width('w');
993 QPushButton* _closeButton = new QPushButton("Close");
994 _closeButton->setMaximumWidth(10*ww);
995 connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
996
997 QGridLayout* dlgLayout = new QGridLayout();
998 QLabel* img = new QLabel();
999 img->setPixmap(QPixmap(":ntrip-logo.png"));
1000 dlgLayout->addWidget(img, 0,0);
1001 dlgLayout->addWidget(new QLabel("BKG Ntrip Client (BNC) Version 1.7"), 0,1);
1002 dlgLayout->addWidget(tb,1,0,1,2);
1003 dlgLayout->addWidget(_closeButton,2,1,Qt::AlignRight);
1004
1005 setLayout(dlgLayout);
1006 resize(60*ww, 60*ww);
1007 show();
1008}
1009
1010//
1011////////////////////////////////////////////////////////////////////////////
1012bncAboutDlg::~bncAboutDlg() {
1013};
1014
1015//
1016////////////////////////////////////////////////////////////////////////////
1017bncFlowchartDlg::bncFlowchartDlg(QWidget* parent) :
1018 QDialog(parent) {
1019
1020 int ww = QFontMetrics(font()).width('w');
1021 QPushButton* _closeButton = new QPushButton("Close");
1022 _closeButton->setMaximumWidth(10*ww);
1023 connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
1024
1025 QGridLayout* dlgLayout = new QGridLayout();
1026 QLabel* img = new QLabel();
1027 img->setPixmap(QPixmap(":bncflowchart.png"));
1028 dlgLayout->addWidget(img, 0,0);
1029 dlgLayout->addWidget(_closeButton,1,0,Qt::AlignLeft);
1030
1031 setLayout(dlgLayout);
1032 show();
1033}
1034
1035//
1036////////////////////////////////////////////////////////////////////////////
1037bncFlowchartDlg::~bncFlowchartDlg() {
1038};
1039
Note: See TracBrowser for help on using the repository browser.