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

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