source: ntrip/trunk/BNC/src/bncwindow.cpp@ 5926

Last change on this file since 5926 was 5926, checked in by mervart, 10 years ago
File size: 118.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 "bnccore.h"
46#include "bncgetthread.h"
47#include "bnctabledlg.h"
48#include "bncipport.h"
49#include "bncudpport.h"
50#include "bncserialport.h"
51#include "bnchlpdlg.h"
52#include "bnchtml.h"
53#include "bnctableitem.h"
54#include "bncsettings.h"
55#include "bncfigure.h"
56#include "bncfigurelate.h"
57#include "bncfigureppp.h"
58#include "bncversion.h"
59#include "bncbytescounter.h"
60#include "bncsslconfig.h"
61#include "upload/bnccustomtrafo.h"
62#include "upload/bncephuploadcaster.h"
63#include "qtfilechooser.h"
64#include "reqcdlg.h"
65#include "bncmap.h"
66#include "rinex/reqcedit.h"
67#include "rinex/reqcanalyze.h"
68#ifdef QT_WEBKIT
69# include "map/bncmapwin.h"
70#endif
71
72using namespace std;
73
74#ifdef GNSSCENTER_PLUGIN
75Q_EXPORT_PLUGIN2(gnsscenter_bnc, t_bncFactory)
76#endif
77
78// Constructor
79////////////////////////////////////////////////////////////////////////////
80bncWindow::bncWindow() {
81
82#ifdef GNSSCENTER_PLUGIN
83 BNC_CORE->setConfFileName("");
84#endif
85
86 _caster = 0;
87 _casterEph = 0;
88
89 _bncFigure = new bncFigure(this);
90 _bncFigureLate = new bncFigureLate(this);
91 _bncFigurePPP = new bncFigurePPP(this);
92 connect(BNC_CORE, SIGNAL(newPosition(bncTime, QVector<double>)),
93 _bncFigurePPP, SLOT(slotNewPosition(bncTime, QVector<double>)));
94 _runningRealTime = false;
95 _runningPostProcessingReqc = false;
96 _runningPostProcessingPPP = false;
97 _reqcActionComboBox = 0; // necessary for enableStartStop()
98
99 _mapWin = 0;
100
101 int ww = QFontMetrics(this->font()).width('w');
102
103 static const QStringList labels = QString("account, Streams: resource loader / mountpoint, decoder, lat, long, nmea, ntrip, bytes").split(",");
104
105 setMinimumSize(85*ww, 65*ww);
106
107 setWindowTitle(tr("BKG Ntrip Client (BNC) Version " BNCVERSION));
108
109 connect(BNC_CORE, SIGNAL(newMessage(QByteArray,bool)),
110 this, SLOT(slotWindowMessage(QByteArray,bool)));
111
112 // Create Actions
113 // --------------
114 _actHelp = new QAction(tr("&Help Contents"),this);
115 connect(_actHelp, SIGNAL(triggered()), SLOT(slotHelp()));
116
117 _actAbout = new QAction(tr("&About BNC"),this);
118 connect(_actAbout, SIGNAL(triggered()), SLOT(slotAbout()));
119
120 _actFlowchart = new QAction(tr("&Flow Chart"),this);
121 connect(_actFlowchart, SIGNAL(triggered()), SLOT(slotFlowchart()));
122
123 _actFontSel = new QAction(tr("Select &Font"),this);
124 connect(_actFontSel, SIGNAL(triggered()), SLOT(slotFontSel()));
125
126 _actSaveOpt = new QAction(tr("&Reread && Save Configuration"),this);
127 connect(_actSaveOpt, SIGNAL(triggered()), SLOT(slotSaveOptions()));
128
129 _actQuit = new QAction(tr("&Quit"),this);
130 connect(_actQuit, SIGNAL(triggered()), SLOT(close()));
131
132 _actAddMountPoints = new QAction(tr("Add &Stream"),this);
133 connect(_actAddMountPoints, SIGNAL(triggered()), SLOT(slotAddMountPoints()));
134
135 _actDeleteMountPoints = new QAction(tr("&Delete Stream"),this);
136 connect(_actDeleteMountPoints, SIGNAL(triggered()), SLOT(slotDeleteMountPoints()));
137 _actDeleteMountPoints->setEnabled(false);
138
139 _actMapMountPoints = new QAction(tr("&Map"),this);
140 connect(_actMapMountPoints, SIGNAL(triggered()), SLOT(slotMapMountPoints()));
141
142 _actStart = new QAction(tr("Sta&rt"),this);
143 connect(_actStart, SIGNAL(triggered()), SLOT(slotStart()));
144
145 _actStop = new QAction(tr("Sto&p"),this);
146 connect(_actStop, SIGNAL(triggered()), SLOT(slotStop()));
147
148 _actwhatsthis= new QAction(tr("Help ?=Shift+F1"),this);
149 connect(_actwhatsthis, SIGNAL(triggered()), SLOT(slotWhatsThis()));
150
151 CreateMenu();
152 AddToolbar();
153
154 bncSettings settings;
155
156 // Network Options
157 // ---------------
158 _proxyHostLineEdit = new QLineEdit(settings.value("proxyHost").toString());
159 _proxyPortLineEdit = new QLineEdit(settings.value("proxyPort").toString());
160
161 connect(_proxyHostLineEdit, SIGNAL(textChanged(const QString &)),
162 this, SLOT(slotBncTextChanged()));
163
164 _sslCaCertPathLineEdit = new QLineEdit(settings.value("sslCaCertPath").toString());
165 _ignoreSslErrorsCheckBox = new QCheckBox();
166 _ignoreSslErrorsCheckBox->setCheckState(Qt::CheckState(
167 settings.value("ignoreSslErrors").toInt()));
168
169 // General Options
170 // ---------------
171 _logFileLineEdit = new QLineEdit(settings.value("logFile").toString());
172 _rawOutFileLineEdit = new QLineEdit(settings.value("rawOutFile").toString());
173 _rnxAppendCheckBox = new QCheckBox();
174 _rnxAppendCheckBox->setCheckState(Qt::CheckState(
175 settings.value("rnxAppend").toInt()));
176 _onTheFlyComboBox = new QComboBox();
177 _onTheFlyComboBox->setEditable(false);
178 _onTheFlyComboBox->addItems(QString("1 day,1 hour,5 min,1 min").split(","));
179 int ii = _onTheFlyComboBox->findText(settings.value("onTheFlyInterval").toString());
180 if (ii != -1) {
181 _onTheFlyComboBox->setCurrentIndex(ii);
182 }
183 _autoStartCheckBox = new QCheckBox();
184 _autoStartCheckBox->setCheckState(Qt::CheckState(
185 settings.value("autoStart").toInt()));
186
187 // RINEX Observations Options
188 // --------------------------
189 _rnxPathLineEdit = new QLineEdit(settings.value("rnxPath").toString());
190 _rnxIntrComboBox = new QComboBox();
191 _rnxIntrComboBox->setEditable(false);
192 _rnxIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
193 ii = _rnxIntrComboBox->findText(settings.value("rnxIntr").toString());
194 if (ii != -1) {
195 _rnxIntrComboBox->setCurrentIndex(ii);
196 }
197 _rnxSamplSpinBox = new QSpinBox();
198 _rnxSamplSpinBox->setMinimum(0);
199 _rnxSamplSpinBox->setMaximum(60);
200 _rnxSamplSpinBox->setSingleStep(5);
201 _rnxSamplSpinBox->setValue(settings.value("rnxSampl").toInt());
202 _rnxSamplSpinBox->setSuffix(" sec");
203 _rnxSkelLineEdit = new QLineEdit(settings.value("rnxSkel").toString());
204 _rnxSkelLineEdit->setMaximumWidth(5*ww);
205 _rnxScrpLineEdit = new QLineEdit(settings.value("rnxScript").toString());
206 _rnxV3CheckBox = new QCheckBox();
207 _rnxV3CheckBox->setCheckState(Qt::CheckState(settings.value("rnxV3").toInt()));
208
209 connect(_rnxPathLineEdit, SIGNAL(textChanged(const QString &)),
210 this, SLOT(slotBncTextChanged()));
211
212 // RINEX Ephemeris Options
213 // -----------------------
214 _ephPathLineEdit = new QLineEdit(settings.value("ephPath").toString());
215 _ephIntrComboBox = new QComboBox();
216 _ephIntrComboBox->setEditable(false);
217 _ephIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
218 int jj = _ephIntrComboBox->findText(settings.value("ephIntr").toString());
219 if (jj != -1) {
220 _ephIntrComboBox->setCurrentIndex(jj);
221 }
222 _outEphPortLineEdit = new QLineEdit(settings.value("outEphPort").toString());
223 _ephV3CheckBox = new QCheckBox();
224 _ephV3CheckBox->setCheckState(Qt::CheckState(settings.value("ephV3").toInt()));
225
226 connect(_outEphPortLineEdit, SIGNAL(textChanged(const QString &)),
227 this, SLOT(slotBncTextChanged()));
228
229 connect(_ephPathLineEdit, SIGNAL(textChanged(const QString &)),
230 this, SLOT(slotBncTextChanged()));
231
232 // Broadcast Corrections Options
233 // -----------------------------
234 _corrPathLineEdit = new QLineEdit(settings.value("corrPath").toString());
235 _corrIntrComboBox = new QComboBox();
236 _corrIntrComboBox->setEditable(false);
237 _corrIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
238 int mm = _corrIntrComboBox->findText(settings.value("corrIntr").toString());
239 if (mm != -1) {
240 _corrIntrComboBox->setCurrentIndex(mm);
241 }
242 _corrPortLineEdit = new QLineEdit(settings.value("corrPort").toString());
243 _corrTimeSpinBox = new QSpinBox();
244 _corrTimeSpinBox->setMinimum(0);
245 _corrTimeSpinBox->setMaximum(60);
246 _corrTimeSpinBox->setSingleStep(1);
247 _corrTimeSpinBox->setSuffix(" sec");
248 _corrTimeSpinBox->setValue(settings.value("corrTime").toInt());
249
250 connect(_corrPathLineEdit, SIGNAL(textChanged(const QString &)),
251 this, SLOT(slotBncTextChanged()));
252
253 connect(_corrPortLineEdit, SIGNAL(textChanged(const QString &)),
254 this, SLOT(slotBncTextChanged()));
255
256 // Feed Engine Options
257 // -------------------
258 _outPortLineEdit = new QLineEdit(settings.value("outPort").toString());
259 _waitTimeSpinBox = new QSpinBox();
260 _waitTimeSpinBox->setMinimum(1);
261 _waitTimeSpinBox->setMaximum(30);
262 _waitTimeSpinBox->setSingleStep(1);
263 _waitTimeSpinBox->setSuffix(" sec");
264 _waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
265 _binSamplSpinBox = new QSpinBox();
266 _binSamplSpinBox->setMinimum(0);
267 _binSamplSpinBox->setMaximum(60);
268 _binSamplSpinBox->setSingleStep(5);
269 _binSamplSpinBox->setValue(settings.value("binSampl").toInt());
270 _binSamplSpinBox->setSuffix(" sec");
271 _outFileLineEdit = new QLineEdit(settings.value("outFile").toString());
272 _outUPortLineEdit = new QLineEdit(settings.value("outUPort").toString());
273
274 connect(_outPortLineEdit, SIGNAL(textChanged(const QString &)),
275 this, SLOT(slotBncTextChanged()));
276
277 connect(_outFileLineEdit, SIGNAL(textChanged(const QString &)),
278 this, SLOT(slotBncTextChanged()));
279
280 // Serial Output Options
281 // ---------------------
282 _serialMountPointLineEdit = new QLineEdit(settings.value("serialMountPoint").toString());
283 _serialPortNameLineEdit = new QLineEdit(settings.value("serialPortName").toString());
284 _serialBaudRateComboBox = new QComboBox();
285 _serialBaudRateComboBox->addItems(QString("110,300,600,"
286 "1200,2400,4800,9600,19200,38400,57600,115200").split(","));
287 int kk = _serialBaudRateComboBox->findText(settings.value("serialBaudRate").toString());
288 if (kk != -1) {
289 _serialBaudRateComboBox->setCurrentIndex(kk);
290 }
291 _serialFlowControlComboBox = new QComboBox();
292 _serialFlowControlComboBox->addItems(QString("OFF,XONXOFF,HARDWARE").split(","));
293 kk = _serialFlowControlComboBox->findText(settings.value("serialFlowControl").toString());
294 if (kk != -1) {
295 _serialFlowControlComboBox->setCurrentIndex(kk);
296 }
297 _serialDataBitsComboBox = new QComboBox();
298 _serialDataBitsComboBox->addItems(QString("5,6,7,8").split(","));
299 kk = _serialDataBitsComboBox->findText(settings.value("serialDataBits").toString());
300 if (kk != -1) {
301 _serialDataBitsComboBox->setCurrentIndex(kk);
302 }
303 _serialParityComboBox = new QComboBox();
304 _serialParityComboBox->addItems(QString("NONE,ODD,EVEN,SPACE").split(","));
305 kk = _serialParityComboBox->findText(settings.value("serialParity").toString());
306 if (kk != -1) {
307 _serialParityComboBox->setCurrentIndex(kk);
308 }
309 _serialStopBitsComboBox = new QComboBox();
310 _serialStopBitsComboBox->addItems(QString("1,2").split(","));
311 kk = _serialStopBitsComboBox->findText(settings.value("serialStopBits").toString());
312 if (kk != -1) {
313 _serialStopBitsComboBox->setCurrentIndex(kk);
314 }
315 _serialAutoNMEAComboBox = new QComboBox();
316 _serialAutoNMEAComboBox->addItems(QString("Auto,Manual").split(","));
317 kk = _serialAutoNMEAComboBox->findText(settings.value("serialAutoNMEA").toString());
318 if (kk != -1) {
319 _serialAutoNMEAComboBox->setCurrentIndex(kk);
320 }
321 _serialFileNMEALineEdit = new QLineEdit(settings.value("serialFileNMEA").toString());
322 _serialHeightNMEALineEdit = new QLineEdit(settings.value("serialHeightNMEA").toString());
323
324 connect(_serialMountPointLineEdit, SIGNAL(textChanged(const QString &)),
325 this, SLOT(slotBncTextChanged()));
326
327 connect(_serialAutoNMEAComboBox, SIGNAL(currentIndexChanged(const QString &)),
328 this, SLOT(slotBncTextChanged()));
329
330 // Outages Options
331 // ---------------
332 _obsRateComboBox = new QComboBox();
333 _obsRateComboBox->setEditable(false);
334 _obsRateComboBox->addItems(QString(",0.1 Hz,0.2 Hz,0.5 Hz,1 Hz,5 Hz").split(","));
335 kk = _obsRateComboBox->findText(settings.value("obsRate").toString());
336 if (kk != -1) {
337 _obsRateComboBox->setCurrentIndex(kk);
338 }
339 _adviseFailSpinBox = new QSpinBox();
340 _adviseFailSpinBox->setMinimum(0);
341 _adviseFailSpinBox->setMaximum(60);
342 _adviseFailSpinBox->setSingleStep(1);
343 _adviseFailSpinBox->setSuffix(" min");
344 _adviseFailSpinBox->setValue(settings.value("adviseFail").toInt());
345 _adviseRecoSpinBox = new QSpinBox();
346 _adviseRecoSpinBox->setMinimum(0);
347 _adviseRecoSpinBox->setMaximum(60);
348 _adviseRecoSpinBox->setSingleStep(1);
349 _adviseRecoSpinBox->setSuffix(" min");
350 _adviseRecoSpinBox->setValue(settings.value("adviseReco").toInt());
351 _adviseScriptLineEdit = new QLineEdit(settings.value("adviseScript").toString());
352
353 connect(_obsRateComboBox, SIGNAL(currentIndexChanged(const QString &)),
354 this, SLOT(slotBncTextChanged()));
355
356 // Miscellaneous Options
357 // ---------------------
358 _miscMountLineEdit = new QLineEdit(settings.value("miscMount").toString());
359 _miscPortLineEdit = new QLineEdit(settings.value("miscPort").toString());
360 _perfIntrComboBox = new QComboBox();
361 _perfIntrComboBox->setEditable(false);
362 _perfIntrComboBox->addItems(QString(",2 sec, 10 sec,1 min,5 min,15 min,1 hour,6 hours,1 day").split(","));
363 int ll = _perfIntrComboBox->findText(settings.value("perfIntr").toString());
364 if (ll != -1) {
365 _perfIntrComboBox->setCurrentIndex(ll);
366 }
367 _scanRTCMCheckBox = new QCheckBox();
368 _scanRTCMCheckBox->setCheckState(Qt::CheckState(
369 settings.value("scanRTCM").toInt()));
370
371 connect(_miscMountLineEdit, SIGNAL(textChanged(const QString &)),
372 this, SLOT(slotBncTextChanged()));
373
374 // Streams
375 // -------
376 _mountPointsTable = new QTableWidget(0,8);
377
378 _mountPointsTable->horizontalHeader()->resizeSection(1,34*ww);
379 _mountPointsTable->horizontalHeader()->resizeSection(2,9*ww);
380 _mountPointsTable->horizontalHeader()->resizeSection(3,7*ww);
381 _mountPointsTable->horizontalHeader()->resizeSection(4,7*ww);
382 _mountPointsTable->horizontalHeader()->resizeSection(5,5*ww);
383 _mountPointsTable->horizontalHeader()->resizeSection(6,5*ww);
384 _mountPointsTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
385 _mountPointsTable->horizontalHeader()->setStretchLastSection(true);
386 _mountPointsTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
387 _mountPointsTable->setHorizontalHeaderLabels(labels);
388 _mountPointsTable->setGridStyle(Qt::NoPen);
389 _mountPointsTable->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
390 _mountPointsTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
391 _mountPointsTable->setSelectionBehavior(QAbstractItemView::SelectRows);
392 _mountPointsTable->hideColumn(0);
393 connect(_mountPointsTable, SIGNAL(itemSelectionChanged()),
394 SLOT(slotSelectionChanged()));
395 populateMountPointsTable();
396
397 _log = new QTextBrowser();
398 _log->setReadOnly(true);
399
400 // Combine Corrections
401 // -------------------
402 _cmbTable = new QTableWidget(0,3);
403 _cmbTable->setHorizontalHeaderLabels(QString("Mountpoint, AC Name, Weight").split(","));
404 _cmbTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
405 _cmbTable->setSelectionBehavior(QAbstractItemView::SelectRows);
406 _cmbTable->setMaximumWidth(30*ww);
407 _cmbTable->horizontalHeader()->resizeSection(0,10*ww);
408 _cmbTable->horizontalHeader()->resizeSection(1,8*ww);
409 _cmbTable->horizontalHeader()->resizeSection(2,8*ww);
410 _cmbTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
411 _cmbTable->horizontalHeader()->setStretchLastSection(true);
412 _cmbTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
413
414 _cmbMaxresLineEdit = new QLineEdit(settings.value("cmbMaxres").toString());
415
416 _cmbSamplSpinBox = new QSpinBox;
417 _cmbSamplSpinBox->setMinimum(10);
418 _cmbSamplSpinBox->setMaximum(60);
419 _cmbSamplSpinBox->setSingleStep(10);
420 _cmbSamplSpinBox->setMaximumWidth(9*ww);
421 _cmbSamplSpinBox->setValue(settings.value("cmbSampl").toInt());
422 _cmbSamplSpinBox->setSuffix(" sec");
423
424 QPushButton* addCmbRowButton = new QPushButton("Add Row");
425 QPushButton* delCmbRowButton = new QPushButton("Delete");
426
427 connect(_cmbTable, SIGNAL(itemSelectionChanged()),
428 SLOT(slotBncTextChanged()));
429
430 _cmbMethodComboBox = new QComboBox();
431 _cmbMethodComboBox->setEditable(false);
432 _cmbMethodComboBox->addItems(QString("Filter,Single-Epoch").split(","));
433 int im = _cmbMethodComboBox->findText(settings.value("cmbMethod").toString());
434 if (im != -1) {
435 _cmbMethodComboBox->setCurrentIndex(im);
436 }
437
438 int iRow = _cmbTable->rowCount();
439 if (iRow > 0) {
440 enableWidget(true, _cmbMethodComboBox);
441 _cmbMaxresLineEdit->setStyleSheet("background-color: white");
442 _cmbMaxresLineEdit->setEnabled(true);
443 _cmbSamplSpinBox->setEnabled(true);
444 }
445 else {
446 enableWidget(false, _cmbMethodComboBox);
447 _cmbMaxresLineEdit->setStyleSheet("background-color: lightGray");
448 _cmbMaxresLineEdit->setEnabled(false);
449 _cmbSamplSpinBox->setEnabled(false);
450 }
451
452 // Upload Results
453 // -------------
454 _uploadTable = new QTableWidget(0,12);
455 _uploadTable->setHorizontalHeaderLabels(QString("Host, Port, Mount, Password, System, CoM, SP3 File, RNX File, PID, SID, IOD, bytes").split(","));
456 _uploadTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
457 _uploadTable->setSelectionBehavior(QAbstractItemView::SelectRows);
458 _uploadTable->horizontalHeader()->resizeSection( 0,13*ww);
459 _uploadTable->horizontalHeader()->resizeSection( 1, 5*ww);
460 _uploadTable->horizontalHeader()->resizeSection( 2, 6*ww);
461 _uploadTable->horizontalHeader()->resizeSection( 3, 8*ww);
462 _uploadTable->horizontalHeader()->resizeSection( 4,11*ww);
463 _uploadTable->horizontalHeader()->resizeSection( 5, 4*ww);
464 _uploadTable->horizontalHeader()->resizeSection( 6,15*ww);
465 _uploadTable->horizontalHeader()->resizeSection( 7,15*ww);
466 _uploadTable->horizontalHeader()->resizeSection( 8, 4*ww);
467 _uploadTable->horizontalHeader()->resizeSection( 9, 4*ww);
468 _uploadTable->horizontalHeader()->resizeSection(10, 4*ww);
469 _uploadTable->horizontalHeader()->resizeSection(11,12*ww);
470 _uploadTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
471 _uploadTable->horizontalHeader()->setStretchLastSection(true);
472 _uploadTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
473
474 connect(_uploadTable, SIGNAL(itemSelectionChanged()),
475 SLOT(slotBncTextChanged()));
476
477 QPushButton* addUploadRowButton = new QPushButton("Add Row");
478 QPushButton* delUploadRowButton = new QPushButton("Del Row");
479 QPushButton* setUploadTrafoButton = new QPushButton("Custom Trafo");
480 _uploadIntrComboBox = new QComboBox;
481 _uploadIntrComboBox->setEditable(false);
482 _uploadIntrComboBox->addItems(QString("1 day,1 hour, 30 min,15 min,10 min,5 min,2 min,1 min").split(","));
483 ii = _uploadIntrComboBox->findText(settings.value("uploadIntr").toString());
484 if (ii != -1) {
485 _uploadIntrComboBox->setCurrentIndex(ii);
486 }
487
488 _uploadSamplRtcmEphCorrSpinBox = new QSpinBox;
489 _uploadSamplRtcmEphCorrSpinBox->setMinimum(0);
490 _uploadSamplRtcmEphCorrSpinBox->setMaximum(60);
491 _uploadSamplRtcmEphCorrSpinBox->setSingleStep(5);
492 _uploadSamplRtcmEphCorrSpinBox->setMaximumWidth(9*ww);
493 _uploadSamplRtcmEphCorrSpinBox->setValue(settings.value("uploadSamplRtcmEphCorr").toInt());
494 _uploadSamplRtcmEphCorrSpinBox->setSuffix(" sec");
495
496 _uploadSamplSp3SpinBox = new QSpinBox;
497 _uploadSamplSp3SpinBox->setMinimum(0);
498 _uploadSamplSp3SpinBox->setMaximum(15);
499 _uploadSamplSp3SpinBox->setSingleStep(1);
500 _uploadSamplSp3SpinBox->setMaximumWidth(9*ww);
501 _uploadSamplSp3SpinBox->setValue(settings.value("uploadSamplSp3").toInt());
502 _uploadSamplSp3SpinBox->setSuffix(" min");
503
504 _uploadSamplClkRnxSpinBox = new QSpinBox;
505 _uploadSamplClkRnxSpinBox->setMinimum(0);
506 _uploadSamplClkRnxSpinBox->setMaximum(60);
507 _uploadSamplClkRnxSpinBox->setSingleStep(5);
508 _uploadSamplClkRnxSpinBox->setMaximumWidth(9*ww);
509 _uploadSamplClkRnxSpinBox->setValue(settings.value("uploadSamplClkRnx").toInt());
510 _uploadSamplClkRnxSpinBox->setSuffix(" sec");
511
512 int iRowT = _uploadTable->rowCount();
513 if (iRowT > 0) {
514 enableWidget(true, _uploadIntrComboBox);
515 enableWidget(true, _uploadSamplRtcmEphCorrSpinBox);
516 enableWidget(true, _uploadSamplSp3SpinBox);
517 enableWidget(true, _uploadSamplClkRnxSpinBox);
518 }
519 else {
520 enableWidget(false, _uploadIntrComboBox);
521 enableWidget(false, _uploadSamplRtcmEphCorrSpinBox);
522 enableWidget(false, _uploadSamplSp3SpinBox);
523 enableWidget(false, _uploadSamplClkRnxSpinBox);
524 }
525
526 // Upload RTCM3 Ephemeris
527 // ----------------------
528 _uploadEphHostLineEdit = new QLineEdit(settings.value("uploadEphHost").toString());
529 _uploadEphPortLineEdit = new QLineEdit(settings.value("uploadEphPort").toString());
530 _uploadEphPasswordLineEdit = new QLineEdit(settings.value("uploadEphPassword").toString());
531 _uploadEphPasswordLineEdit->setEchoMode(QLineEdit::PasswordEchoOnEdit);
532 _uploadEphMountpointLineEdit = new QLineEdit(settings.value("uploadEphMountpoint").toString());
533 _uploadEphSampleSpinBox = new QSpinBox;
534 _uploadEphSampleSpinBox->setMinimum(5);
535 _uploadEphSampleSpinBox->setMaximum(60);
536 _uploadEphSampleSpinBox->setSingleStep(5);
537 _uploadEphSampleSpinBox->setMaximumWidth(9*ww);
538 _uploadEphSampleSpinBox->setValue(settings.value("uploadEphSample").toInt());
539 _uploadEphSampleSpinBox->setSuffix(" sec");
540 _uploadEphBytesCounter = new bncBytesCounter;
541
542 // Canvas with Editable Fields
543 // ---------------------------
544 _canvas = new QWidget;
545 setCentralWidget(_canvas);
546
547 _aogroup = new QTabWidget();
548 QWidget* pgroup = new QWidget();
549 QWidget* ggroup = new QWidget();
550 QWidget* sgroup = new QWidget();
551 QWidget* egroup = new QWidget();
552 QWidget* agroup = new QWidget();
553 QWidget* cgroup = new QWidget();
554 QWidget* ogroup = new QWidget();
555 QWidget* rgroup = new QWidget();
556 QWidget* sergroup = new QWidget();
557 QWidget* pppGroup1 = new QWidget();
558 QWidget* pppGroup2 = new QWidget();
559 QWidget* pppGroup3 = new QWidget();
560 QWidget* pppGroup4 = new QWidget();
561 QWidget* reqcgroup = new QWidget();
562 QWidget* cmbgroup = new QWidget();
563 QWidget* uploadgroup = new QWidget();
564 QWidget* uploadEphgroup = new QWidget();
565 _aogroup->addTab(pgroup,tr("Network"));
566 _aogroup->addTab(ggroup,tr("General"));
567 _aogroup->addTab(ogroup,tr("RINEX Observations"));
568 _aogroup->addTab(egroup,tr("RINEX Ephemeris"));
569 _aogroup->addTab(reqcgroup,tr("RINEX Editing && QC"));
570 _aogroup->addTab(cgroup,tr("Broadcast Corrections"));
571 _aogroup->addTab(sgroup,tr("Feed Engine"));
572 _aogroup->addTab(sergroup,tr("Serial Output"));
573 _aogroup->addTab(agroup,tr("Outages"));
574 _aogroup->addTab(rgroup,tr("Miscellaneous"));
575 _aogroup->addTab(pppGroup1,tr("PPP (1)"));
576 _aogroup->addTab(pppGroup2,tr("PPP (2)"));
577 _aogroup->addTab(pppGroup3,tr("PPP (3)"));
578 _aogroup->addTab(pppGroup4,tr("PPP (4)"));
579
580#ifdef USE_COMBINATION
581 _aogroup->addTab(cmbgroup,tr("Combine Corrections"));
582#endif
583 _aogroup->addTab(uploadgroup,tr("Upload Corrections"));
584 _aogroup->addTab(uploadEphgroup,tr("Upload Ephemeris"));
585
586 // Log Tab
587 // -------
588 _loggroup = new QTabWidget();
589 _loggroup->addTab(_log,tr("Log"));
590 _loggroup->addTab(_bncFigure,tr("Throughput"));
591 _loggroup->addTab(_bncFigureLate,tr("Latency"));
592 _loggroup->addTab(_bncFigurePPP,tr("PPP Plot"));
593
594 // Netowork (Proxy and SSL) Tab
595 // ----------------------------
596 QGridLayout* pLayout = new QGridLayout;
597 pLayout->setColumnMinimumWidth(0,13*ww);
598 _proxyPortLineEdit->setMaximumWidth(9*ww);
599
600 pLayout->addWidget(new QLabel("Settings for proxy in protected networks and for SSL authorization, leave boxes blank if none."),0, 0, 1, 50);
601 pLayout->addWidget(new QLabel("Proxy host"), 1, 0);
602 pLayout->addWidget(_proxyHostLineEdit, 1, 1, 1,10);
603 pLayout->addWidget(new QLabel("Proxy port"), 2, 0);
604 pLayout->addWidget(_proxyPortLineEdit, 2, 1);
605 pLayout->addWidget(new QLabel("Path to SSL Certificates"), 3, 0);
606 pLayout->addWidget(_sslCaCertPathLineEdit, 3, 1, 1,10);
607 pLayout->addWidget(new QLabel("Default: " + bncSslConfig::defaultPath()), 3,11, 1,20);
608 pLayout->addWidget(new QLabel("Ignore SSL Authorization Errors"), 4, 0);
609 pLayout->addWidget(_ignoreSslErrorsCheckBox, 4, 1, 1,10);
610 pLayout->addWidget(new QLabel(" "), 4, 0);
611 pLayout->addWidget(new QLabel(" "), 5, 0);
612 pLayout->addWidget(new QLabel(" "), 6, 0);
613 pgroup->setLayout(pLayout);
614
615 // General Tab
616 // -----------
617 QGridLayout* gLayout = new QGridLayout;
618 gLayout->setColumnMinimumWidth(0,14*ww);
619 _onTheFlyComboBox->setMaximumWidth(9*ww);
620
621 gLayout->addWidget(new QLabel("General settings for logfile, file handling, configuration on-the-fly, and auto-start."),0, 0, 1, 50);
622 gLayout->addWidget(new QLabel("Logfile (full path)"), 1, 0);
623 gLayout->addWidget(_logFileLineEdit, 1, 1, 1,30);
624 gLayout->addWidget(new QLabel("Append files"), 2, 0);
625 gLayout->addWidget(_rnxAppendCheckBox, 2, 1);
626 gLayout->addWidget(new QLabel("Reread configuration"), 3, 0);
627 gLayout->addWidget(_onTheFlyComboBox, 3, 1);
628 gLayout->addWidget(new QLabel("Auto start"), 4, 0);
629 gLayout->addWidget(_autoStartCheckBox, 4, 1);
630 gLayout->addWidget(new QLabel("Raw output file (full path)"), 5, 0);
631 gLayout->addWidget(_rawOutFileLineEdit, 5, 1, 1,30);
632 gLayout->addWidget(new QLabel(" "), 6, 0);
633 ggroup->setLayout(gLayout);
634
635 // RINEX Observations
636 // ------------------
637 QGridLayout* oLayout = new QGridLayout;
638 oLayout->setColumnMinimumWidth(0,14*ww);
639 _rnxIntrComboBox->setMaximumWidth(9*ww);
640 _rnxSamplSpinBox->setMaximumWidth(9*ww);
641
642 oLayout->addWidget(new QLabel("Saving RINEX observation files."),0, 0, 1,50);
643 oLayout->addWidget(new QLabel("Directory"), 1, 0);
644 oLayout->addWidget(_rnxPathLineEdit, 1, 1, 1,24);
645 oLayout->addWidget(new QLabel("Interval"), 2, 0);
646 oLayout->addWidget(_rnxIntrComboBox, 2, 1);
647 oLayout->addWidget(new QLabel(" Sampling"), 2, 2, Qt::AlignRight);
648 oLayout->addWidget(_rnxSamplSpinBox, 2, 3, Qt::AlignLeft);
649 oLayout->addWidget(new QLabel("Skeleton extension"), 3, 0);
650 oLayout->addWidget(_rnxSkelLineEdit, 3, 1, 1, 1, Qt::AlignLeft);
651 oLayout->addWidget(new QLabel("Script (full path)"), 4, 0);
652 oLayout->addWidget(_rnxScrpLineEdit, 4, 1, 1,24);
653 oLayout->addWidget(new QLabel("Version 3"), 5, 0);
654 oLayout->addWidget(_rnxV3CheckBox, 5, 1);
655 oLayout->addWidget(new QLabel(" "), 6, 0);
656 ogroup->setLayout(oLayout);
657
658 // RINEX Ephemeris
659 // ---------------
660 QGridLayout* eLayout = new QGridLayout;
661 eLayout->setColumnMinimumWidth(0,14*ww);
662 _ephIntrComboBox->setMaximumWidth(9*ww);
663 _outEphPortLineEdit->setMaximumWidth(9*ww);
664
665 eLayout->addWidget(new QLabel("Saving RINEX ephemeris files and ephemeris output through IP port."),0,0,1,50);
666 eLayout->addWidget(new QLabel("Directory"), 1, 0);
667 eLayout->addWidget(_ephPathLineEdit, 1, 1, 1,30);
668 eLayout->addWidget(new QLabel("Interval"), 2, 0);
669 eLayout->addWidget(_ephIntrComboBox, 2, 1);
670 eLayout->addWidget(new QLabel("Port"), 3, 0);
671 eLayout->addWidget(_outEphPortLineEdit, 3, 1);
672 eLayout->addWidget(new QLabel("Version 3"), 4, 0);
673 eLayout->addWidget(_ephV3CheckBox, 4, 1);
674 eLayout->addWidget(new QLabel(" "), 5, 0);
675 eLayout->addWidget(new QLabel(" "), 6, 0);
676 egroup->setLayout(eLayout);
677
678
679 // Broadcast Corrections
680 // ---------------------
681 QGridLayout* cLayout = new QGridLayout;
682 cLayout->setColumnMinimumWidth(0,14*ww);
683 _corrIntrComboBox->setMaximumWidth(9*ww);
684 _corrPortLineEdit->setMaximumWidth(9*ww);
685 _corrTimeSpinBox->setMaximumWidth(9*ww);
686
687 cLayout->addWidget(new QLabel("Saving Broadcast Ephemeris correction files and correction output through IP port."),0,0,1,50);
688 cLayout->addWidget(new QLabel("Directory, ASCII"), 1, 0);
689 cLayout->addWidget(_corrPathLineEdit, 1, 1, 1,20);
690 cLayout->addWidget(new QLabel("Interval"), 2, 0);
691 cLayout->addWidget(_corrIntrComboBox, 2, 1);
692 cLayout->addWidget(new QLabel("Port"), 3, 0);
693 cLayout->addWidget(_corrPortLineEdit, 3, 1);
694 cLayout->addWidget(new QLabel(" Wait for full corr epoch"), 3, 2, Qt::AlignRight);
695 cLayout->addWidget(_corrTimeSpinBox, 3, 3, Qt::AlignLeft);
696 cLayout->addWidget(new QLabel(" "), 4, 0);
697 cLayout->addWidget(new QLabel(" "), 5, 0);
698 cLayout->addWidget(new QLabel(" "), 6, 0);
699 cgroup->setLayout(cLayout);
700
701 // Feed Engine
702 // -----------
703 QGridLayout* sLayout = new QGridLayout;
704 sLayout->setColumnMinimumWidth(0,14*ww);
705 _outPortLineEdit->setMaximumWidth(9*ww);
706 _waitTimeSpinBox->setMaximumWidth(9*ww);
707 _binSamplSpinBox->setMaximumWidth(9*ww);
708 _outUPortLineEdit->setMaximumWidth(9*ww);
709
710 sLayout->addWidget(new QLabel("Output decoded observations in ASCII format to feed a real-time GNSS network engine."),0,0,1,50);
711 sLayout->addWidget(new QLabel("Port"), 1, 0);
712 sLayout->addWidget(_outPortLineEdit, 1, 1);
713 sLayout->addWidget(new QLabel("Wait for full obs epoch"), 1, 2, Qt::AlignRight);
714 sLayout->addWidget(_waitTimeSpinBox, 1, 3, Qt::AlignLeft);
715 sLayout->addWidget(new QLabel("Sampling"), 2, 0);
716 sLayout->addWidget(_binSamplSpinBox, 2, 1, Qt::AlignLeft);
717 sLayout->addWidget(new QLabel("File (full path)"), 3, 0);
718 sLayout->addWidget(_outFileLineEdit, 3, 1, 1, 20);
719 sLayout->addWidget(new QLabel("Port (unsynchronized)"), 4, 0);
720 sLayout->addWidget(_outUPortLineEdit, 4, 1);
721 sLayout->addWidget(new QLabel(" "), 5, 0);
722 sLayout->addWidget(new QLabel(" "), 6, 0);
723 sgroup->setLayout(sLayout);
724
725 // Serial Output
726 // -------------
727 QGridLayout* serLayout = new QGridLayout;
728 serLayout->setColumnMinimumWidth(0,14*ww);
729 _serialBaudRateComboBox->setMaximumWidth(9*ww);
730 _serialFlowControlComboBox->setMaximumWidth(11*ww);
731 _serialDataBitsComboBox->setMaximumWidth(5*ww);
732 _serialParityComboBox->setMaximumWidth(9*ww);
733 _serialStopBitsComboBox->setMaximumWidth(5*ww);
734 _serialAutoNMEAComboBox->setMaximumWidth(9*ww);
735 _serialHeightNMEALineEdit->setMaximumWidth(8*ww);
736
737 serLayout->addWidget(new QLabel("Port settings to feed a serial connected receiver."),0,0,1,30);
738 serLayout->addWidget(new QLabel("Mountpoint"), 1, 0, Qt::AlignLeft);
739 serLayout->addWidget(_serialMountPointLineEdit, 1, 1, 1, 2);
740 serLayout->addWidget(new QLabel("Port name"), 2, 0, Qt::AlignLeft);
741 serLayout->addWidget(_serialPortNameLineEdit, 2, 1, 1, 2);
742 serLayout->addWidget(new QLabel("Baud rate"), 3, 0, Qt::AlignLeft);
743 serLayout->addWidget(_serialBaudRateComboBox, 3, 1);
744 serLayout->addWidget(new QLabel("Flow control"), 3, 2, Qt::AlignRight);
745 serLayout->addWidget(_serialFlowControlComboBox, 3, 3);
746 serLayout->addWidget(new QLabel("Data bits"), 4, 0, Qt::AlignLeft);
747 serLayout->addWidget(_serialDataBitsComboBox, 4, 1);
748 serLayout->addWidget(new QLabel("Parity"), 4, 2, Qt::AlignRight);
749 serLayout->addWidget(_serialParityComboBox, 4, 3);
750 serLayout->addWidget(new QLabel(" Stop bits"), 4, 4, Qt::AlignRight);
751 serLayout->addWidget(_serialStopBitsComboBox, 4, 5);
752 serLayout->addWidget(new QLabel("NMEA"), 5, 0);
753 serLayout->addWidget(_serialAutoNMEAComboBox, 5, 1);
754 serLayout->addWidget(new QLabel(" File (full path)"), 5, 2, Qt::AlignRight);
755 serLayout->addWidget(_serialFileNMEALineEdit, 5, 3, 1,15);
756 serLayout->addWidget(new QLabel("Height"), 5,20, Qt::AlignRight);
757 serLayout->addWidget(_serialHeightNMEALineEdit, 5,21, 1,11);
758 serLayout->addWidget(new QLabel(" "), 6, 0);
759
760 sergroup->setLayout(serLayout);
761
762 // Outages
763 // -------
764 QGridLayout* aLayout = new QGridLayout;
765 aLayout->setColumnMinimumWidth(0,14*ww);
766 _obsRateComboBox->setMaximumWidth(9*ww);
767 _adviseFailSpinBox->setMaximumWidth(9*ww);
768 _adviseRecoSpinBox->setMaximumWidth(9*ww);
769
770 aLayout->addWidget(new QLabel("Failure and recovery reports, advisory notes."),0,0,1,50,Qt::AlignLeft);
771 aLayout->addWidget(new QLabel("Observation rate"), 1, 0);
772 aLayout->addWidget(_obsRateComboBox, 1, 1);
773 aLayout->addWidget(new QLabel("Failure threshold"), 2, 0);
774 aLayout->addWidget(_adviseFailSpinBox, 2, 1);
775 aLayout->addWidget(new QLabel("Recovery threshold"), 3, 0);
776 aLayout->addWidget(_adviseRecoSpinBox, 3, 1);
777 aLayout->addWidget(new QLabel("Script (full path)"), 4, 0);
778 aLayout->addWidget(_adviseScriptLineEdit, 4, 1, 1,30);
779 aLayout->addWidget(new QLabel(" "), 5, 0);
780 aLayout->addWidget(new QLabel(" "), 6, 0);
781 agroup->setLayout(aLayout);
782
783 // Miscellaneous
784 // -------------
785 QGridLayout* rLayout = new QGridLayout;
786 rLayout->setColumnMinimumWidth(0,14*ww);
787 _perfIntrComboBox->setMaximumWidth(9*ww);
788 _miscPortLineEdit->setMaximumWidth(9*ww);
789
790 rLayout->addWidget(new QLabel("Log latencies or scan RTCM streams for message types and antenna information or output raw data through TCP/IP port."),0, 0,1,30);
791 rLayout->addWidget(new QLabel("Mountpoint"), 1, 0);
792 rLayout->addWidget(_miscMountLineEdit, 1, 1, 1,7);
793 rLayout->addWidget(new QLabel("Log latency"), 2, 0);
794 rLayout->addWidget(_perfIntrComboBox, 2, 1);
795 rLayout->addWidget(new QLabel("Scan RTCM"), 3, 0);
796 rLayout->addWidget(_scanRTCMCheckBox, 3, 1);
797 rLayout->addWidget(new QLabel("Port"), 4, 0);
798 rLayout->addWidget(_miscPortLineEdit, 4, 1);
799 rLayout->addWidget(new QLabel(" "), 5, 0);
800 rLayout->addWidget(new QLabel(" "), 6, 0);
801 rgroup->setLayout(rLayout);
802
803 // PPP
804 // ---
805 QGridLayout* pppLayout1 = new QGridLayout();
806 int ir = 0;
807 pppLayout1->addWidget(new QLabel("<b>Precise Point Positioning (Input and Output)</b>"), ir, 0, 1, 7, Qt::AlignLeft);
808 ++ir;
809 pppLayout1->addWidget(new QLabel("Data Source"), ir, 0, Qt::AlignLeft);
810 pppLayout1->addWidget(_pppWidgets._dataSource, ir, 1);
811 pppLayout1->addItem(new QSpacerItem(4*ww, 0), ir, 2);
812 pppLayout1->addWidget(new QLabel("RINEX Observations"), ir, 3, Qt::AlignLeft);
813 pppLayout1->addWidget(_pppWidgets._rinexObs, ir, 4, 1, 2);
814 ++ir;
815 pppLayout1->addWidget(new QLabel("RINEX Orbits"), ir, 3, Qt::AlignLeft);
816 pppLayout1->addWidget(_pppWidgets._rinexNav, ir, 4, 1, 2);
817 ++ir;
818 pppLayout1->addWidget(new QLabel("Corrections"), ir, 0, Qt::AlignLeft);
819 pppLayout1->addWidget(_pppWidgets._corrMount, ir, 1);
820 pppLayout1->addWidget(new QLabel("Corrections"), ir, 3, Qt::AlignLeft);
821 pppLayout1->addWidget(_pppWidgets._corrFile, ir, 4, 1, 2);
822 ++ir;
823 pppLayout1->addWidget(new QLabel("<b>Input</b>"), ir, 0, Qt::AlignLeft);
824 pppLayout1->addWidget(new QLabel("<b>Output</b>"), ir, 4, 1, 2, Qt::AlignLeft);
825 ++ir;
826 pppLayout1->addWidget(new QLabel("Coordinates"), ir, 0, Qt::AlignLeft);
827 pppLayout1->addWidget(_pppWidgets._crdFile, ir, 1, 1, 2);
828 pppLayout1->addWidget(new QLabel("Log File"), ir, 4, Qt::AlignLeft);
829 pppLayout1->addWidget(_pppWidgets._logFile, ir, 5);
830 ++ir;
831 pppLayout1->addWidget(new QLabel("ANTEX"), ir, 0, Qt::AlignLeft);
832 pppLayout1->addWidget(_pppWidgets._antexFile, ir, 1, 1, 2);
833 pppLayout1->addWidget(new QLabel("NMEA File"), ir, 4, Qt::AlignLeft);
834 pppLayout1->addWidget(_pppWidgets._nmeaFile, ir, 5);
835 pppLayout1->addWidget(new QLabel("Port"), ir, 6, Qt::AlignLeft);
836 pppLayout1->addWidget(_pppWidgets._nmeaPort, ir, 7);
837
838 pppGroup1->setLayout(pppLayout1);
839
840 QVBoxLayout* pppLayout2 = new QVBoxLayout();
841 pppLayout2->addWidget(new QLabel("<b>Precise Point Positioning (Processed Stations)</b>"));
842 pppLayout2->addWidget(_pppWidgets._staTable, 99);
843 QHBoxLayout* pppLayout2sub = new QHBoxLayout();
844 pppLayout2sub->addWidget(_pppWidgets._addStaButton);
845 pppLayout2sub->addWidget(_pppWidgets._delStaButton);
846 pppLayout2sub->addStretch(99);
847
848 pppLayout2->addLayout(pppLayout2sub);
849
850 pppGroup2->setLayout(pppLayout2);
851
852 QGridLayout* pppLayout3 = new QGridLayout();
853 ir = 0;
854 pppLayout3->addWidget(new QLabel("<b>Precise Point Positioning (Options)</b>"), ir, 0, 1, 2, Qt::AlignLeft);
855 ++ir;
856 pppLayout3->addWidget(new QLabel("GPS LCs"), ir, 0, Qt::AlignLeft);
857 pppLayout3->addWidget(_pppWidgets._lcGPS, ir, 1);
858 pppLayout3->addItem(new QSpacerItem(4*ww, 0), ir, 2);
859 pppLayout3->addWidget(new QLabel("Sigma C1"), ir, 3, Qt::AlignLeft);
860 pppLayout3->addWidget(_pppWidgets._sigmaC1, ir, 4); _pppWidgets._sigmaC1->setMaximumWidth(8*ww);
861 pppLayout3->addWidget(new QLabel("Sigma L1"), ir, 5, Qt::AlignLeft);
862 pppLayout3->addWidget(_pppWidgets._sigmaL1, ir, 6); _pppWidgets._sigmaL1->setMaximumWidth(8*ww);
863 ++ir;
864 pppLayout3->addWidget(new QLabel("GLONASS LCs"), ir, 0, Qt::AlignLeft);
865 pppLayout3->addWidget(_pppWidgets._lcGLONASS, ir, 1);
866 ++ir;
867 pppLayout3->addWidget(new QLabel("Galileo LCs"), ir, 0, Qt::AlignLeft);
868 pppLayout3->addWidget(_pppWidgets._lcGalileo, ir, 1);
869 ++ir;
870 pppLayout3->addWidget(new QLabel("Wait for corrections"), ir, 0, Qt::AlignLeft);
871 pppLayout3->addWidget(_pppWidgets._corrWaitTime, ir, 1);
872 ++ir;
873 pppLayout3->addWidget(new QLabel(""), ir, 0);
874
875 pppGroup3->setLayout(pppLayout3);
876
877 // ------------------------
878 QVBoxLayout* pppLayout4 = new QVBoxLayout;
879 pppLayout4->addWidget(new QLabel("<b>Precise Point Positioning (Plots)</b>"));
880 pppLayout4->addSpacing(ww);
881
882 QHBoxLayout* pppLayout4Hlp1 = new QHBoxLayout;
883 pppLayout4Hlp1->addWidget(new QLabel("PPP Station "));
884 _pppWidgets._plotCoordinates->setMaximumWidth(8*ww);
885 pppLayout4Hlp1->addWidget(_pppWidgets._plotCoordinates);
886 pppLayout4Hlp1->addWidget(new QLabel("Nort-East-Up Time Series"));
887 pppLayout4Hlp1->addStretch();
888 pppLayout4->addLayout(pppLayout4Hlp1);
889 pppLayout4->addSpacing(ww);
890
891 QHBoxLayout* pppLayout4Hlp2 = new QHBoxLayout;
892 pppLayout4Hlp2->addWidget(new QLabel("Track Plot "));
893 connect(_pppWidgets._mapWinButton, SIGNAL(clicked()), SLOT(slotMapPPP()));
894 pppLayout4Hlp2->addWidget(_pppWidgets._mapWinButton);
895
896 pppLayout4Hlp2->addSpacing(1*ww);
897
898 pppLayout4Hlp2->addWidget(new QLabel("Google"));
899 pppLayout4Hlp2->addWidget(_pppWidgets._gmRadioButton);
900
901 pppLayout4Hlp2->addWidget(new QLabel("OSM"));
902 pppLayout4Hlp2->addWidget(_pppWidgets._osmRadioButton);
903
904 pppLayout4Hlp2->addSpacing(3*ww);
905
906 _pppWidgets._mapWinDotSizeLineEdit->setMaximumWidth(5*ww);
907 pppLayout4Hlp2->addWidget(_pppWidgets._mapWinDotSizeLineEdit);
908
909 pppLayout4Hlp2->addSpacing(3*ww);
910
911 pppLayout4Hlp2->addWidget(_pppWidgets._mapWinDotColorComboBox);
912
913 pppLayout4Hlp2->addSpacing(3*ww);
914
915 pppLayout4Hlp2->addWidget(new QLabel("Speed"));
916 pppLayout4Hlp2->addWidget(_pppWidgets._mapSpeedSlider);
917
918 pppLayout4Hlp2->addStretch();
919 pppLayout4->addLayout(pppLayout4Hlp2);
920
921 pppLayout4->addStretch();
922 pppGroup4->setLayout(pppLayout4);
923
924 // Reqc Processing
925 // ---------------
926 _reqcActionComboBox = new QComboBox();
927 _reqcActionComboBox->setEditable(false);
928 _reqcActionComboBox->addItems(QString(",Edit/Concatenate,Analyze").split(","));
929 int ik = _reqcActionComboBox->findText(settings.value("reqcAction").toString());
930 if (ik != -1) {
931 _reqcActionComboBox->setCurrentIndex(ik);
932 }
933 connect(_reqcActionComboBox, SIGNAL(currentIndexChanged(const QString &)),
934 this, SLOT(slotBncTextChanged()));
935
936 QGridLayout* reqcLayout = new QGridLayout;
937 _reqcActionComboBox->setMinimumWidth(15*ww);
938 _reqcActionComboBox->setMaximumWidth(15*ww);
939
940 _reqcObsFileChooser = new qtFileChooser(0, qtFileChooser::Files);
941 _reqcObsFileChooser->setFileName(settings.value("reqcObsFile").toString());
942 _reqcObsFileChooser->setWhatsThis(tr("Specify the full path to an observation file in RINEX v2 or v3 format."));
943 _reqcObsFileChooser->setMinimumWidth(15*ww);
944 _reqcObsFileChooser->setMaximumWidth(15*ww);
945
946 _reqcNavFileChooser = new qtFileChooser(0, qtFileChooser::Files);
947 _reqcNavFileChooser->setFileName(settings.value("reqcNavFile").toString());
948 _reqcNavFileChooser->setWhatsThis(tr("Specify the full path to a RINEX v2 or v3 navigation file."));
949 _reqcNavFileChooser->setMinimumWidth(15*ww);
950 _reqcNavFileChooser->setMaximumWidth(15*ww);
951
952 _reqcOutObsLineEdit = new QLineEdit(settings.value("reqcOutObsFile").toString());
953 _reqcOutObsLineEdit->setWhatsThis(tr("Specify the full path to a RINEX observation output file."));
954 _reqcOutObsLineEdit->setMinimumWidth(15*ww);
955 _reqcOutObsLineEdit->setMaximumWidth(15*ww);
956
957 _reqcOutNavLineEdit = new QLineEdit(settings.value("reqcOutNavFile").toString());
958 _reqcOutNavLineEdit->setWhatsThis(tr("Specify the full path to a RINEX navigation output file."));
959 _reqcOutNavLineEdit->setMinimumWidth(15*ww);
960 _reqcOutNavLineEdit->setMaximumWidth(15*ww);
961
962 _reqcOutLogLineEdit = new QLineEdit(settings.value("reqcOutLogFile").toString());
963 _reqcOutLogLineEdit->setWhatsThis(tr("Specify the full path to a logfile."));
964 _reqcOutLogLineEdit->setMinimumWidth(15*ww);
965 _reqcOutLogLineEdit->setMaximumWidth(15*ww);
966
967 _reqcPlotDirLineEdit = new QLineEdit(settings.value("reqcPlotDir").toString());
968 _reqcPlotDirLineEdit->setWhatsThis(tr("Specify the directory name for saving plots."));
969 _reqcPlotDirLineEdit->setMinimumWidth(15*ww);
970 _reqcPlotDirLineEdit->setMaximumWidth(15*ww);
971
972 _reqcSkyPlotSystems = new QComboBox();
973 _reqcSkyPlotSystems->setEditable(false);
974 _reqcSkyPlotSystems->addItems(QString("ALL,GPS,GLONASS,Galileo").split(","));
975 ik = _reqcSkyPlotSystems->findText(settings.value("reqcSkyPlotSystems").toString());
976 if (ik != -1) {
977 _reqcSkyPlotSystems->setCurrentIndex(ik);
978 }
979
980 ir = 0;
981 reqcLayout->addWidget(new QLabel("RINEX file editing, concatenation and quality check."),ir, 0, 1, 20);
982 ++ir;
983 reqcLayout->addWidget(new QLabel("Action"), ir, 0, Qt::AlignLeft);
984 reqcLayout->addWidget(_reqcActionComboBox, ir, 1, Qt::AlignLeft);
985 _reqcEditOptionButton = new QPushButton("Set Edit Options");
986 reqcLayout->addWidget(_reqcEditOptionButton, ir, 3, Qt::AlignRight);
987 ++ir;
988 reqcLayout->addWidget(new QLabel("Input files (full path)"), ir, 0, Qt::AlignLeft);
989 reqcLayout->addWidget(_reqcObsFileChooser, ir, 1, Qt::AlignRight);
990 reqcLayout->addWidget(new QLabel("Obs"), ir, 2, Qt::AlignLeft);
991 reqcLayout->addWidget(_reqcNavFileChooser, ir, 3, Qt::AlignRight);
992 reqcLayout->addWidget(new QLabel("Nav"), ir, 4, Qt::AlignLeft);
993 ++ir;
994 reqcLayout->addWidget(new QLabel("Output files (full path)"), ir, 0, Qt::AlignLeft);
995 reqcLayout->addWidget(_reqcOutObsLineEdit, ir, 1, Qt::AlignRight);
996 reqcLayout->addWidget(new QLabel("Obs"), ir, 2, Qt::AlignLeft);
997 reqcLayout->addWidget(_reqcOutNavLineEdit, ir, 3, Qt::AlignRight);
998 reqcLayout->addWidget(new QLabel("Nav"), ir, 4, Qt::AlignLeft);
999 ++ir;
1000 reqcLayout->addWidget(_reqcOutLogLineEdit, ir, 1, Qt::AlignRight);
1001 reqcLayout->addWidget(new QLabel("Log"), ir, 2, Qt::AlignLeft);
1002 ++ir;
1003 reqcLayout->addWidget(new QLabel("Directory for plots"), ir, 0, Qt::AlignLeft);
1004 reqcLayout->addWidget(_reqcPlotDirLineEdit, ir, 1, Qt::AlignRight);
1005 ++ir;
1006 reqcLayout->addWidget(new QLabel("Sky plots for"), ir, 0, Qt::AlignLeft);
1007 reqcLayout->addWidget(_reqcSkyPlotSystems, ir, 1, Qt::AlignRight);
1008 ++ir;
1009 reqcLayout->addWidget(new QLabel(""), ir, 1);
1010 reqcLayout->setRowStretch(ir, 999);
1011
1012 reqcLayout->setColumnMinimumWidth(2, 8*ww);
1013 reqcLayout->setColumnMinimumWidth(4, 8*ww);
1014
1015 reqcgroup->setLayout(reqcLayout);
1016
1017 connect(_reqcEditOptionButton, SIGNAL(clicked()),
1018 this, SLOT(slotReqcEditOption()));
1019
1020 // Combine Corrections
1021 // -------------------
1022 QGridLayout* cmbLayout = new QGridLayout;
1023
1024 populateCmbTable();
1025 cmbLayout->addWidget(_cmbTable, 0, 0, 6, 3);
1026 cmbLayout->addWidget(new QLabel(" "), 0, 5);
1027 cmbLayout->addWidget(new QLabel("Combine Broadcast Correction streams."), 0, 6, 1, 50);
1028 cmbLayout->addWidget(new QLabel(" "), 1, 5);
1029 cmbLayout->addWidget(addCmbRowButton, 1, 6);
1030 cmbLayout->addWidget(delCmbRowButton, 1, 7);
1031 cmbLayout->addWidget(new QLabel(" "), 2, 5);
1032 cmbLayout->addWidget(new QLabel("Method"), 2, 6, Qt::AlignRight);
1033 cmbLayout->addWidget(_cmbMethodComboBox, 2, 7, Qt::AlignRight);
1034 cmbLayout->addWidget(new QLabel(" "), 3, 5);
1035 cmbLayout->addWidget(new QLabel("Maximal Residuum"), 3, 6, Qt::AlignRight);
1036 cmbLayout->addWidget(_cmbMaxresLineEdit, 3, 7, Qt::AlignRight);
1037 cmbLayout->addWidget(new QLabel(" "), 4, 5);
1038 cmbLayout->addWidget(new QLabel("Sampling"), 4, 6, Qt::AlignRight);
1039 cmbLayout->addWidget(_cmbSamplSpinBox, 4, 7, Qt::AlignRight);
1040 cmbLayout->addWidget(new QLabel(" "), 5, 0);
1041
1042 connect(addCmbRowButton, SIGNAL(clicked()), this, SLOT(slotAddCmbRow()));
1043 connect(delCmbRowButton, SIGNAL(clicked()), this, SLOT(slotDelCmbRow()));
1044
1045 cmbgroup->setLayout(cmbLayout);
1046
1047 // Upload Layout (Clocks)
1048 // ----------------------
1049 QGridLayout* uploadHlpLayout = new QGridLayout();
1050
1051 connect(addUploadRowButton, SIGNAL(clicked()), this, SLOT(slotAddUploadRow()));
1052 connect(delUploadRowButton, SIGNAL(clicked()), this, SLOT(slotDelUploadRow()));
1053 connect(setUploadTrafoButton, SIGNAL(clicked()), this, SLOT(slotSetUploadTrafo()));
1054
1055 uploadHlpLayout->addWidget(addUploadRowButton, 0, 0);
1056 uploadHlpLayout->addWidget(delUploadRowButton, 0, 1);
1057 uploadHlpLayout->addWidget(new QLabel("Interval"), 0, 2, Qt::AlignRight);
1058 uploadHlpLayout->addWidget(_uploadIntrComboBox, 0, 3);
1059 uploadHlpLayout->addWidget(new QLabel(" Sampling: Orb"), 0, 4, Qt::AlignRight);
1060 uploadHlpLayout->addWidget(_uploadSamplRtcmEphCorrSpinBox, 0, 5);
1061 uploadHlpLayout->addWidget(new QLabel("SP3"), 0, 6, Qt::AlignRight);
1062 uploadHlpLayout->addWidget(_uploadSamplSp3SpinBox, 0, 7);
1063 uploadHlpLayout->addWidget(new QLabel("RNX"), 0, 8, Qt::AlignRight);
1064 uploadHlpLayout->addWidget(_uploadSamplClkRnxSpinBox, 0, 9);
1065 uploadHlpLayout->addWidget(setUploadTrafoButton, 0,10);
1066
1067 QBoxLayout* uploadLayout = new QBoxLayout(QBoxLayout::TopToBottom);
1068 populateUploadTable();
1069
1070 uploadLayout->addWidget(new QLabel("Upload RTCMv3 Broadcast Corrections to caster."));
1071 uploadLayout->addWidget(_uploadTable);
1072 uploadLayout->addLayout(uploadHlpLayout);
1073
1074 uploadgroup->setLayout(uploadLayout);
1075
1076 // Upload Layout (Ephemeris)
1077 // -------------------------
1078 QGridLayout* uploadLayoutEph = new QGridLayout;
1079
1080 uploadLayoutEph->setColumnMinimumWidth(0, 9*ww);
1081 _uploadEphPortLineEdit->setMaximumWidth(9*ww);
1082 _uploadEphPasswordLineEdit->setMaximumWidth(9*ww);
1083 _uploadEphMountpointLineEdit->setMaximumWidth(12*ww);
1084
1085 uploadLayoutEph->addWidget(new QLabel("Upload concatenated RTCMv3 Broadcast Ephemeris to caster."), 0, 0, 1, 50);
1086 uploadLayoutEph->addWidget(new QLabel("Host"), 1, 0);
1087 uploadLayoutEph->addWidget(_uploadEphHostLineEdit, 1, 1, 1, 3);
1088 uploadLayoutEph->addWidget(new QLabel(" Port"), 1, 4, Qt::AlignRight);
1089 uploadLayoutEph->addWidget(_uploadEphPortLineEdit, 1, 5, 1, 1);
1090 uploadLayoutEph->addWidget(new QLabel("Mountpoint "), 2, 0);
1091 uploadLayoutEph->addWidget(_uploadEphMountpointLineEdit, 2, 1);
1092 uploadLayoutEph->addWidget(new QLabel(" Password"), 2, 2, Qt::AlignRight);
1093 uploadLayoutEph->addWidget(_uploadEphPasswordLineEdit, 2, 3);
1094 uploadLayoutEph->addWidget(new QLabel("Sampling"), 3, 0);
1095 uploadLayoutEph->addWidget(_uploadEphSampleSpinBox, 3, 1);
1096 uploadLayoutEph->addWidget(new QLabel("Uploaded"), 4, 0);
1097 uploadLayoutEph->addWidget(_uploadEphBytesCounter, 4, 1);
1098 uploadLayoutEph->addWidget(new QLabel(" "), 5, 0);
1099 uploadLayoutEph->addWidget(new QLabel(" "), 6, 0);
1100
1101 uploadEphgroup->setLayout(uploadLayoutEph);
1102
1103 connect(_uploadEphHostLineEdit, SIGNAL(textChanged(const QString &)),
1104 this, SLOT(slotBncTextChanged()));
1105
1106 // Main Layout
1107 // -----------
1108 QGridLayout* mLayout = new QGridLayout;
1109 _aogroup->setCurrentIndex(settings.value("startTab").toInt());
1110 mLayout->addWidget(_aogroup, 0,0);
1111 mLayout->addWidget(_mountPointsTable, 1,0);
1112 _loggroup->setCurrentIndex(settings.value("statusTab").toInt());
1113 mLayout->addWidget(_loggroup, 2,0);
1114
1115 _canvas->setLayout(mLayout);
1116
1117 // WhatsThis
1118 // ---------
1119 _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>"));
1120 _proxyPortLineEdit->setWhatsThis(tr("<p>Enter your proxy server port number in case a proxy is operated in front of BNC.</p>"));
1121 _sslCaCertPathLineEdit->setWhatsThis(tr("<p>Communication with an NTRIP broadcaster over SSL requires the exchange of client and/or server certificates. Specify the path to a directory where you save certificates on your system. Don't try communication via SSL if you are not sure wheter this is supported by the involved NTRIP broadcaster. Note that SSL communication is usually done over port 443.</p>"));
1122 _ignoreSslErrorsCheckBox->setWhatsThis(tr("<p>SSL communication may involve queries coming from the NTRIP broadcaster. Tick 'Ignore SSL authorization erros' if you don't want to be bothered with this.</p>"));
1123 _waitTimeSpinBox->setWhatsThis(tr("<p>When feeding a real-time GNSS network engine waiting for synchronized input epoch by epoch, BNC drops whatever is received later than 'Wait for full obs 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 network engine or products.</p>"));
1124 _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."));
1125 _outPortLineEdit->setWhatsThis(tr("BNC can produce synchronized observations in a plain ASCII format on your local host through an IP port. Specify a port number here to activate this function."));
1126 _outUPortLineEdit->setWhatsThis(tr("BNC can produce unsynchronized observations in a plain ASCII format on your local host through an IP port. Specify a port number here to activate this function."));
1127 _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."));
1128 _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."));
1129 _corrTimeSpinBox->setWhatsThis(tr("<p>Concerning output through IP port, BNC drops Broadcast Ephemeris Corrections received later than 'Wait for full corr 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.</p><p>Specifying a value of '0' means that BNC immediately outputs all incoming Broadcast Epemeris Corrections and does not drop any of them for latency reasons.</p>"));
1130 _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."));
1131 _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."));
1132 _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."));
1133 _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>"));
1134 _rnxSkelLineEdit->setWhatsThis(tr("<p>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.</p><p>A file in the RINEX Observations 'Directory' with a 'Skeleton extension' suffix is interpreted by BNC as a personal RINEX header skeleton file for the corresponding stream.</p>"));
1135 _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>"));
1136 _autoStartCheckBox->setWhatsThis(tr("<p>Tick 'Auto start' for auto-start of BNC at startup time in window mode with preassigned processing options.</p>"));
1137 _rawOutFileLineEdit->setWhatsThis(tr("<p>Save all data coming in through various streams in the received order and format in one file.</p>"));
1138 _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', '5 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 & Reread Configuration'.</p>"));
1139 _rnxIntrComboBox->setWhatsThis(tr("<p>Select the length of the RINEX Observation file.</p>"));
1140 _ephIntrComboBox->setWhatsThis(tr("<p>Select the length of the RINEX Navigation file.</p>"));
1141 _corrIntrComboBox->setWhatsThis(tr("<p>Select the length of the Broadcast Ephemeris Correction files.</p>"));
1142 _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>"));
1143 _binSamplSpinBox->setWhatsThis(tr("<p>Select the synchronized observation sampling interval in seconds. A value of zero '0' tells BNC to send/store all received epochs.</p>"));
1144 _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>"));
1145 _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>"));
1146 _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>"));
1147 _logFileLineEdit->setWhatsThis(tr("<p>Records of BNC's activities are shown in the 'Log' tab 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.</p><p>The logfile name will automatically be extended by a string '_YYMMDD' carrying the current date."));
1148 _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 may have the task to send the advisory notes by 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>"));
1149 _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' tab 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>"));
1150 _mountPointsTable->setWhatsThis(tr("<p>Streams selected for retrieval are listed in the 'Streams' section. Clicking on 'Add Stream' 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 'Streams' list, highlight it by clicking on it and hit the 'Delete Stream' button. You can also remove multiple streams 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' as given in the sourcetable. BNC allows users to change this selection 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 'RTNET'.</p><p>In case you need to log the raw data as is, BNC allows users to by-pass its decoders 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). VRS streams are indicated by a 'yes' in the 'nmea' column. To initiate these streams, the approximate latitude/longitude rover position is sent to the NTRIP broadcaster. The default values can be change according to your requirement. Double click on 'lat' and 'long' fields, enter the values you wish to send and then hit Enter.</p>"));
1151 _log->setWhatsThis(tr("Records of BNC's activities are shown in the 'Log' tab. 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."));
1152 _bncFigure->setWhatsThis(tr("The bandwidth consumtion per stream is shown in the 'Throughput' tab in bits per second (bps) or kilo bits per second (kbps)."));
1153 _bncFigureLate->setWhatsThis(tr("The individual latency of observations in each incoming stream is shown in the 'Latency' tab. Streams not carrying observations (i.e. those providing only broadcast ephemeris messages) are not considered here. Note that the calculation of correct latencies requires the clock of the host computer to be properly synchronized."));
1154 _ephV3CheckBox->setWhatsThis(tr("The default format for output of RINEX Navigation data containing Broadcast Ephemeris is RINEX Version 2. Select 'Version 3' if you want to output the ephemeris in RINEX Version 3 format."));
1155 _rnxV3CheckBox->setWhatsThis(tr("The default format for RINEX Observation files is RINEX Version 2. Select 'Version 3' if you want to save the observations in RINEX Version 3 format."));
1156 _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>"));
1157 _miscPortLineEdit->setWhatsThis(tr("BNC can output an incoming stream through a TCP/IP port of your local host. Specify a port number here to activate this function."));
1158 _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>In case of RTCM Version 3 MSM streams, BNC will also log contained RINEX Version 3 observation types.</p>."));
1159 _serialMountPointLineEdit->setWhatsThis(tr("<p>Enter a 'Mountpoint' to forward the corresponding stream to a serial connected receiver.</p>"));
1160 _serialPortNameLineEdit->setWhatsThis(tr("<p>Enter the serial 'Port name' selected for communication with your serial connected receiver. 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><p>Note that you must plug a serial cable in the port defined here before you start BNC.</p>"));
1161 _serialBaudRateComboBox->setWhatsThis(tr("<p>Select a 'Baud rate' for the serial output link.</p><p>Note that your selection must equal the baud rate configured to the serial connected receiver. Note further that using a high baud rate is recommended.</p>"));
1162 _serialParityComboBox->setWhatsThis(tr("<p>Select the 'Parity' for the serial output link.</p><p>Note that your selection must equal the parity selection configured to the serial connected receiver. Note further that parity is often set to 'NONE'.</p>"));
1163 _serialDataBitsComboBox->setWhatsThis(tr("<p>Select the number of 'Data bits' for the serial output link.</p><p>Note that your selection must equal the number of data bits configured to the serial connected receiver. Note further that often 8 data bits are used.</p>"));
1164 _serialStopBitsComboBox->setWhatsThis(tr("<p>Select the number of 'Stop bits' for the serial output link.</p><p>Note that your selection must equal the number of stop bits configured to the serial connected receiver. Note further that often 1 stop bit is used.</p>"));
1165 _serialFlowControlComboBox->setWhatsThis(tr("<p>Select a 'Flow control' for the serial output link.</p><p>Note that your selection must equal the flow control configured to the serial connected receiver. Select 'OFF' if you don't know better.</p>"));
1166 _serialAutoNMEAComboBox->setWhatsThis(tr("<p>Select 'Auto' to automatically forward NMEA-GGA messages coming from your serial connected receiver to the NTRIP broadcaster and/or save them in a file.</p><p>Select 'Manual' only when handling a VRS stream and your serial connected receiver doesn't generate NMEA-GGA messages.</p>"));
1167 _serialFileNMEALineEdit->setWhatsThis(tr("<p>Specify the full path to a file where NMEA messages coming from your serial connected receiver are saved.</p>"));
1168 _serialHeightNMEALineEdit->setWhatsThis(tr("<p>Specify an approximate 'Height' above mean sea level in meter for your VRS to simulate an inital NMEA-GGA message.</p><p>The setting of this option is ignored in case of streams coming from physical reference stations.</p>"));
1169 _reqcActionComboBox->setWhatsThis(tr("<p>BNC allows to edit or concatenate RINEX v2 or v3 files or to perform a quality check following UNAVCO's famous 'teqc' program.</p>"));
1170 _reqcEditOptionButton->setWhatsThis(tr("<p>Specify options for editing RINEX v2 or v3 files.</p>"));
1171 _bncFigurePPP->setWhatsThis(tr("PPP time series of North (red), East (green) and Up (blue) coordinate components are shown in the 'PPP Plot' tab when the corresponting option is selected above. Values are either referred to an XYZ reference coordinate (if specified) or referred to the first estimated set of coordinate compoments. The sliding PPP time series window covers the period of the latest 5 minutes."));
1172 _cmbTable->setWhatsThis(tr("<p>BNC allows to process several orbit and clock corrections streams in real-time to produce, encode, upload and save a combination of correctors coming from various providers. Hit the 'Add Row' button, double click on the 'Mountpoint' field to enter a Broadcast Ephemeris corrections mountpoint from the 'Streams' section below and hit Enter. Then double click on the 'AC Name' field to enter your choice of an abbreviation for the Analysis Center (AC) providing the stream. Finally, double click on the 'Weight' field to enter the weight to be applied for this stream in the combination.<ul><li>Note that an appropriate 'Wait for full corr epoch' value needs to be specified for the combination under the 'Broadcast Corrections' tab. A value of 15 seconds would make sense there if the update rate of incoming clock corrections is i.e. 10 seconds.</li><li>Note also that you need to tick 'Use GLONASS' which is part ot the 'PPP (2)' panel in case you want to produce an GPS plus GLONASS combination.</li></ul></p><p>Note further that the orbit information in the final combination stream is just copied from one of the incoming streams. The stream used for providing the orbits may vary over time: if the orbit providing stream has an outage then BNC switches to the next remaining stream for getting hold of the orbit information.</p><p>The combination process requires Broadcast Ephemeris. Besides the orbit and clock corrections stream(s) BNC should therefore pull a stream carrying Broadcast Ephemeris in the form of RTCM Version 3 messages.</p><p>It is possible to specify only one Broadcast Ephemeris corrections stream in the combination table. Instead of combining corrections BNC will then merge them with Broadcast Ephemeris to save results in SP3 and/or Clock RINEX format."));
1173 _cmbMaxresLineEdit->setWhatsThis(tr("<p>BNC combines all incoming clocks according to specified weights. Individual clock estimates that differ by more than 'Maximal Residuum' meters from the average of all clocks will be ignored.<p></p>It is suggested to specify a value of about 0.2 m for the Kalman filter combination approach and a value of about 3.0 meters for the Single-Epoch combination approach.</p><p>Default is a value of '999.0'.</p>"));
1174 _cmbSamplSpinBox->setWhatsThis(tr("<p>Specify a combination sampling interval. Clock and orbit corrections will be produced following that interval. A value of 10 sec may be an appropriate choice.</p>"));
1175 _cmbMethodComboBox->setWhatsThis(tr("<p>Select a clock combination approach. Options are 'Single-Epoch' and Kalman 'Filter'. It is suggested to use the Kalman filter approach for the purpose of Precise Point Positioning.</p>"));
1176 _uploadTable->setWhatsThis(tr("<p>BNC can upload clock and orbit corrections to broadcast ephemeris (Broadcast Corrections) in RTCM Version 3 SSR format. You may have a situation where clocks and orbits come from an external Real-time Network Engine (1) or a situation where clock and orbit corrections are combined within BNC (2).</p><p>(1) BNC identifies a stream as coming from a Real-time Network Engine if its format is specified as 'RTNET' and hence its decoder string in the 'Streams' canvas is 'RTNET'. It encodes and uploads that stream to the specified NTRIP broadcaster</p><p>(2) BNC understands that it is expected to encode and upload combined Broadcast Ephemeris corrections if you specify correction streams in the 'Combine Corrections' stream table.</p><p>Hit the 'Add Row' button, double click on the 'Host' field to enter the IP or URL of an NTRIP broadcaster and hit Enter. Then double click on the 'Port', 'Mount' and 'Password' fields to enter the NTRIP broadcaster IP port (default is 80), the mountpoint and the stream upload password. An empty 'Host' option field means that you don't want to upload corrections.</p><p>Select a target coordinate reference system (e.g. IGS08) for outgoing clock and orbit corrections.</p><p>By default orbit and clock corrections refer to Antenna Phase Center (APC). Tick 'CoM' to refer uploaded corrections to Center of Mass instead of APC.</p><p>Specify a path for saving the generated Broadcast Corrections plus Broadcast Ephemeris as SP3 orbit files. If the specified directory does not exist, BNC will not create SP3 orbit files. The following is a path example for a Linux system:<br>/home/user/BNC${GPSWD}.sp3<br>Note that '${GPSWD}' produces the GPS Week and Day number in the file name.</p><ul><li>As an SP3 file contents should be referred to the satellites Center of Mass (CoM) while correctors are referred to the satellites Antenna Phase Center (APC), an offset has to be applied which is available from an IGS ANTEX file. You should therefore specify the 'ANTEX File' path under tab 'PPP (2)' if you want to save the stream contents in SP3 format. If you don't specify an 'ANTEX File' path there, the SP3 file contents will be referred to the satellites APCs.</li></ul></p><p>Specify a path for saving the generated Broadcast Correction clocks plus Broadcast Ephemeris clocks as Clock RINEX files. If the specified directory does not exist, BNC will not create Clock RINEX files. The following is a path example for a Linux system:<br>/home/user/BNC${GPSWD}.clk<br>Note that '${GPSWD}' produces the GPS Week and Day number in the file name.</p><p>Specify finally an SSR Provider ID number, an SSR Solution ID number and an Issue of Data SSR number.</p><p>In case the 'Combine Corrections' table contains only one Broadcast Corrections stream, BNC will merge that stream with Broadcast Ephemeris to save results in files specified here through SP3 and/or Clock RINEX file path. In such a case you should define only the SP3 and Clock RINEX file path and no further options in the 'Upload Corrections' table.</p>"));
1177 addCmbRowButton->setWhatsThis(tr("Hit 'Add Row' button to add another line to the mountpoints table."));
1178 delCmbRowButton->setWhatsThis(tr("Hit 'Delete' button to delete the highlighted line from the mountpoints table."));
1179 addUploadRowButton->setWhatsThis(tr("Hit 'Add Row' button to add another line to the stream upload table."));
1180 delUploadRowButton->setWhatsThis(tr("Hit 'Del Row' button to delete the highlighted line from the stream upload table."));
1181 _uploadIntrComboBox->setWhatsThis(tr("Select the length of the SP3 and Clock RINEX files."));
1182 _uploadSamplRtcmEphCorrSpinBox->setWhatsThis(tr("Select the stream's orbit correction sampling interval in seconds. A value of zero '0' tells BNC to upload all available orbit and clock correction samples together in combined messages."));
1183 _uploadSamplClkRnxSpinBox->setWhatsThis(tr("Select the Clock RINEX file sampling interval in seconds. A value of zero '0' tells BNC to store all available samples into Clock RINEX files."));
1184 _uploadSamplSp3SpinBox->setWhatsThis(tr("Select the SP3 orbit file sampling interval in minutes. A value of zero '0' tells BNC to store all available samples into SP3 orbit files."));
1185 setUploadTrafoButton->setWhatsThis(tr("Hit 'Custom Trafo' to specify your own 14 parameter Helmert Transformation instead of selecting a predefined transformation through 'System' button."));
1186 _uploadEphHostLineEdit->setWhatsThis(tr("BNC can upload a Broadcast Ephemeris stream in RTCM Version 3 format. Specify the host IP of an NTRIP Broadcaster to upload the stream. An empty option field means that you don't want to upload Broadcast Ephemeris."));
1187 _uploadEphPortLineEdit->setWhatsThis(tr("Specify the IP port of an NTRIP Broadcaster to upload the stream. Default is port 80."));
1188 _uploadEphMountpointLineEdit->setWhatsThis(tr("Specify the mounpoint for stream upload to an NTRIP Broadcaster."));
1189 _uploadEphPasswordLineEdit->setWhatsThis(tr("Specify the stream upload password protecting the mounpoint on an NTRIP Broadcaster."));
1190 _uploadEphSampleSpinBox->setWhatsThis(tr("Select the Broadcast Ephemeris sampling interval in seconds. Defaut is '5' meaning that a complete set of Broadcast Ephemeris is uploaded every 5 seconds."));
1191 _uploadEphBytesCounter->setWhatsThis(tr("BNC shows the amount of data uploaded through this stream."));
1192 _actDeleteMountPoints->setWhatsThis(tr("<p>Delete stream(s) from selection presented in the 'Streams' canvas.</p>"));
1193 _actAddMountPoints->setWhatsThis(tr("<p>Add stream(s) to selection presented in the 'Streams' canvas.</p>"));
1194 _actMapMountPoints->setWhatsThis(tr("<p> Draw distribution map of stream selection presented in the 'Streams' canvas. Use the mouse to zoom in or out.</p><p>Left button: Draw rectangle to zoom in.<br>Right button: Zoom out.<br>Middle button: Zoom back.</p>"));
1195 _actStart->setWhatsThis(tr("<p> Start running BNC.</p>"));
1196 _actStop->setWhatsThis(tr("<p> Stop running BNC.</p>"));
1197// Weber
1198
1199 // Enable/Disable all Widgets
1200 // --------------------------
1201 slotBncTextChanged();
1202 enableStartStop();
1203
1204 // Auto start
1205 // ----------
1206 if ( Qt::CheckState(settings.value("autoStart").toInt()) == Qt::Checked) {
1207 slotStart();
1208 }
1209}
1210
1211// Destructor
1212////////////////////////////////////////////////////////////////////////////
1213bncWindow::~bncWindow() {
1214 delete _caster; BNC_CORE->setCaster(0);
1215 delete _casterEph;
1216}
1217
1218//
1219////////////////////////////////////////////////////////////////////////////
1220void bncWindow::populateMountPointsTable() {
1221
1222 for (int iRow = _mountPointsTable->rowCount()-1; iRow >=0; iRow--) {
1223 _mountPointsTable->removeRow(iRow);
1224 }
1225
1226 bncSettings settings;
1227
1228 QListIterator<QString> it(settings.value("mountPoints").toStringList());
1229 int iRow = 0;
1230 while (it.hasNext()) {
1231 QStringList hlp = it.next().split(" ");
1232 if (hlp.size() < 5) continue;
1233 _mountPointsTable->insertRow(iRow);
1234
1235 QUrl url(hlp[0]);
1236
1237 QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
1238 QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
1239 QString nmea(hlp[4]);
1240 if (hlp[5] == "S") {
1241 fullPath = hlp[0].replace(0,2,"");
1242 }
1243 QString ntripVersion = "2";
1244 if (hlp.size() >= 6) {
1245 ntripVersion = (hlp[5]);
1246 }
1247
1248 QTableWidgetItem* it;
1249 it = new QTableWidgetItem(url.userInfo());
1250 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1251 _mountPointsTable->setItem(iRow, 0, it);
1252
1253 it = new QTableWidgetItem(fullPath);
1254 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1255 _mountPointsTable->setItem(iRow, 1, it);
1256
1257 it = new QTableWidgetItem(format);
1258 _mountPointsTable->setItem(iRow, 2, it);
1259
1260 if (nmea == "yes") {
1261 it = new QTableWidgetItem(latitude);
1262 _mountPointsTable->setItem(iRow, 3, it);
1263 it = new QTableWidgetItem(longitude);
1264 _mountPointsTable->setItem(iRow, 4, it);
1265 } else {
1266 it = new QTableWidgetItem(latitude);
1267 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1268 _mountPointsTable->setItem(iRow, 3, it);
1269 it = new QTableWidgetItem(longitude);
1270 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1271 _mountPointsTable->setItem(iRow, 4, it);
1272 }
1273
1274 it = new QTableWidgetItem(nmea);
1275 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1276 _mountPointsTable->setItem(iRow, 5, it);
1277
1278 it = new QTableWidgetItem(ntripVersion);
1279 //// it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1280 _mountPointsTable->setItem(iRow, 6, it);
1281
1282 bncTableItem* bncIt = new bncTableItem();
1283 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
1284 _mountPointsTable->setItem(iRow, 7, bncIt);
1285
1286 iRow++;
1287 }
1288
1289 _mountPointsTable->sortItems(1);
1290
1291 enableStartStop();
1292}
1293
1294// Retrieve Table
1295////////////////////////////////////////////////////////////////////////////
1296void bncWindow::slotAddMountPoints() {
1297
1298 bncSettings settings;
1299 QString proxyHost = settings.value("proxyHost").toString();
1300 int proxyPort = settings.value("proxyPort").toInt();
1301 if (proxyHost != _proxyHostLineEdit->text() ||
1302 proxyPort != _proxyPortLineEdit->text().toInt()) {
1303 int iRet = QMessageBox::question(this, "Question", "Proxy options "
1304 "changed. Use the new ones?",
1305 QMessageBox::Yes, QMessageBox::No,
1306 QMessageBox::NoButton);
1307 if (iRet == QMessageBox::Yes) {
1308 settings.setValue("proxyHost", _proxyHostLineEdit->text());
1309 settings.setValue("proxyPort", _proxyPortLineEdit->text());
1310 }
1311 }
1312
1313 settings.setValue("sslCaCertPath", _sslCaCertPathLineEdit->text());
1314 settings.setValue("ignoreSslErrors", _ignoreSslErrorsCheckBox->checkState());
1315
1316 QMessageBox msgBox;
1317 msgBox.setIcon(QMessageBox::Question);
1318 msgBox.setWindowTitle("Add Stream");
1319 msgBox.setText("Add stream(s) coming from:");
1320
1321 QPushButton* buttonNtrip = msgBox.addButton(tr("Caster"), QMessageBox::ActionRole);
1322 QPushButton* buttonIP = msgBox.addButton(tr("TCP/IP port"), QMessageBox::ActionRole);
1323 QPushButton* buttonUDP = msgBox.addButton(tr("UDP port"), QMessageBox::ActionRole);
1324 QPushButton* buttonSerial = msgBox.addButton(tr("Serial port"), QMessageBox::ActionRole);
1325 QPushButton* buttonCancel = msgBox.addButton(tr("Cancel"), QMessageBox::ActionRole);
1326
1327 msgBox.exec();
1328
1329 if (msgBox.clickedButton() == buttonNtrip) {
1330 bncTableDlg* dlg = new bncTableDlg(this);
1331 dlg->move(this->pos().x()+50, this->pos().y()+50);
1332 connect(dlg, SIGNAL(newMountPoints(QStringList*)),
1333 this, SLOT(slotNewMountPoints(QStringList*)));
1334 dlg->exec();
1335 delete dlg;
1336 } else if (msgBox.clickedButton() == buttonIP) {
1337 bncIpPort* ipp = new bncIpPort(this);
1338 connect(ipp, SIGNAL(newMountPoints(QStringList*)),
1339 this, SLOT(slotNewMountPoints(QStringList*)));
1340 ipp->exec();
1341 delete ipp;
1342 } else if (msgBox.clickedButton() == buttonUDP) {
1343 bncUdpPort* udp = new bncUdpPort(this);
1344 connect(udp, SIGNAL(newMountPoints(QStringList*)),
1345 this, SLOT(slotNewMountPoints(QStringList*)));
1346 udp->exec();
1347 delete udp;
1348 } else if (msgBox.clickedButton() == buttonSerial) {
1349 bncSerialPort* sep = new bncSerialPort(this);
1350 connect(sep, SIGNAL(newMountPoints(QStringList*)),
1351 this, SLOT(slotNewMountPoints(QStringList*)));
1352 sep->exec();
1353 delete sep;
1354 } else if (msgBox.clickedButton() == buttonCancel) {
1355 // Cancel
1356 }
1357
1358 enableStartStop();
1359}
1360
1361// Delete Selected Mount Points
1362////////////////////////////////////////////////////////////////////////////
1363void bncWindow::slotDeleteMountPoints() {
1364
1365 int nRows = _mountPointsTable->rowCount();
1366 bool flg[nRows];
1367 for (int iRow = 0; iRow < nRows; iRow++) {
1368 if (_mountPointsTable->isItemSelected(_mountPointsTable->item(iRow,1))) {
1369 flg[iRow] = true;
1370 }
1371 else {
1372 flg[iRow] = false;
1373 }
1374 }
1375 for (int iRow = nRows-1; iRow >= 0; iRow--) {
1376 if (flg[iRow]) {
1377 _mountPointsTable->removeRow(iRow);
1378 }
1379 }
1380 _actDeleteMountPoints->setEnabled(false);
1381
1382 enableStartStop();
1383}
1384
1385// New Mount Points Selected
1386////////////////////////////////////////////////////////////////////////////
1387void bncWindow::slotNewMountPoints(QStringList* mountPoints) {
1388 int iRow = 0;
1389 QListIterator<QString> it(*mountPoints);
1390 while (it.hasNext()) {
1391 QStringList hlp = it.next().split(" ");
1392 QUrl url(hlp[0]);
1393 QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
1394 QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
1395 QString nmea(hlp[4]);
1396 if (hlp[5] == "S") {
1397 fullPath = hlp[0].replace(0,2,"");
1398 }
1399 QString ntripVersion = "2";
1400 if (hlp.size() >= 6) {
1401 ntripVersion = (hlp[5]);
1402 }
1403
1404 _mountPointsTable->insertRow(iRow);
1405
1406 QTableWidgetItem* it;
1407 it = new QTableWidgetItem(url.userInfo());
1408 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1409 _mountPointsTable->setItem(iRow, 0, it);
1410
1411 it = new QTableWidgetItem(fullPath);
1412 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1413 _mountPointsTable->setItem(iRow, 1, it);
1414
1415 it = new QTableWidgetItem(format);
1416 _mountPointsTable->setItem(iRow, 2, it);
1417
1418 if (nmea == "yes") {
1419 it = new QTableWidgetItem(latitude);
1420 _mountPointsTable->setItem(iRow, 3, it);
1421 it = new QTableWidgetItem(longitude);
1422 _mountPointsTable->setItem(iRow, 4, it);
1423 } else {
1424 it = new QTableWidgetItem(latitude);
1425 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1426 _mountPointsTable->setItem(iRow, 3, it);
1427 it = new QTableWidgetItem(longitude);
1428 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1429 _mountPointsTable->setItem(iRow, 4, it);
1430 }
1431
1432 it = new QTableWidgetItem(nmea);
1433 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1434 _mountPointsTable->setItem(iRow, 5, it);
1435
1436 it = new QTableWidgetItem(ntripVersion);
1437 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1438 _mountPointsTable->setItem(iRow, 6, it);
1439
1440 bncTableItem* bncIt = new bncTableItem();
1441 _mountPointsTable->setItem(iRow, 7, bncIt);
1442
1443 iRow++;
1444 }
1445 _mountPointsTable->hideColumn(0);
1446 _mountPointsTable->sortItems(1);
1447 delete mountPoints;
1448
1449 enableStartStop();
1450}
1451
1452// Save Options (serialize)
1453////////////////////////////////////////////////////////////////////////////
1454void bncWindow::slotSaveOptions() {
1455 saveOptions();
1456 bncSettings settings;
1457 settings.sync();
1458}
1459
1460// Save Options (memory only)
1461////////////////////////////////////////////////////////////////////////////
1462void bncWindow::saveOptions() {
1463
1464 QStringList mountPoints;
1465 for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
1466
1467 if (_mountPointsTable->item(iRow, 6)->text() != "S") {
1468 QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
1469 "@" + _mountPointsTable->item(iRow, 1)->text() );
1470
1471 mountPoints.append(url.toString() + " " +
1472 _mountPointsTable->item(iRow, 2)->text()
1473 + " " + _mountPointsTable->item(iRow, 3)->text()
1474 + " " + _mountPointsTable->item(iRow, 4)->text()
1475 + " " + _mountPointsTable->item(iRow, 5)->text()
1476 + " " + _mountPointsTable->item(iRow, 6)->text());
1477 } else {
1478 mountPoints.append(
1479 "//" + _mountPointsTable->item(iRow, 1)->text()
1480 + " " + _mountPointsTable->item(iRow, 2)->text()
1481 + " " + _mountPointsTable->item(iRow, 3)->text()
1482 + " " + _mountPointsTable->item(iRow, 4)->text()
1483 + " " + _mountPointsTable->item(iRow, 5)->text()
1484 + " " + _mountPointsTable->item(iRow, 6)->text());
1485 }
1486 }
1487
1488 QStringList combineStreams;
1489 for (int iRow = 0; iRow < _cmbTable->rowCount(); iRow++) {
1490 QString hlp;
1491 for (int iCol = 0; iCol < _cmbTable->columnCount(); iCol++) {
1492 if (_cmbTable->item(iRow, iCol)) {
1493 hlp += _cmbTable->item(iRow, iCol)->text() + " ";
1494 }
1495 }
1496 if (!hlp.isEmpty()) {
1497 combineStreams << hlp;
1498 }
1499 }
1500
1501 QStringList uploadMountpointsOut;
1502 for (int iRow = 0; iRow < _uploadTable->rowCount(); iRow++) {
1503 QString hlp;
1504 for (int iCol = 0; iCol < _uploadTable->columnCount(); iCol++) {
1505 if (_uploadTable->cellWidget(iRow, iCol) &&
1506 (iCol == 3 || iCol == 4 || iCol == 5)) {
1507 if (iCol == 3) {
1508 QLineEdit* passwd = (QLineEdit*)(_uploadTable->cellWidget(iRow, iCol));
1509 hlp += passwd->text() + ",";
1510 }
1511 else if (iCol == 4) {
1512 QComboBox* system = (QComboBox*)(_uploadTable->cellWidget(iRow, iCol));
1513 hlp += system->currentText() + ",";
1514 }
1515 else if (iCol == 5) {
1516 QCheckBox* com = (QCheckBox*)(_uploadTable->cellWidget(iRow, iCol));
1517 QString state; state.setNum(com->checkState());
1518 hlp += state + ",";
1519 }
1520 }
1521 else if (_uploadTable->item(iRow, iCol)) {
1522 hlp += _uploadTable->item(iRow, iCol)->text() + ",";
1523 }
1524 }
1525 if (!hlp.isEmpty()) {
1526 uploadMountpointsOut << hlp;
1527 }
1528 }
1529
1530 bncSettings settings;
1531
1532 settings.setValue("startTab", _aogroup->currentIndex());
1533 settings.setValue("statusTab", _loggroup->currentIndex());
1534 settings.setValue("mountPoints", mountPoints);
1535// Network
1536 settings.setValue("proxyHost", _proxyHostLineEdit->text());
1537 settings.setValue("proxyPort", _proxyPortLineEdit->text());
1538 settings.setValue("sslCaCertPath", _sslCaCertPathLineEdit->text());
1539 settings.setValue("ignoreSslErrors", _ignoreSslErrorsCheckBox->checkState());
1540// General
1541 settings.setValue("logFile", _logFileLineEdit->text());
1542 settings.setValue("rnxAppend", _rnxAppendCheckBox->checkState());
1543 settings.setValue("onTheFlyInterval", _onTheFlyComboBox->currentText());
1544 settings.setValue("autoStart", _autoStartCheckBox->checkState());
1545 settings.setValue("rawOutFile", _rawOutFileLineEdit->text());
1546// RINEX Observations
1547 settings.setValue("rnxPath", _rnxPathLineEdit->text());
1548 settings.setValue("rnxIntr", _rnxIntrComboBox->currentText());
1549 settings.setValue("rnxSampl", _rnxSamplSpinBox->value());
1550 settings.setValue("rnxSkel", _rnxSkelLineEdit->text());
1551 settings.setValue("rnxScript", _rnxScrpLineEdit->text());
1552 settings.setValue("rnxV3", _rnxV3CheckBox->checkState());
1553// RINEX Ephemeris
1554 settings.setValue("ephPath", _ephPathLineEdit->text());
1555 settings.setValue("ephIntr", _ephIntrComboBox->currentText());
1556 settings.setValue("outEphPort", _outEphPortLineEdit->text());
1557 settings.setValue("ephV3", _ephV3CheckBox->checkState());
1558// Broadcast Corrections
1559 settings.setValue("corrPath", _corrPathLineEdit->text());
1560 settings.setValue("corrIntr", _corrIntrComboBox->currentText());
1561 settings.setValue("corrPort", _corrPortLineEdit->text());
1562 settings.setValue("corrTime", _corrTimeSpinBox->value());
1563// Feed Engine
1564 settings.setValue("outPort", _outPortLineEdit->text());
1565 settings.setValue("waitTime", _waitTimeSpinBox->value());
1566 settings.setValue("binSampl", _binSamplSpinBox->value());
1567 settings.setValue("outFile", _outFileLineEdit->text());
1568 settings.setValue("outUPort", _outUPortLineEdit->text());
1569// Serial Output
1570 settings.setValue("serialMountPoint",_serialMountPointLineEdit->text());
1571 settings.setValue("serialPortName", _serialPortNameLineEdit->text());
1572 settings.setValue("serialBaudRate", _serialBaudRateComboBox->currentText());
1573 settings.setValue("serialFlowControl",_serialFlowControlComboBox->currentText());
1574 settings.setValue("serialDataBits", _serialDataBitsComboBox->currentText());
1575 settings.setValue("serialParity", _serialParityComboBox->currentText());
1576 settings.setValue("serialStopBits", _serialStopBitsComboBox->currentText());
1577 settings.setValue("serialAutoNMEA", _serialAutoNMEAComboBox->currentText());
1578 settings.setValue("serialFileNMEA",_serialFileNMEALineEdit->text());
1579 settings.setValue("serialHeightNMEA",_serialHeightNMEALineEdit->text());
1580// Outages
1581 settings.setValue("obsRate", _obsRateComboBox->currentText());
1582 settings.setValue("adviseFail", _adviseFailSpinBox->value());
1583 settings.setValue("adviseReco", _adviseRecoSpinBox->value());
1584 settings.setValue("adviseScript",_adviseScriptLineEdit->text());
1585// Miscellaneous
1586 settings.setValue("miscMount", _miscMountLineEdit->text());
1587 settings.setValue("miscPort", _miscPortLineEdit->text());
1588 settings.setValue("perfIntr", _perfIntrComboBox->currentText());
1589 settings.setValue("scanRTCM", _scanRTCMCheckBox->checkState());
1590// Reqc
1591 settings.setValue("reqcAction", _reqcActionComboBox->currentText());
1592 settings.setValue("reqcObsFile", _reqcObsFileChooser->fileName());
1593 settings.setValue("reqcNavFile", _reqcNavFileChooser->fileName());
1594 settings.setValue("reqcOutObsFile", _reqcOutObsLineEdit->text());
1595 settings.setValue("reqcOutNavFile", _reqcOutNavLineEdit->text());
1596 settings.setValue("reqcOutLogFile", _reqcOutLogLineEdit->text());
1597 settings.setValue("reqcPlotDir", _reqcPlotDirLineEdit->text());
1598 settings.setValue("reqcSkyPlotSystems", _reqcSkyPlotSystems->currentText());
1599// Combine Corrections
1600 if (!combineStreams.isEmpty()) {
1601 settings.setValue("combineStreams", combineStreams);
1602 }
1603 else {
1604 settings.setValue("combineStreams", "");
1605 }
1606 settings.setValue("cmbMethod", _cmbMethodComboBox->currentText());
1607 settings.setValue("cmbMaxres", _cmbMaxresLineEdit->text());
1608 settings.setValue("cmbSampl", _cmbSamplSpinBox->value());
1609// Upload Corrections
1610 if (!uploadMountpointsOut.isEmpty()) {
1611 settings.setValue("uploadMountpointsOut", uploadMountpointsOut);
1612 }
1613 else {
1614 settings.setValue("uploadMountpointsOut", "");
1615 }
1616 settings.setValue("uploadIntr", _uploadIntrComboBox->currentText());
1617 settings.setValue("uploadSamplRtcmEphCorr", _uploadSamplRtcmEphCorrSpinBox->value());
1618 settings.setValue("uploadSamplSp3", _uploadSamplSp3SpinBox->value());
1619 settings.setValue("uploadSamplClkRnx", _uploadSamplClkRnxSpinBox->value());
1620// Upload Ephemeris
1621 settings.setValue("uploadEphHost", _uploadEphHostLineEdit->text());
1622 settings.setValue("uploadEphPort", _uploadEphPortLineEdit->text());
1623 settings.setValue("uploadEphMountpoint",_uploadEphMountpointLineEdit->text());
1624 settings.setValue("uploadEphPassword", _uploadEphPasswordLineEdit->text());
1625 settings.setValue("uploadEphSample", _uploadEphSampleSpinBox->value());
1626
1627 if (_caster) {
1628 _caster->readMountPoints();
1629 }
1630
1631 _pppWidgets.saveOptions();
1632}
1633
1634// All get slots terminated
1635////////////////////////////////////////////////////////////////////////////
1636void bncWindow::slotGetThreadsFinished() {
1637 BNC_CORE->slotMessage("All Get Threads Terminated", true);
1638 delete _caster; _caster = 0; BNC_CORE->setCaster(0);
1639 delete _casterEph; _casterEph = 0;
1640 _runningRealTime = false;
1641}
1642
1643// Start It!
1644////////////////////////////////////////////////////////////////////////////
1645void bncWindow::slotStart() {
1646 saveOptions();
1647 if ( _pppWidgets._dataSource->currentText() == "RINEX Files") {
1648 _runningPostProcessingPPP = true;
1649 BNC_CORE->startPPP(true);
1650 }
1651 else if ( !_reqcActionComboBox->currentText().isEmpty() ) {
1652 startPostProcessingReqc();
1653 }
1654 else {
1655 startRealTime();
1656 BNC_CORE->startPPP(true);
1657 }
1658}
1659
1660// Start Real-Time (Retrieve Data etc.)
1661////////////////////////////////////////////////////////////////////////////
1662void bncWindow::startRealTime() {
1663
1664 _runningRealTime = true;
1665
1666 _bncFigurePPP->reset();
1667
1668 _actDeleteMountPoints->setEnabled(false);
1669
1670 enableStartStop();
1671
1672 _caster = new bncCaster();
1673
1674 BNC_CORE->setCaster(_caster);
1675 BNC_CORE->setPort(_outEphPortLineEdit->text().toInt());
1676 BNC_CORE->setPortCorr(_corrPortLineEdit->text().toInt());
1677 BNC_CORE->initCombination();
1678
1679 connect(_caster, SIGNAL(getThreadsFinished()),
1680 this, SLOT(slotGetThreadsFinished()));
1681
1682 connect (_caster, SIGNAL(mountPointsRead(QList<bncGetThread*>)),
1683 this, SLOT(slotMountPointsRead(QList<bncGetThread*>)));
1684
1685 BNC_CORE->slotMessage("========== Start BNC v" BNCVERSION " =========", true);
1686
1687 bncSettings settings;
1688
1689 QDir rnxdir(settings.value("rnxPath").toString());
1690 if (!rnxdir.exists()) BNC_CORE->slotMessage("Cannot find RINEX Observations directory", true);
1691
1692 QString rnx_file = settings.value("rnxScript").toString();
1693 if ( !rnx_file.isEmpty() ) {
1694 QFile rnxfile(settings.value("rnxScript").toString());
1695 if (!rnxfile.exists()) BNC_CORE->slotMessage("Cannot find RINEX Observations script", true);
1696 }
1697
1698 QDir ephdir(settings.value("ephPath").toString());
1699 if (!ephdir.exists()) BNC_CORE->slotMessage("Cannot find RINEX Ephemeris directory", true);
1700
1701 QDir corrdir(settings.value("corrPath").toString());
1702 if (!corrdir.exists()) BNC_CORE->slotMessage("Cannot find Broadcast Corrections directory", true);
1703
1704 QString advise_file = settings.value("adviseScript").toString();
1705 if ( !advise_file.isEmpty() ) {
1706 QFile advisefile(settings.value("adviseScript").toString());
1707 if (!advisefile.exists()) BNC_CORE->slotMessage("Cannot find Outages script", true);
1708 }
1709
1710 QString ant_file = settings.value("pppAntex").toString();
1711 if ( !ant_file.isEmpty() ) {
1712 QFile anxfile(settings.value("pppAntex").toString());
1713 if (!anxfile.exists()) BNC_CORE->slotMessage("Cannot find IGS ANTEX file", true);
1714 }
1715
1716 _caster->readMountPoints();
1717
1718 _casterEph = new bncEphUploadCaster();
1719 connect(_casterEph, SIGNAL(newBytes(QByteArray,double)),
1720 _uploadEphBytesCounter, SLOT(slotNewBytes(QByteArray,double)));
1721}
1722
1723// Retrieve Data
1724////////////////////////////////////////////////////////////////////////////
1725void bncWindow::slotStop() {
1726 int iRet = QMessageBox::question(this, "Stop", "Stop retrieving data?",
1727 QMessageBox::Yes, QMessageBox::No,
1728 QMessageBox::NoButton);
1729 if (iRet == QMessageBox::Yes) {
1730 BNC_CORE->stopPPP();
1731 BNC_CORE->stopCombination();
1732 delete _caster; _caster = 0; BNC_CORE->setCaster(0);
1733 delete _casterEph; _casterEph = 0;
1734 _runningRealTime = false;
1735 enableStartStop();
1736 }
1737}
1738
1739// Close Application gracefully
1740////////////////////////////////////////////////////////////////////////////
1741void bncWindow::closeEvent(QCloseEvent* event) {
1742
1743 int iRet = QMessageBox::question(this, "Close", "Save Options?",
1744 QMessageBox::Yes, QMessageBox::No,
1745 QMessageBox::Cancel);
1746
1747 if (iRet == QMessageBox::Cancel) {
1748 event->ignore();
1749 return;
1750 }
1751 else if (iRet == QMessageBox::Yes) {
1752 slotSaveOptions();
1753 }
1754
1755 BNC_CORE->stopPPP();
1756
1757 QMainWindow::closeEvent(event);
1758}
1759
1760// User changed the selection of mountPoints
1761////////////////////////////////////////////////////////////////////////////
1762void bncWindow::slotSelectionChanged() {
1763 if (_mountPointsTable->selectedItems().isEmpty()) {
1764 _actDeleteMountPoints->setEnabled(false);
1765 }
1766 else {
1767 _actDeleteMountPoints->setEnabled(true);
1768 }
1769}
1770
1771// Display Program Messages
1772////////////////////////////////////////////////////////////////////////////
1773void bncWindow::slotWindowMessage(const QByteArray msg, bool showOnScreen) {
1774
1775#ifdef DEBUG_RTCM2_2021
1776 const int maxBufferSize = 1000;
1777#else
1778 const int maxBufferSize = 10000;
1779#endif
1780
1781 if (! showOnScreen ) {
1782 return;
1783 }
1784
1785 QString txt = _log->toPlainText() + "\n" +
1786 QDateTime::currentDateTime().toUTC().toString("yy-MM-dd hh:mm:ss ") + msg;
1787 _log->clear();
1788 _log->append(txt.right(maxBufferSize));
1789}
1790
1791// About Message
1792////////////////////////////////////////////////////////////////////////////
1793void bncWindow::slotAbout() {
1794 new bncAboutDlg(0);
1795}
1796
1797//Flowchart
1798////////////////////////////////////////////////////////////////////////////
1799void bncWindow::slotFlowchart() {
1800 new bncFlowchartDlg(0);
1801}
1802
1803// Help Window
1804////////////////////////////////////////////////////////////////////////////
1805void bncWindow::slotHelp() {
1806 QUrl url;
1807 url.setPath(":bnchelp.html");
1808 new bncHlpDlg(0, url);
1809}
1810
1811// Select Fonts
1812////////////////////////////////////////////////////////////////////////////
1813void bncWindow::slotFontSel() {
1814 bool ok;
1815 QFont newFont = QFontDialog::getFont(&ok, this->font(), this);
1816 if (ok) {
1817 bncSettings settings;
1818 settings.setValue("font", newFont.toString());
1819 QApplication::setFont(newFont);
1820 int ww = QFontMetrics(newFont).width('w');
1821 setMinimumSize(60*ww, 80*ww);
1822 resize(60*ww, 80*ww);
1823 }
1824}
1825
1826// Whats This Help
1827void bncWindow::slotWhatsThis() {
1828 QWhatsThis::enterWhatsThisMode();
1829}
1830
1831//
1832////////////////////////////////////////////////////////////////////////////
1833void bncWindow::slotMountPointsRead(QList<bncGetThread*> threads) {
1834 _threads = threads;
1835
1836 _bncFigure->updateMountPoints();
1837 _bncFigureLate->updateMountPoints();
1838
1839 populateMountPointsTable();
1840 bncSettings settings;
1841 _binSamplSpinBox->setValue(settings.value("binSampl").toInt());
1842 _waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
1843 QListIterator<bncGetThread*> iTh(threads);
1844 while (iTh.hasNext()) {
1845 bncGetThread* thread = iTh.next();
1846 for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
1847 QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
1848 "@" + _mountPointsTable->item(iRow, 1)->text() );
1849 if (url == thread->mountPoint() &&
1850 _mountPointsTable->item(iRow, 3)->text() == thread->latitude() &&
1851 _mountPointsTable->item(iRow, 4)->text() == thread->longitude() ) {
1852 ((bncTableItem*) _mountPointsTable->item(iRow, 7))->setGetThread(thread);
1853 disconnect(thread, SIGNAL(newBytes(QByteArray, double)),
1854 _bncFigure, SLOT(slotNewData(QByteArray, double)));
1855 connect(thread, SIGNAL(newBytes(QByteArray, double)),
1856 _bncFigure, SLOT(slotNewData(QByteArray, double)));
1857 disconnect(thread, SIGNAL(newLatency(QByteArray, double)),
1858 _bncFigureLate, SLOT(slotNewLatency(QByteArray, double)));
1859 connect(thread, SIGNAL(newLatency(QByteArray, double)),
1860 _bncFigureLate, SLOT(slotNewLatency(QByteArray, double)));
1861 break;
1862 }
1863 }
1864 }
1865}
1866
1867//
1868////////////////////////////////////////////////////////////////////////////
1869void bncWindow::CreateMenu() {
1870 // Create Menus
1871 // ------------
1872 _menuFile = menuBar()->addMenu(tr("&File"));
1873 _menuFile->addAction(_actFontSel);
1874 _menuFile->addSeparator();
1875 _menuFile->addAction(_actSaveOpt);
1876 _menuFile->addSeparator();
1877 _menuFile->addAction(_actQuit);
1878
1879 _menuHlp = menuBar()->addMenu(tr("&Help"));
1880 _menuHlp->addAction(_actHelp);
1881 _menuHlp->addAction(_actFlowchart);
1882 _menuHlp->addAction(_actAbout);
1883}
1884
1885// Toolbar
1886////////////////////////////////////////////////////////////////////////////
1887void bncWindow::AddToolbar() {
1888 QToolBar* toolBar = new QToolBar;
1889 addToolBar(Qt::BottomToolBarArea, toolBar);
1890 toolBar->setMovable(false);
1891 toolBar->addAction(_actAddMountPoints);
1892 toolBar->addAction(_actDeleteMountPoints);
1893 toolBar->addAction(_actMapMountPoints);
1894 toolBar->addAction(_actStart);
1895 toolBar->addAction(_actStop);
1896 toolBar->addWidget(new QLabel(" "));
1897 toolBar->addAction(_actwhatsthis);
1898}
1899
1900// About
1901////////////////////////////////////////////////////////////////////////////
1902bncAboutDlg::bncAboutDlg(QWidget* parent) :
1903 QDialog(parent) {
1904
1905 QTextBrowser* tb = new QTextBrowser;
1906 QUrl url; url.setPath(":bncabout.html");
1907 tb->setSource(url);
1908 tb->setReadOnly(true);
1909
1910 int ww = QFontMetrics(font()).width('w');
1911 QPushButton* _closeButton = new QPushButton("Close");
1912 _closeButton->setMaximumWidth(10*ww);
1913 connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
1914
1915 QGridLayout* dlgLayout = new QGridLayout();
1916 QLabel* img = new QLabel();
1917 img->setPixmap(QPixmap(":ntrip-logo.png"));
1918 dlgLayout->addWidget(img, 0,0);
1919 dlgLayout->addWidget(new QLabel("BKG Ntrip Client (BNC) Version "BNCVERSION), 0,1);
1920 dlgLayout->addWidget(tb,1,0,1,2);
1921 dlgLayout->addWidget(_closeButton,2,1,Qt::AlignRight);
1922
1923 setLayout(dlgLayout);
1924 resize(60*ww, 60*ww);
1925 setWindowTitle("About BNC");
1926 show();
1927}
1928
1929//
1930////////////////////////////////////////////////////////////////////////////
1931bncAboutDlg::~bncAboutDlg() {
1932};
1933
1934// Flowchart
1935////////////////////////////////////////////////////////////////////////////
1936bncFlowchartDlg::bncFlowchartDlg(QWidget* parent) :
1937 QDialog(parent) {
1938
1939 int ww = QFontMetrics(font()).width('w');
1940 QPushButton* _closeButton = new QPushButton("Close");
1941 _closeButton->setMaximumWidth(10*ww);
1942 connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
1943
1944 QGridLayout* dlgLayout = new QGridLayout();
1945 QLabel* img = new QLabel();
1946 img->setPixmap(QPixmap(":bncflowchart.png"));
1947 dlgLayout->addWidget(img, 0,0);
1948 dlgLayout->addWidget(_closeButton,1,0,Qt::AlignLeft);
1949
1950 setLayout(dlgLayout);
1951 setWindowTitle("Flow Chart");
1952 show();
1953}
1954
1955//
1956////////////////////////////////////////////////////////////////////////////
1957bncFlowchartDlg::~bncFlowchartDlg() {
1958};
1959
1960// Enable/Disable Widget (and change its color)
1961////////////////////////////////////////////////////////////////////////////
1962void bncWindow::enableWidget(bool enable, QWidget* widget) {
1963 const static QPalette paletteWhite(QColor(255, 255, 255));
1964 const static QPalette paletteGray(QColor(230, 230, 230));
1965
1966 widget->setEnabled(enable);
1967 if (enable) {
1968 widget->setPalette(paletteWhite);
1969 }
1970 else {
1971 widget->setPalette(paletteGray);
1972 }
1973}
1974
1975// Bnc Text
1976////////////////////////////////////////////////////////////////////////////
1977void bncWindow::slotBncTextChanged(){
1978
1979 bool enable = true;
1980
1981 // Proxy
1982 //------
1983 if (sender() == 0 || sender() == _proxyHostLineEdit) {
1984 enable = !_proxyHostLineEdit->text().isEmpty();
1985 enableWidget(enable, _proxyPortLineEdit);
1986 }
1987
1988 // RINEX Observations
1989 // ------------------
1990 if (sender() == 0 || sender() == _rnxPathLineEdit) {
1991 enable = !_rnxPathLineEdit->text().isEmpty();
1992 enableWidget(enable, _rnxSamplSpinBox);
1993 enableWidget(enable, _rnxSkelLineEdit);
1994 enableWidget(enable, _rnxScrpLineEdit);
1995 enableWidget(enable, _rnxV3CheckBox);
1996 enableWidget(enable, _rnxIntrComboBox);
1997 }
1998
1999 // RINEX Ephemeris
2000 // ---------------
2001 if (sender() == 0 || sender() == _ephPathLineEdit || sender() == _outEphPortLineEdit) {
2002 enable = !_ephPathLineEdit->text().isEmpty() || !_outEphPortLineEdit->text().isEmpty();
2003 enableWidget(enable, _ephIntrComboBox);
2004 enableWidget(enable, _ephV3CheckBox);
2005 }
2006
2007 // Broadcast Corrections
2008 // ---------------------
2009 if (sender() == 0 || sender() == _corrPathLineEdit || sender() == _corrPortLineEdit) {
2010 enable = !_corrPathLineEdit->text().isEmpty() || !_corrPortLineEdit->text().isEmpty();
2011 enableWidget(enable, _corrIntrComboBox);
2012 }
2013
2014 // Feed Engine
2015 // -----------
2016 if (sender() == 0 || sender() == _outPortLineEdit || sender() == _outFileLineEdit) {
2017 enable = !_outPortLineEdit->text().isEmpty() || !_outFileLineEdit->text().isEmpty();
2018 enableWidget(enable, _waitTimeSpinBox);
2019 enableWidget(enable, _binSamplSpinBox);
2020 }
2021
2022 // Serial Output
2023 // -------------
2024 if (sender() == 0 || sender() == _serialMountPointLineEdit ||
2025 sender() == _serialAutoNMEAComboBox) {
2026 enable = !_serialMountPointLineEdit->text().isEmpty();
2027 enableWidget(enable, _serialPortNameLineEdit);
2028 enableWidget(enable, _serialBaudRateComboBox);
2029 enableWidget(enable, _serialParityComboBox);
2030 enableWidget(enable, _serialDataBitsComboBox);
2031 enableWidget(enable, _serialStopBitsComboBox);
2032 enableWidget(enable, _serialFlowControlComboBox);
2033 enableWidget(enable, _serialAutoNMEAComboBox);
2034
2035 bool enable2 = enable && _serialAutoNMEAComboBox->currentText() != "Auto";
2036 enableWidget(enable2, _serialFileNMEALineEdit);
2037 }
2038
2039 // Outages
2040 // -------
2041 if (sender() == 0 || sender() == _obsRateComboBox) {
2042 enable = !_obsRateComboBox->currentText().isEmpty();
2043 enableWidget(enable, _adviseFailSpinBox);
2044 enableWidget(enable, _adviseRecoSpinBox);
2045 enableWidget(enable, _adviseScriptLineEdit);
2046 }
2047
2048 // Miscellaneous
2049 // -------------
2050 if (sender() == 0 || sender() == _miscMountLineEdit) {
2051 enable = !_miscMountLineEdit->text().isEmpty();
2052 enableWidget(enable, _perfIntrComboBox);
2053 enableWidget(enable, _scanRTCMCheckBox);
2054 enableWidget(enable, _miscPortLineEdit);
2055 }
2056
2057 // Enable/disable Broadcast Ephemerides
2058 // ------------------------------------
2059 if (sender() == 0 || sender() == _uploadEphHostLineEdit) {
2060 if (!_uploadEphHostLineEdit->text().isEmpty()) {
2061 _uploadEphPortLineEdit->setStyleSheet("background-color: white");
2062 _uploadEphMountpointLineEdit->setStyleSheet("background-color: white");
2063 _uploadEphPasswordLineEdit->setStyleSheet("background-color: white");
2064 _uploadEphSampleSpinBox->setStyleSheet("background-color: white");
2065 _uploadEphPortLineEdit->setEnabled(true);
2066 _uploadEphMountpointLineEdit->setEnabled(true);
2067 _uploadEphPasswordLineEdit->setEnabled(true);
2068 _uploadEphSampleSpinBox->setEnabled(true);
2069 }
2070 else {
2071 _uploadEphPortLineEdit->setStyleSheet("background-color: lightGray");
2072 _uploadEphMountpointLineEdit->setStyleSheet("background-color: lightGray");
2073 _uploadEphPasswordLineEdit->setStyleSheet("background-color: lightGray");
2074 _uploadEphSampleSpinBox->setStyleSheet("background-color: lightGray");
2075 _uploadEphPortLineEdit->setEnabled(false);
2076 _uploadEphMountpointLineEdit->setEnabled(false);
2077 _uploadEphPasswordLineEdit->setEnabled(false);
2078 _uploadEphSampleSpinBox->setEnabled(false);
2079 }
2080 }
2081
2082 // Combine Corrections
2083 // -------------------
2084 if (sender() == 0 || sender() == _cmbTable) {
2085 int iRow = _cmbTable->rowCount();
2086 if (iRow > 0) {
2087 enableWidget(true, _cmbMethodComboBox);
2088 _cmbMaxresLineEdit->setStyleSheet("background-color: white");
2089 _cmbMaxresLineEdit->setEnabled(true);
2090 _cmbSamplSpinBox->setEnabled(true);
2091 }
2092 else {
2093 enableWidget(false, _cmbMethodComboBox);
2094 _cmbMaxresLineEdit->setStyleSheet("background-color: lightGray");
2095 _cmbMaxresLineEdit->setEnabled(false);
2096 _cmbSamplSpinBox->setEnabled(false);
2097 }
2098 }
2099
2100 // Upload(clk)
2101 // -----------
2102 int iRow = _uploadTable->rowCount();
2103 if (iRow > 0) {
2104 enableWidget(true, _uploadIntrComboBox);
2105 enableWidget(true, _uploadSamplRtcmEphCorrSpinBox);
2106 enableWidget(true, _uploadSamplClkRnxSpinBox);
2107 enableWidget(true, _uploadSamplSp3SpinBox);
2108 }
2109 else {
2110 enableWidget(false, _uploadIntrComboBox);
2111 enableWidget(false, _uploadSamplRtcmEphCorrSpinBox);
2112 enableWidget(false, _uploadSamplClkRnxSpinBox);
2113 enableWidget(false, _uploadSamplSp3SpinBox);
2114 }
2115
2116 // QC
2117 // --
2118 if (sender() == 0 || sender() == _reqcActionComboBox) {
2119 enable = !_reqcActionComboBox->currentText().isEmpty();
2120 bool enable10 = _reqcActionComboBox->currentText() == "Edit/Concatenate";
2121 enableWidget(enable && enable10, _reqcEditOptionButton);
2122 enableWidget(enable, _reqcObsFileChooser);
2123 enableWidget(enable, _reqcNavFileChooser);
2124 enableWidget(enable && enable10, _reqcOutObsLineEdit);
2125 enableWidget(enable && enable10, _reqcOutNavLineEdit);
2126 enableWidget(enable, _reqcOutLogLineEdit);
2127 enableWidget(enable && !enable10, _reqcPlotDirLineEdit);
2128 enableWidget(enable && !enable10, _reqcSkyPlotSystems);
2129 }
2130
2131 enableStartStop();
2132}
2133
2134//
2135////////////////////////////////////////////////////////////////////////////
2136void bncWindow::slotAddCmbRow() {
2137 int iRow = _cmbTable->rowCount();
2138 _cmbTable->insertRow(iRow);
2139 for (int iCol = 0; iCol < _cmbTable->columnCount(); iCol++) {
2140 _cmbTable->setItem(iRow, iCol, new QTableWidgetItem(""));
2141 }
2142}
2143
2144//
2145////////////////////////////////////////////////////////////////////////////
2146void bncWindow::slotDelCmbRow() {
2147 int nRows = _cmbTable->rowCount();
2148 bool flg[nRows];
2149 for (int iRow = 0; iRow < nRows; iRow++) {
2150 if (_cmbTable->isItemSelected(_cmbTable->item(iRow,1))) {
2151 flg[iRow] = true;
2152 }
2153 else {
2154 flg[iRow] = false;
2155 }
2156 }
2157 for (int iRow = nRows-1; iRow >= 0; iRow--) {
2158 if (flg[iRow]) {
2159 _cmbTable->removeRow(iRow);
2160 }
2161 }
2162 nRows = _cmbTable->rowCount();
2163 if (nRows < 1) {
2164 enableWidget(false, _cmbMethodComboBox);
2165 _cmbMaxresLineEdit->setStyleSheet("background-color: lightGray");
2166 _cmbMaxresLineEdit->setEnabled(false);
2167 _cmbSamplSpinBox->setEnabled(false);
2168 }
2169}
2170
2171//
2172////////////////////////////////////////////////////////////////////////////
2173void bncWindow::populateCmbTable() {
2174
2175 for (int iRow = _cmbTable->rowCount()-1; iRow >=0; iRow--) {
2176 _cmbTable->removeRow(iRow);
2177 }
2178
2179 bncSettings settings;
2180
2181 int iRow = -1;
2182 QListIterator<QString> it(settings.value("combineStreams").toStringList());
2183 while (it.hasNext()) {
2184 QStringList hlp = it.next().split(" ");
2185 if (hlp.size() > 2) {
2186 ++iRow;
2187 _cmbTable->insertRow(iRow);
2188 }
2189 for (int iCol = 0; iCol < hlp.size(); iCol++) {
2190 _cmbTable->setItem(iRow, iCol, new QTableWidgetItem(hlp[iCol]));
2191 }
2192 }
2193}
2194
2195//
2196////////////////////////////////////////////////////////////////////////////
2197void bncWindow::slotAddUploadRow() {
2198 int iRow = _uploadTable->rowCount();
2199 _uploadTable->insertRow(iRow);
2200 for (int iCol = 0; iCol < _uploadTable->columnCount(); iCol++) {
2201 if (iCol == 3) {
2202 QLineEdit* passwd = new QLineEdit();
2203 passwd->setFrame(false);
2204 passwd->setEchoMode(QLineEdit::PasswordEchoOnEdit);
2205 _uploadTable->setCellWidget(iRow, iCol, passwd);
2206 }
2207 else if (iCol == 4) {
2208 QComboBox* system = new QComboBox();
2209 system->setEditable(false);
2210 system->addItems(QString(",IGS08,ETRF2000,NAD83,GDA94,SIRGAS95,SIRGAS2000,DREF91,Custom").split(","));
2211 system->setFrame(false);
2212 _uploadTable->setCellWidget(iRow, iCol, system);
2213 }
2214 else if (iCol == 5) {
2215 QCheckBox* com = new QCheckBox();
2216 _uploadTable->setCellWidget(iRow, iCol, com);
2217 }
2218 else if (iCol == 11) {
2219 bncTableItem* bncIt = new bncTableItem();
2220 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
2221 _uploadTable->setItem(iRow, iCol, bncIt);
2222 BNC_CORE->_uploadTableItems[iRow] = bncIt;
2223 }
2224 else {
2225 _uploadTable->setItem(iRow, iCol, new QTableWidgetItem(""));
2226 }
2227 }
2228}
2229
2230//
2231////////////////////////////////////////////////////////////////////////////
2232void bncWindow::slotDelUploadRow() {
2233 BNC_CORE->_uploadTableItems.clear();
2234 int nRows = _uploadTable->rowCount();
2235 bool flg[nRows];
2236 for (int iRow = 0; iRow < nRows; iRow++) {
2237 if (_uploadTable->isItemSelected(_uploadTable->item(iRow,1))) {
2238 flg[iRow] = true;
2239 }
2240 else {
2241 flg[iRow] = false;
2242 }
2243 }
2244 for (int iRow = nRows-1; iRow >= 0; iRow--) {
2245 if (flg[iRow]) {
2246 _uploadTable->removeRow(iRow);
2247 }
2248 }
2249 for (int iRow = 0; iRow < _uploadTable->rowCount(); iRow++) {
2250 BNC_CORE->_uploadTableItems[iRow] =
2251 (bncTableItem*) _uploadTable->item(iRow, 11);
2252 }
2253 nRows = _uploadTable->rowCount();
2254 if (nRows < 1) {
2255 enableWidget(false, _uploadIntrComboBox);
2256 enableWidget(false, _uploadSamplRtcmEphCorrSpinBox);
2257 enableWidget(false, _uploadSamplSp3SpinBox);
2258 enableWidget(false, _uploadSamplClkRnxSpinBox);
2259 }
2260}
2261
2262//
2263////////////////////////////////////////////////////////////////////////////
2264void bncWindow::populateUploadTable() {
2265 for (int iRow = _uploadTable->rowCount()-1; iRow >=0; iRow--) {
2266 _uploadTable->removeRow(iRow);
2267 }
2268
2269 bncSettings settings;
2270
2271 int iRow = -1;
2272 QListIterator<QString> it(settings.value("uploadMountpointsOut").toStringList());
2273 while (it.hasNext()) {
2274 QStringList hlp = it.next().split(",");
2275 if (hlp.size() > 6) {
2276 ++iRow;
2277 _uploadTable->insertRow(iRow);
2278 }
2279 for (int iCol = 0; iCol < hlp.size(); iCol++) {
2280 if (iCol == 3) {
2281 QLineEdit* passwd = new QLineEdit();
2282 passwd->setFrame(false);
2283 passwd->setEchoMode(QLineEdit::PasswordEchoOnEdit);
2284 passwd->setText(hlp[iCol]);
2285 _uploadTable->setCellWidget(iRow, iCol, passwd);
2286 }
2287 else if (iCol == 4) {
2288 QComboBox* system = new QComboBox();
2289 system->setEditable(false);
2290 system->addItems(QString(",IGS08,ETRF2000,NAD83,GDA94,SIRGAS95,SIRGAS2000,DREF91,Custom").split(","));
2291 system->setFrame(false);
2292 system->setCurrentIndex(system->findText(hlp[iCol]));
2293 _uploadTable->setCellWidget(iRow, iCol, system);
2294 }
2295 else if (iCol == 5) {
2296 QCheckBox* com = new QCheckBox();
2297 if (hlp[iCol].toInt() == Qt::Checked) {
2298 com->setCheckState(Qt::Checked);
2299 }
2300 _uploadTable->setCellWidget(iRow, iCol, com);
2301 }
2302 else if (iCol == 11) {
2303 bncTableItem* bncIt = new bncTableItem();
2304 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
2305 _uploadTable->setItem(iRow, iCol, bncIt);
2306 BNC_CORE->_uploadTableItems[iRow] = bncIt;
2307 }
2308 else {
2309 _uploadTable->setItem(iRow, iCol, new QTableWidgetItem(hlp[iCol]));
2310 }
2311 }
2312 }
2313}
2314
2315//
2316////////////////////////////////////////////////////////////////////////////
2317void bncWindow::slotSetUploadTrafo() {
2318 bncCustomTrafo* dlg = new bncCustomTrafo(this);
2319 dlg->exec();
2320 delete dlg;
2321}
2322
2323// Progress Bar Change
2324////////////////////////////////////////////////////////////////////////////
2325void bncWindow::slotPostProgress(int nEpo) {
2326 if (_actStart) {
2327 _actStart->setText(QString("%1 Epochs").arg(nEpo));
2328 }
2329}
2330
2331// Start Post-Processing Reqc
2332////////////////////////////////////////////////////////////////////////////
2333void bncWindow::startPostProcessingReqc() {
2334 _runningPostProcessingReqc = true;
2335 enableStartStop();
2336 if (_reqcActionComboBox->currentText() == "Analyze") {
2337 t_reqcAnalyze* reqcAnalyze = new t_reqcAnalyze(this);
2338 connect(reqcAnalyze, SIGNAL(finished()),
2339 this, SLOT(slotFinishedPostProcessingReqc()));
2340 reqcAnalyze->start();
2341 }
2342 else {
2343 t_reqcEdit* reqcEdit = new t_reqcEdit(this);
2344 connect(reqcEdit, SIGNAL(finished()),
2345 this, SLOT(slotFinishedPostProcessingReqc()));
2346 reqcEdit->start();
2347 }
2348}
2349
2350// Post-Processing Reqc Finished
2351////////////////////////////////////////////////////////////////////////////
2352void bncWindow::slotFinishedPostProcessingReqc() {
2353 _runningPostProcessingReqc = false;
2354 if (_reqcActionComboBox->currentText() != "Analyze") {
2355 QMessageBox::information(this, "Information",
2356 "RINEX Processing Thread Finished");
2357 }
2358 enableStartStop();
2359}
2360
2361// Edit teqc-like editing options
2362////////////////////////////////////////////////////////////////////////////
2363void bncWindow::slotReqcEditOption() {
2364 reqcDlg* dlg = new reqcDlg(this);
2365 dlg->move(this->pos().x()+50, this->pos().y()+50);
2366 dlg->exec();
2367 delete dlg;
2368}
2369
2370// Enable/Disable Start and Stop Buttons
2371////////////////////////////////////////////////////////////////////////////
2372void bncWindow::enableStartStop() {
2373
2374 if ( _reqcActionComboBox && !_reqcActionComboBox->currentText().isEmpty() ) {
2375 if (_runningPostProcessingReqc) {
2376 _actStart->setEnabled(false);
2377 }
2378 else {
2379 _actStart->setEnabled(true);
2380 }
2381 _actStop->setEnabled(false);
2382 }
2383 else {
2384 if (_runningRealTime) {
2385 _actStart->setEnabled(false);
2386 _actStop->setEnabled(true);
2387 }
2388 else {
2389 _actStop->setEnabled(false);
2390 if (_mountPointsTable->rowCount() == 0) {
2391 _actStart->setEnabled(false);
2392 }
2393 else {
2394 _actStart->setEnabled(true);
2395 }
2396 }
2397 }
2398}
2399
2400// Show Map
2401////////////////////////////////////////////////////////////////////////////
2402void bncWindow::slotMapMountPoints() {
2403 saveOptions();
2404 t_bncMap* bncMap = new t_bncMap(this);
2405 bncMap->setMinimumSize(800, 600);
2406 bncMap->setWindowTitle("Selected Mountpoints");
2407
2408 bncSettings settings;
2409 QListIterator<QString> it(settings.value("mountPoints").toStringList());
2410 while (it.hasNext()) {
2411 QStringList hlp = it.next().split(" ");
2412 if (hlp.size() < 5) continue;
2413 QUrl url(hlp[0]);
2414 double latDeg = hlp[2].toDouble();
2415 double lonDeg = hlp[3].toDouble();
2416 bncMap->slotNewPoint(QFileInfo(url.path()).fileName(), latDeg, lonDeg);
2417 }
2418
2419 bncMap->show();
2420}
2421
2422// Show Map
2423////////////////////////////////////////////////////////////////////////////
2424void bncWindow::slotMapPPP() {
2425#ifdef QT_WEBKIT
2426 saveOptions();
2427 enableWidget(false, _pppWidgets._mapWinButton);
2428 enableWidget(false, _pppWidgets._gmRadioButton);
2429 enableWidget(false, _pppWidgets._osmRadioButton);
2430 enableWidget(false, _pppWidgets._mapWinDotSizeLineEdit);
2431 enableWidget(false, _pppWidgets._mapWinDotColorComboBox);
2432
2433 if (!_mapWin) {
2434 _mapWin = new bncMapWin(this);
2435 connect(_mapWin, SIGNAL(mapClosed()), this, SLOT(slotMapPPPClosed()));
2436 connect(BNC_CORE, SIGNAL(newPosition(bncTime, QVector<double>)),
2437 _mapWin, SLOT(slotNewPosition(bncTime, QVector<double>)));
2438 }
2439 _mapWin->show();
2440#else
2441 QMessageBox::information(this, "Information",
2442 "Qt Library compiled without QtWebKit");
2443#endif
2444}
2445
2446// Show Map
2447////////////////////////////////////////////////////////////////////////////
2448void bncWindow::slotMapPPPClosed() {
2449#ifdef QT_WEBKIT
2450 enableWidget(true, _pppWidgets._mapWinButton);
2451 enableWidget(true, _pppWidgets._gmRadioButton);
2452 enableWidget(true, _pppWidgets._osmRadioButton);
2453 enableWidget(true, _pppWidgets._mapWinDotSizeLineEdit);
2454 enableWidget(true, _pppWidgets._mapWinDotColorComboBox);
2455 if (_mapWin) {
2456 QListIterator<bncGetThread*> it(_threads);
2457 while (it.hasNext()) {
2458 bncGetThread* thread = it.next();
2459 thread->disconnect(_mapWin);
2460 }
2461 _mapWin->deleteLater();
2462 _mapWin = 0;
2463 }
2464#endif
2465}
Note: See TracBrowser for help on using the repository browser.