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

Last change on this file since 7172 was 7172, checked in by stuerze, 9 years ago

Check box for the optional choice of RINEX V3 navigation file names is added

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