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

Last change on this file since 3788 was 3788, checked in by mervart, 12 years ago
File size: 122.5 KB
Line 
1// Part of BNC, a utility for retrieving decoding and
2// converting GNSS data streams from NTRIP broadcasters.
3//
4// Copyright (C) 2007
5// German Federal Agency for Cartography and Geodesy (BKG)
6// http://www.bkg.bund.de
7// Czech Technical University Prague, Department of Geodesy
8// http://www.fsv.cvut.cz
9//
10// Email: euref-ip@bkg.bund.de
11//
12// This program is free software; you can redistribute it and/or
13// modify it under the terms of the GNU General Public License
14// as published by the Free Software Foundation, version 2.
15//
16// This program is distributed in the hope that it will be useful,
17// but WITHOUT ANY WARRANTY; without even the implied warranty of
18// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
19// GNU General Public License for more details.
20//
21// You should have received a copy of the GNU General Public License
22// along with this program; if not, write to the Free Software
23// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
24
25/* -------------------------------------------------------------------------
26 * BKG NTRIP Client
27 * -------------------------------------------------------------------------
28 *
29 * Class: bncWindow
30 *
31 * Purpose: This class implements the main application window.
32 *
33 * Author: L. Mervart
34 *
35 * Created: 24-Dec-2005
36 *
37 * Changes:
38 *
39 * -----------------------------------------------------------------------*/
40
41#include <iostream>
42
43#include <unistd.h>
44#include "bncwindow.h"
45#include "bncapp.h"
46#include "bncgetthread.h"
47#include "bnctabledlg.h"
48#include "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 "teqcdlg.h"
65#ifdef USE_POSTPROCESSING
66# include "rinex/bncpostprocess.h"
67#endif
68
69using namespace std;
70
71// Constructor
72////////////////////////////////////////////////////////////////////////////
73bncWindow::bncWindow() {
74
75 _caster = 0;
76 _casterEph = 0;
77
78 _bncFigure = new bncFigure(this);
79 _bncFigureLate = new bncFigureLate(this);
80 _bncFigurePPP = new bncFigurePPP(this);
81 _runningRealTime = false;
82 _runningPostProcessingPPP = false;
83 _runningPostProcessingTeqc = false;
84
85 int ww = QFontMetrics(this->font()).width('w');
86
87 static const QStringList labels = QString("account, Streams: resource loader / mountpoint, decoder, lat, long, nmea, ntrip, bytes").split(",");
88
89 setMinimumSize(85*ww, 65*ww);
90
91 setWindowTitle(tr("BKG Ntrip Client (BNC) Version " BNCVERSION));
92
93 connect((bncApp*)qApp, SIGNAL(newMessage(QByteArray,bool)),
94 this, SLOT(slotWindowMessage(QByteArray,bool)));
95
96 // Create Actions
97 // --------------
98 _actHelp = new QAction(tr("&Help Contents"),this);
99 connect(_actHelp, SIGNAL(triggered()), SLOT(slotHelp()));
100
101 _actAbout = new QAction(tr("&About BNC"),this);
102 connect(_actAbout, SIGNAL(triggered()), SLOT(slotAbout()));
103
104 _actFlowchart = new QAction(tr("&Flow Chart"),this);
105 connect(_actFlowchart, SIGNAL(triggered()), SLOT(slotFlowchart()));
106
107 _actFontSel = new QAction(tr("Select &Font"),this);
108 connect(_actFontSel, SIGNAL(triggered()), SLOT(slotFontSel()));
109
110 _actSaveOpt = new QAction(tr("&Save && Reread Configuration"),this);
111 connect(_actSaveOpt, SIGNAL(triggered()), SLOT(slotSaveOptions()));
112
113 _actQuit = new QAction(tr("&Quit"),this);
114 connect(_actQuit, SIGNAL(triggered()), SLOT(close()));
115
116 _actAddMountPoints = new QAction(tr("Add &Stream"),this);
117 connect(_actAddMountPoints, SIGNAL(triggered()), SLOT(slotAddMountPoints()));
118
119 _actDeleteMountPoints = new QAction(tr("&Delete Stream"),this);
120 connect(_actDeleteMountPoints, SIGNAL(triggered()), SLOT(slotDeleteMountPoints()));
121 _actDeleteMountPoints->setEnabled(false);
122
123 _actStart = new QAction(tr("Sta&rt"),this);
124 connect(_actStart, SIGNAL(triggered()), SLOT(slotStart()));
125
126 _actStop = new QAction(tr("Sto&p"),this);
127 connect(_actStop, SIGNAL(triggered()), SLOT(slotStop()));
128
129 _actwhatsthis= new QAction(tr("Help ?=Shift+F1"),this);
130 connect(_actwhatsthis, SIGNAL(triggered()), SLOT(slotWhatsThis()));
131
132 CreateMenu();
133 AddToolbar();
134
135 bncSettings settings;
136
137 // Network Options
138 // ---------------
139 _proxyHostLineEdit = new QLineEdit(settings.value("proxyHost").toString());
140 _proxyPortLineEdit = new QLineEdit(settings.value("proxyPort").toString());
141
142 connect(_proxyHostLineEdit, SIGNAL(textChanged(const QString &)),
143 this, SLOT(slotBncTextChanged()));
144
145 _sslCaCertPathLineEdit = new QLineEdit(settings.value("sslCaCertPath").toString());
146 _ignoreSslErrorsCheckBox = new QCheckBox();
147 _ignoreSslErrorsCheckBox->setCheckState(Qt::CheckState(
148 settings.value("ignoreSslErrors").toInt()));
149
150 // General Options
151 // ---------------
152 _logFileLineEdit = new QLineEdit(settings.value("logFile").toString());
153 _rawOutFileLineEdit = new QLineEdit(settings.value("rawOutFile").toString());
154 _rnxAppendCheckBox = new QCheckBox();
155 _rnxAppendCheckBox->setCheckState(Qt::CheckState(
156 settings.value("rnxAppend").toInt()));
157 _onTheFlyComboBox = new QComboBox();
158 _onTheFlyComboBox->setEditable(false);
159 _onTheFlyComboBox->addItems(QString("1 day,1 hour,1 min").split(","));
160 int ii = _onTheFlyComboBox->findText(settings.value("onTheFlyInterval").toString());
161 if (ii != -1) {
162 _onTheFlyComboBox->setCurrentIndex(ii);
163 }
164 _autoStartCheckBox = new QCheckBox();
165 _autoStartCheckBox->setCheckState(Qt::CheckState(
166 settings.value("autoStart").toInt()));
167
168 // RINEX Observations Options
169 // --------------------------
170 _rnxPathLineEdit = new QLineEdit(settings.value("rnxPath").toString());
171 _rnxIntrComboBox = new QComboBox();
172 _rnxIntrComboBox->setEditable(false);
173 _rnxIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
174 ii = _rnxIntrComboBox->findText(settings.value("rnxIntr").toString());
175 if (ii != -1) {
176 _rnxIntrComboBox->setCurrentIndex(ii);
177 }
178 _rnxSamplSpinBox = new QSpinBox();
179 _rnxSamplSpinBox->setMinimum(0);
180 _rnxSamplSpinBox->setMaximum(60);
181 _rnxSamplSpinBox->setSingleStep(5);
182 _rnxSamplSpinBox->setValue(settings.value("rnxSampl").toInt());
183 _rnxSamplSpinBox->setSuffix(" sec");
184 _rnxSkelLineEdit = new QLineEdit(settings.value("rnxSkel").toString());
185 _rnxSkelLineEdit->setMaximumWidth(5*ww);
186 _rnxScrpLineEdit = new QLineEdit(settings.value("rnxScript").toString());
187 _rnxV3CheckBox = new QCheckBox();
188 _rnxV3CheckBox->setCheckState(Qt::CheckState(settings.value("rnxV3").toInt()));
189
190 connect(_rnxPathLineEdit, SIGNAL(textChanged(const QString &)),
191 this, SLOT(slotBncTextChanged()));
192
193 // RINEX Ephemeris Options
194 // -----------------------
195 _ephPathLineEdit = new QLineEdit(settings.value("ephPath").toString());
196 _ephIntrComboBox = new QComboBox();
197 _ephIntrComboBox->setEditable(false);
198 _ephIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
199 int jj = _ephIntrComboBox->findText(settings.value("ephIntr").toString());
200 if (jj != -1) {
201 _ephIntrComboBox->setCurrentIndex(jj);
202 }
203 _outEphPortLineEdit = new QLineEdit(settings.value("outEphPort").toString());
204 _ephV3CheckBox = new QCheckBox();
205 _ephV3CheckBox->setCheckState(Qt::CheckState(settings.value("ephV3").toInt()));
206
207 connect(_outEphPortLineEdit, SIGNAL(textChanged(const QString &)),
208 this, SLOT(slotBncTextChanged()));
209
210 connect(_ephPathLineEdit, SIGNAL(textChanged(const QString &)),
211 this, SLOT(slotBncTextChanged()));
212
213 // Broadcast Corrections Options
214 // -----------------------------
215 _corrPathLineEdit = new QLineEdit(settings.value("corrPath").toString());
216 _corrIntrComboBox = new QComboBox();
217 _corrIntrComboBox->setEditable(false);
218 _corrIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
219 int mm = _corrIntrComboBox->findText(settings.value("corrIntr").toString());
220 if (mm != -1) {
221 _corrIntrComboBox->setCurrentIndex(mm);
222 }
223 _corrPortLineEdit = new QLineEdit(settings.value("corrPort").toString());
224 _corrTimeSpinBox = new QSpinBox();
225 _corrTimeSpinBox->setMinimum(0);
226 _corrTimeSpinBox->setMaximum(60);
227 _corrTimeSpinBox->setSingleStep(1);
228 _corrTimeSpinBox->setSuffix(" sec");
229 _corrTimeSpinBox->setValue(settings.value("corrTime").toInt());
230
231 connect(_corrPathLineEdit, SIGNAL(textChanged(const QString &)),
232 this, SLOT(slotBncTextChanged()));
233
234 connect(_corrPortLineEdit, SIGNAL(textChanged(const QString &)),
235 this, SLOT(slotBncTextChanged()));
236
237 // Feed Engine Options
238 // -------------------
239 _outPortLineEdit = new QLineEdit(settings.value("outPort").toString());
240 _waitTimeSpinBox = new QSpinBox();
241 _waitTimeSpinBox->setMinimum(1);
242 _waitTimeSpinBox->setMaximum(30);
243 _waitTimeSpinBox->setSingleStep(1);
244 _waitTimeSpinBox->setSuffix(" sec");
245 _waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
246 _binSamplSpinBox = new QSpinBox();
247 _binSamplSpinBox->setMinimum(0);
248 _binSamplSpinBox->setMaximum(60);
249 _binSamplSpinBox->setSingleStep(5);
250 _binSamplSpinBox->setValue(settings.value("binSampl").toInt());
251 _binSamplSpinBox->setSuffix(" sec");
252 _outFileLineEdit = new QLineEdit(settings.value("outFile").toString());
253 _outUPortLineEdit = new QLineEdit(settings.value("outUPort").toString());
254
255 connect(_outPortLineEdit, SIGNAL(textChanged(const QString &)),
256 this, SLOT(slotBncTextChanged()));
257
258 connect(_outFileLineEdit, SIGNAL(textChanged(const QString &)),
259 this, SLOT(slotBncTextChanged()));
260
261 // Serial Output Options
262 // ---------------------
263 _serialMountPointLineEdit = new QLineEdit(settings.value("serialMountPoint").toString());
264 _serialPortNameLineEdit = new QLineEdit(settings.value("serialPortName").toString());
265 _serialBaudRateComboBox = new QComboBox();
266 _serialBaudRateComboBox->addItems(QString("110,300,600,"
267 "1200,2400,4800,9600,19200,38400,57600,115200").split(","));
268 int kk = _serialBaudRateComboBox->findText(settings.value("serialBaudRate").toString());
269 if (kk != -1) {
270 _serialBaudRateComboBox->setCurrentIndex(kk);
271 }
272 _serialFlowControlComboBox = new QComboBox();
273 _serialFlowControlComboBox->addItems(QString("OFF,XONXOFF,HARDWARE").split(","));
274 kk = _serialFlowControlComboBox->findText(settings.value("serialFlowControl").toString());
275 if (kk != -1) {
276 _serialFlowControlComboBox->setCurrentIndex(kk);
277 }
278 _serialDataBitsComboBox = new QComboBox();
279 _serialDataBitsComboBox->addItems(QString("5,6,7,8").split(","));
280 kk = _serialDataBitsComboBox->findText(settings.value("serialDataBits").toString());
281 if (kk != -1) {
282 _serialDataBitsComboBox->setCurrentIndex(kk);
283 }
284 _serialParityComboBox = new QComboBox();
285 _serialParityComboBox->addItems(QString("NONE,ODD,EVEN,SPACE").split(","));
286 kk = _serialParityComboBox->findText(settings.value("serialParity").toString());
287 if (kk != -1) {
288 _serialParityComboBox->setCurrentIndex(kk);
289 }
290 _serialStopBitsComboBox = new QComboBox();
291 _serialStopBitsComboBox->addItems(QString("1,2").split(","));
292 kk = _serialStopBitsComboBox->findText(settings.value("serialStopBits").toString());
293 if (kk != -1) {
294 _serialStopBitsComboBox->setCurrentIndex(kk);
295 }
296 _serialAutoNMEAComboBox = new QComboBox();
297 _serialAutoNMEAComboBox->addItems(QString("Auto,Manual").split(","));
298 kk = _serialAutoNMEAComboBox->findText(settings.value("serialAutoNMEA").toString());
299 if (kk != -1) {
300 _serialAutoNMEAComboBox->setCurrentIndex(kk);
301 }
302 _serialFileNMEALineEdit = new QLineEdit(settings.value("serialFileNMEA").toString());
303 _serialHeightNMEALineEdit = new QLineEdit(settings.value("serialHeightNMEA").toString());
304
305 connect(_serialMountPointLineEdit, SIGNAL(textChanged(const QString &)),
306 this, SLOT(slotBncTextChanged()));
307
308 connect(_serialAutoNMEAComboBox, SIGNAL(currentIndexChanged(const QString &)),
309 this, SLOT(slotBncTextChanged()));
310
311 // Outages Options
312 // ---------------
313 _obsRateComboBox = new QComboBox();
314 _obsRateComboBox->setEditable(false);
315 _obsRateComboBox->addItems(QString(",0.1 Hz,0.2 Hz,0.5 Hz,1 Hz,5 Hz").split(","));
316 kk = _obsRateComboBox->findText(settings.value("obsRate").toString());
317 if (kk != -1) {
318 _obsRateComboBox->setCurrentIndex(kk);
319 }
320 _adviseFailSpinBox = new QSpinBox();
321 _adviseFailSpinBox->setMinimum(0);
322 _adviseFailSpinBox->setMaximum(60);
323 _adviseFailSpinBox->setSingleStep(1);
324 _adviseFailSpinBox->setSuffix(" min");
325 _adviseFailSpinBox->setValue(settings.value("adviseFail").toInt());
326 _adviseRecoSpinBox = new QSpinBox();
327 _adviseRecoSpinBox->setMinimum(0);
328 _adviseRecoSpinBox->setMaximum(60);
329 _adviseRecoSpinBox->setSingleStep(1);
330 _adviseRecoSpinBox->setSuffix(" min");
331 _adviseRecoSpinBox->setValue(settings.value("adviseReco").toInt());
332 _adviseScriptLineEdit = new QLineEdit(settings.value("adviseScript").toString());
333
334 connect(_obsRateComboBox, SIGNAL(currentIndexChanged(const QString &)),
335 this, SLOT(slotBncTextChanged()));
336
337 // Miscellaneous Options
338 // ---------------------
339 _miscMountLineEdit = new QLineEdit(settings.value("miscMount").toString());
340 _perfIntrComboBox = new QComboBox();
341 _perfIntrComboBox->setEditable(false);
342 _perfIntrComboBox->addItems(QString(",2 sec, 10 sec,1 min,5 min,15 min,1 hour,6 hours,1 day").split(","));
343 int ll = _perfIntrComboBox->findText(settings.value("perfIntr").toString());
344 if (ll != -1) {
345 _perfIntrComboBox->setCurrentIndex(ll);
346 }
347 _scanRTCMCheckBox = new QCheckBox();
348 _scanRTCMCheckBox->setCheckState(Qt::CheckState(
349 settings.value("scanRTCM").toInt()));
350
351 connect(_miscMountLineEdit, SIGNAL(textChanged(const QString &)),
352 this, SLOT(slotBncTextChanged()));
353
354 // PPP Options
355 // -----------
356 _pppMountLineEdit = new QLineEdit(settings.value("pppMount").toString());
357 _pppMountLineEdit->setMaximumWidth(8*ww);
358 _pppCorrMountLineEdit = new QLineEdit(settings.value("pppCorrMount").toString());
359 _pppMountLineEdit->setMinimumWidth(8*ww);
360 _pppCorrMountLineEdit->setMinimumWidth(8*ww);
361 _pppCorrMountLineEdit->setMaximumWidth(8*ww);
362 _pppNMEALineEdit = new QLineEdit(settings.value("nmeaFile").toString());
363 _pppNMEAPortLineEdit = new QLineEdit(settings.value("nmeaPort").toString());
364 _pppSigCLineEdit = new QLineEdit(settings.value("pppSigmaCode").toString());
365 _pppSigPLineEdit = new QLineEdit(settings.value("pppSigmaPhase").toString());
366 _pppSigCrd0 = new QLineEdit(settings.value("pppSigCrd0").toString());
367 _pppSigCrdP = new QLineEdit(settings.value("pppSigCrdP").toString());
368 _pppSigTrp0 = new QLineEdit(settings.value("pppSigTrp0").toString());
369 _pppSigTrpP = new QLineEdit(settings.value("pppSigTrpP").toString());
370 _pppAverageLineEdit = new QLineEdit(settings.value("pppAverage").toString());
371 _pppQuickStartLineEdit = new QLineEdit(settings.value("pppQuickStart").toString());
372 _pppMaxSolGapLineEdit = new QLineEdit(settings.value("pppMaxSolGap").toString());
373 _pppRefCrdXLineEdit = new QLineEdit(settings.value("pppRefCrdX").toString());
374 _pppRefCrdYLineEdit = new QLineEdit(settings.value("pppRefCrdY").toString());
375 _pppRefCrdZLineEdit = new QLineEdit(settings.value("pppRefCrdZ").toString());
376 _pppRefdNLineEdit = new QLineEdit(settings.value("pppRefdN").toString());
377 _pppRefdELineEdit = new QLineEdit(settings.value("pppRefdE").toString());
378 _pppRefdULineEdit = new QLineEdit(settings.value("pppRefdU").toString());
379 _pppSync = new QLineEdit(settings.value("pppSync").toString());
380 _pppAntexFileChooser = new qtFileChooser;
381 _pppAntexFileChooser->setMinimumWidth(12*ww);
382 _pppAntennaLineEdit = new QLineEdit(settings.value("pppAntenna").toString());
383 _pppAntexFileChooser->setFileName(settings.value("pppAntex").toString());
384
385 _pppSPPComboBox = new QComboBox();
386 _pppSPPComboBox->setEditable(false);
387 _pppSPPComboBox->addItems(QString(",Realtime-PPP,Realtime-SPP,Post-Processing").split(","));
388 int ik = _pppSPPComboBox->findText(settings.value("pppSPP").toString());
389 if (ik != -1) {
390 _pppSPPComboBox->setCurrentIndex(ik);
391 }
392 _pppUsePhaseCheckBox = new QCheckBox();
393 _pppUsePhaseCheckBox->setCheckState(Qt::CheckState(
394 settings.value("pppUsePhase").toInt()));
395 _pppEstTropoCheckBox = new QCheckBox();
396 _pppEstTropoCheckBox->setCheckState(Qt::CheckState(
397 settings.value("pppEstTropo").toInt()));
398 _pppGLONASSCheckBox = new QCheckBox();
399 _pppGLONASSCheckBox->setCheckState(Qt::CheckState(
400 settings.value("pppGLONASS").toInt()));
401 _pppGalileoCheckBox = new QCheckBox();
402 _pppGalileoCheckBox->setCheckState(Qt::CheckState(
403 settings.value("pppGalileo").toInt()));
404
405 _pppPlotCoordinates = new QCheckBox();
406 _pppPlotCoordinates->setCheckState(Qt::CheckState(
407 settings.value("pppPlotCoordinates").toInt()));
408
409 _pppApplySatAntCheckBox = new QCheckBox();
410 _pppApplySatAntCheckBox->setCheckState(Qt::CheckState(
411 settings.value("pppApplySatAnt").toInt()));
412
413 connect(_pppMountLineEdit, SIGNAL(textChanged(const QString &)),
414 this, SLOT(slotBncTextChanged()));
415
416 connect(_pppCorrMountLineEdit, SIGNAL(textChanged(const QString &)),
417 this, SLOT(slotBncTextChanged()));
418
419 connect(_pppUsePhaseCheckBox, SIGNAL(stateChanged(int)),
420 this, SLOT(slotBncTextChanged()));
421
422 connect(_pppRefCrdXLineEdit, SIGNAL(textChanged(const QString &)),
423 this, SLOT(slotBncTextChanged()));
424 connect(_pppRefCrdYLineEdit, SIGNAL(textChanged(const QString &)),
425 this, SLOT(slotBncTextChanged()));
426 connect(_pppRefCrdZLineEdit, SIGNAL(textChanged(const QString &)),
427 this, SLOT(slotBncTextChanged()));
428 connect(_pppRefdNLineEdit, SIGNAL(textChanged(const QString &)),
429 this, SLOT(slotBncTextChanged()));
430 connect(_pppRefdELineEdit, SIGNAL(textChanged(const QString &)),
431 this, SLOT(slotBncTextChanged()));
432 connect(_pppRefdULineEdit, SIGNAL(textChanged(const QString &)),
433 this, SLOT(slotBncTextChanged()));
434
435 connect(_pppEstTropoCheckBox, SIGNAL(stateChanged(int)),
436 this, SLOT(slotBncTextChanged()));
437
438 connect(_pppSync, SIGNAL(textChanged(const QString &)),
439 this, SLOT(slotBncTextChanged()));
440
441 connect(_pppSPPComboBox, SIGNAL(currentIndexChanged(const QString &)),
442 this, SLOT(slotBncTextChanged()));
443
444 connect(_pppAntexFileChooser, SIGNAL(fileNameChanged(const QString &)),
445 this, SLOT(slotBncTextChanged()));
446
447 connect(_pppQuickStartLineEdit, SIGNAL(textChanged(const QString &)),
448 this, SLOT(slotBncTextChanged()));
449
450 // Streams
451 // -------
452 _mountPointsTable = new QTableWidget(0,8);
453
454 _mountPointsTable->horizontalHeader()->resizeSection(1,34*ww);
455 _mountPointsTable->horizontalHeader()->resizeSection(2,9*ww);
456 _mountPointsTable->horizontalHeader()->resizeSection(3,7*ww);
457 _mountPointsTable->horizontalHeader()->resizeSection(4,7*ww);
458 _mountPointsTable->horizontalHeader()->resizeSection(5,5*ww);
459 _mountPointsTable->horizontalHeader()->resizeSection(6,5*ww);
460 _mountPointsTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
461 _mountPointsTable->horizontalHeader()->setStretchLastSection(true);
462 _mountPointsTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
463 _mountPointsTable->setHorizontalHeaderLabels(labels);
464 _mountPointsTable->setGridStyle(Qt::NoPen);
465 _mountPointsTable->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
466 _mountPointsTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
467 _mountPointsTable->setSelectionBehavior(QAbstractItemView::SelectRows);
468 _mountPointsTable->hideColumn(0);
469 connect(_mountPointsTable, SIGNAL(itemSelectionChanged()),
470 SLOT(slotSelectionChanged()));
471 populateMountPointsTable();
472
473 _log = new QTextBrowser();
474 _log->setReadOnly(true);
475
476 // Combination
477 // -----------
478 _cmbTable = new QTableWidget(0,3);
479 _cmbTable->setHorizontalHeaderLabels(QString("Mountpoint, AC Name, Weight").split(","));
480 _cmbTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
481 _cmbTable->setSelectionBehavior(QAbstractItemView::SelectRows);
482 _cmbTable->setMaximumWidth(30*ww);
483 _cmbTable->horizontalHeader()->resizeSection(0,10*ww);
484 _cmbTable->horizontalHeader()->resizeSection(1,8*ww);
485 _cmbTable->horizontalHeader()->resizeSection(2,8*ww);
486 _cmbTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
487 _cmbTable->horizontalHeader()->setStretchLastSection(true);
488 _cmbTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
489
490 _cmbMaxresLineEdit = new QLineEdit(settings.value("cmbMaxres").toString());
491
492 QPushButton* addCmbRowButton = new QPushButton("Add Row");
493 QPushButton* delCmbRowButton = new QPushButton("Delete");
494
495 connect(_cmbTable, SIGNAL(itemSelectionChanged()),
496 SLOT(slotBncTextChanged()));
497
498 _cmbMethodComboBox = new QComboBox();
499 _cmbMethodComboBox->setEditable(false);
500 _cmbMethodComboBox->addItems(QString("Filter,Single-Epoch").split(","));
501 int im = _cmbMethodComboBox->findText(settings.value("cmbMethod").toString());
502 if (im != -1) {
503 _cmbMethodComboBox->setCurrentIndex(im);
504 }
505
506 // Upload Results
507 // -------------
508 _uploadTable = new QTableWidget(0,9);
509 _uploadTable->setHorizontalHeaderLabels(QString("Host, Port, Mount, Password, System, CoM, SP3 File, RNX File, bytes").split(","));
510 _uploadTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
511 _uploadTable->setSelectionBehavior(QAbstractItemView::SelectRows);
512 _uploadTable->horizontalHeader()->resizeSection(0,13*ww);
513 _uploadTable->horizontalHeader()->resizeSection(1,5*ww);
514 _uploadTable->horizontalHeader()->resizeSection(2,6*ww);
515 _uploadTable->horizontalHeader()->resizeSection(3,8*ww);
516 _uploadTable->horizontalHeader()->resizeSection(4,11*ww);
517 _uploadTable->horizontalHeader()->resizeSection(5,4*ww);
518 _uploadTable->horizontalHeader()->resizeSection(6,15*ww);
519 _uploadTable->horizontalHeader()->resizeSection(7,15*ww);
520 _uploadTable->horizontalHeader()->resizeSection(8,10*ww);
521 _uploadTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
522 _uploadTable->horizontalHeader()->setStretchLastSection(true);
523 _uploadTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
524
525 connect(_uploadTable, SIGNAL(itemSelectionChanged()),
526 SLOT(slotBncTextChanged()));
527
528 QPushButton* addUploadRowButton = new QPushButton("Add Row");
529 QPushButton* delUploadRowButton = new QPushButton("Del Row");
530 QPushButton* setUploadTrafoButton = new QPushButton("Custom Trafo");
531 _uploadIntrComboBox = new QComboBox;
532 _uploadIntrComboBox->setEditable(false);
533 _uploadIntrComboBox->addItems(QString("1 day,1 hour, 30 min,15 min,10 min,5 min,2 min,1 min").split(","));
534 ii = _uploadIntrComboBox->findText(settings.value("uploadIntr").toString());
535 if (ii != -1) {
536 _uploadIntrComboBox->setCurrentIndex(ii);
537 }
538 _uploadSamplSpinBox = new QSpinBox;
539 _uploadSamplSpinBox->setMinimum(5);
540 _uploadSamplSpinBox->setMaximum(60);
541 _uploadSamplSpinBox->setSingleStep(5);
542 _uploadSamplSpinBox->setMaximumWidth(9*ww);
543 _uploadSamplSpinBox->setValue(settings.value("uploadSampl").toInt());
544 _uploadSamplSpinBox->setSuffix(" sec");
545
546 _uploadSamplOrbSpinBox = new QSpinBox;
547 _uploadSamplOrbSpinBox->setMinimum(0);
548 _uploadSamplOrbSpinBox->setMaximum(60);
549 _uploadSamplOrbSpinBox->setSingleStep(5);
550 _uploadSamplOrbSpinBox->setMaximumWidth(9*ww);
551 _uploadSamplOrbSpinBox->setValue(settings.value("uploadSamplOrb").toInt());
552 _uploadSamplOrbSpinBox->setSuffix(" sec");
553
554 // Upload RTCM3 Ephemeris
555 // ----------------------
556 _uploadEphHostLineEdit = new QLineEdit(settings.value("uploadEphHost").toString());
557 _uploadEphPortLineEdit = new QLineEdit(settings.value("uploadEphPort").toString());
558 _uploadEphPasswordLineEdit = new QLineEdit(settings.value("uploadEphPassword").toString());
559 _uploadEphPasswordLineEdit->setEchoMode(QLineEdit::PasswordEchoOnEdit);
560 _uploadEphMountpointLineEdit = new QLineEdit(settings.value("uploadEphMountpoint").toString());
561 _uploadEphSampleSpinBox = new QSpinBox;
562 _uploadEphSampleSpinBox->setMinimum(5);
563 _uploadEphSampleSpinBox->setMaximum(60);
564 _uploadEphSampleSpinBox->setSingleStep(5);
565 _uploadEphSampleSpinBox->setMaximumWidth(9*ww);
566 _uploadEphSampleSpinBox->setValue(settings.value("uploadEphSample").toInt());
567 _uploadEphSampleSpinBox->setSuffix(" sec");
568 _uploadEphBytesCounter = new bncBytesCounter;
569
570 // Canvas with Editable Fields
571 // ---------------------------
572 _canvas = new QWidget;
573 setCentralWidget(_canvas);
574
575 _aogroup = new QTabWidget();
576 QWidget* pgroup = new QWidget();
577 QWidget* ggroup = new QWidget();
578 QWidget* sgroup = new QWidget();
579 QWidget* egroup = new QWidget();
580 QWidget* agroup = new QWidget();
581 QWidget* cgroup = new QWidget();
582 QWidget* ogroup = new QWidget();
583 QWidget* rgroup = new QWidget();
584 QWidget* sergroup = new QWidget();
585 QWidget* pppgroup = new QWidget();
586 QWidget* ppp2group = new QWidget();
587 QWidget* teqcgroup = new QWidget();
588 QWidget* cmbgroup = new QWidget();
589 QWidget* uploadgroup = new QWidget();
590 QWidget* uploadEphgroup = new QWidget();
591 _aogroup->addTab(pgroup,tr("Network"));
592 _aogroup->addTab(ggroup,tr("General"));
593 _aogroup->addTab(ogroup,tr("RINEX Observations"));
594 _aogroup->addTab(egroup,tr("RINEX Ephemeris"));
595 _aogroup->addTab(cgroup,tr("Broadcast Corrections"));
596 _aogroup->addTab(sgroup,tr("Feed Engine"));
597 _aogroup->addTab(sergroup,tr("Serial Output"));
598 _aogroup->addTab(agroup,tr("Outages"));
599 _aogroup->addTab(rgroup,tr("Miscellaneous"));
600 _aogroup->addTab(pppgroup,tr("PPP (1)")); _tabIndexPPP1 = _aogroup->count() - 1;
601 _aogroup->addTab(ppp2group,tr("PPP (2)")); _tabIndexPPP2 = _aogroup->count() - 1;
602 _aogroup->addTab(teqcgroup,tr("Teqc")); _tabIndexTeqc = _aogroup->count() - 1;
603#ifdef USE_COMBINATION
604 _aogroup->addTab(cmbgroup,tr("Combination"));
605#endif
606 _aogroup->addTab(uploadgroup,tr("Upload (clk)"));
607 _aogroup->addTab(uploadEphgroup,tr("Upload (eph)"));
608
609 // Log Tab
610 // -------
611 _loggroup = new QTabWidget();
612 _loggroup->addTab(_log,tr("Log"));
613 _loggroup->addTab(_bncFigure,tr("Throughput"));
614 _loggroup->addTab(_bncFigureLate,tr("Latency"));
615 _loggroup->addTab(_bncFigurePPP,tr("PPP Plot"));
616
617 // Netowork (Proxy and SSL) Tab
618 // ----------------------------
619 QGridLayout* pLayout = new QGridLayout;
620 pLayout->setColumnMinimumWidth(0,13*ww);
621 _proxyPortLineEdit->setMaximumWidth(9*ww);
622
623 pLayout->addWidget(new QLabel("Settings for proxy in protected networks and for SSL authorization, leave boxes blank if none."),0, 0, 1, 50);
624 pLayout->addWidget(new QLabel("Proxy host"), 1, 0);
625 pLayout->addWidget(_proxyHostLineEdit, 1, 1, 1,10);
626 pLayout->addWidget(new QLabel("Proxy port"), 2, 0);
627 pLayout->addWidget(_proxyPortLineEdit, 2, 1);
628 pLayout->addWidget(new QLabel("Path to SSL Certificates"), 3, 0);
629 pLayout->addWidget(_sslCaCertPathLineEdit, 3, 1, 1,10);
630 pLayout->addWidget(new QLabel("Default: " + bncSslConfig::defaultPath()), 3,11, 1,20);
631 pLayout->addWidget(new QLabel("Ignore SSL Authorization Errors"), 4, 0);
632 pLayout->addWidget(_ignoreSslErrorsCheckBox, 4, 1, 1,10);
633 pLayout->addWidget(new QLabel(" "), 4, 0);
634 pLayout->addWidget(new QLabel(" "), 5, 0);
635 pLayout->addWidget(new QLabel(" "), 6, 0);
636 pgroup->setLayout(pLayout);
637
638 // General Tab
639 // -----------
640 QGridLayout* gLayout = new QGridLayout;
641 gLayout->setColumnMinimumWidth(0,14*ww);
642 _onTheFlyComboBox->setMaximumWidth(9*ww);
643
644 gLayout->addWidget(new QLabel("General settings for logfile, file handling, configuration on-the-fly, and auto-start."),0, 0, 1, 50);
645 gLayout->addWidget(new QLabel("Logfile (full path)"), 1, 0);
646 gLayout->addWidget(_logFileLineEdit, 1, 1, 1,30);
647 gLayout->addWidget(new QLabel("Append files"), 2, 0);
648 gLayout->addWidget(_rnxAppendCheckBox, 2, 1);
649 gLayout->addWidget(new QLabel("Reread configuration"), 3, 0);
650 gLayout->addWidget(_onTheFlyComboBox, 3, 1);
651 gLayout->addWidget(new QLabel("Auto start"), 4, 0);
652 gLayout->addWidget(_autoStartCheckBox, 4, 1);
653 gLayout->addWidget(new QLabel("Raw output file (full path)"), 5, 0);
654 gLayout->addWidget(_rawOutFileLineEdit, 5, 1, 1,30);
655 gLayout->addWidget(new QLabel(" "), 6, 0);
656 ggroup->setLayout(gLayout);
657
658 // RINEX Observations
659 // ------------------
660 QGridLayout* oLayout = new QGridLayout;
661 oLayout->setColumnMinimumWidth(0,14*ww);
662 _rnxIntrComboBox->setMaximumWidth(9*ww);
663 _rnxSamplSpinBox->setMaximumWidth(9*ww);
664
665 oLayout->addWidget(new QLabel("Saving RINEX observation files."),0, 0, 1,50);
666 oLayout->addWidget(new QLabel("Directory"), 1, 0);
667 oLayout->addWidget(_rnxPathLineEdit, 1, 1, 1,24);
668 oLayout->addWidget(new QLabel("Interval"), 2, 0);
669 oLayout->addWidget(_rnxIntrComboBox, 2, 1);
670 oLayout->addWidget(new QLabel(" Sampling"), 2, 2, Qt::AlignRight);
671 oLayout->addWidget(_rnxSamplSpinBox, 2, 3, Qt::AlignLeft);
672 oLayout->addWidget(new QLabel("Skeleton extension"), 3, 0);
673 oLayout->addWidget(_rnxSkelLineEdit, 3, 1, 1, 1, Qt::AlignLeft);
674 oLayout->addWidget(new QLabel("Script (full path)"), 4, 0);
675 oLayout->addWidget(_rnxScrpLineEdit, 4, 1, 1,24);
676 oLayout->addWidget(new QLabel("Version 3"), 5, 0);
677 oLayout->addWidget(_rnxV3CheckBox, 5, 1);
678 oLayout->addWidget(new QLabel(" "), 6, 0);
679 ogroup->setLayout(oLayout);
680
681 // RINEX Ephemeris
682 // ---------------
683 QGridLayout* eLayout = new QGridLayout;
684 eLayout->setColumnMinimumWidth(0,14*ww);
685 _ephIntrComboBox->setMaximumWidth(9*ww);
686 _outEphPortLineEdit->setMaximumWidth(9*ww);
687
688 eLayout->addWidget(new QLabel("Saving RINEX ephemeris files and ephemeris output through IP port."),0,0,1,50);
689 eLayout->addWidget(new QLabel("Directory"), 1, 0);
690 eLayout->addWidget(_ephPathLineEdit, 1, 1, 1,30);
691 eLayout->addWidget(new QLabel("Interval"), 2, 0);
692 eLayout->addWidget(_ephIntrComboBox, 2, 1);
693 eLayout->addWidget(new QLabel("Port"), 3, 0);
694 eLayout->addWidget(_outEphPortLineEdit, 3, 1);
695 eLayout->addWidget(new QLabel("Version 3"), 4, 0);
696 eLayout->addWidget(_ephV3CheckBox, 4, 1);
697 eLayout->addWidget(new QLabel(" "), 5, 0);
698 eLayout->addWidget(new QLabel(" "), 6, 0);
699 egroup->setLayout(eLayout);
700
701
702 // Broadcast Corrections
703 // ---------------------
704 QGridLayout* cLayout = new QGridLayout;
705 cLayout->setColumnMinimumWidth(0,14*ww);
706 _corrIntrComboBox->setMaximumWidth(9*ww);
707 _corrPortLineEdit->setMaximumWidth(9*ww);
708 _corrTimeSpinBox->setMaximumWidth(9*ww);
709
710 cLayout->addWidget(new QLabel("Saving Broadcast Ephemeris correction files and correction output through IP port."),0,0,1,50);
711 cLayout->addWidget(new QLabel("Directory, ASCII"), 1, 0);
712 cLayout->addWidget(_corrPathLineEdit, 1, 1, 1,20);
713 cLayout->addWidget(new QLabel("Interval"), 2, 0);
714 cLayout->addWidget(_corrIntrComboBox, 2, 1);
715 cLayout->addWidget(new QLabel("Port"), 3, 0);
716 cLayout->addWidget(_corrPortLineEdit, 3, 1);
717 cLayout->addWidget(new QLabel(" Wait for full epoch"), 3, 2, Qt::AlignRight);
718 cLayout->addWidget(_corrTimeSpinBox, 3, 3, Qt::AlignLeft);
719 cLayout->addWidget(new QLabel(" "), 4, 0);
720 cLayout->addWidget(new QLabel(" "), 5, 0);
721 cLayout->addWidget(new QLabel(" "), 6, 0);
722 cgroup->setLayout(cLayout);
723
724 // Feed Engine
725 // -----------
726 QGridLayout* sLayout = new QGridLayout;
727 sLayout->setColumnMinimumWidth(0,14*ww);
728 _outPortLineEdit->setMaximumWidth(9*ww);
729 _waitTimeSpinBox->setMaximumWidth(9*ww);
730 _binSamplSpinBox->setMaximumWidth(9*ww);
731 _outUPortLineEdit->setMaximumWidth(9*ww);
732
733 sLayout->addWidget(new QLabel("Output decoded observations in a binary format to feed a real-time GNSS network engine."),0,0,1,50);
734 sLayout->addWidget(new QLabel("Port"), 1, 0);
735 sLayout->addWidget(_outPortLineEdit, 1, 1);
736 sLayout->addWidget(new QLabel("Wait for full epoch"), 1, 2, Qt::AlignRight);
737 sLayout->addWidget(_waitTimeSpinBox, 1, 3, Qt::AlignLeft);
738 sLayout->addWidget(new QLabel("Sampling"), 2, 0);
739 sLayout->addWidget(_binSamplSpinBox, 2, 1, Qt::AlignLeft);
740 sLayout->addWidget(new QLabel("File (full path)"), 3, 0);
741 sLayout->addWidget(_outFileLineEdit, 3, 1, 1, 20);
742 sLayout->addWidget(new QLabel("Port (unsynchronized)"), 4, 0);
743 sLayout->addWidget(_outUPortLineEdit, 4, 1);
744 sLayout->addWidget(new QLabel(" "), 5, 0);
745 sLayout->addWidget(new QLabel(" "), 6, 0);
746 sgroup->setLayout(sLayout);
747
748 // Serial Output
749 // -------------
750 QGridLayout* serLayout = new QGridLayout;
751 serLayout->setColumnMinimumWidth(0,14*ww);
752 _serialBaudRateComboBox->setMaximumWidth(9*ww);
753 _serialFlowControlComboBox->setMaximumWidth(11*ww);
754 _serialDataBitsComboBox->setMaximumWidth(5*ww);
755 _serialParityComboBox->setMaximumWidth(9*ww);
756 _serialStopBitsComboBox->setMaximumWidth(5*ww);
757 _serialAutoNMEAComboBox->setMaximumWidth(9*ww);
758 _serialHeightNMEALineEdit->setMaximumWidth(8*ww);
759
760 serLayout->addWidget(new QLabel("Port settings to feed a serial connected receiver."),0,0,1,30);
761 serLayout->addWidget(new QLabel("Mountpoint"), 1, 0, Qt::AlignLeft);
762 serLayout->addWidget(_serialMountPointLineEdit, 1, 1, 1, 2);
763 serLayout->addWidget(new QLabel("Port name"), 2, 0, Qt::AlignLeft);
764 serLayout->addWidget(_serialPortNameLineEdit, 2, 1, 1, 2);
765 serLayout->addWidget(new QLabel("Baud rate"), 3, 0, Qt::AlignLeft);
766 serLayout->addWidget(_serialBaudRateComboBox, 3, 1);
767 serLayout->addWidget(new QLabel("Flow control"), 3, 2, Qt::AlignRight);
768 serLayout->addWidget(_serialFlowControlComboBox, 3, 3);
769 serLayout->addWidget(new QLabel("Data bits"), 4, 0, Qt::AlignLeft);
770 serLayout->addWidget(_serialDataBitsComboBox, 4, 1);
771 serLayout->addWidget(new QLabel("Parity"), 4, 2, Qt::AlignRight);
772 serLayout->addWidget(_serialParityComboBox, 4, 3);
773 serLayout->addWidget(new QLabel(" Stop bits"), 4, 4, Qt::AlignRight);
774 serLayout->addWidget(_serialStopBitsComboBox, 4, 5);
775 serLayout->addWidget(new QLabel("NMEA"), 5, 0);
776 serLayout->addWidget(_serialAutoNMEAComboBox, 5, 1);
777 serLayout->addWidget(new QLabel(" File (full path)"), 5, 2, Qt::AlignRight);
778 serLayout->addWidget(_serialFileNMEALineEdit, 5, 3, 1,15);
779 serLayout->addWidget(new QLabel("Height"), 5,20, Qt::AlignRight);
780 serLayout->addWidget(_serialHeightNMEALineEdit, 5,21, 1,11);
781 serLayout->addWidget(new QLabel(" "), 6, 0);
782
783 sergroup->setLayout(serLayout);
784
785 // Outages
786 // -------
787 QGridLayout* aLayout = new QGridLayout;
788 aLayout->setColumnMinimumWidth(0,14*ww);
789 _obsRateComboBox->setMaximumWidth(9*ww);
790 _adviseFailSpinBox->setMaximumWidth(9*ww);
791 _adviseRecoSpinBox->setMaximumWidth(9*ww);
792
793 aLayout->addWidget(new QLabel("Failure and recovery reports, advisory notes."),0,0,1,50,Qt::AlignLeft);
794 aLayout->addWidget(new QLabel("Observation rate"), 1, 0);
795 aLayout->addWidget(_obsRateComboBox, 1, 1);
796 aLayout->addWidget(new QLabel("Failure threshold"), 2, 0);
797 aLayout->addWidget(_adviseFailSpinBox, 2, 1);
798 aLayout->addWidget(new QLabel("Recovery threshold"), 3, 0);
799 aLayout->addWidget(_adviseRecoSpinBox, 3, 1);
800 aLayout->addWidget(new QLabel("Script (full path)"), 4, 0);
801 aLayout->addWidget(_adviseScriptLineEdit, 4, 1, 1,30);
802 aLayout->addWidget(new QLabel(" "), 5, 0);
803 aLayout->addWidget(new QLabel(" "), 6, 0);
804 agroup->setLayout(aLayout);
805
806 // Miscellaneous
807 // -------------
808 QGridLayout* rLayout = new QGridLayout;
809 rLayout->setColumnMinimumWidth(0,14*ww);
810 _perfIntrComboBox->setMaximumWidth(9*ww);
811
812 rLayout->addWidget(new QLabel("Log latencies or scan RTCM streams for numbers of message types and antenna information."),0, 0,1,30);
813 rLayout->addWidget(new QLabel("Mountpoint"), 1, 0);
814 rLayout->addWidget(_miscMountLineEdit, 1, 1, 1,7);
815 rLayout->addWidget(new QLabel("Log latency"), 2, 0);
816 rLayout->addWidget(_perfIntrComboBox, 2, 1);
817 rLayout->addWidget(new QLabel("Scan RTCM"), 3, 0);
818 rLayout->addWidget(_scanRTCMCheckBox, 3, 1);
819 rLayout->addWidget(new QLabel(" "), 4, 0);
820 rLayout->addWidget(new QLabel(" "), 5, 0);
821 rLayout->addWidget(new QLabel(" "), 6, 0);
822 rgroup->setLayout(rLayout);
823
824 // PPP Client
825 // ----------
826 QGridLayout* pppLayout = new QGridLayout;
827 pppLayout->setColumnMinimumWidth(0,14*ww);
828 _pppSigCLineEdit->setMaximumWidth(6*ww);
829 _pppSigPLineEdit->setMaximumWidth(6*ww);
830 _pppSigCrd0->setMaximumWidth(6*ww);
831 _pppSigCrdP->setMaximumWidth(6*ww);
832 _pppSigTrp0->setMaximumWidth(6*ww);
833 _pppSigTrpP->setMaximumWidth(6*ww);
834 _pppAverageLineEdit->setMaximumWidth(6*ww);
835 _pppQuickStartLineEdit->setMaximumWidth(6*ww);
836 _pppMaxSolGapLineEdit->setMaximumWidth(6*ww);
837 _pppRefCrdXLineEdit->setMaximumWidth(10*ww);
838 _pppRefCrdYLineEdit->setMaximumWidth(10*ww);
839 _pppRefCrdZLineEdit->setMaximumWidth(10*ww);
840 _pppRefdNLineEdit->setMaximumWidth(5*ww);
841 _pppRefdELineEdit->setMaximumWidth(5*ww);
842 _pppRefdULineEdit->setMaximumWidth(5*ww);
843 _pppSync->setMaximumWidth(6*ww);
844 _pppSPPComboBox->setMinimumWidth(15*ww);
845 _pppNMEAPortLineEdit->setMaximumWidth(10*ww);
846
847 _postObsFileChooser = new qtFileChooser;
848 _postObsFileChooser->setFileName(settings.value("postObsFile").toString());
849 _postNavFileChooser = new qtFileChooser;
850 _postNavFileChooser->setFileName(settings.value("postNavFile").toString());
851 _postCorrFileChooser = new qtFileChooser;
852 _postCorrFileChooser->setFileName(settings.value("postCorrFile").toString());
853 _postOutLineEdit = new QLineEdit(settings.value("postOutFile").toString());
854
855 int ir = 0;
856 pppLayout->addWidget(new QLabel("Precise Point Positioning, Panel 1."), ir, 0, 1, 100);
857 ++ir;
858 pppLayout->addWidget(new QLabel("Obs. Mountpnt"), ir, 0, 1, 10);
859 pppLayout->addWidget(_pppMountLineEdit, ir,10, 1, 10);
860 pppLayout->addWidget(_pppSPPComboBox, ir,20, 1, 15);
861 pppLayout->addWidget(new QLabel(" X "), ir,35, 1, 5, Qt::AlignRight);
862 pppLayout->addWidget(_pppRefCrdXLineEdit, ir,40, 1, 13);
863 pppLayout->addWidget(new QLabel(" Y "), ir,50, 1, 10, Qt::AlignRight);
864 pppLayout->addWidget(_pppRefCrdYLineEdit, ir,60, 1, 13);
865 pppLayout->addWidget(new QLabel(" Z "), ir,70, 1, 10, Qt::AlignRight);
866 pppLayout->addWidget(_pppRefCrdZLineEdit, ir,80, 1, 13);
867 ++ir;
868 pppLayout->addWidget(new QLabel("Corr. Mountpnt "), ir, 0, 1, 10);
869 pppLayout->addWidget(_pppCorrMountLineEdit, ir,10, 1, 20);
870 pppLayout->addWidget(new QLabel(" dN "), ir,30, 1, 10, Qt::AlignRight);
871 pppLayout->addWidget(_pppRefdNLineEdit, ir,40, 1, 10);
872 pppLayout->addWidget(new QLabel(" dE "), ir,50, 1, 10, Qt::AlignRight);
873 pppLayout->addWidget(_pppRefdELineEdit, ir,60, 1, 10);
874 pppLayout->addWidget(new QLabel(" dU "), ir,70, 1, 10, Qt::AlignRight);
875 pppLayout->addWidget(_pppRefdULineEdit, ir,80, 1, 10);
876 ++ir;
877 pppLayout->addWidget(new QLabel("Output"), ir, 0, 1, 10);
878 pppLayout->addWidget(_pppNMEALineEdit, ir,10, 1, 20);
879 pppLayout->addWidget(new QLabel("NMEA File"), ir,30, 1, 10, Qt::AlignLeft);
880 pppLayout->addWidget(_pppNMEAPortLineEdit, ir,40, 1, 10);
881 pppLayout->addWidget(new QLabel("NMEA Port"), ir,50, 1, 10);
882 pppLayout->addWidget(_pppPlotCoordinates, ir,60, 1, 3);
883 pppLayout->addWidget(new QLabel("PPP Plot"), ir,63, 1, 10);
884 ++ir;
885 pppLayout->addWidget(new QLabel("Post-Processing"), ir, 0, 1, 10);
886 pppLayout->addWidget(_postObsFileChooser, ir,10, 1, 25);
887 pppLayout->addWidget(new QLabel("Obs "), ir,35, 1, 5);
888 pppLayout->addWidget(_postNavFileChooser, ir,40, 1, 20);
889 pppLayout->addWidget(new QLabel("Nav "), ir,60, 1, 5);
890 pppLayout->addWidget(_postCorrFileChooser, ir,65, 1, 22);
891 pppLayout->addWidget(new QLabel("Corr"), ir,87, 1, 5);
892 ++ir;
893 pppLayout->addWidget(new QLabel("Post-Proc. cont'd"),ir, 0, 1, 10);
894 pppLayout->addWidget(_postOutLineEdit, ir,10, 1, 40);
895 pppLayout->addWidget(new QLabel("Output"), ir,50);
896 ++ir;
897 pppLayout->addWidget(new QLabel(" "), ir, 0, 1, 10);
898
899 pppgroup->setLayout(pppLayout);
900
901 // PPP Client (second panel)
902 // -------------------------
903 QGridLayout* ppp2Layout = new QGridLayout;
904 ppp2Layout->setColumnMinimumWidth(0,14*ww);
905 ir = 0;
906 ppp2Layout->addWidget(new QLabel("Precise Point Positioning, Panel 2."), ir, 0, 1, 10);
907 ++ir;
908 ppp2Layout->addWidget(new QLabel("Antennas"), ir, 0);
909 ppp2Layout->addWidget(_pppAntexFileChooser, ir, 1,1,3);
910 ppp2Layout->addWidget(new QLabel("ANTEX File"), ir, 4);
911 ppp2Layout->addWidget(_pppAntennaLineEdit, ir, 5,1,3);
912 ppp2Layout->addWidget(new QLabel("Antenna Name"), ir, 8);
913 ++ir;
914 ppp2Layout->addWidget(new QLabel("Antennas cont'd"), ir, 0);
915 ppp2Layout->addWidget(_pppApplySatAntCheckBox, ir, 1, Qt::AlignRight);
916 ppp2Layout->addWidget(new QLabel("Apply Sat. Ant. Offsets"),ir, 2);
917 ++ir;
918 ppp2Layout->addWidget(new QLabel("Sigmas"), ir, 0);
919 ppp2Layout->addWidget(_pppSigCLineEdit, ir, 1, Qt::AlignRight);
920 ppp2Layout->addWidget(new QLabel("Code"), ir, 2);
921 ppp2Layout->addWidget(_pppSigPLineEdit, ir, 3);
922 ppp2Layout->addWidget(new QLabel("Phase"), ir, 4);
923 ppp2Layout->addWidget(_pppSigTrp0, ir, 5, Qt::AlignRight);
924 ppp2Layout->addWidget(new QLabel("Tropo Init "), ir, 6);
925 ppp2Layout->addWidget(_pppSigTrpP, ir, 7);
926 ppp2Layout->addWidget(new QLabel("Tropo White Noise"), ir, 8);
927 ++ir;
928 ppp2Layout->addWidget(new QLabel("Options"), ir, 0, 1, 5);
929 ppp2Layout->addWidget(_pppUsePhaseCheckBox, ir, 1, Qt::AlignRight);
930 ppp2Layout->addWidget(new QLabel("Use phase obs"), ir, 2);
931 ppp2Layout->addWidget(_pppEstTropoCheckBox, ir, 3, Qt::AlignRight);
932 ppp2Layout->addWidget(new QLabel("Estimate tropo"), ir, 4);
933 ppp2Layout->addWidget(_pppGLONASSCheckBox, ir, 5, Qt::AlignRight);
934 ppp2Layout->addWidget(new QLabel("Use GLONASS"), ir, 6);
935 ppp2Layout->addWidget(_pppGalileoCheckBox, ir, 7, Qt::AlignRight);
936 ppp2Layout->addWidget(new QLabel("Use Galileo "), ir, 8);
937 ++ir;
938 ppp2Layout->addWidget(new QLabel("Options cont'd"), ir, 0);
939 ppp2Layout->addWidget(_pppSigCrd0, ir, 1, Qt::AlignRight);
940 ppp2Layout->addWidget(new QLabel("Sigma XYZ Init "), ir, 2);
941 ppp2Layout->addWidget(_pppSigCrdP, ir, 3, Qt::AlignRight);
942 ppp2Layout->addWidget(new QLabel("Sigma XYZ Noise "), ir, 4);
943 ppp2Layout->addWidget(_pppQuickStartLineEdit, ir, 5, Qt::AlignRight);
944 ppp2Layout->addWidget(new QLabel("Quick-Start (sec) "), ir, 6);
945 ppp2Layout->addWidget(_pppMaxSolGapLineEdit, ir, 7, Qt::AlignRight);
946 ppp2Layout->addWidget(new QLabel("Max Sol. Gap (sec)"), ir, 8);
947 ++ir;
948 ppp2Layout->addWidget(new QLabel("Options cont'd"), ir, 0);
949 ppp2Layout->addWidget(_pppSync, ir, 1);
950 ppp2Layout->addWidget(new QLabel("Sync Corr (sec) "), ir, 2);
951 ppp2Layout->addWidget(_pppAverageLineEdit, ir, 3, Qt::AlignRight);
952 ppp2Layout->addWidget(new QLabel("Averaging (min)") , ir, 4);
953 ++ir;
954 ppp2Layout->addWidget(new QLabel(""), ir, 0);
955
956 ppp2group->setLayout(ppp2Layout);
957
958 // Teqc Processing
959 // ---------------
960 _teqcActionComboBox = new QComboBox();
961 _teqcActionComboBox->setEditable(false);
962 _teqcActionComboBox->addItems(QString(",Edit,Analyze").split(","));
963 ik = _teqcActionComboBox->findText(settings.value("teqcAction").toString());
964 if (ik != -1) {
965 _teqcActionComboBox->setCurrentIndex(ik);
966 }
967 connect(_teqcActionComboBox, SIGNAL(currentIndexChanged(const QString &)),
968 this, SLOT(slotBncTextChanged()));
969
970 QGridLayout* teqcLayout = new QGridLayout;
971 teqcLayout->setColumnMinimumWidth(0,14*ww);
972 _teqcActionComboBox->setMaximumWidth(10*ww);
973
974 _teqcObsFileChooser = new qtFileChooser;
975 _teqcObsFileChooser->setFileName(settings.value("teqcObsFile").toString());
976 _teqcObsFileChooser->setWhatsThis(tr("Specify the full path to an observation file in RINEX v2 or v3 format."));
977
978 _teqcNavFileChooser = new qtFileChooser;
979 _teqcNavFileChooser->setFileName(settings.value("teqcNavFile").toString());
980 _teqcNavFileChooser->setWhatsThis(tr("Specify the full path to a RINEX v2 or v3 navigation file."));
981
982 _teqcOutLineEdit = new QLineEdit(settings.value("teqcOutFile").toString());
983 _teqcOutLineEdit->setWhatsThis(tr("Specify the full path to an output file."));
984
985 ir = 0;
986 teqcLayout->addWidget(new QLabel("RINEX file editing or concatination and file or stream quality control."), ir, 0, 1, 20);
987 ++ir;
988 teqcLayout->addWidget(new QLabel("Action"), ir, 0);
989 teqcLayout->addWidget(_teqcActionComboBox, ir, 1);
990 _teqcEditOptionButton = new QPushButton("Set Edit Options");
991 teqcLayout->addWidget(_teqcEditOptionButton, ir, 2);
992 ++ir;
993 teqcLayout->addWidget(new QLabel("Input files (full path)"), ir, 0);
994 teqcLayout->addWidget(_teqcObsFileChooser, ir, 1, 1, 4);
995 teqcLayout->addWidget(new QLabel("Obs "), ir, 5);
996 teqcLayout->addWidget(_teqcNavFileChooser, ir, 6, 1, 8);
997 teqcLayout->addWidget(new QLabel("Nav"), ir, 14);
998 ++ir;
999 teqcLayout->addWidget(new QLabel("Output file (full path)"), ir, 0);
1000 teqcLayout->addWidget(_teqcOutLineEdit, ir, 1, 1, 4);
1001 ++ir;
1002 teqcLayout->addWidget(new QLabel(" "), ir, 0);
1003 ++ir;
1004 teqcLayout->addWidget(new QLabel(" "), ir, 0);
1005 ++ir;
1006 teqcLayout->addWidget(new QLabel(" "), ir, 0);
1007
1008 teqcgroup->setLayout(teqcLayout);
1009
1010 connect(_teqcEditOptionButton, SIGNAL(clicked()),
1011 this, SLOT(slotTeqcEditOption()));
1012
1013 // Combination
1014 // -----------
1015 QGridLayout* cmbLayout = new QGridLayout;
1016
1017 populateCmbTable();
1018 cmbLayout->addWidget(_cmbTable, 0, 0, 6, 3);
1019 cmbLayout->addWidget(new QLabel("Combine Broadcast Ephemeris correction streams."),0, 5, 1, 5);
1020 cmbLayout->addWidget(addCmbRowButton, 1, 5);
1021 cmbLayout->addWidget(delCmbRowButton, 1, 6);
1022 cmbLayout->addWidget(new QLabel("Method"), 2, 5);
1023 cmbLayout->addWidget(_cmbMethodComboBox, 2, 6, Qt::AlignRight);
1024 cmbLayout->addWidget(new QLabel("Maximal Residuum"), 3, 5);
1025 cmbLayout->addWidget(_cmbMaxresLineEdit, 3, 6, Qt::AlignRight);
1026 cmbLayout->addWidget(new QLabel(" "), 4, 0);
1027 cmbLayout->addWidget(new QLabel(" "), 5, 0);
1028 cmbLayout->addWidget(new QLabel(" "), 6, 0);
1029
1030 connect(addCmbRowButton, SIGNAL(clicked()), this, SLOT(slotAddCmbRow()));
1031 connect(delCmbRowButton, SIGNAL(clicked()), this, SLOT(slotDelCmbRow()));
1032
1033 cmbgroup->setLayout(cmbLayout);
1034
1035 // Upload Layout (Clocks)
1036 // ----------------------
1037 QGridLayout* uploadHlpLayout = new QGridLayout();
1038
1039 connect(addUploadRowButton, SIGNAL(clicked()), this, SLOT(slotAddUploadRow()));
1040 connect(delUploadRowButton, SIGNAL(clicked()), this, SLOT(slotDelUploadRow()));
1041 connect(setUploadTrafoButton, SIGNAL(clicked()), this, SLOT(slotSetUploadTrafo()));
1042
1043 uploadHlpLayout->addWidget(addUploadRowButton, 0,0);
1044 uploadHlpLayout->addWidget(delUploadRowButton, 0,1);
1045 uploadHlpLayout->addWidget(new QLabel("Interval"), 0,2, Qt::AlignRight);
1046 uploadHlpLayout->addWidget(_uploadIntrComboBox, 0,3);
1047 uploadHlpLayout->addWidget(new QLabel(" Sampling (Clk)"),0,4, Qt::AlignRight);
1048 uploadHlpLayout->addWidget(_uploadSamplSpinBox, 0,5);
1049 uploadHlpLayout->addWidget(new QLabel(" Sampling (Orb)"),0,6, Qt::AlignRight);
1050 uploadHlpLayout->addWidget(_uploadSamplOrbSpinBox, 0,7);
1051 uploadHlpLayout->addWidget(setUploadTrafoButton, 0,8);
1052
1053 QBoxLayout* uploadLayout = new QBoxLayout(QBoxLayout::TopToBottom);
1054 populateUploadTable();
1055
1056 uploadLayout->addWidget(new QLabel("Upload orbit/clock stream coming from Real-time Network Engine or upload orbit/clock combination stream."));
1057 uploadLayout->addWidget(_uploadTable);
1058 uploadLayout->addLayout(uploadHlpLayout);
1059
1060 uploadgroup->setLayout(uploadLayout);
1061
1062 // Upload Layout (Ephemeris)
1063 // -------------------------
1064 QGridLayout* uploadLayoutEph = new QGridLayout;
1065
1066 uploadLayoutEph->setColumnMinimumWidth(0, 9*ww);
1067 _uploadEphPortLineEdit->setMaximumWidth(9*ww);
1068 _uploadEphPasswordLineEdit->setMaximumWidth(9*ww);
1069 _uploadEphMountpointLineEdit->setMaximumWidth(12*ww);
1070
1071 uploadLayoutEph->addWidget(new QLabel("Upload concatenated RTCMv3 Broadcast Ephemeris to caster."), 0, 0, 1, 50);
1072 uploadLayoutEph->addWidget(new QLabel("Host"), 1, 0);
1073 uploadLayoutEph->addWidget(_uploadEphHostLineEdit, 1, 1, 1, 3);
1074 uploadLayoutEph->addWidget(new QLabel(" Port"), 1, 4, Qt::AlignRight);
1075 uploadLayoutEph->addWidget(_uploadEphPortLineEdit, 1, 5, 1, 1);
1076 uploadLayoutEph->addWidget(new QLabel("Mountpoint "), 2, 0);
1077 uploadLayoutEph->addWidget(_uploadEphMountpointLineEdit, 2, 1);
1078 uploadLayoutEph->addWidget(new QLabel(" Password"), 2, 2, Qt::AlignRight);
1079 uploadLayoutEph->addWidget(_uploadEphPasswordLineEdit, 2, 3);
1080 uploadLayoutEph->addWidget(new QLabel("Sampling"), 3, 0);
1081 uploadLayoutEph->addWidget(_uploadEphSampleSpinBox, 3, 1);
1082 uploadLayoutEph->addWidget(new QLabel("Uploaded"), 4, 0);
1083 uploadLayoutEph->addWidget(_uploadEphBytesCounter, 4, 1);
1084 uploadLayoutEph->addWidget(new QLabel(" "), 5, 0);
1085 uploadLayoutEph->addWidget(new QLabel(" "), 6, 0);
1086
1087 uploadEphgroup->setLayout(uploadLayoutEph);
1088
1089 connect(_uploadEphHostLineEdit, SIGNAL(textChanged(const QString &)),
1090 this, SLOT(slotBncTextChanged()));
1091
1092 // Main Layout
1093 // -----------
1094 QGridLayout* mLayout = new QGridLayout;
1095 _aogroup->setCurrentIndex(settings.value("startTab").toInt());
1096 mLayout->addWidget(_aogroup, 0,0);
1097 mLayout->addWidget(_mountPointsTable, 1,0);
1098 _loggroup->setCurrentIndex(settings.value("statusTab").toInt());
1099 mLayout->addWidget(_loggroup, 2,0);
1100
1101 _canvas->setLayout(mLayout);
1102
1103 // WhatsThis
1104 // ---------
1105 _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>"));
1106 _proxyPortLineEdit->setWhatsThis(tr("<p>Enter your proxy server port number in case a proxy is operated in front of BNC.</p>"));
1107 _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>"));
1108 _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>"));
1109 _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 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>"));
1110 _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."));
1111 _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."));
1112 _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."));
1113 _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."));
1114 _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."));
1115 _corrTimeSpinBox->setWhatsThis(tr("<p>Concerning output through IP port, BNC drops Broadcast Ephemeris Corrections received later than 'Wait for full epoch' seconds. A value of 2 to 5 seconds is recommended, depending on the latency of the incoming correction stream(s) and the delay acceptable to your real-time application.</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>"));
1116 _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."));
1117 _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."));
1118 _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."));
1119 _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>"));
1120 _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>"));
1121 _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>"));
1122 _autoStartCheckBox->setWhatsThis(tr("<p>Tick 'Auto start' for auto-start of BNC at startup time in window mode with preassigned processing options.</p>"));
1123 _rawOutFileLineEdit->setWhatsThis(tr("<p>Save all data coming in through various streams in the received order and format in one file.</p>"));
1124
1125 _onTheFlyComboBox->setWhatsThis(tr("<p>When operating BNC online in 'no window' mode, some configuration parameters can be changed on-the-fly without interrupting the running process. For that BNC rereads parts of its configuration in pre-defined intervals.<p></p>Select '1 min', '1 hour', or '1 day' to force BNC to reread its configuration every full minute, hour, or day and let in between edited configuration options become effective on-the-fly without terminating uninvolved threads.</p><p>Note that when operating BNC in window mode, on-the-fly changeable configuration options become effective immediately through 'Save & Reread Configuration'.</p>"));
1126 _rnxIntrComboBox->setWhatsThis(tr("<p>Select the length of the RINEX Observation file.</p>"));
1127 _ephIntrComboBox->setWhatsThis(tr("<p>Select the length of the RINEX Navigation file.</p>"));
1128 _corrIntrComboBox->setWhatsThis(tr("<p>Select the length of the Broadcast Ephemeris Correction files.</p>"));
1129 _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>"));
1130 _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>"));
1131 _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>"));
1132 _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>"));
1133 _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>"));
1134 _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."));
1135 _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>"));
1136 _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>"));
1137 _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' and 'RTCM_3.x'.</p><p>In case you need to log the raw data as is, BNC allows users to by-pass its decoders and and directly save the input in daily log files. To do this specify the decoder string as 'ZERO'.</p><p>BNC can also retrieve streams from virtual reference stations (VRS). 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>"));
1138 _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."));
1139 _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)."));
1140 _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."));
1141 _ephV3CheckBox->setWhatsThis(tr("The default format for output of RINEX Navigation data containing Broadcast Ephemeris is RINEX Version 2.11. Select 'Version 3' if you want to output the ephemeris in RINEX Version 3 format."));
1142 _rnxV3CheckBox->setWhatsThis(tr("The default format for RINEX Observation files is RINEX Version 2.11. Select 'Version 3' if you want to save the observations in RINEX Version 3 format."));
1143 _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>"));
1144 _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>"));
1145 _serialMountPointLineEdit->setWhatsThis(tr("<p>Enter a 'Mountpoint' to forward the corresponding stream to a serial connected receiver.</p>"));
1146 _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>"));
1147 _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>"));
1148 _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>"));
1149 _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>"));
1150 _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>"));
1151 _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>"));
1152 _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>"));
1153 _serialFileNMEALineEdit->setWhatsThis(tr("<p>Specify the full path to a file where NMEA messages coming from your serial connected receiver are saved.</p>"));
1154 _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>"));
1155 _pppMountLineEdit->setWhatsThis(tr("<p>Specify an observations stream by its mountpoint from the 'Streams' list compiled below if you want BNC to estimate coordinates for the affected receiver position through a PPP solution. Example: 'FFMJ1'</p><p>Note that PPP in BNC requires to also pull a stream carrying RTCM Version 3 satellite orbit and clock corrections to Broadcast Ephemeris referring to the satellites' Antenna Phase Centers (APC). Stream CLK11 on NTRIP broadcaster products.igs-ip.net is an example.</p><p>Pulling in addition a third stream carrying Broadcast Ephemeris messages in high repetition rate is suggested if such messages are comeing from the receiver in low repetition rate or don't come at all from there.</p>"));
1156 _pppCorrMountLineEdit->setWhatsThis(tr("<p>You must specify an orbit/clock corrections stream by its mountpoint from the 'Streams' list compiled below. Example: 'CLK10'</p>"));
1157 _pppSPPComboBox->setWhatsThis(tr("<p>Choose between plain Single Point Positioning (SPP) and Precise Point Positioning (PPP) in real-time and post processing mode.</p>"));
1158 _teqcActionComboBox->setWhatsThis(tr("<p>Select an editing action.</p>"));
1159 _pppUsePhaseCheckBox->setWhatsThis(tr("<p>By default BNC applies a PPP solution using an ionosphere free P3 linear combination of code observations.</p><p>Tick 'Use phase obs' for an ionosphere free L3 linear combination of phase observations.</p>"));
1160 _pppEstTropoCheckBox->setWhatsThis(tr("<p>By default BNC does not introduce troposphere parameters when estimating coordinates.</p><p>Tick 'Estimate tropo' to introduce troposphere parameters when estimating coordinates.</p>"));
1161 _pppGLONASSCheckBox->setWhatsThis(tr("<p>By default BNC does not use GLONASS observations in PPP mode.</p><p>Tick 'Use GLONASS' for adding GLONASS observations to GPS and Galileo (optional) in a PPP solution.</p>"));
1162 _pppGalileoCheckBox->setWhatsThis(tr("<p>By default BNC does not use Galileo observations in PPP mode.</p><p>Tick 'Use Galileo' for adding Galileo observations to GPS and GLONASS (optional) in a PPP solution.</p>"));
1163 _pppPlotCoordinates->setWhatsThis(tr("<p>BNC will plot PPP results in the 'PPP Plot' tab as North (red), East (green) and Up (blue) displacements when this option is selected. Values will be either referred to an XYZ reference coordinate (if specified) or referred to the first estimated coordinate. The sliding PPP time series window will cover the period of the latest 5 minutes.</p><p>Note that a PPP time series makes only sense for a stationary operated receiver."));
1164 _pppNMEALineEdit->setWhatsThis(tr("<p>Specify the full path to a file where PPP results are saved as NMEA messages.</p>"));
1165 _pppNMEAPortLineEdit->setWhatsThis(tr("<p>Specify an IP port number to output PPP results as NMEA messages through an IP port.</p>"));
1166 _pppSigCLineEdit->setWhatsThis(tr("<p>Enter a sigma for your code observations in meters.</p><p>The higher the sigma you enter, the less the contribution of code observations to a PPP solution based on a combination of code and phase data. 5.0 (default) is likely to be an appropriate choice.</p>"));
1167 _pppQuickStartLineEdit->setWhatsThis(tr("<p>Enter the lenght of a startup period in seconds for which you want to fix the PPP solution to a known XYZ coordinate as introduced above and adjust a sigma 'XYZ Ini' according to the coordinate's precision. Fixing the coordinate is done in BNC through setting the 'Sigma XYZ Noise' you define below temporarily to zero.</p><p>This so-called Quick-Start option allows the PPP solution to rapidly converge. It requires that the antenna remains unmoved on the know position throughout the startup period.</p><p>A value of 120 is likely to be an appropriate choice for 'Quick-Start'. Default is an empty option field, meaning that you don't want BNC to operate in Quick-Start mode.</p>"));
1168 _pppMaxSolGapLineEdit->setWhatsThis(tr("<p>Specify a 'Maximum Solution Gap' in seconds. Should the time span between two consecutive solutions exceed this limit, the algorithm returns into the Quick-Start mode and fixes the introduced reference coordinate for the specified Quick-Start period. A value of '120' seconds could be an appropriate choice.</p><p>This option makes only sense for a stationary operated receiver where solution convergence can be enforced because a good approximation for the rover position is known. Default is an empty option field, meaning that you don't want BNC to return into the Quick-Start mode after failures caused i.e. by longer lasting outages.</p>"));
1169 _pppSigPLineEdit->setWhatsThis(tr("<p>Enter a sigma for your phase observations in meters.</p><p>The higher the sigma you enter, the less the contribution of phase observations to a PPP solutions based on a combination of code and phase data. 0.02 (default) is likely to be an appropriate choice.</p>"));
1170 _pppAverageLineEdit->setWhatsThis(tr("<p>Enter the length of a sliding time window in minutes. BNC will continuously output moving average positions computed from those individual positions obtained most recently throughout this period.</p><p>An empty option field (default) means that you don't want BNC to output moving average positions.</p>"));
1171 _pppSigCrd0->setWhatsThis(tr("<p>Enter a sigma in meters for the initial XYZ coordinate componentes. A value of 100.0 (default) may be an appropriate choice. However, this value may be significantly smaller (i.e. 0.01) when starting for example from a station with known XZY position in Quick-Start mode."));
1172 _pppSigCrdP->setWhatsThis(tr("<p>Enter a sigma in meters for the white noise of estimated XYZ coordinate components. A value of 100.0 (default) may be appropriate considering the potential movement of a rover position.</p>"));
1173 _pppSigTrp0->setWhatsThis(tr("<p>Enter a sigma in meters for the a-priory model based tropospheric delay estimation. A value of 0.1 (default) may be an appropriate choice.</p>"));
1174 _pppSigTrpP->setWhatsThis(tr("<p>Enter a sigma in meters per second to describe the expected variation of the tropospheric effect.</p><p>Supposing 1Hz observation data, a value of 3e-6 (default) would mean that the tropospheric effect may vary for 3600 * 3e-6 = 0.01 meters per hour.</p>"));
1175 _pppRefCrdXLineEdit->setWhatsThis(tr("<p>Enter reference coordinate X of the receiver's position.</p>"));
1176 _pppRefCrdYLineEdit->setWhatsThis(tr("<p>Enter reference coordinate Y of the receiver's position.</p>"));
1177 _pppRefCrdZLineEdit->setWhatsThis(tr("<p>Enter reference coordinate Z of the receiver's position.</p>"));
1178 _pppRefdNLineEdit->setWhatsThis(tr("<p>Enter north antenna excentricity.</p>"));
1179 _pppRefdELineEdit->setWhatsThis(tr("<p>Enter east antenna excentricity.</p>"));
1180 _pppRefdULineEdit->setWhatsThis(tr("<p>Enter up antenna excentricity.</p>"));
1181 _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."));
1182 _pppSync->setWhatsThis(tr(
1183 "<p> Zero value (or empty field, default) means that BNC processes each epoch of data "
1184 "immediately after its arrival using satellite clock corrections available at "
1185 "that time.</p><p> Non-zero value 'Sync Corr' (i.e. 5) means that the epochs of data "
1186 "are buffered and the processing of each epoch is postponed till the satellite clock "
1187 "corrections not older than 'Sync Corr' seconds are available.<p>"));
1188 _pppAntexFileChooser->setWhatsThis(tr("<p>IGS provides a file containing absolute phase center corrections for GNSS satellite and receiver antennas in ANTEX format. Entering the full path to such an ANTEX file is required for correcting observations for antenna phase center offsets and variations. It allows you to specify the name of your receiver's antenna (as contained in the ANTEX file) to apply such corrections.</p><p>Default is an empty option field meaning that you don't want to correct observations for antenna phase center offsets and variations.</p>"));
1189 _pppAntennaLineEdit->setWhatsThis(tr("<p>Specify the receiver's antenna name as defined in your ANTEX file. Observations will be corrected for the antenna phase center's offset which may result in a reduction of a few centimeters at max. Corrections for phase center variations are not yet applied by BNC. The specified name must consist of 20 characters. Add trailing blanks if the antenna name has less then 20 characters.</p><p>Default is an empty option field meaning that you don't want to correct observations for antenna phase center offsets.</p>"));
1190 _pppApplySatAntCheckBox->setWhatsThis(tr("<p>This option is not yet working.</p><p>Satellite orbit and clock corrections refer to the satellite's antenna phase centers and hence observations are actually <u>not</u> to be corrected for satellite antenna phase center offsets. However, you may like to tick 'Apply Offsets' to force BNC to correct observations for satellite antenna phase center offsets.</p><p>Default is to <u>not</u> correct observations for satellite antenna phase center offsets."));
1191 _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.</p><p>Note that an appropriate 'Wait for full 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.</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 streams to get hold of the orbit information.</p><p>The combination process requires Broadcast Ephemeris. Besides the orbit and clock corrections stream(s) BNC must therefore pull a stream carrying Broadcast Ephemeris in the form of RTCM Version 3 messages.</p>"));
1192 _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>"));
1193 _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>"));
1194 _uploadTable->setWhatsThis(tr("<p>BNC can upload clock and orbit corrections to broadcast ephemeris (Broadcast Corrections) in RTCM Version 3 SSR format. The clock and orbit corrections may either come from a Real-time Network Engine or from a combination of incoming orbit/clock streams.</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' field to enter the NTRIP broadcaster IP port.</p>"));
1195 _postObsFileChooser->setWhatsThis(tr("Full path to RINEX v2/v3 Observation file."));
1196 _postNavFileChooser->setWhatsThis(tr("Full path to RINEX v2/v3 Navigation file."));
1197 _postCorrFileChooser->setWhatsThis(tr("Full path to Broadcast Corrections file as previously saved with BNC in plain ASCII format."));
1198 _postOutLineEdit->setWhatsThis(tr("Full path to file with post processing PPP results. "));
1199 addCmbRowButton->setWhatsThis(tr("Hit 'Add Row' button to add another line to the mountpoints table."));
1200 delCmbRowButton->setWhatsThis(tr("Hit 'Delete' button to delete the highlighted line from the mountpoints table."));
1201
1202 // Enable/Disable all Widgets
1203 // --------------------------
1204 slotBncTextChanged();
1205 enableStartStop();
1206
1207 // Auto start
1208 // ----------
1209 if ( Qt::CheckState(settings.value("autoStart").toInt()) == Qt::Checked) {
1210 slotStart();
1211 }
1212}
1213
1214// Destructor
1215////////////////////////////////////////////////////////////////////////////
1216bncWindow::~bncWindow() {
1217 delete _caster;
1218 delete _casterEph;
1219}
1220
1221//
1222////////////////////////////////////////////////////////////////////////////
1223void bncWindow::populateMountPointsTable() {
1224
1225 for (int iRow = _mountPointsTable->rowCount()-1; iRow >=0; iRow--) {
1226 _mountPointsTable->removeRow(iRow);
1227 }
1228
1229 bncSettings settings;
1230
1231 QListIterator<QString> it(settings.value("mountPoints").toStringList());
1232 int iRow = 0;
1233 while (it.hasNext()) {
1234 QStringList hlp = it.next().split(" ");
1235 if (hlp.size() < 5) continue;
1236 _mountPointsTable->insertRow(iRow);
1237
1238 QUrl url(hlp[0]);
1239
1240 QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
1241 QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
1242 QString nmea(hlp[4]);
1243 if (hlp[5] == "S") {
1244 fullPath = hlp[0].replace(0,2,"");
1245 }
1246 QString ntripVersion = "2";
1247 if (hlp.size() >= 6) {
1248 ntripVersion = (hlp[5]);
1249 }
1250
1251 QTableWidgetItem* it;
1252 it = new QTableWidgetItem(url.userInfo());
1253 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1254 _mountPointsTable->setItem(iRow, 0, it);
1255
1256 it = new QTableWidgetItem(fullPath);
1257 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1258 _mountPointsTable->setItem(iRow, 1, it);
1259
1260 it = new QTableWidgetItem(format);
1261 _mountPointsTable->setItem(iRow, 2, it);
1262
1263 if (nmea == "yes") {
1264 it = new QTableWidgetItem(latitude);
1265 _mountPointsTable->setItem(iRow, 3, it);
1266 it = new QTableWidgetItem(longitude);
1267 _mountPointsTable->setItem(iRow, 4, it);
1268 } else {
1269 it = new QTableWidgetItem(latitude);
1270 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1271 _mountPointsTable->setItem(iRow, 3, it);
1272 it = new QTableWidgetItem(longitude);
1273 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1274 _mountPointsTable->setItem(iRow, 4, it);
1275 }
1276
1277 it = new QTableWidgetItem(nmea);
1278 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1279 _mountPointsTable->setItem(iRow, 5, it);
1280
1281 it = new QTableWidgetItem(ntripVersion);
1282 //// it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1283 _mountPointsTable->setItem(iRow, 6, it);
1284
1285 bncTableItem* bncIt = new bncTableItem();
1286 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
1287 _mountPointsTable->setItem(iRow, 7, bncIt);
1288
1289 iRow++;
1290 }
1291
1292 _mountPointsTable->sortItems(1);
1293
1294 enableStartStop();
1295}
1296
1297// Retrieve Table
1298////////////////////////////////////////////////////////////////////////////
1299void bncWindow::slotAddMountPoints() {
1300
1301 bncSettings settings;
1302 QString proxyHost = settings.value("proxyHost").toString();
1303 int proxyPort = settings.value("proxyPort").toInt();
1304 if (proxyHost != _proxyHostLineEdit->text() ||
1305 proxyPort != _proxyPortLineEdit->text().toInt()) {
1306 int iRet = QMessageBox::question(this, "Question", "Proxy options "
1307 "changed. Use the new ones?",
1308 QMessageBox::Yes, QMessageBox::No,
1309 QMessageBox::NoButton);
1310 if (iRet == QMessageBox::Yes) {
1311 settings.setValue("proxyHost", _proxyHostLineEdit->text());
1312 settings.setValue("proxyPort", _proxyPortLineEdit->text());
1313 settings.sync();
1314 }
1315 }
1316
1317 settings.setValue("sslCaCertPath", _sslCaCertPathLineEdit->text());
1318 settings.setValue("ignoreSslErrors", _ignoreSslErrorsCheckBox->checkState());
1319 settings.sync();
1320
1321 QMessageBox msgBox;
1322 msgBox.setIcon(QMessageBox::Question);
1323 msgBox.setWindowTitle("Add Stream");
1324 msgBox.setText("Add stream(s) coming from:");
1325
1326 QPushButton* buttonNtrip = msgBox.addButton(tr("Caster"), QMessageBox::ActionRole);
1327 QPushButton* buttonIP = msgBox.addButton(tr("TCP/IP port"), QMessageBox::ActionRole);
1328 QPushButton* buttonUDP = msgBox.addButton(tr("UDP port"), QMessageBox::ActionRole);
1329 QPushButton* buttonSerial = msgBox.addButton(tr("Serial port"), QMessageBox::ActionRole);
1330 QPushButton* buttonCancel = msgBox.addButton(tr("Cancel"), QMessageBox::ActionRole);
1331
1332 msgBox.exec();
1333
1334 if (msgBox.clickedButton() == buttonNtrip) {
1335 bncTableDlg* dlg = new bncTableDlg(this);
1336 dlg->move(this->pos().x()+50, this->pos().y()+50);
1337 connect(dlg, SIGNAL(newMountPoints(QStringList*)),
1338 this, SLOT(slotNewMountPoints(QStringList*)));
1339 dlg->exec();
1340 delete dlg;
1341 } else if (msgBox.clickedButton() == buttonIP) {
1342 bncIpPort* ipp = new bncIpPort(this);
1343 connect(ipp, SIGNAL(newMountPoints(QStringList*)),
1344 this, SLOT(slotNewMountPoints(QStringList*)));
1345 ipp->exec();
1346 delete ipp;
1347 } else if (msgBox.clickedButton() == buttonUDP) {
1348 bncUdpPort* udp = new bncUdpPort(this);
1349 connect(udp, SIGNAL(newMountPoints(QStringList*)),
1350 this, SLOT(slotNewMountPoints(QStringList*)));
1351 udp->exec();
1352 delete udp;
1353 } else if (msgBox.clickedButton() == buttonSerial) {
1354 bncSerialPort* sep = new bncSerialPort(this);
1355 connect(sep, SIGNAL(newMountPoints(QStringList*)),
1356 this, SLOT(slotNewMountPoints(QStringList*)));
1357 sep->exec();
1358 delete sep;
1359 } else if (msgBox.clickedButton() == buttonCancel) {
1360 // Cancel
1361 }
1362
1363 enableStartStop();
1364}
1365
1366// Delete Selected Mount Points
1367////////////////////////////////////////////////////////////////////////////
1368void bncWindow::slotDeleteMountPoints() {
1369
1370 int nRows = _mountPointsTable->rowCount();
1371 bool flg[nRows];
1372 for (int iRow = 0; iRow < nRows; iRow++) {
1373 if (_mountPointsTable->isItemSelected(_mountPointsTable->item(iRow,1))) {
1374 flg[iRow] = true;
1375 }
1376 else {
1377 flg[iRow] = false;
1378 }
1379 }
1380 for (int iRow = nRows-1; iRow >= 0; iRow--) {
1381 if (flg[iRow]) {
1382 _mountPointsTable->removeRow(iRow);
1383 }
1384 }
1385 _actDeleteMountPoints->setEnabled(false);
1386
1387 enableStartStop();
1388}
1389
1390// New Mount Points Selected
1391////////////////////////////////////////////////////////////////////////////
1392void bncWindow::slotNewMountPoints(QStringList* mountPoints) {
1393 int iRow = 0;
1394 QListIterator<QString> it(*mountPoints);
1395 while (it.hasNext()) {
1396 QStringList hlp = it.next().split(" ");
1397 QUrl url(hlp[0]);
1398 QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
1399 QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
1400 QString nmea(hlp[4]);
1401 if (hlp[5] == "S") {
1402 fullPath = hlp[0].replace(0,2,"");
1403 }
1404 QString ntripVersion = "2";
1405 if (hlp.size() >= 6) {
1406 ntripVersion = (hlp[5]);
1407 }
1408
1409 _mountPointsTable->insertRow(iRow);
1410
1411 QTableWidgetItem* it;
1412 it = new QTableWidgetItem(url.userInfo());
1413 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1414 _mountPointsTable->setItem(iRow, 0, it);
1415
1416 it = new QTableWidgetItem(fullPath);
1417 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1418 _mountPointsTable->setItem(iRow, 1, it);
1419
1420 it = new QTableWidgetItem(format);
1421 _mountPointsTable->setItem(iRow, 2, it);
1422
1423 if (nmea == "yes") {
1424 it = new QTableWidgetItem(latitude);
1425 _mountPointsTable->setItem(iRow, 3, it);
1426 it = new QTableWidgetItem(longitude);
1427 _mountPointsTable->setItem(iRow, 4, it);
1428 } else {
1429 it = new QTableWidgetItem(latitude);
1430 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1431 _mountPointsTable->setItem(iRow, 3, it);
1432 it = new QTableWidgetItem(longitude);
1433 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1434 _mountPointsTable->setItem(iRow, 4, it);
1435 }
1436
1437 it = new QTableWidgetItem(nmea);
1438 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1439 _mountPointsTable->setItem(iRow, 5, it);
1440
1441 it = new QTableWidgetItem(ntripVersion);
1442 it->setFlags(it->flags() & ~Qt::ItemIsEditable);
1443 _mountPointsTable->setItem(iRow, 6, it);
1444
1445 bncTableItem* bncIt = new bncTableItem();
1446 _mountPointsTable->setItem(iRow, 7, bncIt);
1447
1448 iRow++;
1449 }
1450 _mountPointsTable->hideColumn(0);
1451 _mountPointsTable->sortItems(1);
1452 delete mountPoints;
1453
1454 enableStartStop();
1455}
1456
1457// Save Options
1458////////////////////////////////////////////////////////////////////////////
1459void bncWindow::slotSaveOptions() {
1460
1461 QStringList mountPoints;
1462 for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
1463
1464 if (_mountPointsTable->item(iRow, 6)->text() != "S") {
1465 QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
1466 "@" + _mountPointsTable->item(iRow, 1)->text() );
1467
1468 mountPoints.append(url.toString() + " " +
1469 _mountPointsTable->item(iRow, 2)->text()
1470 + " " + _mountPointsTable->item(iRow, 3)->text()
1471 + " " + _mountPointsTable->item(iRow, 4)->text()
1472 + " " + _mountPointsTable->item(iRow, 5)->text()
1473 + " " + _mountPointsTable->item(iRow, 6)->text());
1474 } else {
1475 mountPoints.append(
1476 "//" + _mountPointsTable->item(iRow, 1)->text()
1477 + " " + _mountPointsTable->item(iRow, 2)->text()
1478 + " " + _mountPointsTable->item(iRow, 3)->text()
1479 + " " + _mountPointsTable->item(iRow, 4)->text()
1480 + " " + _mountPointsTable->item(iRow, 5)->text()
1481 + " " + _mountPointsTable->item(iRow, 6)->text());
1482 }
1483 }
1484
1485 QStringList combineStreams;
1486 for (int iRow = 0; iRow < _cmbTable->rowCount(); iRow++) {
1487 QString hlp;
1488 for (int iCol = 0; iCol < _cmbTable->columnCount(); iCol++) {
1489 if (_cmbTable->item(iRow, iCol)) {
1490 hlp += _cmbTable->item(iRow, iCol)->text() + " ";
1491 }
1492 }
1493 if (!hlp.isEmpty()) {
1494 combineStreams << hlp;
1495 }
1496 }
1497
1498 QStringList uploadMountpointsOut;
1499 for (int iRow = 0; iRow < _uploadTable->rowCount(); iRow++) {
1500 QString hlp;
1501 for (int iCol = 0; iCol < _uploadTable->columnCount(); iCol++) {
1502 if (_uploadTable->cellWidget(iRow, iCol) &&
1503 (iCol == 3 || iCol == 4 || iCol == 5)) {
1504 if (iCol == 3) {
1505 QLineEdit* passwd = (QLineEdit*)(_uploadTable->cellWidget(iRow, iCol));
1506 hlp += passwd->text() + ",";
1507 }
1508 else if (iCol == 4) {
1509 QComboBox* system = (QComboBox*)(_uploadTable->cellWidget(iRow, iCol));
1510 hlp += system->currentText() + ",";
1511 }
1512 else if (iCol == 5) {
1513 QCheckBox* com = (QCheckBox*)(_uploadTable->cellWidget(iRow, iCol));
1514 QString state; state.setNum(com->checkState());
1515 hlp += state + ",";
1516 }
1517 }
1518 else if (_uploadTable->item(iRow, iCol)) {
1519 hlp += _uploadTable->item(iRow, iCol)->text() + ",";
1520 }
1521 }
1522 if (!hlp.isEmpty()) {
1523 uploadMountpointsOut << hlp;
1524 }
1525 }
1526
1527 bncSettings settings;
1528
1529 settings.setValue("adviseFail", _adviseFailSpinBox->value());
1530 settings.setValue("adviseReco", _adviseRecoSpinBox->value());
1531 settings.setValue("adviseScript",_adviseScriptLineEdit->text());
1532 settings.setValue("autoStart", _autoStartCheckBox->checkState());
1533 settings.setValue("binSampl", _binSamplSpinBox->value());
1534 settings.setValue("corrIntr", _corrIntrComboBox->currentText());
1535 settings.setValue("corrPath", _corrPathLineEdit->text());
1536 settings.setValue("corrPort", _corrPortLineEdit->text());
1537 settings.setValue("corrTime", _corrTimeSpinBox->value());
1538 settings.setValue("ephIntr", _ephIntrComboBox->currentText());
1539 settings.setValue("ephPath", _ephPathLineEdit->text());
1540 settings.setValue("ephV3", _ephV3CheckBox->checkState());
1541 settings.setValue("logFile", _logFileLineEdit->text());
1542 settings.setValue("rawOutFile", _rawOutFileLineEdit->text());
1543 settings.setValue("miscMount", _miscMountLineEdit->text());
1544 settings.setValue("pppMount", _pppMountLineEdit->text());
1545 settings.setValue("pppCorrMount",_pppCorrMountLineEdit->text());
1546 settings.setValue("pppSPP", _pppSPPComboBox->currentText());
1547 settings.setValue("nmeaFile", _pppNMEALineEdit->text());
1548 settings.setValue("nmeaPort", _pppNMEAPortLineEdit->text());
1549 settings.setValue("pppSigmaCode",_pppSigCLineEdit->text());
1550 settings.setValue("pppSigmaPhase",_pppSigPLineEdit->text());
1551 settings.setValue("pppSigCrd0",_pppSigCrd0->text());
1552 settings.setValue("pppSigCrdP",_pppSigCrdP->text());
1553 settings.setValue("pppSigTrp0",_pppSigTrp0->text());
1554 settings.setValue("pppSigTrpP",_pppSigTrpP->text());
1555 settings.setValue("pppAverage", _pppAverageLineEdit->text());
1556 settings.setValue("pppQuickStart", _pppQuickStartLineEdit->text());
1557 settings.setValue("pppMaxSolGap", _pppMaxSolGapLineEdit->text());
1558 settings.setValue("pppRefCrdX", _pppRefCrdXLineEdit->text());
1559 settings.setValue("pppRefCrdY", _pppRefCrdYLineEdit->text());
1560 settings.setValue("pppRefCrdZ", _pppRefCrdZLineEdit->text());
1561 settings.setValue("pppRefdN", _pppRefdNLineEdit->text());
1562 settings.setValue("pppRefdE", _pppRefdELineEdit->text());
1563 settings.setValue("pppRefdU", _pppRefdULineEdit->text());
1564 settings.setValue("pppSync", _pppSync->text());
1565 settings.setValue("pppUsePhase", _pppUsePhaseCheckBox->checkState());
1566 settings.setValue("pppPlotCoordinates", _pppPlotCoordinates->checkState());
1567 settings.setValue("pppEstTropo", _pppEstTropoCheckBox->checkState());
1568 settings.setValue("pppGLONASS", _pppGLONASSCheckBox->checkState());
1569 settings.setValue("pppGalileo", _pppGalileoCheckBox->checkState());
1570 settings.setValue("pppAntenna", _pppAntennaLineEdit->text());
1571 settings.setValue("pppAntex", _pppAntexFileChooser->fileName());
1572 settings.setValue("pppApplySatAnt", _pppApplySatAntCheckBox->checkState());
1573 settings.setValue("mountPoints", mountPoints);
1574 settings.setValue("obsRate", _obsRateComboBox->currentText());
1575 settings.setValue("onTheFlyInterval", _onTheFlyComboBox->currentText());
1576 settings.setValue("outEphPort", _outEphPortLineEdit->text());
1577 settings.setValue("outFile", _outFileLineEdit->text());
1578 settings.setValue("outPort", _outPortLineEdit->text());
1579 settings.setValue("outUPort", _outUPortLineEdit->text());
1580 settings.setValue("perfIntr", _perfIntrComboBox->currentText());
1581 settings.setValue("proxyHost", _proxyHostLineEdit->text());
1582 settings.setValue("proxyPort", _proxyPortLineEdit->text());
1583 settings.setValue("sslCaCertPath", _sslCaCertPathLineEdit->text());
1584 settings.setValue("ignoreSslErrors", _ignoreSslErrorsCheckBox->checkState());
1585 settings.setValue("rnxAppend", _rnxAppendCheckBox->checkState());
1586 settings.setValue("rnxIntr", _rnxIntrComboBox->currentText());
1587 settings.setValue("rnxPath", _rnxPathLineEdit->text());
1588 settings.setValue("rnxSampl", _rnxSamplSpinBox->value());
1589 settings.setValue("rnxScript", _rnxScrpLineEdit->text());
1590 settings.setValue("rnxSkel", _rnxSkelLineEdit->text());
1591 settings.setValue("rnxV3", _rnxV3CheckBox->checkState());
1592 settings.setValue("scanRTCM", _scanRTCMCheckBox->checkState());
1593 settings.setValue("serialFileNMEA",_serialFileNMEALineEdit->text());
1594 settings.setValue("serialHeightNMEA",_serialHeightNMEALineEdit->text());
1595 settings.setValue("serialAutoNMEA", _serialAutoNMEAComboBox->currentText());
1596 settings.setValue("serialBaudRate", _serialBaudRateComboBox->currentText());
1597 settings.setValue("serialDataBits", _serialDataBitsComboBox->currentText());
1598 settings.setValue("serialMountPoint",_serialMountPointLineEdit->text());
1599 settings.setValue("serialParity", _serialParityComboBox->currentText());
1600 settings.setValue("serialPortName", _serialPortNameLineEdit->text());
1601 settings.setValue("serialStopBits", _serialStopBitsComboBox->currentText());
1602 settings.setValue("serialFlowControl",_serialFlowControlComboBox->currentText());
1603 settings.setValue("startTab", _aogroup->currentIndex());
1604 settings.setValue("statusTab", _loggroup->currentIndex());
1605 settings.setValue("waitTime", _waitTimeSpinBox->value());
1606 if (!combineStreams.isEmpty()) {
1607 settings.setValue("combineStreams", combineStreams);
1608 }
1609 else {
1610 settings.setValue("combineStreams", "");
1611 }
1612 settings.setValue("cmbMaxres", _cmbMaxresLineEdit->text());
1613 settings.setValue("cmbMethod", _cmbMethodComboBox->currentText());
1614
1615 if (!uploadMountpointsOut.isEmpty()) {
1616 settings.setValue("uploadMountpointsOut", uploadMountpointsOut);
1617 }
1618 else {
1619 settings.setValue("uploadMountpointsOut", "");
1620 }
1621 settings.setValue("uploadIntr", _uploadIntrComboBox->currentText());
1622 settings.setValue("uploadSampl", _uploadSamplSpinBox->value());
1623 settings.setValue("uploadSamplOrb", _uploadSamplOrbSpinBox->value());
1624
1625 settings.setValue("uploadEphHost", _uploadEphHostLineEdit->text());
1626 settings.setValue("uploadEphPort", _uploadEphPortLineEdit->text());
1627 settings.setValue("uploadEphPassword", _uploadEphPasswordLineEdit->text());
1628 settings.setValue("uploadEphMountpoint",_uploadEphMountpointLineEdit->text());
1629 settings.setValue("uploadEphSample", _uploadEphSampleSpinBox->value());
1630
1631 settings.setValue("postObsFile", _postObsFileChooser->fileName());
1632 settings.setValue("postNavFile", _postNavFileChooser->fileName());
1633 settings.setValue("postCorrFile", _postCorrFileChooser->fileName());
1634 settings.setValue("postOutFile", _postOutLineEdit->text());
1635
1636 settings.setValue("teqcObsFile", _teqcObsFileChooser->fileName());
1637 settings.setValue("teqcNavFile", _teqcNavFileChooser->fileName());
1638 settings.setValue("teqcOutFile", _teqcOutLineEdit->text());
1639
1640 if (_caster) {
1641 _caster->slotReadMountPoints();
1642 }
1643 settings.sync();
1644}
1645
1646// All get slots terminated
1647////////////////////////////////////////////////////////////////////////////
1648void bncWindow::slotGetThreadsFinished() {
1649 ((bncApp*)qApp)->slotMessage("All Get Threads Terminated", true);
1650 delete _caster; _caster = 0;
1651 delete _casterEph; _casterEph = 0;
1652 _runningRealTime = false;
1653}
1654
1655// Start It!
1656////////////////////////////////////////////////////////////////////////////
1657void bncWindow::slotStart() {
1658 slotSaveOptions();
1659 if ( _pppSPPComboBox->currentText() == "Post-Processing" ) {
1660 startPostProcessingPPP();
1661 }
1662 else if ( !_teqcActionComboBox->currentText().isEmpty() ) {
1663 startPostProcessingTeqc();
1664 }
1665 else {
1666 startRealTime();
1667 }
1668}
1669
1670// Start Real-Time (Retrieve Data etc.)
1671////////////////////////////////////////////////////////////////////////////
1672void bncWindow::startRealTime() {
1673
1674 _runningRealTime = true;
1675
1676 _bncFigurePPP->reset();
1677
1678 _actDeleteMountPoints->setEnabled(false);
1679
1680 enableStartStop();
1681
1682 _caster = new bncCaster(_outFileLineEdit->text(),
1683 _outPortLineEdit->text().toInt());
1684
1685 ((bncApp*)qApp)->setPort(_outEphPortLineEdit->text().toInt());
1686 ((bncApp*)qApp)->setPortCorr(_corrPortLineEdit->text().toInt());
1687 ((bncApp*)qApp)->initCombination();
1688
1689 connect(_caster, SIGNAL(getThreadsFinished()),
1690 this, SLOT(slotGetThreadsFinished()));
1691
1692 connect (_caster, SIGNAL(mountPointsRead(QList<bncGetThread*>)),
1693 this, SLOT(slotMountPointsRead(QList<bncGetThread*>)));
1694
1695 ((bncApp*)qApp)->slotMessage("========== Start BNC v" BNCVERSION " =========", true);
1696
1697 bncSettings settings;
1698
1699 QDir rnxdir(settings.value("rnxPath").toString());
1700 if (!rnxdir.exists()) ((bncApp*)qApp)->slotMessage("Cannot find RINEX Observations directory", true);
1701
1702 QString rnx_file = settings.value("rnxScript").toString();
1703 if ( !rnx_file.isEmpty() ) {
1704 QFile rnxfile(settings.value("rnxScript").toString());
1705 if (!rnxfile.exists()) ((bncApp*)qApp)->slotMessage("Cannot find RINEX Observations script", true);
1706 }
1707
1708 QDir ephdir(settings.value("ephPath").toString());
1709 if (!ephdir.exists()) ((bncApp*)qApp)->slotMessage("Cannot find RINEX Ephemeris directory", true);
1710
1711 QDir corrdir(settings.value("corrPath").toString());
1712 if (!corrdir.exists()) ((bncApp*)qApp)->slotMessage("Cannot find Broadcast Corrections directory", true);
1713
1714 QString advise_file = settings.value("adviseScript").toString();
1715 if ( !advise_file.isEmpty() ) {
1716 QFile advisefile(settings.value("adviseScript").toString());
1717 if (!advisefile.exists()) ((bncApp*)qApp)->slotMessage("Cannot find Outages script", true);
1718 }
1719
1720 QString ant_file = settings.value("pppAntex").toString();
1721 if ( !ant_file.isEmpty() ) {
1722 QFile anxfile(settings.value("pppAntex").toString());
1723 if (!anxfile.exists()) ((bncApp*)qApp)->slotMessage("Cannot find IGS ANTEX file", true);
1724 }
1725
1726 _caster->slotReadMountPoints();
1727
1728 _casterEph = new bncEphUploadCaster();
1729 connect(_casterEph, SIGNAL(newBytes(QByteArray,double)),
1730 _uploadEphBytesCounter, SLOT(slotNewBytes(QByteArray,double)));
1731}
1732
1733// Retrieve Data
1734////////////////////////////////////////////////////////////////////////////
1735void bncWindow::slotStop() {
1736 int iRet = QMessageBox::question(this, "Stop", "Stop retrieving data?",
1737 QMessageBox::Yes, QMessageBox::No,
1738 QMessageBox::NoButton);
1739 if (iRet == QMessageBox::Yes) {
1740 ((bncApp*)qApp)->stopCombination();
1741 delete _caster; _caster = 0;
1742 delete _casterEph; _casterEph = 0;
1743 _runningRealTime = false;
1744 }
1745}
1746
1747// Close Application gracefully
1748////////////////////////////////////////////////////////////////////////////
1749void bncWindow::closeEvent(QCloseEvent* event) {
1750
1751 int iRet = QMessageBox::question(this, "Close", "Save Options?",
1752 QMessageBox::Yes, QMessageBox::No,
1753 QMessageBox::Cancel);
1754
1755 if (iRet == QMessageBox::Cancel) {
1756 event->ignore();
1757 return;
1758 }
1759 else if (iRet == QMessageBox::Yes) {
1760 slotSaveOptions();
1761 }
1762
1763 QMainWindow::closeEvent(event);
1764}
1765
1766// User changed the selection of mountPoints
1767////////////////////////////////////////////////////////////////////////////
1768void bncWindow::slotSelectionChanged() {
1769 if (_mountPointsTable->selectedItems().isEmpty()) {
1770 _actDeleteMountPoints->setEnabled(false);
1771 }
1772 else {
1773 _actDeleteMountPoints->setEnabled(true);
1774 }
1775}
1776
1777// Display Program Messages
1778////////////////////////////////////////////////////////////////////////////
1779void bncWindow::slotWindowMessage(const QByteArray msg, bool showOnScreen) {
1780
1781#ifdef DEBUG_RTCM2_2021
1782 const int maxBufferSize = 1000;
1783#else
1784 const int maxBufferSize = 10000;
1785#endif
1786
1787 if (! showOnScreen ) {
1788 return;
1789 }
1790
1791 QString txt = _log->toPlainText() + "\n" +
1792 QDateTime::currentDateTime().toUTC().toString("yy-MM-dd hh:mm:ss ") + msg;
1793 _log->clear();
1794 _log->append(txt.right(maxBufferSize));
1795}
1796
1797// About Message
1798////////////////////////////////////////////////////////////////////////////
1799void bncWindow::slotAbout() {
1800 new bncAboutDlg(0);
1801}
1802
1803//Flowchart
1804////////////////////////////////////////////////////////////////////////////
1805void bncWindow::slotFlowchart() {
1806 new bncFlowchartDlg(0);
1807}
1808
1809// Help Window
1810////////////////////////////////////////////////////////////////////////////
1811void bncWindow::slotHelp() {
1812 QUrl url;
1813 url.setPath(":bnchelp.html");
1814 new bncHlpDlg(0, url);
1815}
1816
1817// Select Fonts
1818////////////////////////////////////////////////////////////////////////////
1819void bncWindow::slotFontSel() {
1820 bool ok;
1821 QFont newFont = QFontDialog::getFont(&ok, this->font(), this);
1822 if (ok) {
1823 bncSettings settings;
1824 settings.setValue("font", newFont.toString());
1825 QApplication::setFont(newFont);
1826 int ww = QFontMetrics(newFont).width('w');
1827 setMinimumSize(60*ww, 80*ww);
1828 resize(60*ww, 80*ww);
1829 }
1830}
1831
1832// Whats This Help
1833void bncWindow::slotWhatsThis() {
1834 QWhatsThis::enterWhatsThisMode();
1835}
1836
1837//
1838////////////////////////////////////////////////////////////////////////////
1839void bncWindow::slotMountPointsRead(QList<bncGetThread*> threads) {
1840 _bncFigure->updateMountPoints();
1841 _bncFigureLate->updateMountPoints();
1842
1843 populateMountPointsTable();
1844 bncSettings settings;
1845 _binSamplSpinBox->setValue(settings.value("binSampl").toInt());
1846 _waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
1847 QListIterator<bncGetThread*> iTh(threads);
1848 while (iTh.hasNext()) {
1849 bncGetThread* thread = iTh.next();
1850 for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
1851 QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
1852 "@" + _mountPointsTable->item(iRow, 1)->text() );
1853 if (url == thread->mountPoint() &&
1854 _mountPointsTable->item(iRow, 3)->text() == thread->latitude() &&
1855 _mountPointsTable->item(iRow, 4)->text() == thread->longitude() ) {
1856 ((bncTableItem*) _mountPointsTable->item(iRow, 7))->setGetThread(thread);
1857 disconnect(thread, SIGNAL(newBytes(QByteArray, double)),
1858 _bncFigure, SLOT(slotNewData(QByteArray, double)));
1859 connect(thread, SIGNAL(newBytes(QByteArray, double)),
1860 _bncFigure, SLOT(slotNewData(QByteArray, double)));
1861 disconnect(thread, SIGNAL(newLatency(QByteArray, double)),
1862 _bncFigureLate, SLOT(slotNewLatency(QByteArray, double)));
1863 connect(thread, SIGNAL(newLatency(QByteArray, double)),
1864 _bncFigureLate, SLOT(slotNewLatency(QByteArray, double)));
1865 if ( Qt::CheckState(settings.value("pppPlotCoordinates").toInt()) == Qt::Checked) {
1866 disconnect(thread,
1867 SIGNAL(newPosition(bncTime, double, double, double)),
1868 _bncFigurePPP,
1869 SLOT(slotNewPosition(bncTime, double, double, double)));
1870 connect(thread, SIGNAL(newPosition(bncTime, double, double, double)),
1871 _bncFigurePPP,
1872 SLOT(slotNewPosition(bncTime, double, double, double)));
1873 }
1874 break;
1875 }
1876 }
1877 }
1878}
1879
1880//
1881////////////////////////////////////////////////////////////////////////////
1882void bncWindow::CreateMenu() {
1883 // Create Menus
1884 // ------------
1885 _menuFile = menuBar()->addMenu(tr("&File"));
1886 _menuFile->addAction(_actFontSel);
1887 _menuFile->addSeparator();
1888 _menuFile->addAction(_actSaveOpt);
1889 _menuFile->addSeparator();
1890 _menuFile->addAction(_actQuit);
1891
1892 _menuHlp = menuBar()->addMenu(tr("&Help"));
1893 _menuHlp->addAction(_actHelp);
1894 _menuHlp->addAction(_actFlowchart);
1895 _menuHlp->addAction(_actAbout);
1896}
1897
1898// Toolbar
1899////////////////////////////////////////////////////////////////////////////
1900void bncWindow::AddToolbar() {
1901 QToolBar* toolBar = new QToolBar;
1902 addToolBar(Qt::BottomToolBarArea, toolBar);
1903 toolBar->setMovable(false);
1904 toolBar->addAction(_actAddMountPoints);
1905 toolBar->addAction(_actDeleteMountPoints);
1906 toolBar->addAction(_actStart);
1907 toolBar->addAction(_actStop);
1908 toolBar->addWidget(new QLabel(" "));
1909 toolBar->addAction(_actwhatsthis);
1910}
1911
1912// About
1913////////////////////////////////////////////////////////////////////////////
1914bncAboutDlg::bncAboutDlg(QWidget* parent) :
1915 QDialog(parent) {
1916
1917 QTextBrowser* tb = new QTextBrowser;
1918 QUrl url; url.setPath(":bncabout.html");
1919 tb->setSource(url);
1920 tb->setReadOnly(true);
1921
1922 int ww = QFontMetrics(font()).width('w');
1923 QPushButton* _closeButton = new QPushButton("Close");
1924 _closeButton->setMaximumWidth(10*ww);
1925 connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
1926
1927 QGridLayout* dlgLayout = new QGridLayout();
1928 QLabel* img = new QLabel();
1929 img->setPixmap(QPixmap(":ntrip-logo.png"));
1930 dlgLayout->addWidget(img, 0,0);
1931 dlgLayout->addWidget(new QLabel("BKG Ntrip Client (BNC) Version "BNCVERSION), 0,1);
1932 dlgLayout->addWidget(tb,1,0,1,2);
1933 dlgLayout->addWidget(_closeButton,2,1,Qt::AlignRight);
1934
1935 setLayout(dlgLayout);
1936 resize(60*ww, 60*ww);
1937 setWindowTitle("About BNC");
1938 show();
1939}
1940
1941//
1942////////////////////////////////////////////////////////////////////////////
1943bncAboutDlg::~bncAboutDlg() {
1944};
1945
1946// Flowchart
1947////////////////////////////////////////////////////////////////////////////
1948bncFlowchartDlg::bncFlowchartDlg(QWidget* parent) :
1949 QDialog(parent) {
1950
1951 int ww = QFontMetrics(font()).width('w');
1952 QPushButton* _closeButton = new QPushButton("Close");
1953 _closeButton->setMaximumWidth(10*ww);
1954 connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
1955
1956 QGridLayout* dlgLayout = new QGridLayout();
1957 QLabel* img = new QLabel();
1958 img->setPixmap(QPixmap(":bncflowchart.png"));
1959 dlgLayout->addWidget(img, 0,0);
1960 dlgLayout->addWidget(_closeButton,1,0,Qt::AlignLeft);
1961
1962 setLayout(dlgLayout);
1963 setWindowTitle("Flow Chart");
1964 show();
1965}
1966
1967//
1968////////////////////////////////////////////////////////////////////////////
1969bncFlowchartDlg::~bncFlowchartDlg() {
1970};
1971
1972// Enable/Disable Widget (and change its color)
1973////////////////////////////////////////////////////////////////////////////
1974void bncWindow::enableWidget(bool enable, QWidget* widget) {
1975 const static QPalette paletteWhite(QColor(255, 255, 255));
1976 const static QPalette paletteGray(QColor(230, 230, 230));
1977
1978 widget->setEnabled(enable);
1979 if (enable) {
1980 widget->setPalette(paletteWhite);
1981 }
1982 else {
1983 widget->setPalette(paletteGray);
1984 }
1985}
1986
1987// Bnc Text
1988////////////////////////////////////////////////////////////////////////////
1989void bncWindow::slotBncTextChanged(){
1990
1991 bool enable = true;
1992
1993 // Proxy
1994 //------
1995 if (sender() == 0 || sender() == _proxyHostLineEdit) {
1996 enable = !_proxyHostLineEdit->text().isEmpty();
1997 enableWidget(enable, _proxyPortLineEdit);
1998 }
1999
2000 // RINEX Observations
2001 // ------------------
2002 if (sender() == 0 || sender() == _rnxPathLineEdit) {
2003 enable = !_rnxPathLineEdit->text().isEmpty();
2004 enableWidget(enable, _rnxSamplSpinBox);
2005 enableWidget(enable, _rnxSkelLineEdit);
2006 enableWidget(enable, _rnxScrpLineEdit);
2007 enableWidget(enable, _rnxV3CheckBox);
2008 enableWidget(enable, _rnxIntrComboBox);
2009 }
2010
2011 // RINEX Ephemeris
2012 // ---------------
2013 if (sender() == 0 || sender() == _ephPathLineEdit || sender() == _outEphPortLineEdit) {
2014 enable = !_ephPathLineEdit->text().isEmpty() || !_outEphPortLineEdit->text().isEmpty();
2015 enableWidget(enable, _ephIntrComboBox);
2016 enableWidget(enable, _ephV3CheckBox);
2017 }
2018
2019 // Broadcast Corrections
2020 // ---------------------
2021 if (sender() == 0 || sender() == _corrPathLineEdit || sender() == _corrPortLineEdit) {
2022 enable = !_corrPathLineEdit->text().isEmpty() || !_corrPortLineEdit->text().isEmpty();
2023 enableWidget(enable, _corrIntrComboBox);
2024 }
2025
2026 // Feed Engine
2027 // -----------
2028 if (sender() == 0 || sender() == _outPortLineEdit || sender() == _outFileLineEdit) {
2029 enable = !_outPortLineEdit->text().isEmpty() || !_outFileLineEdit->text().isEmpty();
2030 enableWidget(enable, _waitTimeSpinBox);
2031 enableWidget(enable, _binSamplSpinBox);
2032 }
2033
2034 // Serial Output
2035 // -------------
2036 if (sender() == 0 || sender() == _serialMountPointLineEdit ||
2037 sender() == _serialAutoNMEAComboBox) {
2038 enable = !_serialMountPointLineEdit->text().isEmpty();
2039 enableWidget(enable, _serialPortNameLineEdit);
2040 enableWidget(enable, _serialBaudRateComboBox);
2041 enableWidget(enable, _serialParityComboBox);
2042 enableWidget(enable, _serialDataBitsComboBox);
2043 enableWidget(enable, _serialStopBitsComboBox);
2044 enableWidget(enable, _serialFlowControlComboBox);
2045 enableWidget(enable, _serialAutoNMEAComboBox);
2046
2047 bool enable2 = enable && _serialAutoNMEAComboBox->currentText() != "Auto";
2048 enableWidget(enable2, _serialFileNMEALineEdit);
2049 }
2050
2051 // Outages
2052 // -------
2053 if (sender() == 0 || sender() == _obsRateComboBox) {
2054 enable = !_obsRateComboBox->currentText().isEmpty();
2055 enableWidget(enable, _adviseFailSpinBox);
2056 enableWidget(enable, _adviseRecoSpinBox);
2057 enableWidget(enable, _adviseScriptLineEdit);
2058 }
2059
2060 // Miscellaneous
2061 // -------------
2062 if (sender() == 0 || sender() == _miscMountLineEdit) {
2063 enable = !_miscMountLineEdit->text().isEmpty();
2064 enableWidget(enable, _perfIntrComboBox);
2065 enableWidget(enable, _scanRTCMCheckBox);
2066 }
2067
2068 // PPP Client
2069 // ----------
2070 if (sender() == 0
2071 || sender() == _pppMountLineEdit
2072 || sender() == _pppCorrMountLineEdit
2073 || sender() == _pppRefCrdXLineEdit
2074 || sender() == _pppRefCrdYLineEdit
2075 || sender() == _pppRefCrdZLineEdit
2076 || sender() == _pppRefdNLineEdit
2077 || sender() == _pppRefdELineEdit
2078 || sender() == _pppRefdULineEdit
2079 || sender() == _pppSync
2080 || sender() == _pppSPPComboBox
2081 || sender() == _pppQuickStartLineEdit
2082 || sender() == _pppEstTropoCheckBox
2083 || sender() == _pppUsePhaseCheckBox
2084 || sender() == _pppAntexFileChooser ) {
2085
2086 enable = (!_pppMountLineEdit->text().isEmpty() && !_pppCorrMountLineEdit->text().isEmpty()) ||
2087 (!_pppMountLineEdit->text().isEmpty() && _pppSPPComboBox->currentText() == "Realtime-SPP") ||
2088 (_pppSPPComboBox->currentText() == "Post-Processing");
2089
2090 enableWidget(enable, _pppNMEALineEdit);
2091 enableWidget(enable, _pppNMEAPortLineEdit);
2092 enableWidget(enable, _pppRefCrdXLineEdit);
2093 enableWidget(enable, _pppRefCrdYLineEdit);
2094 enableWidget(enable, _pppRefCrdZLineEdit);
2095 enableWidget(enable, _pppRefdNLineEdit);
2096 enableWidget(enable, _pppRefdELineEdit);
2097 enableWidget(enable, _pppRefdULineEdit);
2098 enableWidget(enable, _pppUsePhaseCheckBox);
2099 enableWidget(enable, _pppPlotCoordinates);
2100 enableWidget(enable, _pppEstTropoCheckBox);
2101 enableWidget(enable, _pppGLONASSCheckBox);
2102 enableWidget(enable, _pppGalileoCheckBox);
2103 enableWidget(enable, _pppAntexFileChooser);
2104 enableWidget(enable, _pppSigCLineEdit);
2105 enableWidget(enable, _pppSigCrd0);
2106 enableWidget(enable, _pppSigCrdP);
2107
2108 bool enable2 = enable && !_pppRefCrdXLineEdit->text().isEmpty() &&
2109 !_pppRefCrdYLineEdit->text().isEmpty() &&
2110 !_pppRefCrdZLineEdit->text().isEmpty();
2111
2112 enableWidget(enable2, _pppAverageLineEdit);
2113 enableWidget(enable2, _pppQuickStartLineEdit);
2114
2115 bool enable3 = enable2 && !_pppQuickStartLineEdit->text().isEmpty();
2116 enableWidget(enable3, _pppMaxSolGapLineEdit);
2117
2118 bool enable4 = enable && !_pppAntexFileChooser->fileName().isEmpty();
2119 enableWidget(enable4, _pppAntennaLineEdit);
2120 enableWidget(enable4, _pppApplySatAntCheckBox);
2121
2122 bool enable5 = enable && _pppEstTropoCheckBox->isChecked() && !_pppMountLineEdit->text().isEmpty();
2123 enableWidget(enable5, _pppSigTrp0);
2124 enableWidget(enable5, _pppSigTrpP);
2125
2126 bool enable6 = enable && _pppUsePhaseCheckBox->isChecked();
2127 enableWidget(enable6, _pppSigPLineEdit);
2128
2129 bool enable7 = enable && _pppSPPComboBox->currentText() == "Realtime-PPP";
2130 enableWidget(enable7, _pppSync);
2131
2132 bool enable8 = _pppSPPComboBox->currentText() == "Realtime-PPP";
2133 enableWidget(enable8, _pppCorrMountLineEdit);
2134
2135 bool enable9 = _pppSPPComboBox->currentText() == "Post-Processing";
2136 enableWidget(enable9, _postObsFileChooser);
2137 enableWidget(enable9, _postNavFileChooser);
2138 enableWidget(enable9, _postCorrFileChooser);
2139 enableWidget(enable9, _postOutLineEdit);
2140 enableWidget(!enable9, _pppMountLineEdit);
2141 }
2142
2143 if (sender() == 0 || sender() == _teqcActionComboBox) {
2144 bool enable10 = _teqcActionComboBox->currentText() == "Edit";
2145 enableWidget(enable10, _teqcEditOptionButton);
2146 enableWidget(!enable10, _teqcNavFileChooser);
2147 enableWidget(!enable10, _teqcOutLineEdit);
2148 }
2149}
2150
2151//
2152////////////////////////////////////////////////////////////////////////////
2153void bncWindow::slotAddCmbRow() {
2154 int iRow = _cmbTable->rowCount();
2155 _cmbTable->insertRow(iRow);
2156 for (int iCol = 0; iCol < _cmbTable->columnCount(); iCol++) {
2157 _cmbTable->setItem(iRow, iCol, new QTableWidgetItem(""));
2158 }
2159}
2160
2161//
2162////////////////////////////////////////////////////////////////////////////
2163void bncWindow::slotDelCmbRow() {
2164 int nRows = _cmbTable->rowCount();
2165 bool flg[nRows];
2166 for (int iRow = 0; iRow < nRows; iRow++) {
2167 if (_cmbTable->isItemSelected(_cmbTable->item(iRow,1))) {
2168 flg[iRow] = true;
2169 }
2170 else {
2171 flg[iRow] = false;
2172 }
2173 }
2174 for (int iRow = nRows-1; iRow >= 0; iRow--) {
2175 if (flg[iRow]) {
2176 _cmbTable->removeRow(iRow);
2177 }
2178 }
2179}
2180
2181//
2182////////////////////////////////////////////////////////////////////////////
2183void bncWindow::populateCmbTable() {
2184
2185 for (int iRow = _cmbTable->rowCount()-1; iRow >=0; iRow--) {
2186 _cmbTable->removeRow(iRow);
2187 }
2188
2189 bncSettings settings;
2190
2191 int iRow = -1;
2192 QListIterator<QString> it(settings.value("combineStreams").toStringList());
2193 while (it.hasNext()) {
2194 QStringList hlp = it.next().split(" ");
2195 if (hlp.size() > 2) {
2196 ++iRow;
2197 _cmbTable->insertRow(iRow);
2198 }
2199 for (int iCol = 0; iCol < hlp.size(); iCol++) {
2200 _cmbTable->setItem(iRow, iCol, new QTableWidgetItem(hlp[iCol]));
2201 }
2202 }
2203}
2204
2205//
2206////////////////////////////////////////////////////////////////////////////
2207void bncWindow::slotAddUploadRow() {
2208 int iRow = _uploadTable->rowCount();
2209 _uploadTable->insertRow(iRow);
2210 for (int iCol = 0; iCol < _uploadTable->columnCount(); iCol++) {
2211 if (iCol == 3) {
2212 QLineEdit* passwd = new QLineEdit();
2213 passwd->setFrame(false);
2214 passwd->setEchoMode(QLineEdit::PasswordEchoOnEdit);
2215 _uploadTable->setCellWidget(iRow, iCol, passwd);
2216 }
2217 else if (iCol == 4) {
2218 QComboBox* system = new QComboBox();
2219 system->setEditable(false);
2220 system->addItems(QString("IGS05,ETRF2000,NAD83,GDA94,SIRGAS95,SIRGAS2000,Custom").split(","));
2221 system->setFrame(false);
2222 _uploadTable->setCellWidget(iRow, iCol, system);
2223 }
2224 else if (iCol == 5) {
2225 QCheckBox* com = new QCheckBox();
2226 _uploadTable->setCellWidget(iRow, iCol, com);
2227 }
2228 else if (iCol == 8) {
2229 bncTableItem* bncIt = new bncTableItem();
2230 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
2231 _uploadTable->setItem(iRow, iCol, bncIt);
2232 ((bncApp*)qApp)->_uploadTableItems[iRow] = bncIt;
2233 }
2234 else {
2235 _uploadTable->setItem(iRow, iCol, new QTableWidgetItem(""));
2236 }
2237 }
2238}
2239
2240//
2241////////////////////////////////////////////////////////////////////////////
2242void bncWindow::slotDelUploadRow() {
2243 ((bncApp*)qApp)->_uploadTableItems.clear();
2244 int nRows = _uploadTable->rowCount();
2245 bool flg[nRows];
2246 for (int iRow = 0; iRow < nRows; iRow++) {
2247 if (_uploadTable->isItemSelected(_uploadTable->item(iRow,1))) {
2248 flg[iRow] = true;
2249 }
2250 else {
2251 flg[iRow] = false;
2252 }
2253 }
2254 for (int iRow = nRows-1; iRow >= 0; iRow--) {
2255 if (flg[iRow]) {
2256 _uploadTable->removeRow(iRow);
2257 }
2258 }
2259 for (int iRow = 0; iRow < _uploadTable->rowCount(); iRow++) {
2260 ((bncApp*)qApp)->_uploadTableItems[iRow] =
2261 (bncTableItem*) _uploadTable->item(iRow, 8);
2262 }
2263}
2264
2265//
2266////////////////////////////////////////////////////////////////////////////
2267void bncWindow::populateUploadTable() {
2268 for (int iRow = _uploadTable->rowCount()-1; iRow >=0; iRow--) {
2269 _uploadTable->removeRow(iRow);
2270 }
2271
2272 bncSettings settings;
2273
2274 int iRow = -1;
2275 QListIterator<QString> it(settings.value("uploadMountpointsOut").toStringList());
2276 while (it.hasNext()) {
2277 QStringList hlp = it.next().split(",");
2278 if (hlp.size() > 6) {
2279 ++iRow;
2280 _uploadTable->insertRow(iRow);
2281 }
2282 for (int iCol = 0; iCol < hlp.size(); iCol++) {
2283 if (iCol == 3) {
2284 QLineEdit* passwd = new QLineEdit();
2285 passwd->setFrame(false);
2286 passwd->setEchoMode(QLineEdit::PasswordEchoOnEdit);
2287 passwd->setText(hlp[iCol]);
2288 _uploadTable->setCellWidget(iRow, iCol, passwd);
2289 }
2290 else if (iCol == 4) {
2291 QComboBox* system = new QComboBox();
2292 system->setEditable(false);
2293 system->addItems(QString("IGS05,ETRF2000,NAD83,GDA94,SIRGAS95,SIRGAS2000,Custom").split(","));
2294 system->setFrame(false);
2295 system->setCurrentIndex(system->findText(hlp[iCol]));
2296 _uploadTable->setCellWidget(iRow, iCol, system);
2297 }
2298 else if (iCol == 5) {
2299 QCheckBox* com = new QCheckBox();
2300 if (hlp[iCol].toInt() == Qt::Checked) {
2301 com->setCheckState(Qt::Checked);
2302 }
2303 _uploadTable->setCellWidget(iRow, iCol, com);
2304 }
2305 else if (iCol == 8) {
2306 bncTableItem* bncIt = new bncTableItem();
2307 bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
2308 _uploadTable->setItem(iRow, iCol, bncIt);
2309 ((bncApp*)qApp)->_uploadTableItems[iRow] = bncIt;
2310 }
2311 else {
2312 _uploadTable->setItem(iRow, iCol, new QTableWidgetItem(hlp[iCol]));
2313 }
2314 }
2315 }
2316}
2317
2318//
2319////////////////////////////////////////////////////////////////////////////
2320void bncWindow::slotSetUploadTrafo() {
2321 bncCustomTrafo* dlg = new bncCustomTrafo(this);
2322 dlg->exec();
2323 delete dlg;
2324}
2325
2326// Start Post-Processing PPP
2327////////////////////////////////////////////////////////////////////////////
2328void bncWindow::startPostProcessingPPP() {
2329#ifdef USE_POSTPROCESSING
2330 _runningPostProcessingPPP = true;
2331 _actStart->setText("0 Epochs");
2332
2333 slotSaveOptions();
2334
2335 t_postProcessing* postProcessing = new t_postProcessing(this);
2336 connect(postProcessing, SIGNAL(finished()), this, SLOT(slotFinishedPostProcessingPPP()));
2337 connect(postProcessing, SIGNAL(progress(int)), this, SLOT(slotPostProgress(int)));
2338
2339 postProcessing->start();
2340#else
2341 QMessageBox::information(this, "Information",
2342 "Post-Processing Not Permitted");
2343#endif
2344}
2345
2346// Post-Processing PPP Finished
2347////////////////////////////////////////////////////////////////////////////
2348void bncWindow::slotFinishedPostProcessingPPP() {
2349 _runningPostProcessingPPP = false;
2350 QMessageBox::information(this, "Information",
2351 "Post-Processing Thread Finished");
2352 _actStart->setText("Start");
2353 enableStartStop();
2354}
2355
2356// Progress Bar Change
2357////////////////////////////////////////////////////////////////////////////
2358void bncWindow::slotPostProgress(int nEpo) {
2359 if (_actStart) {
2360 _actStart->setText(QString("%1 Epochs").arg(nEpo));
2361 }
2362}
2363
2364// Start Post-Processing Teqc
2365////////////////////////////////////////////////////////////////////////////
2366void bncWindow::startPostProcessingTeqc() {
2367 _runningPostProcessingTeqc = false; // TODO
2368 QMessageBox::information(this, "Information",
2369 "Teqc-Processing Not Yet Implemented");
2370}
2371
2372// Post-Processing Teqc Finished
2373////////////////////////////////////////////////////////////////////////////
2374void bncWindow::slotFinishedPostProcessingTeqc() {
2375 _runningPostProcessingTeqc = false;
2376 QMessageBox::information(this, "Information",
2377 "Teqc-Processing Thread Finished");
2378}
2379
2380// Edit teqc-like editing options
2381////////////////////////////////////////////////////////////////////////////
2382void bncWindow::slotTeqcEditOption() {
2383 teqcDlg* dlg = new teqcDlg(this);
2384 dlg->move(this->pos().x()+50, this->pos().y()+50);
2385 dlg->exec();
2386 delete dlg;
2387}
2388
2389// Enable/Disable Start and Stop Buttons
2390////////////////////////////////////////////////////////////////////////////
2391void bncWindow::enableStartStop() {
2392
2393 if ( _pppSPPComboBox->currentText() == "Post-Processing" ) {
2394 _actStart->setEnabled(true);
2395 _actStop->setEnabled(false);
2396 }
2397 else if ( !_teqcActionComboBox->currentText().isEmpty() ) {
2398 _actStart->setEnabled(true);
2399 _actStop->setEnabled(false);
2400 }
2401 else {
2402 if (_runningRealTime) {
2403 _actStart->setEnabled(false);
2404 _actStop->setEnabled(true);
2405 }
2406 else {
2407 _actStop->setEnabled(false);
2408 if (_mountPointsTable->rowCount() == 0) {
2409 _actStart->setEnabled(false);
2410 }
2411 else {
2412 _actStart->setEnabled(true);
2413 }
2414 }
2415 }
2416}
Note: See TracBrowser for help on using the repository browser.