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

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