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

Last change on this file since 6764 was 6764, checked in by weber, 9 years ago

Documentation completed

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