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

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