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

Last change on this file since 6754 was 6754, checked in by weber, 7 years ago

Documentation completed

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