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

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

Design of Rinex 2/3 observation output specification

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