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

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