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

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

Documentation completed

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