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

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

Option 'Directory for plots' enabled

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