// Part of BNC, a utility for retrieving decoding and
// converting GNSS data streams from NTRIP broadcasters.
//
// Copyright (C) 2007
// German Federal Agency for Cartography and Geodesy (BKG)
// http://www.bkg.bund.de
// Czech Technical University Prague, Department of Geodesy
// http://www.fsv.cvut.cz
//
// Email: euref-ip@bkg.bund.de
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation, version 2.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
/* -------------------------------------------------------------------------
* BKG NTRIP Client
* -------------------------------------------------------------------------
*
* Class: bncWindow
*
* Purpose: This class implements the main application window.
*
* Author: L. Mervart
*
* Created: 24-Dec-2005
*
* Changes:
*
* -----------------------------------------------------------------------*/
#include
#include
#include "bncwindow.h"
#include "bncapp.h"
#include "bncgetthread.h"
#include "bnctabledlg.h"
#include "bncipport.h"
#include "bncudpport.h"
#include "bncserialport.h"
#include "bnchlpdlg.h"
#include "bnchtml.h"
#include "bnctableitem.h"
#include "bncsettings.h"
#include "bncfigure.h"
#include "bncfigurelate.h"
#include "bncfigureppp.h"
#include "bncversion.h"
#include "bncbytescounter.h"
#include "bncsslconfig.h"
#include "upload/bnccustomtrafo.h"
#include "upload/bncephuploadcaster.h"
#include "qtfilechooser.h"
using namespace std;
// Constructor
////////////////////////////////////////////////////////////////////////////
bncWindow::bncWindow() {
_caster = 0;
_casterEph = 0;
_bncFigure = new bncFigure(this);
_bncFigureLate = new bncFigureLate(this);
_bncFigurePPP = new bncFigurePPP(this);
int ww = QFontMetrics(this->font()).width('w');
static const QStringList labels = QString("account, Streams: resource loader / mountpoint, decoder, lat, long, nmea, ntrip, bytes").split(",");
setMinimumSize(85*ww, 65*ww);
setWindowTitle(tr("BKG Ntrip Client (BNC) Version " BNCVERSION));
connect((bncApp*)qApp, SIGNAL(newMessage(QByteArray,bool)),
this, SLOT(slotWindowMessage(QByteArray,bool)));
// Create Actions
// --------------
_actHelp = new QAction(tr("&Help Contents"),this);
connect(_actHelp, SIGNAL(triggered()), SLOT(slotHelp()));
_actAbout = new QAction(tr("&About BNC"),this);
connect(_actAbout, SIGNAL(triggered()), SLOT(slotAbout()));
_actFlowchart = new QAction(tr("&Flow Chart"),this);
connect(_actFlowchart, SIGNAL(triggered()), SLOT(slotFlowchart()));
_actFontSel = new QAction(tr("Select &Font"),this);
connect(_actFontSel, SIGNAL(triggered()), SLOT(slotFontSel()));
_actSaveOpt = new QAction(tr("&Save && Reread Configuration"),this);
connect(_actSaveOpt, SIGNAL(triggered()), SLOT(slotSaveOptions()));
_actQuit = new QAction(tr("&Quit"),this);
connect(_actQuit, SIGNAL(triggered()), SLOT(close()));
_actAddMountPoints = new QAction(tr("Add &Stream"),this);
connect(_actAddMountPoints, SIGNAL(triggered()), SLOT(slotAddMountPoints()));
_actDeleteMountPoints = new QAction(tr("&Delete Stream"),this);
connect(_actDeleteMountPoints, SIGNAL(triggered()), SLOT(slotDeleteMountPoints()));
_actDeleteMountPoints->setEnabled(false);
_actGetData = new QAction(tr("Sta&rt"),this);
connect(_actGetData, SIGNAL(triggered()), SLOT(slotGetData()));
_actPostProcessing = new QAction(tr("Start PP"),this);
connect(_actPostProcessing, SIGNAL(triggered()), SLOT(slotPostProcessing()));
_postProgressBar = new QProgressBar;
_postProgressBar->setTextVisible(true);
_postProgressBar->setMinimum(0);
_postProgressBar->setMaximum(100);
_postProgressBar->setMinimumWidth(20*ww);
_postProgressBar->setMaximumWidth(20*ww);
_actStop = new QAction(tr("Sto&p"),this);
connect(_actStop, SIGNAL(triggered()), SLOT(slotStop()));
_actStop->setEnabled(false);
_actwhatsthis= new QAction(tr("Help ?=Shift+F1"),this);
connect(_actwhatsthis, SIGNAL(triggered()), SLOT(slotWhatsThis()));
CreateMenu();
AddToolbar();
bncSettings settings;
// Netowrk Options
// ---------------
_proxyHostLineEdit = new QLineEdit(settings.value("proxyHost").toString());
_proxyPortLineEdit = new QLineEdit(settings.value("proxyPort").toString());
connect(_proxyHostLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
_sslCaCertPathLineEdit = new QLineEdit(settings.value("sslCaCertPath").toString());
_ignoreSslErrorsCheckBox = new QCheckBox();
_ignoreSslErrorsCheckBox->setCheckState(Qt::CheckState(
settings.value("ignoreSslErrors").toInt()));
// General Options
// ---------------
_logFileLineEdit = new QLineEdit(settings.value("logFile").toString());
_rawOutFileLineEdit = new QLineEdit(settings.value("rawOutFile").toString());
_rnxAppendCheckBox = new QCheckBox();
_rnxAppendCheckBox->setCheckState(Qt::CheckState(
settings.value("rnxAppend").toInt()));
_onTheFlyComboBox = new QComboBox();
_onTheFlyComboBox->setEditable(false);
_onTheFlyComboBox->addItems(QString("1 day,1 hour,1 min").split(","));
int ii = _onTheFlyComboBox->findText(settings.value("onTheFlyInterval").toString());
if (ii != -1) {
_onTheFlyComboBox->setCurrentIndex(ii);
}
_autoStartCheckBox = new QCheckBox();
_autoStartCheckBox->setCheckState(Qt::CheckState(
settings.value("autoStart").toInt()));
// RINEX Observations Options
// --------------------------
_rnxPathLineEdit = new QLineEdit(settings.value("rnxPath").toString());
_rnxIntrComboBox = new QComboBox();
_rnxIntrComboBox->setEditable(false);
_rnxIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
ii = _rnxIntrComboBox->findText(settings.value("rnxIntr").toString());
if (ii != -1) {
_rnxIntrComboBox->setCurrentIndex(ii);
}
_rnxSamplSpinBox = new QSpinBox();
_rnxSamplSpinBox->setMinimum(0);
_rnxSamplSpinBox->setMaximum(60);
_rnxSamplSpinBox->setSingleStep(5);
_rnxSamplSpinBox->setValue(settings.value("rnxSampl").toInt());
_rnxSamplSpinBox->setSuffix(" sec");
_rnxSkelLineEdit = new QLineEdit(settings.value("rnxSkel").toString());
_rnxSkelLineEdit->setMaximumWidth(5*ww);
_rnxScrpLineEdit = new QLineEdit(settings.value("rnxScript").toString());
_rnxV3CheckBox = new QCheckBox();
_rnxV3CheckBox->setCheckState(Qt::CheckState(settings.value("rnxV3").toInt()));
connect(_rnxPathLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// RINEX Ephemeris Options
// -----------------------
_ephPathLineEdit = new QLineEdit(settings.value("ephPath").toString());
_ephIntrComboBox = new QComboBox();
_ephIntrComboBox->setEditable(false);
_ephIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
int jj = _ephIntrComboBox->findText(settings.value("ephIntr").toString());
if (jj != -1) {
_ephIntrComboBox->setCurrentIndex(jj);
}
_outEphPortLineEdit = new QLineEdit(settings.value("outEphPort").toString());
_ephV3CheckBox = new QCheckBox();
_ephV3CheckBox->setCheckState(Qt::CheckState(settings.value("ephV3").toInt()));
connect(_outEphPortLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_ephPathLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Broadcast Corrections Options
// -----------------------------
_corrPathLineEdit = new QLineEdit(settings.value("corrPath").toString());
_corrIntrComboBox = new QComboBox();
_corrIntrComboBox->setEditable(false);
_corrIntrComboBox->addItems(QString("1 min,2 min,5 min,10 min,15 min,30 min,1 hour,1 day").split(","));
int mm = _corrIntrComboBox->findText(settings.value("corrIntr").toString());
if (mm != -1) {
_corrIntrComboBox->setCurrentIndex(mm);
}
_corrPortLineEdit = new QLineEdit(settings.value("corrPort").toString());
_corrTimeSpinBox = new QSpinBox();
_corrTimeSpinBox->setMinimum(0);
_corrTimeSpinBox->setMaximum(60);
_corrTimeSpinBox->setSingleStep(1);
_corrTimeSpinBox->setSuffix(" sec");
_corrTimeSpinBox->setValue(settings.value("corrTime").toInt());
connect(_corrPathLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_corrPortLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Feed Engine Options
// -------------------
_outPortLineEdit = new QLineEdit(settings.value("outPort").toString());
_waitTimeSpinBox = new QSpinBox();
_waitTimeSpinBox->setMinimum(1);
_waitTimeSpinBox->setMaximum(30);
_waitTimeSpinBox->setSingleStep(1);
_waitTimeSpinBox->setSuffix(" sec");
_waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
_binSamplSpinBox = new QSpinBox();
_binSamplSpinBox->setMinimum(0);
_binSamplSpinBox->setMaximum(60);
_binSamplSpinBox->setSingleStep(5);
_binSamplSpinBox->setValue(settings.value("binSampl").toInt());
_binSamplSpinBox->setSuffix(" sec");
_outFileLineEdit = new QLineEdit(settings.value("outFile").toString());
_outUPortLineEdit = new QLineEdit(settings.value("outUPort").toString());
connect(_outPortLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_outFileLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Serial Output Options
// ---------------------
_serialMountPointLineEdit = new QLineEdit(settings.value("serialMountPoint").toString());
_serialPortNameLineEdit = new QLineEdit(settings.value("serialPortName").toString());
_serialBaudRateComboBox = new QComboBox();
_serialBaudRateComboBox->addItems(QString("110,300,600,"
"1200,2400,4800,9600,19200,38400,57600,115200").split(","));
int kk = _serialBaudRateComboBox->findText(settings.value("serialBaudRate").toString());
if (kk != -1) {
_serialBaudRateComboBox->setCurrentIndex(kk);
}
_serialFlowControlComboBox = new QComboBox();
_serialFlowControlComboBox->addItems(QString("OFF,XONXOFF,HARDWARE").split(","));
kk = _serialFlowControlComboBox->findText(settings.value("serialFlowControl").toString());
if (kk != -1) {
_serialFlowControlComboBox->setCurrentIndex(kk);
}
_serialDataBitsComboBox = new QComboBox();
_serialDataBitsComboBox->addItems(QString("5,6,7,8").split(","));
kk = _serialDataBitsComboBox->findText(settings.value("serialDataBits").toString());
if (kk != -1) {
_serialDataBitsComboBox->setCurrentIndex(kk);
}
_serialParityComboBox = new QComboBox();
_serialParityComboBox->addItems(QString("NONE,ODD,EVEN,SPACE").split(","));
kk = _serialParityComboBox->findText(settings.value("serialParity").toString());
if (kk != -1) {
_serialParityComboBox->setCurrentIndex(kk);
}
_serialStopBitsComboBox = new QComboBox();
_serialStopBitsComboBox->addItems(QString("1,2").split(","));
kk = _serialStopBitsComboBox->findText(settings.value("serialStopBits").toString());
if (kk != -1) {
_serialStopBitsComboBox->setCurrentIndex(kk);
}
_serialAutoNMEAComboBox = new QComboBox();
_serialAutoNMEAComboBox->addItems(QString("Auto,Manual").split(","));
kk = _serialAutoNMEAComboBox->findText(settings.value("serialAutoNMEA").toString());
if (kk != -1) {
_serialAutoNMEAComboBox->setCurrentIndex(kk);
}
_serialFileNMEALineEdit = new QLineEdit(settings.value("serialFileNMEA").toString());
_serialHeightNMEALineEdit = new QLineEdit(settings.value("serialHeightNMEA").toString());
connect(_serialMountPointLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_serialAutoNMEAComboBox, SIGNAL(currentIndexChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Outages Options
// ---------------
_obsRateComboBox = new QComboBox();
_obsRateComboBox->setEditable(false);
_obsRateComboBox->addItems(QString(",0.1 Hz,0.2 Hz,0.5 Hz,1 Hz,5 Hz").split(","));
kk = _obsRateComboBox->findText(settings.value("obsRate").toString());
if (kk != -1) {
_obsRateComboBox->setCurrentIndex(kk);
}
_adviseFailSpinBox = new QSpinBox();
_adviseFailSpinBox->setMinimum(0);
_adviseFailSpinBox->setMaximum(60);
_adviseFailSpinBox->setSingleStep(1);
_adviseFailSpinBox->setSuffix(" min");
_adviseFailSpinBox->setValue(settings.value("adviseFail").toInt());
_adviseRecoSpinBox = new QSpinBox();
_adviseRecoSpinBox->setMinimum(0);
_adviseRecoSpinBox->setMaximum(60);
_adviseRecoSpinBox->setSingleStep(1);
_adviseRecoSpinBox->setSuffix(" min");
_adviseRecoSpinBox->setValue(settings.value("adviseReco").toInt());
_adviseScriptLineEdit = new QLineEdit(settings.value("adviseScript").toString());
connect(_obsRateComboBox, SIGNAL(currentIndexChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Miscellaneous Options
// ---------------------
_miscMountLineEdit = new QLineEdit(settings.value("miscMount").toString());
_perfIntrComboBox = new QComboBox();
_perfIntrComboBox->setEditable(false);
_perfIntrComboBox->addItems(QString(",2 sec, 10 sec,1 min,5 min,15 min,1 hour,6 hours,1 day").split(","));
int ll = _perfIntrComboBox->findText(settings.value("perfIntr").toString());
if (ll != -1) {
_perfIntrComboBox->setCurrentIndex(ll);
}
_scanRTCMCheckBox = new QCheckBox();
_scanRTCMCheckBox->setCheckState(Qt::CheckState(
settings.value("scanRTCM").toInt()));
connect(_miscMountLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// PPP Options
// -----------
_pppMountLineEdit = new QLineEdit(settings.value("pppMount").toString());
_pppCorrMountLineEdit = new QLineEdit(settings.value("pppCorrMount").toString());
_pppNMEALineEdit = new QLineEdit(settings.value("nmeaFile").toString());
_pppNMEAPortLineEdit = new QLineEdit(settings.value("nmeaPort").toString());
_pppSigCLineEdit = new QLineEdit(settings.value("pppSigmaCode").toString());
_pppSigPLineEdit = new QLineEdit(settings.value("pppSigmaPhase").toString());
_pppSigCrd0 = new QLineEdit(settings.value("pppSigCrd0").toString());
_pppSigCrdP = new QLineEdit(settings.value("pppSigCrdP").toString());
_pppSigTrp0 = new QLineEdit(settings.value("pppSigTrp0").toString());
_pppSigTrpP = new QLineEdit(settings.value("pppSigTrpP").toString());
_pppAverageLineEdit = new QLineEdit(settings.value("pppAverage").toString());
_pppQuickStartLineEdit = new QLineEdit(settings.value("pppQuickStart").toString());
_pppMaxSolGapLineEdit = new QLineEdit(settings.value("pppMaxSolGap").toString());
_pppRefCrdXLineEdit = new QLineEdit(settings.value("pppRefCrdX").toString());
_pppRefCrdYLineEdit = new QLineEdit(settings.value("pppRefCrdY").toString());
_pppRefCrdZLineEdit = new QLineEdit(settings.value("pppRefCrdZ").toString());
_pppRefdNLineEdit = new QLineEdit(settings.value("pppRefdN").toString());
_pppRefdELineEdit = new QLineEdit(settings.value("pppRefdE").toString());
_pppRefdULineEdit = new QLineEdit(settings.value("pppRefdU").toString());
_pppSync = new QLineEdit(settings.value("pppSync").toString());
_pppAntennaLineEdit = new QLineEdit(settings.value("pppAntenna").toString());
_pppAntexLineEdit = new QLineEdit(settings.value("pppAntex").toString());
_pppSPPComboBox = new QComboBox();
_pppSPPComboBox->setEditable(false);
_pppSPPComboBox->addItems(QString("PPP,SPP,Post-Processing").split(","));
int ik = _pppSPPComboBox->findText(settings.value("pppSPP").toString());
if (ik != -1) {
_pppSPPComboBox->setCurrentIndex(ik);
}
_pppUsePhaseCheckBox = new QCheckBox();
_pppUsePhaseCheckBox->setCheckState(Qt::CheckState(
settings.value("pppUsePhase").toInt()));
_pppEstTropoCheckBox = new QCheckBox();
_pppEstTropoCheckBox->setCheckState(Qt::CheckState(
settings.value("pppEstTropo").toInt()));
_pppGLONASSCheckBox = new QCheckBox();
_pppGLONASSCheckBox->setCheckState(Qt::CheckState(
settings.value("pppGLONASS").toInt()));
_pppGalileoCheckBox = new QCheckBox();
_pppGalileoCheckBox->setCheckState(Qt::CheckState(
settings.value("pppGalileo").toInt()));
_pppPlotCoordinates = new QCheckBox();
_pppPlotCoordinates->setCheckState(Qt::CheckState(
settings.value("pppPlotCoordinates").toInt()));
_pppApplySatAntCheckBox = new QCheckBox();
_pppApplySatAntCheckBox->setCheckState(Qt::CheckState(
settings.value("pppApplySatAnt").toInt()));
connect(_pppMountLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppCorrMountLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppUsePhaseCheckBox, SIGNAL(stateChanged(int)),
this, SLOT(slotBncTextChanged()));
connect(_pppRefCrdXLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppRefCrdYLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppRefCrdZLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppRefdNLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppRefdELineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppRefdULineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppEstTropoCheckBox, SIGNAL(stateChanged(int)),
this, SLOT(slotBncTextChanged()));
connect(_pppSync, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppSPPComboBox, SIGNAL(currentIndexChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppAntexLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
connect(_pppQuickStartLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Streams
// -------
_mountPointsTable = new QTableWidget(0,8);
_mountPointsTable->horizontalHeader()->resizeSection(1,34*ww);
_mountPointsTable->horizontalHeader()->resizeSection(2,9*ww);
_mountPointsTable->horizontalHeader()->resizeSection(3,7*ww);
_mountPointsTable->horizontalHeader()->resizeSection(4,7*ww);
_mountPointsTable->horizontalHeader()->resizeSection(5,5*ww);
_mountPointsTable->horizontalHeader()->resizeSection(6,5*ww);
_mountPointsTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
_mountPointsTable->horizontalHeader()->setStretchLastSection(true);
_mountPointsTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
_mountPointsTable->setHorizontalHeaderLabels(labels);
_mountPointsTable->setGridStyle(Qt::NoPen);
_mountPointsTable->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
_mountPointsTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
_mountPointsTable->setSelectionBehavior(QAbstractItemView::SelectRows);
_mountPointsTable->hideColumn(0);
connect(_mountPointsTable, SIGNAL(itemSelectionChanged()),
SLOT(slotSelectionChanged()));
populateMountPointsTable();
_log = new QTextBrowser();
_log->setReadOnly(true);
// Combination
// -----------
_cmbTable = new QTableWidget(0,3);
_cmbTable->setHorizontalHeaderLabels(QString("Mountpoint, AC Name, Weight").split(","));
_cmbTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
_cmbTable->setSelectionBehavior(QAbstractItemView::SelectRows);
_cmbTable->setMaximumWidth(30*ww);
_cmbTable->horizontalHeader()->resizeSection(0,10*ww);
_cmbTable->horizontalHeader()->resizeSection(1,8*ww);
_cmbTable->horizontalHeader()->resizeSection(2,8*ww);
_cmbTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
_cmbTable->horizontalHeader()->setStretchLastSection(true);
_cmbTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
_cmbMaxresLineEdit = new QLineEdit(settings.value("cmbMaxres").toString());
QPushButton* addCmbRowButton = new QPushButton("Add Row");
QPushButton* delCmbRowButton = new QPushButton("Delete");
connect(_cmbTable, SIGNAL(itemSelectionChanged()),
SLOT(slotBncTextChanged()));
_cmbMethodComboBox = new QComboBox();
_cmbMethodComboBox->setEditable(false);
_cmbMethodComboBox->addItems(QString("Filter,Single-Epoch").split(","));
int im = _cmbMethodComboBox->findText(settings.value("cmbMethod").toString());
if (im != -1) {
_cmbMethodComboBox->setCurrentIndex(im);
}
// Upload Results
// -------------
_uploadTable = new QTableWidget(0,9);
_uploadTable->setHorizontalHeaderLabels(QString("Host, Port, Mount, Password, System, CoM, SP3 File, RNX File, bytes").split(","));
_uploadTable->setSelectionMode(QAbstractItemView::ExtendedSelection);
_uploadTable->setSelectionBehavior(QAbstractItemView::SelectRows);
_uploadTable->horizontalHeader()->resizeSection(0,13*ww);
_uploadTable->horizontalHeader()->resizeSection(1,5*ww);
_uploadTable->horizontalHeader()->resizeSection(2,6*ww);
_uploadTable->horizontalHeader()->resizeSection(3,8*ww);
_uploadTable->horizontalHeader()->resizeSection(4,11*ww);
_uploadTable->horizontalHeader()->resizeSection(5,4*ww);
_uploadTable->horizontalHeader()->resizeSection(6,15*ww);
_uploadTable->horizontalHeader()->resizeSection(7,15*ww);
_uploadTable->horizontalHeader()->resizeSection(8,10*ww);
_uploadTable->horizontalHeader()->setResizeMode(QHeaderView::Interactive);
_uploadTable->horizontalHeader()->setStretchLastSection(true);
_uploadTable->horizontalHeader()->setDefaultAlignment(Qt::AlignLeft);
connect(_uploadTable, SIGNAL(itemSelectionChanged()),
SLOT(slotBncTextChanged()));
QPushButton* addUploadRowButton = new QPushButton("Add Row");
QPushButton* delUploadRowButton = new QPushButton("Del Row");
QPushButton* setUploadTrafoButton = new QPushButton("Custom Trafo");
_uploadIntrComboBox = new QComboBox;
_uploadIntrComboBox->setEditable(false);
_uploadIntrComboBox->addItems(QString("1 day,1 hour, 30 min,15 min,10 min,5 min,2 min,1 min").split(","));
ii = _uploadIntrComboBox->findText(settings.value("uploadIntr").toString());
if (ii != -1) {
_uploadIntrComboBox->setCurrentIndex(ii);
}
_uploadSamplSpinBox = new QSpinBox;
_uploadSamplSpinBox->setMinimum(5);
_uploadSamplSpinBox->setMaximum(60);
_uploadSamplSpinBox->setSingleStep(5);
_uploadSamplSpinBox->setMaximumWidth(9*ww);
_uploadSamplSpinBox->setValue(settings.value("uploadSampl").toInt());
_uploadSamplSpinBox->setSuffix(" sec");
_uploadSamplOrbSpinBox = new QSpinBox;
_uploadSamplOrbSpinBox->setMinimum(0);
_uploadSamplOrbSpinBox->setMaximum(60);
_uploadSamplOrbSpinBox->setSingleStep(5);
_uploadSamplOrbSpinBox->setMaximumWidth(9*ww);
_uploadSamplOrbSpinBox->setValue(settings.value("uploadSamplOrb").toInt());
_uploadSamplOrbSpinBox->setSuffix(" sec");
// Upload RTCM3 Ephemeris
// ----------------------
_uploadEphHostLineEdit = new QLineEdit(settings.value("uploadEphHost").toString());
_uploadEphPortLineEdit = new QLineEdit(settings.value("uploadEphPort").toString());
_uploadEphPasswordLineEdit = new QLineEdit(settings.value("uploadEphPassword").toString());
_uploadEphPasswordLineEdit->setEchoMode(QLineEdit::PasswordEchoOnEdit);
_uploadEphMountpointLineEdit = new QLineEdit(settings.value("uploadEphMountpoint").toString());
_uploadEphSampleSpinBox = new QSpinBox;
_uploadEphSampleSpinBox->setMinimum(5);
_uploadEphSampleSpinBox->setMaximum(60);
_uploadEphSampleSpinBox->setSingleStep(5);
_uploadEphSampleSpinBox->setMaximumWidth(9*ww);
_uploadEphSampleSpinBox->setValue(settings.value("uploadEphSample").toInt());
_uploadEphSampleSpinBox->setSuffix(" sec");
_uploadEphBytesCounter = new bncBytesCounter;
// WhatsThis
// ---------
_proxyHostLineEdit->setWhatsThis(tr("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.
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.
"));
_proxyPortLineEdit->setWhatsThis(tr("Enter your proxy server port number in case a proxy is operated in front of BNC.
"));
_waitTimeSpinBox->setWhatsThis(tr("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 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.
"));
_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."));
_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."));
_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."));
_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."));
_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."));
_corrTimeSpinBox->setWhatsThis(tr("Concerning output through IP port, BNC drops Broadcast Ephemeris Corrections received later than 'Wait for full epoch' seconds. A value of 2 to 5 seconds is recommended, depending on the latency of the incoming correction stream(s) and the delay acceptable to your real-time application.
Specifying a value of '0' means that BNC immediately outputs all incoming Broadcast Epemeris Corrections and does not drop any of them for latency reasons.
"));
_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."));
_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."));
_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."));
_rnxScrpLineEdit->setWhatsThis(tr("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).
"));
_rnxSkelLineEdit->setWhatsThis(tr("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.
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.
"));
_rnxAppendCheckBox->setWhatsThis(tr("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.
"));
_autoStartCheckBox->setWhatsThis(tr("Tick 'Auto start' for auto-start of BNC at startup time in window mode with preassigned processing options.
"));
_rawOutFileLineEdit->setWhatsThis(tr("Save all data coming in through various streams in the received order and format in one file.
"));
_onTheFlyComboBox->setWhatsThis(tr("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.
Select '1 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.
Note that when operating BNC in window mode, on-the-fly changeable configuration options become effective immediately through 'Save & Reread Configuration'.
"));
_rnxIntrComboBox->setWhatsThis(tr("Select the length of the RINEX Observation file.
"));
_ephIntrComboBox->setWhatsThis(tr("Select the length of the RINEX Navigation file.
"));
_corrIntrComboBox->setWhatsThis(tr("Select the length of the Broadcast Ephemeris Correction files.
"));
_rnxSamplSpinBox->setWhatsThis(tr("Select the RINEX Observation sampling interval in seconds. A value of zero '0' tells BNC to store all received epochs into RINEX.
"));
_binSamplSpinBox->setWhatsThis(tr("Select the synchronized observation sampling interval in seconds. A value of zero '0' tells BNC to send/store all received epochs.
"));
_obsRateComboBox->setWhatsThis(tr("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.
"));
_adviseRecoSpinBox->setWhatsThis(tr("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.
A value of zero '0' means that for any stream recovery, however short, BNC immediately generates an advisory note.
"));
_adviseFailSpinBox->setWhatsThis(tr("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.
A value of zero '0' means that for any stream failure, however short, BNC immediately generates an advisory note.
"));
_logFileLineEdit->setWhatsThis(tr("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.
The logfile name will automatically be extended by a string '_YYMMDD' carrying the current date."));
_adviseScriptLineEdit->setWhatsThis(tr("
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.
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.
"));
_perfIntrComboBox->setWhatsThis(tr("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).
Select a 'Log latency' interval or select the empty option field if you do not want BNC to log latencies and statistical information.
"));
_mountPointsTable->setWhatsThis(tr("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.
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' and 'RTCM_3.x'.
In case you need to log the raw data as is, BNC allows users to by-pass its decoders and and directly save the input in daily log files. To do this specify the decoder string as 'ZERO'.
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.
"));
_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."));
_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)."));
_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."));
_ephV3CheckBox->setWhatsThis(tr("The default format for output of RINEX Navigation data containing Broadcast Ephemeris is RINEX Version 2.11. Select 'Version 3' if you want to output the ephemeris in RINEX Version 3 format."));
_rnxV3CheckBox->setWhatsThis(tr("The default format for RINEX Observation files is RINEX Version 2.11. Select 'Version 3' if you want to save the observations in RINEX Version 3 format."));
_miscMountLineEdit->setWhatsThis(tr("Specify a mountpoint to apply any of the options shown below. Enter 'ALL' if you want to apply these options to all configured streams.
An empty option field (default) means that you don't want BNC to apply any of these options.
"));
_scanRTCMCheckBox->setWhatsThis(tr("Tick 'Scan RTCM' to log the numbers of incomming message types as well as contained antenna coordinates, antenna heigt, and antenna descriptor.
"));
_serialMountPointLineEdit->setWhatsThis(tr("Enter a 'Mountpoint' to forward the corresponding stream to a serial connected receiver.
"));
_serialPortNameLineEdit->setWhatsThis(tr("Enter the serial 'Port name' selected for communication with your serial connected receiver. Valid port names are
Windows: COM1, COM2
Linux: /dev/ttyS0, /dev/ttyS1
FreeBSD: /dev/ttyd0, /dev/ttyd1
Digital Unix: /dev/tty01, /dev/tty02
HP-UX: /dev/tty1p0, /dev/tty2p0
SGI/IRIX: /dev/ttyf1, /dev/ttyf2
SunOS/Solaris: /dev/ttya, /dev/ttyb
Note that you must plug a serial cable in the port defined here before you start BNC.
"));
_serialBaudRateComboBox->setWhatsThis(tr("Select a 'Baud rate' for the serial output link.
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.
"));
_serialParityComboBox->setWhatsThis(tr("Select the 'Parity' for the serial output link.
Note that your selection must equal the parity selection configured to the serial connected receiver. Note further that parity is often set to 'NONE'.
"));
_serialDataBitsComboBox->setWhatsThis(tr("Select the number of 'Data bits' for the serial output link.
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.
"));
_serialStopBitsComboBox->setWhatsThis(tr("Select the number of 'Stop bits' for the serial output link.
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.
"));
_serialFlowControlComboBox->setWhatsThis(tr("Select a 'Flow control' for the serial output link.
Note that your selection must equal the flow control configured to the serial connected receiver. Select 'OFF' if you don't know better.
"));
_serialAutoNMEAComboBox->setWhatsThis(tr("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.
Select 'Manual' only when handling a VRS stream and your serial connected receiver doesn't generate NMEA-GGA messages.
"));
_serialFileNMEALineEdit->setWhatsThis(tr("Specify the full path to a file where NMEA messages coming from your serial connected receiver are saved.
"));
_serialHeightNMEALineEdit->setWhatsThis(tr("Specify an approximate 'Height' above mean sea level in meter for your VRS to simulate an inital NMEA-GGA message.
The setting of this option is ignored in case of streams coming from physical reference stations.
"));
_pppMountLineEdit->setWhatsThis(tr("Specify an observations stream by its mountpoint from the 'Streams' list compiled below if you want BNC to estimate coordinates for the affected receiver position through a PPP solution. Example: 'FFMJ1'
Note that PPP in BNC requires to also pull a stream carrying RTCM Version 3 satellite orbit and clock corrections to Broadcast Ephemeris referring to the satellites' Antenna Phase Centers (APC). Stream CLK11 on NTRIP broadcaster products.igs-ip.net is an example.
Pulling in addition a third stream carrying Broadcast Ephemeris messages in high repetition rate is suggested if such messages are comeing from the receiver in low repetition rate or don't come at all from there.
"));
_pppCorrMountLineEdit->setWhatsThis(tr("You must specify an orbit/clock corrections stream by its mountpoint from the 'Streams' list compiled below. Example: 'CLK10'
"));
_pppSPPComboBox->setWhatsThis(tr("Choose between plain Single Point Positioning (SPP) and Precise Point Positioning (PPP).
"));
_pppUsePhaseCheckBox->setWhatsThis(tr("By default BNC applies a PPP solution using an ionosphere free P3 linear combination of code observations.
Tick 'Use phase obs' for an ionosphere free L3 linear combination of phase observations.
"));
_pppEstTropoCheckBox->setWhatsThis(tr("By default BNC does not introduce troposphere parameters when estimating coordinates.
Tick 'Estimate tropo' to introduce troposphere parameters when estimating coordinates.
"));
_pppGLONASSCheckBox->setWhatsThis(tr("By default BNC does not use GLONASS observations in PPP mode.
Tick 'Use GLONASS' for a combined processing of both, GPS and GLONASS observations in PPP mode.
"));
_pppGalileoCheckBox->setWhatsThis(tr("By default BNC does not use Galileo observations in PPP mode.
Tick 'Use Galileo' for a combined processing of both, GPS and Galileo observations in PPP mode.
"));
_pppPlotCoordinates->setWhatsThis(tr("BNC will plot PPP results in the 'PPP Plot' tab as North (red), East (green) and Up (blue) displacements when this option is selected. Values will be either referred to an XYZ reference coordinate (if specified) or referred to the first estimated coordinate. The sliding PPP time series window will cover the period of the latest 5 minutes.
Note that a PPP time series makes only sense for a stationary operated receiver."));
_pppNMEALineEdit->setWhatsThis(tr("
Specify the full path to a file where PPP results are saved as NMEA messages.
"));
_pppNMEAPortLineEdit->setWhatsThis(tr("Specify an IP port number to output PPP results as NMEA messages through an IP port.
"));
_pppSigCLineEdit->setWhatsThis(tr("Enter a sigma for your code observations in meters.
The higher the sigma you enter, the less the contribution of code observations to a PPP solution based on a combination of code and phase data. 5.0 (default) is likely to be an appropriate choice.
"));
_pppQuickStartLineEdit->setWhatsThis(tr("Enter the lenght of a startup period in seconds for which you want to fix the PPP solution to a known XYZ coordinate as introduced above and adjust a sigma 'XYZ Ini' according to the coordinate's precision. Fixing the coordinate is done in BNC through setting the 'Sigma XYZ Noise' you define below temporarily to zero.
This so-called Quick-Start option allows the PPP solution to rapidly converge. It requires that the antenna remains unmoved on the know position throughout the startup period.
A value of 120 is likely to be an appropriate choice for 'Quick-Start'. Default is an empty option field, meaning that you don't want BNC to operate in Quick-Start mode.
"));
_pppMaxSolGapLineEdit->setWhatsThis(tr("Specify a 'Maximum Solution Gap' in seconds. Should the time span between two consecutive solutions exceed this limit, the algorithm returns into the Quick-Start mode and fixes the introduced reference coordinate for the specified Quick-Start period. A value of '120' seconds could be an appropriate choice.
This option makes only sense for a stationary operated receiver where solution convergence can be enforced because a good approximation for the rover position is known. Default is an empty option field, meaning that you don't want BNC to return into the Quick-Start mode after failures caused i.e. by longer lasting outages.
"));
_pppSigPLineEdit->setWhatsThis(tr("Enter a sigma for your phase observations in meters.
The higher the sigma you enter, the less the contribution of phase observations to a PPP solutions based on a combination of code and phase data. 0.02 (default) is likely to be an appropriate choice.
"));
_pppAverageLineEdit->setWhatsThis(tr("Enter the length of a sliding time window in minutes. BNC will continuously output moving average positions computed from those individual positions obtained most recently throughout this period.
An empty option field (default) means that you don't want BNC to output moving average positions.
"));
_pppSigCrd0->setWhatsThis(tr("Enter a sigma in meters for the initial XYZ coordinate componentes. A value of 100.0 (default) may be an appropriate choice. However, this value may be significantly smaller (i.e. 0.01) when starting for example from a station with known XZY position in Quick-Start mode."));
_pppSigCrdP->setWhatsThis(tr("
Enter a sigma in meters for the white noise of estimated XYZ coordinate components. A value of 100.0 (default) may be appropriate considering the potential movement of a rover position.
"));
_pppSigTrp0->setWhatsThis(tr("Enter a sigma in meters for the a-priory model based tropospheric delay estimation. A value of 0.1 (default) may be an appropriate choice.
"));
_pppSigTrpP->setWhatsThis(tr("Enter a sigma in meters per second to describe the expected variation of the tropospheric effect.
Supposing 1Hz observation data, a value of 1e-6 (default) would mean that the tropospheric effect may vary for 3600 * 1e-6 = 0.0036 meters per hour.
"));
_pppRefCrdXLineEdit->setWhatsThis(tr("Enter reference coordinate X of the receiver's position.
"));
_pppRefCrdYLineEdit->setWhatsThis(tr("Enter reference coordinate Y of the receiver's position.
"));
_pppRefCrdZLineEdit->setWhatsThis(tr("Enter reference coordinate Z of the receiver's position.
"));
_pppRefdNLineEdit->setWhatsThis(tr("Enter north antenna excentricity.
"));
_pppRefdELineEdit->setWhatsThis(tr("Enter east antenna excentricity.
"));
_pppRefdULineEdit->setWhatsThis(tr("Enter up antenna excentricity.
"));
_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."));
_pppSync->setWhatsThis(tr(
" Zero value (or empty field, default) means that BNC processes each epoch of data "
"immediately after its arrival using satellite clock corrections available at "
"that time.
Non-zero value 'Sync Corr' (i.e. 5) means that the epochs of data "
"are buffered and the processing of each epoch is postponed till the satellite clock "
"corrections not older than 'Sync Corr' seconds are available.
"));
_pppAntexLineEdit->setWhatsThis(tr("
IGS provides a file containing absolute phase center corrections for GNSS satellite and receiver antennas in ANTEX format. Entering the full path to such an ANTEX file is required for correcting observations for antenna phase center offsets and variations. It allows you to specify the name of your receiver's antenna (as contained in the ANTEX file) to apply such corrections.
Default is an empty option field meaning that you don't want to correct observations for antenna phase center offsets and variations.
"));
_pppAntennaLineEdit->setWhatsThis(tr("Specify the receiver's antenna name as defined in your ANTEX file. Observations will be corrected for the antenna phase center's offset which may result in a reduction of a few centimeters at max. Corrections for phase center variations are not yet applied by BNC. The specified name must consist of 20 characters. Add trailing blanks if the antenna name has less then 20 characters.
Default is an empty option field meaning that you don't want to correct observations for antenna phase center offsets.
"));
_pppApplySatAntCheckBox->setWhatsThis(tr("This option is not yet working.
Satellite orbit and clock corrections refer to the satellite's antenna phase centers and hence observations are actually not to be corrected for satellite antenna phase center offsets. However, you may like to tick 'Apply Offsets' to force BNC to correct observations for satellite antenna phase center offsets.
Default is to not correct observations for satellite antenna phase center offsets."));
_cmbTable->setWhatsThis(tr("
BNC allows to process several orbit and clock corrections streams in real-time to produce, encode, upload and save a combination of correctors 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.
Note that an appropriate 'Wait for full 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.
Note further that the sequence of rows in the 'Combination Table' is of importance because the orbit information in the final combination stream is just copied from the stream listed in the first row. Hence the first line in the 'Combination Table' defines a kind of 'Master AC'. The update rate of the combination product follows the 'Master AC's update rate.
The combination process requires Broadcast Ephemeris. Besides the orbit and clock corrections stream(s) BNC must therefore pull a stream carrying Broadcast Ephemeris in the form of RTCM Version 3 messages.
"));
addCmbRowButton->setWhatsThis(tr("Hit 'Add Row' button to add another line to the mountpoints table."));
delCmbRowButton->setWhatsThis(tr("Hit 'Delete' button to delete the highlighted line from the mountpoints table."));
// Canvas with Editable Fields
// ---------------------------
_canvas = new QWidget;
setCentralWidget(_canvas);
_aogroup = new QTabWidget();
QWidget* pgroup = new QWidget();
QWidget* ggroup = new QWidget();
QWidget* sgroup = new QWidget();
QWidget* egroup = new QWidget();
QWidget* agroup = new QWidget();
QWidget* cgroup = new QWidget();
QWidget* ogroup = new QWidget();
QWidget* rgroup = new QWidget();
QWidget* sergroup = new QWidget();
QWidget* pppgroup = new QWidget();
QWidget* ppp2group = new QWidget();
QWidget* cmbgroup = new QWidget();
QWidget* uploadgroup = new QWidget();
QWidget* uploadEphgroup = new QWidget();
_aogroup->addTab(pgroup,tr("Network"));
_aogroup->addTab(ggroup,tr("General"));
_aogroup->addTab(ogroup,tr("RINEX Observations"));
_aogroup->addTab(egroup,tr("RINEX Ephemeris"));
_aogroup->addTab(cgroup,tr("Broadcast Corrections"));
_aogroup->addTab(sgroup,tr("Feed Engine"));
_aogroup->addTab(sergroup,tr("Serial Output"));
_aogroup->addTab(agroup,tr("Outages"));
_aogroup->addTab(rgroup,tr("Miscellaneous"));
_aogroup->addTab(pppgroup,tr("PPP (1)"));
_aogroup->addTab(ppp2group,tr("PPP (2)"));
#ifdef USE_COMBINATION
_aogroup->addTab(cmbgroup,tr("Combination"));
#endif
_aogroup->addTab(uploadgroup,tr("Upload (clk)"));
_aogroup->addTab(uploadEphgroup,tr("Upload (eph)"));
// Log Tab
// -------
_loggroup = new QTabWidget();
_loggroup->addTab(_log,tr("Log"));
_loggroup->addTab(_bncFigure,tr("Throughput"));
_loggroup->addTab(_bncFigureLate,tr("Latency"));
_loggroup->addTab(_bncFigurePPP,tr("PPP Plot"));
// Netowork (Proxy and SSL) Tab
// ----------------------------
QGridLayout* pLayout = new QGridLayout;
pLayout->setColumnMinimumWidth(0,13*ww);
_proxyPortLineEdit->setMaximumWidth(9*ww);
pLayout->addWidget(new QLabel("Proxy host"), 0, 0);
pLayout->addWidget(_proxyHostLineEdit, 0, 1, 1,10);
pLayout->addWidget(new QLabel("Proxy port"), 1, 0);
pLayout->addWidget(_proxyPortLineEdit, 1, 1);
pLayout->addWidget(new QLabel("Settings for proxy in protected networks, leave boxes blank if none."),2, 0, 1, 50, Qt::AlignLeft);
pLayout->addWidget(new QLabel(" "),3,0);
pLayout->addWidget(new QLabel(" "),4,0);
pLayout->addWidget(new QLabel("Path to SSL Certificates"), 5, 0);
pLayout->addWidget(_sslCaCertPathLineEdit, 5, 1, 1,10);
pLayout->addWidget(new QLabel("default: " + bncSslConfig::defaultPath()), 5, 12, 1,20);
pLayout->addWidget(new QLabel("Ignore SSL Authorization Errors"), 6,0);
pLayout->addWidget(_ignoreSslErrorsCheckBox, 6, 1, 1,10);
pLayout->addWidget(new QLabel("Settings for SSL Authorization."),7, 0, 1, 50, Qt::AlignLeft);
pgroup->setLayout(pLayout);
// General Tab
// -----------
QGridLayout* gLayout = new QGridLayout;
gLayout->setColumnMinimumWidth(0,14*ww);
_onTheFlyComboBox->setMaximumWidth(9*ww);
gLayout->addWidget(new QLabel("Logfile (full path)"), 0, 0);
gLayout->addWidget(_logFileLineEdit, 0, 1, 1,30); // 1
gLayout->addWidget(new QLabel("Append files"), 1, 0);
gLayout->addWidget(_rnxAppendCheckBox, 1, 1);
gLayout->addWidget(new QLabel("Reread configuration"), 2, 0);
gLayout->addWidget(_onTheFlyComboBox, 2, 1);
gLayout->addWidget(new QLabel("Auto start"), 3, 0);
gLayout->addWidget(_autoStartCheckBox, 3, 1);
gLayout->addWidget(new QLabel("Raw output file (full path)"), 4, 0);
gLayout->addWidget(_rawOutFileLineEdit, 4, 1, 1,30);
gLayout->addWidget(new QLabel("General settings for logfile, file handling, configuration on-the-fly, and auto-start."),5, 0, 1, 50, Qt::AlignLeft);
ggroup->setLayout(gLayout);
// RINEX Observations
// ------------------
QGridLayout* oLayout = new QGridLayout;
oLayout->setColumnMinimumWidth(0,14*ww);
_rnxIntrComboBox->setMaximumWidth(9*ww);
_rnxSamplSpinBox->setMaximumWidth(9*ww);
oLayout->addWidget(new QLabel("Directory"), 0, 0);
oLayout->addWidget(_rnxPathLineEdit, 0, 1,1,24);
oLayout->addWidget(new QLabel("Interval"), 1, 0);
oLayout->addWidget(_rnxIntrComboBox, 1, 1);
oLayout->addWidget(new QLabel(" Sampling"), 1, 2, Qt::AlignRight);
oLayout->addWidget(_rnxSamplSpinBox, 1, 3, Qt::AlignLeft);
oLayout->addWidget(new QLabel("Skeleton extension"), 2, 0);
oLayout->addWidget(_rnxSkelLineEdit, 2, 1,1,1, Qt::AlignLeft);
oLayout->addWidget(new QLabel("Script (full path)"), 3, 0);
oLayout->addWidget(_rnxScrpLineEdit, 3, 1,1,24);
oLayout->addWidget(new QLabel("Version 3"), 4, 0);
oLayout->addWidget(_rnxV3CheckBox, 4, 1);
oLayout->addWidget(new QLabel("Saving RINEX observation files."),5,0,1,50, Qt::AlignLeft);
ogroup->setLayout(oLayout);
// RINEX Ephemeris
// ---------------
QGridLayout* eLayout = new QGridLayout;
eLayout->setColumnMinimumWidth(0,14*ww);
_ephIntrComboBox->setMaximumWidth(9*ww);
_outEphPortLineEdit->setMaximumWidth(9*ww);
eLayout->addWidget(new QLabel("Directory"), 0, 0);
eLayout->addWidget(_ephPathLineEdit, 0, 1, 1,30);
eLayout->addWidget(new QLabel("Interval"), 1, 0);
eLayout->addWidget(_ephIntrComboBox, 1, 1);
eLayout->addWidget(new QLabel("Port"), 2, 0);
eLayout->addWidget(_outEphPortLineEdit, 2, 1);
eLayout->addWidget(new QLabel("Version 3"), 3, 0);
eLayout->addWidget(_ephV3CheckBox, 3, 1);
eLayout->addWidget(new QLabel("Saving RINEX ephemeris files and ephemeris output through IP port."),4,0,1,50,Qt::AlignLeft);
eLayout->addWidget(new QLabel(" "),5,0);
egroup->setLayout(eLayout);
// Broadcast Corrections
// ---------------------
QGridLayout* cLayout = new QGridLayout;
cLayout->setColumnMinimumWidth(0,14*ww);
_corrIntrComboBox->setMaximumWidth(9*ww);
_corrPortLineEdit->setMaximumWidth(9*ww);
_corrTimeSpinBox->setMaximumWidth(9*ww);
cLayout->addWidget(new QLabel("Directory, ASCII"), 0, 0);
cLayout->addWidget(_corrPathLineEdit, 0, 1,1,20);
cLayout->addWidget(new QLabel("Interval"), 1, 0);
cLayout->addWidget(_corrIntrComboBox, 1, 1);
cLayout->addWidget(new QLabel("Port"), 2, 0);
cLayout->addWidget(_corrPortLineEdit, 2, 1);
cLayout->addWidget(new QLabel(" Wait for full epoch"), 2, 2, Qt::AlignRight);
cLayout->addWidget(_corrTimeSpinBox, 2, 3, Qt::AlignLeft);
cLayout->addWidget(new QLabel("Saving Broadcast Ephemeris correction files and correction output through IP port."),3,0,1,50);
cLayout->addWidget(new QLabel(" "),4,0);
cLayout->addWidget(new QLabel(" "),5,0);
cgroup->setLayout(cLayout);
// Feed Engine
// -----------
QGridLayout* sLayout = new QGridLayout;
sLayout->setColumnMinimumWidth(0,14*ww);
_outPortLineEdit->setMaximumWidth(9*ww);
_waitTimeSpinBox->setMaximumWidth(9*ww);
_binSamplSpinBox->setMaximumWidth(9*ww);
_outUPortLineEdit->setMaximumWidth(9*ww);
sLayout->addWidget(new QLabel("Port"), 0, 0);
sLayout->addWidget(_outPortLineEdit, 0, 1);
sLayout->addWidget(new QLabel("Wait for full epoch"), 0, 2, Qt::AlignRight);
sLayout->addWidget(_waitTimeSpinBox, 0, 3, Qt::AlignLeft);
sLayout->addWidget(new QLabel("Sampling"), 1, 0);
sLayout->addWidget(_binSamplSpinBox, 1, 1, Qt::AlignLeft);
sLayout->addWidget(new QLabel("File (full path)"), 2, 0);
sLayout->addWidget(_outFileLineEdit, 2, 1, 1, 20);
sLayout->addWidget(new QLabel("Port (unsynchronized)"), 3, 0);
sLayout->addWidget(_outUPortLineEdit, 3, 1);
sLayout->addWidget(new QLabel("Output decoded observations in a binary format to feed a real-time GNSS network engine."),4,0,1,50);
sLayout->addWidget(new QLabel(" "),5,0);
sgroup->setLayout(sLayout);
// Serial Output
// -------------
QGridLayout* serLayout = new QGridLayout;
serLayout->setColumnMinimumWidth(0,14*ww);
_serialBaudRateComboBox->setMaximumWidth(9*ww);
_serialFlowControlComboBox->setMaximumWidth(11*ww);
_serialDataBitsComboBox->setMaximumWidth(5*ww);
_serialParityComboBox->setMaximumWidth(9*ww);
_serialStopBitsComboBox->setMaximumWidth(5*ww);
_serialAutoNMEAComboBox->setMaximumWidth(9*ww);
_serialHeightNMEALineEdit->setMaximumWidth(8*ww);
serLayout->addWidget(new QLabel("Mountpoint"), 0,0, Qt::AlignLeft);
serLayout->addWidget(_serialMountPointLineEdit, 0,1,1,2);
serLayout->addWidget(new QLabel("Port name"), 1,0, Qt::AlignLeft);
serLayout->addWidget(_serialPortNameLineEdit, 1,1,1,2);
serLayout->addWidget(new QLabel("Baud rate"), 2,0, Qt::AlignLeft);
serLayout->addWidget(_serialBaudRateComboBox, 2,1);
serLayout->addWidget(new QLabel("Flow control"), 2,2, Qt::AlignRight);
serLayout->addWidget(_serialFlowControlComboBox, 2,3);
serLayout->addWidget(new QLabel("Data bits"), 3,0, Qt::AlignLeft);
serLayout->addWidget(_serialDataBitsComboBox, 3,1);
serLayout->addWidget(new QLabel("Parity"), 3,2, Qt::AlignRight);
serLayout->addWidget(_serialParityComboBox, 3,3);
serLayout->addWidget(new QLabel(" Stop bits"), 3,4, Qt::AlignRight);
serLayout->addWidget(_serialStopBitsComboBox, 3,5);
serLayout->addWidget(new QLabel("NMEA"), 4,0);
serLayout->addWidget(_serialAutoNMEAComboBox, 4,1);
serLayout->addWidget(new QLabel(" File (full path)"), 4,2, Qt::AlignRight);
serLayout->addWidget(_serialFileNMEALineEdit, 4,3,1,15);
serLayout->addWidget(new QLabel("Height"), 4,20, Qt::AlignRight);
serLayout->addWidget(_serialHeightNMEALineEdit, 4,21,1,11);
serLayout->addWidget(new QLabel("Port settings to feed a serial connected receiver."),5,0,1,30);
sergroup->setLayout(serLayout);
// Outages
// -------
QGridLayout* aLayout = new QGridLayout;
aLayout->setColumnMinimumWidth(0,14*ww);
_obsRateComboBox->setMaximumWidth(9*ww);
_adviseFailSpinBox->setMaximumWidth(9*ww);
_adviseRecoSpinBox->setMaximumWidth(9*ww);
aLayout->addWidget(new QLabel("Observation rate"), 0, 0);
aLayout->addWidget(_obsRateComboBox, 0, 1);
aLayout->addWidget(new QLabel("Failure threshold"), 1, 0);
aLayout->addWidget(_adviseFailSpinBox, 1, 1);
aLayout->addWidget(new QLabel("Recovery threshold"), 2, 0);
aLayout->addWidget(_adviseRecoSpinBox, 2, 1);
aLayout->addWidget(new QLabel("Script (full path)"), 3, 0);
aLayout->addWidget(_adviseScriptLineEdit, 3, 1,1,30);
aLayout->addWidget(new QLabel("Failure and recovery reports, advisory notes."),4,0,1,50,Qt::AlignLeft);
aLayout->addWidget(new QLabel(" "), 5, 0);
agroup->setLayout(aLayout);
// Miscellaneous
// -------------
QGridLayout* rLayout = new QGridLayout;
rLayout->setColumnMinimumWidth(0,14*ww);
_perfIntrComboBox->setMaximumWidth(9*ww);
rLayout->addWidget(new QLabel("Mountpoint"), 0, 0);
rLayout->addWidget(_miscMountLineEdit, 0, 1, 1,7);
rLayout->addWidget(new QLabel("Log latency"), 1, 0);
rLayout->addWidget(_perfIntrComboBox, 1, 1);
rLayout->addWidget(new QLabel("Scan RTCM"), 2, 0);
rLayout->addWidget(_scanRTCMCheckBox, 2, 1);
rLayout->addWidget(new QLabel("Log latencies or scan RTCM streams for numbers of message types and antenna information."),3, 0,1,30);
rLayout->addWidget(new QLabel(" "), 4, 0);
rLayout->addWidget(new QLabel(" "), 5, 0);
rgroup->setLayout(rLayout);
// PPP Client
// ----------
QGridLayout* pppLayout = new QGridLayout;
_pppSigCLineEdit->setMaximumWidth(6*ww);
_pppSigPLineEdit->setMaximumWidth(6*ww);
_pppSigCrd0->setMaximumWidth(6*ww);
_pppSigCrdP->setMaximumWidth(6*ww);
_pppSigTrp0->setMaximumWidth(6*ww);
_pppSigTrpP->setMaximumWidth(6*ww);
_pppAverageLineEdit->setMaximumWidth(6*ww);
_pppQuickStartLineEdit->setMaximumWidth(6*ww);
_pppMaxSolGapLineEdit->setMaximumWidth(6*ww);
_pppRefCrdXLineEdit->setMaximumWidth(10*ww);
_pppRefCrdYLineEdit->setMaximumWidth(10*ww);
_pppRefCrdZLineEdit->setMaximumWidth(10*ww);
_pppRefdNLineEdit->setMaximumWidth(5*ww);
_pppRefdELineEdit->setMaximumWidth(5*ww);
_pppRefdULineEdit->setMaximumWidth(5*ww);
_pppSync->setMaximumWidth(6*ww);
_pppSPPComboBox->setMaximumWidth(15*ww);
_pppNMEAPortLineEdit->setMaximumWidth(6*ww);
_postObsFileChooser = new qtFileChooser;
_postObsFileChooser->setFileName(settings.value("postObsFile").toString());
_postObsFileChooser->setWhatsThis(tr("Full Path to RINEX Observation File"));
_postNavFileChooser = new qtFileChooser;
_postNavFileChooser->setFileName(settings.value("postNavFile").toString());
_postNavFileChooser->setWhatsThis(tr("Full Path to RINEX Navigation File"));
_postCorrFileChooser = new qtFileChooser;
_postNavFileChooser->setFileName(settings.value("postCorrFile").toString());
_postNavFileChooser->setWhatsThis(tr("Full Path to DGPS Correction File"));
int ir = 0;
pppLayout->addWidget(new QLabel("Precise Point Positioning (Panel 1)"), ir, 0, 1, 8);
++ir;
pppLayout->addWidget(new QLabel("Obs Mountpoint"), ir, 0);
pppLayout->addWidget(_pppMountLineEdit, ir, 1);
pppLayout->addWidget(_pppSPPComboBox, ir, 2);
pppLayout->addWidget(new QLabel(" X "), ir, 3, Qt::AlignRight);
pppLayout->addWidget(_pppRefCrdXLineEdit, ir, 4);
pppLayout->addWidget(new QLabel(" Y "), ir, 5, Qt::AlignRight);
pppLayout->addWidget(_pppRefCrdYLineEdit, ir, 6);
pppLayout->addWidget(new QLabel(" Z "), ir, 7, Qt::AlignRight);
pppLayout->addWidget(_pppRefCrdZLineEdit, ir, 8);
++ir;
pppLayout->addWidget(new QLabel("Corr Mountpoint "), ir, 0);
pppLayout->addWidget(_pppCorrMountLineEdit, ir, 1);
pppLayout->addWidget(new QLabel(" dN "), ir, 3, Qt::AlignRight);
pppLayout->addWidget(_pppRefdNLineEdit, ir, 4);
pppLayout->addWidget(new QLabel(" dE "), ir, 5, Qt::AlignRight);
pppLayout->addWidget(_pppRefdELineEdit, ir, 6);
pppLayout->addWidget(new QLabel(" dU "), ir, 7, Qt::AlignRight);
pppLayout->addWidget(_pppRefdULineEdit, ir, 8);
++ir;
pppLayout->addWidget(new QLabel("Options"), ir, 0, 1, 5);
pppLayout->addWidget(_pppUsePhaseCheckBox, ir, 1, Qt::AlignRight);
pppLayout->addWidget(new QLabel("Use phase obs"), ir, 2);
pppLayout->addWidget(_pppEstTropoCheckBox, ir, 3, Qt::AlignRight);
pppLayout->addWidget(new QLabel("Estimate tropo"), ir, 4);
pppLayout->addWidget(_pppGLONASSCheckBox, ir, 5, Qt::AlignRight);
pppLayout->addWidget(new QLabel("Use GLONASS"), ir, 6);
pppLayout->addWidget(_pppGalileoCheckBox, ir, 7, Qt::AlignRight);
pppLayout->addWidget(new QLabel("Use Galileo "), ir, 8);
++ir;
pppLayout->addWidget(new QLabel("Output"), ir, 0);
pppLayout->addWidget(_pppNMEALineEdit, ir, 1, 1, 3);
pppLayout->addWidget(new QLabel("NMEA File"), ir, 4);
pppLayout->addWidget(_pppNMEAPortLineEdit, ir, 5, Qt::AlignRight);
pppLayout->addWidget(new QLabel("NMEA Port"), ir, 6);
pppLayout->addWidget(_pppPlotCoordinates, ir, 7, Qt::AlignRight);
pppLayout->addWidget(new QLabel("PPP Plot"), ir, 8);
++ir;
pppLayout->addWidget(new QLabel("Post-Processing"));
QHBoxLayout* hlpLayout = new QHBoxLayout;
{
hlpLayout->addWidget(_postObsFileChooser);
hlpLayout->addWidget(new QLabel("Obs"));
hlpLayout->addWidget(_postNavFileChooser);
hlpLayout->addWidget(new QLabel("Nav"));
hlpLayout->addWidget(_postCorrFileChooser);
hlpLayout->addWidget(new QLabel("Corr"));
}
pppLayout->addLayout(hlpLayout, ir, 1, 1, 7);
pppgroup->setLayout(pppLayout);
// PPP Client (second panel)
// -------------------------
QGridLayout* ppp2Layout = new QGridLayout;
ir = 0;
ppp2Layout->addWidget(new QLabel("Precise Point Positioning (Panel 2)"), ir, 0, 1, 8);
++ir;
ppp2Layout->addWidget(new QLabel("Antennas"), ir, 0);
ppp2Layout->addWidget(_pppAntexLineEdit, ir, 1, 1, 3);
ppp2Layout->addWidget(new QLabel("ANTEX File "), ir, 4);
ppp2Layout->addWidget(_pppAntennaLineEdit, ir, 5, 1, 3);
ppp2Layout->addWidget(new QLabel("Rec. Ant. Name"), ir, 8);
++ir;
ppp2Layout->addWidget(new QLabel("Satellite Antenna "), ir, 0);
ppp2Layout->addWidget(_pppApplySatAntCheckBox, ir, 1, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Apply Offsets"), ir, 2, Qt::AlignLeft);
++ir;
ppp2Layout->addWidget(new QLabel("Sigmas"), ir, 0);
ppp2Layout->addWidget(_pppSigCLineEdit, ir, 1, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Code"), ir, 2);
ppp2Layout->addWidget(_pppSigPLineEdit, ir, 3);
ppp2Layout->addWidget(new QLabel("Phase"), ir, 4);
ppp2Layout->addWidget(_pppSigTrp0, ir, 5, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Tropo Init "), ir, 6);
ppp2Layout->addWidget(_pppSigTrpP, ir, 7);
ppp2Layout->addWidget(new QLabel("Tropo White Noise"), ir, 8);
++ir;
ppp2Layout->addWidget(new QLabel("Options cont'd"), ir, 0);
ppp2Layout->addWidget(_pppSigCrd0, ir, 1, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Sigma XYZ Init "), ir, 2);
ppp2Layout->addWidget(_pppSigCrdP, ir, 3, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Sigma XYZ Noise "), ir, 4);
ppp2Layout->addWidget(_pppQuickStartLineEdit, ir, 5, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Quick-Start (sec) "), ir, 6);
ppp2Layout->addWidget(_pppMaxSolGapLineEdit, ir, 7, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Max Sol. Gap (sec)"), ir, 8);
++ir;
ppp2Layout->addWidget(new QLabel("Options cont'd"), ir, 0);
ppp2Layout->addWidget(_pppSync, ir, 1);
ppp2Layout->addWidget(new QLabel("Sync Corr (sec) "), ir, 2);
ppp2Layout->addWidget(_pppAverageLineEdit, ir, 3, Qt::AlignRight);
ppp2Layout->addWidget(new QLabel("Averaging (min)") , ir, 4);
++ir;
ppp2Layout->addWidget(new QLabel(" "), ir, 0);
ppp2Layout->setRowStretch(ir, 99);
ppp2group->setLayout(ppp2Layout);
// Combination
// -----------
QGridLayout* cmbLayout = new QGridLayout;
populateCmbTable();
cmbLayout->addWidget(_cmbTable,0,0,6,3);
cmbLayout->addWidget(addCmbRowButton,1,3);
connect(addCmbRowButton, SIGNAL(clicked()), this, SLOT(slotAddCmbRow()));
cmbLayout->addWidget(delCmbRowButton,2,3);
cmbLayout->addWidget(new QLabel("Method"), 3, 3);
cmbLayout->addWidget(_cmbMethodComboBox, 3, 4, Qt::AlignRight);
cmbLayout->addWidget(new QLabel("Maximal Residuum"), 4, 3);
cmbLayout->addWidget(_cmbMaxresLineEdit, 4, 4, Qt::AlignRight);
connect(delCmbRowButton, SIGNAL(clicked()), this, SLOT(slotDelCmbRow()));
cmbLayout->addWidget(new QLabel(" Combine Broadcast Ephemeris corrections streams."),5,3,1,3);
cmbgroup->setLayout(cmbLayout);
// Upload Layout (Clocks)
// ----------------------
QGridLayout* uploadHlpLayout = new QGridLayout();
uploadHlpLayout->addWidget(new QLabel("Upload RTNet or Combination Results"),0,0);
uploadHlpLayout->addWidget(addUploadRowButton,0,1);
connect(addUploadRowButton, SIGNAL(clicked()), this, SLOT(slotAddUploadRow()));
uploadHlpLayout->addWidget(delUploadRowButton,0,2);
connect(delUploadRowButton, SIGNAL(clicked()), this, SLOT(slotDelUploadRow()));
uploadHlpLayout->addWidget(setUploadTrafoButton,1,1);
connect(setUploadTrafoButton, SIGNAL(clicked()), this, SLOT(slotSetUploadTrafo()));
uploadHlpLayout->addWidget(new QLabel("Interval"),0,3, Qt::AlignRight);
uploadHlpLayout->addWidget(_uploadIntrComboBox,0,4);
uploadHlpLayout->addWidget(new QLabel("Sampling"),1,3, Qt::AlignRight);
uploadHlpLayout->addWidget(_uploadSamplSpinBox,1,4);
uploadHlpLayout->addWidget(new QLabel("Sampling (Orb)"),1,5, Qt::AlignRight);
uploadHlpLayout->addWidget(_uploadSamplOrbSpinBox,1,6);
QBoxLayout* uploadLayout = new QBoxLayout(QBoxLayout::TopToBottom);
populateUploadTable();
uploadLayout->addWidget(_uploadTable);
uploadLayout->addLayout(uploadHlpLayout);
uploadgroup->setLayout(uploadLayout);
// Upload Layout (Ephemeris)
// -------------------------
QGridLayout* uploadLayoutEph = new QGridLayout;
uploadLayoutEph->setColumnMinimumWidth(0, 9*ww);
_uploadEphPortLineEdit->setMaximumWidth(9*ww);
_uploadEphPasswordLineEdit->setMaximumWidth(9*ww);
_uploadEphMountpointLineEdit->setMaximumWidth(12*ww);
uploadLayoutEph->addWidget(new QLabel("Host"), 0, 0);
uploadLayoutEph->addWidget(_uploadEphHostLineEdit, 0, 1, 1, 3);
uploadLayoutEph->addWidget(new QLabel(" Port"), 0, 4, Qt::AlignRight);
uploadLayoutEph->addWidget(_uploadEphPortLineEdit, 0, 5, 1, 1);
uploadLayoutEph->addWidget(new QLabel("Mountpoint "), 1, 0);
uploadLayoutEph->addWidget(_uploadEphMountpointLineEdit, 1, 1);
uploadLayoutEph->addWidget(new QLabel(" Password"), 1, 2, Qt::AlignRight);
uploadLayoutEph->addWidget(_uploadEphPasswordLineEdit, 1, 3);
uploadLayoutEph->addWidget(new QLabel("Sampling"), 2, 0);
uploadLayoutEph->addWidget(_uploadEphSampleSpinBox, 2, 1);
uploadLayoutEph->addWidget(new QLabel("Upload concatenated RTCMv3 Broadcast Ephemeris to caster."), 3, 0, 1, 5);
uploadLayoutEph->addWidget(_uploadEphBytesCounter, 3, 5);
uploadEphgroup->setLayout(uploadLayoutEph);
connect(_uploadEphHostLineEdit, SIGNAL(textChanged(const QString &)),
this, SLOT(slotBncTextChanged()));
// Main Layout
// -----------
QGridLayout* mLayout = new QGridLayout;
_aogroup->setCurrentIndex(settings.value("startTab").toInt());
mLayout->addWidget(_aogroup, 0,0);
mLayout->addWidget(_mountPointsTable, 1,0);
_loggroup->setCurrentIndex(settings.value("statusTab").toInt());
mLayout->addWidget(_loggroup, 2,0);
_canvas->setLayout(mLayout);
// Enable/Disable all Widgets
// --------------------------
slotBncTextChanged();
// Auto start
// ----------
if ( Qt::CheckState(settings.value("autoStart").toInt()) == Qt::Checked) {
slotGetData();
}
}
// Destructor
////////////////////////////////////////////////////////////////////////////
bncWindow::~bncWindow() {
delete _caster;
delete _casterEph;
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::populateMountPointsTable() {
for (int iRow = _mountPointsTable->rowCount()-1; iRow >=0; iRow--) {
_mountPointsTable->removeRow(iRow);
}
bncSettings settings;
QListIterator it(settings.value("mountPoints").toStringList());
if (!it.hasNext()) {
_actGetData->setEnabled(false);
}
int iRow = 0;
while (it.hasNext()) {
QStringList hlp = it.next().split(" ");
if (hlp.size() < 5) continue;
_mountPointsTable->insertRow(iRow);
QUrl url(hlp[0]);
QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
QString nmea(hlp[4]);
if (hlp[5] == "S") {
fullPath = hlp[0].replace(0,2,"");
}
QString ntripVersion = "2";
if (hlp.size() >= 6) {
ntripVersion = (hlp[5]);
}
QTableWidgetItem* it;
it = new QTableWidgetItem(url.userInfo());
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 0, it);
it = new QTableWidgetItem(fullPath);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 1, it);
it = new QTableWidgetItem(format);
_mountPointsTable->setItem(iRow, 2, it);
if (nmea == "yes") {
it = new QTableWidgetItem(latitude);
_mountPointsTable->setItem(iRow, 3, it);
it = new QTableWidgetItem(longitude);
_mountPointsTable->setItem(iRow, 4, it);
} else {
it = new QTableWidgetItem(latitude);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 3, it);
it = new QTableWidgetItem(longitude);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 4, it);
}
it = new QTableWidgetItem(nmea);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 5, it);
it = new QTableWidgetItem(ntripVersion);
//// it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 6, it);
bncTableItem* bncIt = new bncTableItem();
bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 7, bncIt);
iRow++;
}
_mountPointsTable->sortItems(1);
}
// Retrieve Table
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotAddMountPoints() {
bncSettings settings;
QString proxyHost = settings.value("proxyHost").toString();
int proxyPort = settings.value("proxyPort").toInt();
if (proxyHost != _proxyHostLineEdit->text() ||
proxyPort != _proxyPortLineEdit->text().toInt()) {
int iRet = QMessageBox::question(this, "Question", "Proxy options "
"changed. Use the new ones?",
QMessageBox::Yes, QMessageBox::No,
QMessageBox::NoButton);
if (iRet == QMessageBox::Yes) {
settings.setValue("proxyHost", _proxyHostLineEdit->text());
settings.setValue("proxyPort", _proxyPortLineEdit->text());
settings.sync();
}
}
settings.setValue("sslCaCertPath", _sslCaCertPathLineEdit->text());
settings.setValue("ignoreSslErrors", _ignoreSslErrorsCheckBox->checkState());
settings.sync();
QMessageBox msgBox;
msgBox.setIcon(QMessageBox::Question);
msgBox.setWindowTitle("Add Stream");
msgBox.setText("Add stream(s) coming from:");
QPushButton* buttonNtrip = msgBox.addButton(tr("Caster"), QMessageBox::ActionRole);
QPushButton* buttonIP = msgBox.addButton(tr("TCP/IP port"), QMessageBox::ActionRole);
QPushButton* buttonUDP = msgBox.addButton(tr("UDP port"), QMessageBox::ActionRole);
QPushButton* buttonSerial = msgBox.addButton(tr("Serial port"), QMessageBox::ActionRole);
QPushButton* buttonCancel = msgBox.addButton(tr("Cancel"), QMessageBox::ActionRole);
msgBox.exec();
if (msgBox.clickedButton() == buttonNtrip) {
bncTableDlg* dlg = new bncTableDlg(this);
dlg->move(this->pos().x()+50, this->pos().y()+50);
connect(dlg, SIGNAL(newMountPoints(QStringList*)),
this, SLOT(slotNewMountPoints(QStringList*)));
dlg->exec();
delete dlg;
} else if (msgBox.clickedButton() == buttonIP) {
bncIpPort* ipp = new bncIpPort(this);
connect(ipp, SIGNAL(newMountPoints(QStringList*)),
this, SLOT(slotNewMountPoints(QStringList*)));
ipp->exec();
delete ipp;
} else if (msgBox.clickedButton() == buttonUDP) {
bncUdpPort* udp = new bncUdpPort(this);
connect(udp, SIGNAL(newMountPoints(QStringList*)),
this, SLOT(slotNewMountPoints(QStringList*)));
udp->exec();
delete udp;
} else if (msgBox.clickedButton() == buttonSerial) {
bncSerialPort* sep = new bncSerialPort(this);
connect(sep, SIGNAL(newMountPoints(QStringList*)),
this, SLOT(slotNewMountPoints(QStringList*)));
sep->exec();
delete sep;
} else if (msgBox.clickedButton() == buttonCancel) {
// Cancel
}
}
// Delete Selected Mount Points
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotDeleteMountPoints() {
int nRows = _mountPointsTable->rowCount();
bool flg[nRows];
for (int iRow = 0; iRow < nRows; iRow++) {
if (_mountPointsTable->isItemSelected(_mountPointsTable->item(iRow,1))) {
flg[iRow] = true;
}
else {
flg[iRow] = false;
}
}
for (int iRow = nRows-1; iRow >= 0; iRow--) {
if (flg[iRow]) {
_mountPointsTable->removeRow(iRow);
}
}
_actDeleteMountPoints->setEnabled(false);
if (_mountPointsTable->rowCount() == 0) {
_actGetData->setEnabled(false);
}
}
// New Mount Points Selected
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotNewMountPoints(QStringList* mountPoints) {
int iRow = 0;
QListIterator it(*mountPoints);
while (it.hasNext()) {
QStringList hlp = it.next().split(" ");
QUrl url(hlp[0]);
QString fullPath = url.host() + QString(":%1").arg(url.port()) + url.path();
QString format(hlp[1]); QString latitude(hlp[2]); QString longitude(hlp[3]);
QString nmea(hlp[4]);
if (hlp[5] == "S") {
fullPath = hlp[0].replace(0,2,"");
}
QString ntripVersion = "2";
if (hlp.size() >= 6) {
ntripVersion = (hlp[5]);
}
_mountPointsTable->insertRow(iRow);
QTableWidgetItem* it;
it = new QTableWidgetItem(url.userInfo());
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 0, it);
it = new QTableWidgetItem(fullPath);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 1, it);
it = new QTableWidgetItem(format);
_mountPointsTable->setItem(iRow, 2, it);
if (nmea == "yes") {
it = new QTableWidgetItem(latitude);
_mountPointsTable->setItem(iRow, 3, it);
it = new QTableWidgetItem(longitude);
_mountPointsTable->setItem(iRow, 4, it);
} else {
it = new QTableWidgetItem(latitude);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 3, it);
it = new QTableWidgetItem(longitude);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 4, it);
}
it = new QTableWidgetItem(nmea);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 5, it);
it = new QTableWidgetItem(ntripVersion);
it->setFlags(it->flags() & ~Qt::ItemIsEditable);
_mountPointsTable->setItem(iRow, 6, it);
bncTableItem* bncIt = new bncTableItem();
_mountPointsTable->setItem(iRow, 7, bncIt);
iRow++;
}
_mountPointsTable->hideColumn(0);
_mountPointsTable->sortItems(1);
if (mountPoints->count() > 0 && !_actStop->isEnabled()) {
_actGetData->setEnabled(true);
}
delete mountPoints;
}
// Save Options
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotSaveOptions() {
QStringList mountPoints;
for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
if (_mountPointsTable->item(iRow, 6)->text() != "S") {
QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
"@" + _mountPointsTable->item(iRow, 1)->text() );
mountPoints.append(url.toString() + " " +
_mountPointsTable->item(iRow, 2)->text()
+ " " + _mountPointsTable->item(iRow, 3)->text()
+ " " + _mountPointsTable->item(iRow, 4)->text()
+ " " + _mountPointsTable->item(iRow, 5)->text()
+ " " + _mountPointsTable->item(iRow, 6)->text());
} else {
mountPoints.append(
"//" + _mountPointsTable->item(iRow, 1)->text()
+ " " + _mountPointsTable->item(iRow, 2)->text()
+ " " + _mountPointsTable->item(iRow, 3)->text()
+ " " + _mountPointsTable->item(iRow, 4)->text()
+ " " + _mountPointsTable->item(iRow, 5)->text()
+ " " + _mountPointsTable->item(iRow, 6)->text());
}
}
QStringList combineStreams;
for (int iRow = 0; iRow < _cmbTable->rowCount(); iRow++) {
QString hlp;
for (int iCol = 0; iCol < _cmbTable->columnCount(); iCol++) {
if (_cmbTable->item(iRow, iCol)) {
hlp += _cmbTable->item(iRow, iCol)->text() + " ";
}
}
if (!hlp.isEmpty()) {
combineStreams << hlp;
}
}
QStringList uploadMountpointsOut;
for (int iRow = 0; iRow < _uploadTable->rowCount(); iRow++) {
QString hlp;
for (int iCol = 0; iCol < _uploadTable->columnCount(); iCol++) {
if (_uploadTable->cellWidget(iRow, iCol) &&
(iCol == 3 || iCol == 4 || iCol == 5)) {
if (iCol == 3) {
QLineEdit* passwd = (QLineEdit*)(_uploadTable->cellWidget(iRow, iCol));
hlp += passwd->text() + ",";
}
else if (iCol == 4) {
QComboBox* system = (QComboBox*)(_uploadTable->cellWidget(iRow, iCol));
hlp += system->currentText() + ",";
}
else if (iCol == 5) {
QCheckBox* com = (QCheckBox*)(_uploadTable->cellWidget(iRow, iCol));
QString state; state.setNum(com->checkState());
hlp += state + ",";
}
}
else if (_uploadTable->item(iRow, iCol)) {
hlp += _uploadTable->item(iRow, iCol)->text() + ",";
}
}
if (!hlp.isEmpty()) {
uploadMountpointsOut << hlp;
}
}
bncSettings settings;
settings.setValue("adviseFail", _adviseFailSpinBox->value());
settings.setValue("adviseReco", _adviseRecoSpinBox->value());
settings.setValue("adviseScript",_adviseScriptLineEdit->text());
settings.setValue("autoStart", _autoStartCheckBox->checkState());
settings.setValue("binSampl", _binSamplSpinBox->value());
settings.setValue("corrIntr", _corrIntrComboBox->currentText());
settings.setValue("corrPath", _corrPathLineEdit->text());
settings.setValue("corrPort", _corrPortLineEdit->text());
settings.setValue("corrTime", _corrTimeSpinBox->value());
settings.setValue("ephIntr", _ephIntrComboBox->currentText());
settings.setValue("ephPath", _ephPathLineEdit->text());
settings.setValue("ephV3", _ephV3CheckBox->checkState());
settings.setValue("logFile", _logFileLineEdit->text());
settings.setValue("rawOutFile", _rawOutFileLineEdit->text());
settings.setValue("miscMount", _miscMountLineEdit->text());
settings.setValue("pppMount", _pppMountLineEdit->text());
settings.setValue("pppCorrMount",_pppCorrMountLineEdit->text());
settings.setValue("pppSPP", _pppSPPComboBox->currentText());
settings.setValue("nmeaFile", _pppNMEALineEdit->text());
settings.setValue("nmeaPort", _pppNMEAPortLineEdit->text());
settings.setValue("pppSigmaCode",_pppSigCLineEdit->text());
settings.setValue("pppSigmaPhase",_pppSigPLineEdit->text());
settings.setValue("pppSigCrd0",_pppSigCrd0->text());
settings.setValue("pppSigCrdP",_pppSigCrdP->text());
settings.setValue("pppSigTrp0",_pppSigTrp0->text());
settings.setValue("pppSigTrpP",_pppSigTrpP->text());
settings.setValue("pppAverage", _pppAverageLineEdit->text());
settings.setValue("pppQuickStart", _pppQuickStartLineEdit->text());
settings.setValue("pppMaxSolGap", _pppMaxSolGapLineEdit->text());
settings.setValue("pppRefCrdX", _pppRefCrdXLineEdit->text());
settings.setValue("pppRefCrdY", _pppRefCrdYLineEdit->text());
settings.setValue("pppRefCrdZ", _pppRefCrdZLineEdit->text());
settings.setValue("pppRefdN", _pppRefdNLineEdit->text());
settings.setValue("pppRefdE", _pppRefdELineEdit->text());
settings.setValue("pppRefdU", _pppRefdULineEdit->text());
settings.setValue("pppSync", _pppSync->text());
settings.setValue("pppUsePhase", _pppUsePhaseCheckBox->checkState());
settings.setValue("pppPlotCoordinates", _pppPlotCoordinates->checkState());
settings.setValue("pppEstTropo", _pppEstTropoCheckBox->checkState());
settings.setValue("pppGLONASS", _pppGLONASSCheckBox->checkState());
settings.setValue("pppGalileo", _pppGalileoCheckBox->checkState());
settings.setValue("pppAntenna", _pppAntennaLineEdit->text());
settings.setValue("pppAntex", _pppAntexLineEdit->text());
settings.setValue("pppApplySatAnt", _pppApplySatAntCheckBox->checkState());
settings.setValue("mountPoints", mountPoints);
settings.setValue("obsRate", _obsRateComboBox->currentText());
settings.setValue("onTheFlyInterval", _onTheFlyComboBox->currentText());
settings.setValue("outEphPort", _outEphPortLineEdit->text());
settings.setValue("outFile", _outFileLineEdit->text());
settings.setValue("outPort", _outPortLineEdit->text());
settings.setValue("outUPort", _outUPortLineEdit->text());
settings.setValue("perfIntr", _perfIntrComboBox->currentText());
settings.setValue("proxyHost", _proxyHostLineEdit->text());
settings.setValue("proxyPort", _proxyPortLineEdit->text());
settings.setValue("sslCaCertPath", _sslCaCertPathLineEdit->text());
settings.setValue("ignoreSslErrors", _ignoreSslErrorsCheckBox->checkState());
settings.setValue("rnxAppend", _rnxAppendCheckBox->checkState());
settings.setValue("rnxIntr", _rnxIntrComboBox->currentText());
settings.setValue("rnxPath", _rnxPathLineEdit->text());
settings.setValue("rnxSampl", _rnxSamplSpinBox->value());
settings.setValue("rnxScript", _rnxScrpLineEdit->text());
settings.setValue("rnxSkel", _rnxSkelLineEdit->text());
settings.setValue("rnxV3", _rnxV3CheckBox->checkState());
settings.setValue("scanRTCM", _scanRTCMCheckBox->checkState());
settings.setValue("serialFileNMEA",_serialFileNMEALineEdit->text());
settings.setValue("serialHeightNMEA",_serialHeightNMEALineEdit->text());
settings.setValue("serialAutoNMEA", _serialAutoNMEAComboBox->currentText());
settings.setValue("serialBaudRate", _serialBaudRateComboBox->currentText());
settings.setValue("serialDataBits", _serialDataBitsComboBox->currentText());
settings.setValue("serialMountPoint",_serialMountPointLineEdit->text());
settings.setValue("serialParity", _serialParityComboBox->currentText());
settings.setValue("serialPortName", _serialPortNameLineEdit->text());
settings.setValue("serialStopBits", _serialStopBitsComboBox->currentText());
settings.setValue("serialFlowControl",_serialFlowControlComboBox->currentText());
settings.setValue("startTab", _aogroup->currentIndex());
settings.setValue("statusTab", _loggroup->currentIndex());
settings.setValue("waitTime", _waitTimeSpinBox->value());
if (!combineStreams.isEmpty()) {
settings.setValue("combineStreams", combineStreams);
}
else {
settings.setValue("combineStreams", "");
}
settings.setValue("cmbMaxres", _cmbMaxresLineEdit->text());
settings.setValue("cmbMethod", _cmbMethodComboBox->currentText());
if (!uploadMountpointsOut.isEmpty()) {
settings.setValue("uploadMountpointsOut", uploadMountpointsOut);
}
else {
settings.setValue("uploadMountpointsOut", "");
}
settings.setValue("uploadIntr", _uploadIntrComboBox->currentText());
settings.setValue("uploadSampl", _uploadSamplSpinBox->value());
settings.setValue("uploadSamplOrb", _uploadSamplOrbSpinBox->value());
settings.setValue("uploadEphHost", _uploadEphHostLineEdit->text());
settings.setValue("uploadEphPort", _uploadEphPortLineEdit->text());
settings.setValue("uploadEphPassword", _uploadEphPasswordLineEdit->text());
settings.setValue("uploadEphMountpoint",_uploadEphMountpointLineEdit->text());
settings.setValue("uploadEphSample", _uploadEphSampleSpinBox->value());
settings.setValue("postObsFile", _postObsFileChooser->fileName());
settings.setValue("postNavFile", _postNavFileChooser->fileName());
settings.setValue("postCorrFile", _postCorrFileChooser->fileName());
if (_caster) {
_caster->slotReadMountPoints();
}
settings.sync();
}
// All get slots terminated
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotGetThreadsFinished() {
((bncApp*)qApp)->slotMessage("All Get Threads Terminated", true);
delete _caster; _caster = 0;
delete _casterEph; _casterEph = 0;
_actGetData->setEnabled(true);
_actStop->setEnabled(false);
}
// Retrieve Data
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotGetData() {
slotSaveOptions();
_bncFigurePPP->reset();
_actDeleteMountPoints->setEnabled(false);
_actGetData->setEnabled(false);
_actStop->setEnabled(true);
_caster = new bncCaster(_outFileLineEdit->text(),
_outPortLineEdit->text().toInt());
((bncApp*)qApp)->setPort(_outEphPortLineEdit->text().toInt());
((bncApp*)qApp)->setPortCorr(_corrPortLineEdit->text().toInt());
((bncApp*)qApp)->initCombination();
connect(_caster, SIGNAL(getThreadsFinished()),
this, SLOT(slotGetThreadsFinished()));
connect (_caster, SIGNAL(mountPointsRead(QList)),
this, SLOT(slotMountPointsRead(QList)));
((bncApp*)qApp)->slotMessage("========== Start BNC v" BNCVERSION " =========", true);
bncSettings settings;
QDir rnxdir(settings.value("rnxPath").toString());
if (!rnxdir.exists()) ((bncApp*)qApp)->slotMessage("Cannot find RINEX Observations directory", true);
QString rnx_file = settings.value("rnxScript").toString();
if ( !rnx_file.isEmpty() ) {
QFile rnxfile(settings.value("rnxScript").toString());
if (!rnxfile.exists()) ((bncApp*)qApp)->slotMessage("Cannot find RINEX Observations script", true);
}
QDir ephdir(settings.value("ephPath").toString());
if (!ephdir.exists()) ((bncApp*)qApp)->slotMessage("Cannot find RINEX Ephemeris directory", true);
QDir corrdir(settings.value("corrPath").toString());
if (!corrdir.exists()) ((bncApp*)qApp)->slotMessage("Cannot find Broadcast Corrections directory", true);
QString advise_file = settings.value("adviseScript").toString();
if ( !advise_file.isEmpty() ) {
QFile advisefile(settings.value("adviseScript").toString());
if (!advisefile.exists()) ((bncApp*)qApp)->slotMessage("Cannot find Outages script", true);
}
QString ant_file = settings.value("pppAntex").toString();
if ( !ant_file.isEmpty() ) {
QFile anxfile(settings.value("pppAntex").toString());
if (!anxfile.exists()) ((bncApp*)qApp)->slotMessage("Cannot find IGS ANTEX file", true);
}
_caster->slotReadMountPoints();
_casterEph = new bncEphUploadCaster();
connect(_casterEph, SIGNAL(newBytes(QByteArray,double)),
_uploadEphBytesCounter, SLOT(slotNewBytes(QByteArray,double)));
}
// Retrieve Data
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotStop() {
int iRet = QMessageBox::question(this, "Stop", "Stop retrieving data?",
QMessageBox::Yes, QMessageBox::No,
QMessageBox::NoButton);
if (iRet == QMessageBox::Yes) {
((bncApp*)qApp)->stopCombination();
delete _caster; _caster = 0;
delete _casterEph; _casterEph = 0;
_actGetData->setEnabled(true);
_actStop->setEnabled(false);
}
}
// Close Application gracefully
////////////////////////////////////////////////////////////////////////////
void bncWindow::closeEvent(QCloseEvent* event) {
int iRet = QMessageBox::question(this, "Close", "Save Options?",
QMessageBox::Yes, QMessageBox::No,
QMessageBox::Cancel);
if (iRet == QMessageBox::Cancel) {
event->ignore();
return;
}
else if (iRet == QMessageBox::Yes) {
slotSaveOptions();
}
QMainWindow::closeEvent(event);
}
// User changed the selection of mountPoints
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotSelectionChanged() {
if (_mountPointsTable->selectedItems().isEmpty()) {
_actDeleteMountPoints->setEnabled(false);
}
else {
_actDeleteMountPoints->setEnabled(true);
}
}
// Display Program Messages
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotWindowMessage(const QByteArray msg, bool showOnScreen) {
#ifdef DEBUG_RTCM2_2021
const int maxBufferSize = 1000;
#else
const int maxBufferSize = 10000;
#endif
if (! showOnScreen ) {
return;
}
QString txt = _log->toPlainText() + "\n" +
QDateTime::currentDateTime().toUTC().toString("yy-MM-dd hh:mm:ss ") + msg;
_log->clear();
_log->append(txt.right(maxBufferSize));
}
// About Message
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotAbout() {
new bncAboutDlg(0);
}
//Flowchart
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotFlowchart() {
new bncFlowchartDlg(0);
}
// Help Window
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotHelp() {
QUrl url;
url.setPath(":bnchelp.html");
new bncHlpDlg(0, url);
}
// Select Fonts
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotFontSel() {
bool ok;
QFont newFont = QFontDialog::getFont(&ok, this->font(), this);
if (ok) {
bncSettings settings;
settings.setValue("font", newFont.toString());
QApplication::setFont(newFont);
int ww = QFontMetrics(newFont).width('w');
setMinimumSize(60*ww, 80*ww);
resize(60*ww, 80*ww);
}
}
// Whats This Help
void bncWindow::slotWhatsThis() {
QWhatsThis::enterWhatsThisMode();
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotMountPointsRead(QList threads) {
_bncFigure->updateMountPoints();
_bncFigureLate->updateMountPoints();
populateMountPointsTable();
bncSettings settings;
_binSamplSpinBox->setValue(settings.value("binSampl").toInt());
_waitTimeSpinBox->setValue(settings.value("waitTime").toInt());
QListIterator iTh(threads);
while (iTh.hasNext()) {
bncGetThread* thread = iTh.next();
for (int iRow = 0; iRow < _mountPointsTable->rowCount(); iRow++) {
QUrl url( "//" + _mountPointsTable->item(iRow, 0)->text() +
"@" + _mountPointsTable->item(iRow, 1)->text() );
if (url == thread->mountPoint() &&
_mountPointsTable->item(iRow, 3)->text() == thread->latitude() &&
_mountPointsTable->item(iRow, 4)->text() == thread->longitude() ) {
((bncTableItem*) _mountPointsTable->item(iRow, 7))->setGetThread(thread);
disconnect(thread, SIGNAL(newBytes(QByteArray, double)),
_bncFigure, SLOT(slotNewData(QByteArray, double)));
connect(thread, SIGNAL(newBytes(QByteArray, double)),
_bncFigure, SLOT(slotNewData(QByteArray, double)));
disconnect(thread, SIGNAL(newLatency(QByteArray, double)),
_bncFigureLate, SLOT(slotNewLatency(QByteArray, double)));
connect(thread, SIGNAL(newLatency(QByteArray, double)),
_bncFigureLate, SLOT(slotNewLatency(QByteArray, double)));
if ( Qt::CheckState(settings.value("pppPlotCoordinates").toInt()) == Qt::Checked) {
disconnect(thread,
SIGNAL(newPosition(bncTime, double, double, double)),
_bncFigurePPP,
SLOT(slotNewPosition(bncTime, double, double, double)));
connect(thread, SIGNAL(newPosition(bncTime, double, double, double)),
_bncFigurePPP,
SLOT(slotNewPosition(bncTime, double, double, double)));
}
break;
}
}
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::CreateMenu() {
// Create Menus
// ------------
_menuFile = menuBar()->addMenu(tr("&File"));
_menuFile->addAction(_actFontSel);
_menuFile->addSeparator();
_menuFile->addAction(_actSaveOpt);
_menuFile->addSeparator();
_menuFile->addAction(_actQuit);
_menuHlp = menuBar()->addMenu(tr("&Help"));
_menuHlp->addAction(_actHelp);
_menuHlp->addAction(_actFlowchart);
_menuHlp->addAction(_actAbout);
}
// Toolbar
////////////////////////////////////////////////////////////////////////////
void bncWindow::AddToolbar() {
// Tool (Command) Bar
// ------------------
QToolBar* toolBar = new QToolBar;
addToolBar(Qt::BottomToolBarArea, toolBar);
toolBar->setMovable(false);
toolBar->addAction(_actAddMountPoints);
toolBar->addAction(_actDeleteMountPoints);
toolBar->addAction(_actGetData);
toolBar->addAction(_actStop);
toolBar->addAction(_actPostProcessing);
_postProgressLabel = new QLabel(" Progress");
toolBar->addWidget(_postProgressLabel);
toolBar->addWidget(_postProgressBar);
enableWidget(false, _postProgressLabel);
enableWidget(false, _postProgressBar);
toolBar->addWidget(new QLabel(" "));
toolBar->addAction(_actwhatsthis);
}
// About
////////////////////////////////////////////////////////////////////////////
bncAboutDlg::bncAboutDlg(QWidget* parent) :
QDialog(parent) {
QTextBrowser* tb = new QTextBrowser;
QUrl url; url.setPath(":bncabout.html");
tb->setSource(url);
tb->setReadOnly(true);
int ww = QFontMetrics(font()).width('w');
QPushButton* _closeButton = new QPushButton("Close");
_closeButton->setMaximumWidth(10*ww);
connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
QGridLayout* dlgLayout = new QGridLayout();
QLabel* img = new QLabel();
img->setPixmap(QPixmap(":ntrip-logo.png"));
dlgLayout->addWidget(img, 0,0);
dlgLayout->addWidget(new QLabel("BKG Ntrip Client (BNC) Version "BNCVERSION), 0,1);
dlgLayout->addWidget(tb,1,0,1,2);
dlgLayout->addWidget(_closeButton,2,1,Qt::AlignRight);
setLayout(dlgLayout);
resize(60*ww, 60*ww);
setWindowTitle("About BNC");
show();
}
//
////////////////////////////////////////////////////////////////////////////
bncAboutDlg::~bncAboutDlg() {
};
// Flowchart
////////////////////////////////////////////////////////////////////////////
bncFlowchartDlg::bncFlowchartDlg(QWidget* parent) :
QDialog(parent) {
int ww = QFontMetrics(font()).width('w');
QPushButton* _closeButton = new QPushButton("Close");
_closeButton->setMaximumWidth(10*ww);
connect(_closeButton, SIGNAL(clicked()), this, SLOT(close()));
QGridLayout* dlgLayout = new QGridLayout();
QLabel* img = new QLabel();
img->setPixmap(QPixmap(":bncflowchart.png"));
dlgLayout->addWidget(img, 0,0);
dlgLayout->addWidget(_closeButton,1,0,Qt::AlignLeft);
setLayout(dlgLayout);
setWindowTitle("Flow Chart");
show();
}
//
////////////////////////////////////////////////////////////////////////////
bncFlowchartDlg::~bncFlowchartDlg() {
};
// Enable/Disable Widget (and change its color)
////////////////////////////////////////////////////////////////////////////
void bncWindow::enableWidget(bool enable, QWidget* widget) {
const static QPalette paletteWhite(QColor(255, 255, 255));
const static QPalette paletteGray(QColor(230, 230, 230));
widget->setEnabled(enable);
if (enable) {
widget->setPalette(paletteWhite);
}
else {
widget->setPalette(paletteGray);
}
}
// Bnc Text
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotBncTextChanged(){
bool enable = true;
// Proxy
//------
if (sender() == 0 || sender() == _proxyHostLineEdit) {
enable = !_proxyHostLineEdit->text().isEmpty();
enableWidget(enable, _proxyPortLineEdit);
}
// RINEX Observations
// ------------------
if (sender() == 0 || sender() == _rnxPathLineEdit) {
enable = !_rnxPathLineEdit->text().isEmpty();
enableWidget(enable, _rnxSamplSpinBox);
enableWidget(enable, _rnxSkelLineEdit);
enableWidget(enable, _rnxScrpLineEdit);
enableWidget(enable, _rnxV3CheckBox);
enableWidget(enable, _rnxIntrComboBox);
}
// RINEX Ephemeris
// ---------------
if (sender() == 0 || sender() == _ephPathLineEdit || sender() == _outEphPortLineEdit) {
enable = !_ephPathLineEdit->text().isEmpty() || !_outEphPortLineEdit->text().isEmpty();
enableWidget(enable, _ephIntrComboBox);
enableWidget(enable, _ephV3CheckBox);
}
// Broadcast Corrections
// ---------------------
if (sender() == 0 || sender() == _corrPathLineEdit || sender() == _corrPortLineEdit) {
enable = !_corrPathLineEdit->text().isEmpty() || !_corrPortLineEdit->text().isEmpty();
enableWidget(enable, _corrIntrComboBox);
}
// Feed Engine
// -----------
if (sender() == 0 || sender() == _outPortLineEdit || sender() == _outFileLineEdit) {
enable = !_outPortLineEdit->text().isEmpty() || !_outFileLineEdit->text().isEmpty();
enableWidget(enable, _waitTimeSpinBox);
enableWidget(enable, _binSamplSpinBox);
}
// Serial Output
// -------------
if (sender() == 0 || sender() == _serialMountPointLineEdit ||
sender() == _serialAutoNMEAComboBox) {
enable = !_serialMountPointLineEdit->text().isEmpty();
enableWidget(enable, _serialPortNameLineEdit);
enableWidget(enable, _serialBaudRateComboBox);
enableWidget(enable, _serialParityComboBox);
enableWidget(enable, _serialDataBitsComboBox);
enableWidget(enable, _serialStopBitsComboBox);
enableWidget(enable, _serialFlowControlComboBox);
enableWidget(enable, _serialAutoNMEAComboBox);
bool enable2 = enable && _serialAutoNMEAComboBox->currentText() != "Auto";
enableWidget(enable2, _serialFileNMEALineEdit);
}
// Outages
// -------
if (sender() == 0 || sender() == _obsRateComboBox) {
enable = !_obsRateComboBox->currentText().isEmpty();
enableWidget(enable, _adviseFailSpinBox);
enableWidget(enable, _adviseRecoSpinBox);
enableWidget(enable, _adviseScriptLineEdit);
}
// Miscellaneous
// -------------
if (sender() == 0 || sender() == _miscMountLineEdit) {
enable = !_miscMountLineEdit->text().isEmpty();
enableWidget(enable, _perfIntrComboBox);
enableWidget(enable, _scanRTCMCheckBox);
}
// PPP Client
// ----------
if (sender() == 0
|| sender() == _pppMountLineEdit
|| sender() == _pppCorrMountLineEdit
|| sender() == _pppRefCrdXLineEdit
|| sender() == _pppRefCrdYLineEdit
|| sender() == _pppRefCrdZLineEdit
|| sender() == _pppRefdNLineEdit
|| sender() == _pppRefdELineEdit
|| sender() == _pppRefdULineEdit
|| sender() == _pppSync
|| sender() == _pppSPPComboBox
|| sender() == _pppQuickStartLineEdit
|| sender() == _pppEstTropoCheckBox
|| sender() == _pppUsePhaseCheckBox
|| sender() == _pppAntexLineEdit ) {
enable = (!_pppMountLineEdit->text().isEmpty() && !_pppCorrMountLineEdit->text().isEmpty()) ||
(!_pppMountLineEdit->text().isEmpty() && _pppSPPComboBox->currentText() == "SPP") ||
(_pppSPPComboBox->currentText() == "Post-Processing");
enableWidget(enable, _pppNMEALineEdit);
enableWidget(enable, _pppNMEAPortLineEdit);
enableWidget(enable, _pppRefCrdXLineEdit);
enableWidget(enable, _pppRefCrdYLineEdit);
enableWidget(enable, _pppRefCrdZLineEdit);
enableWidget(enable, _pppRefdNLineEdit);
enableWidget(enable, _pppRefdELineEdit);
enableWidget(enable, _pppRefdULineEdit);
enableWidget(enable, _pppUsePhaseCheckBox);
enableWidget(enable, _pppPlotCoordinates);
enableWidget(enable, _pppEstTropoCheckBox);
enableWidget(enable, _pppGLONASSCheckBox);
enableWidget(enable, _pppGalileoCheckBox);
enableWidget(enable, _pppAntexLineEdit);
enableWidget(enable, _pppSigCLineEdit);
enableWidget(enable, _pppSigCrd0);
enableWidget(enable, _pppSigCrdP);
bool enable2 = enable && !_pppRefCrdXLineEdit->text().isEmpty() &&
!_pppRefCrdYLineEdit->text().isEmpty() &&
!_pppRefCrdZLineEdit->text().isEmpty();
enableWidget(enable2, _pppAverageLineEdit);
enableWidget(enable2, _pppQuickStartLineEdit);
bool enable3 = enable2 && !_pppQuickStartLineEdit->text().isEmpty();
enableWidget(enable3, _pppMaxSolGapLineEdit);
bool enable4 = enable && !_pppAntexLineEdit->text().isEmpty();
enableWidget(enable4, _pppAntennaLineEdit);
enableWidget(enable4, _pppApplySatAntCheckBox);
bool enable5 = enable && _pppEstTropoCheckBox->isChecked() && !_pppMountLineEdit->text().isEmpty();
enableWidget(enable5, _pppSigTrp0);
enableWidget(enable5, _pppSigTrpP);
bool enable6 = enable && _pppUsePhaseCheckBox->isChecked();
enableWidget(enable6, _pppSigPLineEdit);
bool enable7 = enable && _pppSPPComboBox->currentText() == "PPP";
enableWidget(enable7, _pppSync);
enableWidget(enable7, _pppCorrMountLineEdit);
bool enable8 = _pppSPPComboBox->currentText() == "Post-Processing";
enableWidget(enable8, _postObsFileChooser);
enableWidget(enable8, _postNavFileChooser);
enableWidget(enable8, _postCorrFileChooser);
_actPostProcessing->setEnabled(enable8);
enableWidget(!enable8, _pppMountLineEdit);
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotAddCmbRow() {
int iRow = _cmbTable->rowCount();
_cmbTable->insertRow(iRow);
for (int iCol = 0; iCol < _cmbTable->columnCount(); iCol++) {
_cmbTable->setItem(iRow, iCol, new QTableWidgetItem(""));
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotDelCmbRow() {
int nRows = _cmbTable->rowCount();
bool flg[nRows];
for (int iRow = 0; iRow < nRows; iRow++) {
if (_cmbTable->isItemSelected(_cmbTable->item(iRow,1))) {
flg[iRow] = true;
}
else {
flg[iRow] = false;
}
}
for (int iRow = nRows-1; iRow >= 0; iRow--) {
if (flg[iRow]) {
_cmbTable->removeRow(iRow);
}
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::populateCmbTable() {
for (int iRow = _cmbTable->rowCount()-1; iRow >=0; iRow--) {
_cmbTable->removeRow(iRow);
}
bncSettings settings;
int iRow = -1;
QListIterator it(settings.value("combineStreams").toStringList());
while (it.hasNext()) {
QStringList hlp = it.next().split(" ");
if (hlp.size() > 2) {
++iRow;
_cmbTable->insertRow(iRow);
}
for (int iCol = 0; iCol < hlp.size(); iCol++) {
_cmbTable->setItem(iRow, iCol, new QTableWidgetItem(hlp[iCol]));
}
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotAddUploadRow() {
int iRow = _uploadTable->rowCount();
_uploadTable->insertRow(iRow);
for (int iCol = 0; iCol < _uploadTable->columnCount(); iCol++) {
if (iCol == 3) {
QLineEdit* passwd = new QLineEdit();
passwd->setFrame(false);
passwd->setEchoMode(QLineEdit::PasswordEchoOnEdit);
_uploadTable->setCellWidget(iRow, iCol, passwd);
}
else if (iCol == 4) {
QComboBox* system = new QComboBox();
system->setEditable(false);
system->addItems(QString("IGS05,ETRF2000,NAD83,GDA94,SIRGAS95,SIRGAS2000,Custom").split(","));
system->setFrame(false);
_uploadTable->setCellWidget(iRow, iCol, system);
}
else if (iCol == 5) {
QCheckBox* com = new QCheckBox();
_uploadTable->setCellWidget(iRow, iCol, com);
}
else if (iCol == 8) {
bncTableItem* bncIt = new bncTableItem();
bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
_uploadTable->setItem(iRow, iCol, bncIt);
((bncApp*)qApp)->_uploadTableItems[iRow] = bncIt;
}
else {
_uploadTable->setItem(iRow, iCol, new QTableWidgetItem(""));
}
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotDelUploadRow() {
((bncApp*)qApp)->_uploadTableItems.clear();
int nRows = _uploadTable->rowCount();
bool flg[nRows];
for (int iRow = 0; iRow < nRows; iRow++) {
if (_uploadTable->isItemSelected(_uploadTable->item(iRow,1))) {
flg[iRow] = true;
}
else {
flg[iRow] = false;
}
}
for (int iRow = nRows-1; iRow >= 0; iRow--) {
if (flg[iRow]) {
_uploadTable->removeRow(iRow);
}
}
for (int iRow = 0; iRow < _uploadTable->rowCount(); iRow++) {
((bncApp*)qApp)->_uploadTableItems[iRow] =
(bncTableItem*) _uploadTable->item(iRow, 8);
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::populateUploadTable() {
for (int iRow = _uploadTable->rowCount()-1; iRow >=0; iRow--) {
_uploadTable->removeRow(iRow);
}
bncSettings settings;
int iRow = -1;
QListIterator it(settings.value("uploadMountpointsOut").toStringList());
while (it.hasNext()) {
QStringList hlp = it.next().split(",");
if (hlp.size() > 6) {
++iRow;
_uploadTable->insertRow(iRow);
}
for (int iCol = 0; iCol < hlp.size(); iCol++) {
if (iCol == 3) {
QLineEdit* passwd = new QLineEdit();
passwd->setFrame(false);
passwd->setEchoMode(QLineEdit::PasswordEchoOnEdit);
passwd->setText(hlp[iCol]);
_uploadTable->setCellWidget(iRow, iCol, passwd);
}
else if (iCol == 4) {
QComboBox* system = new QComboBox();
system->setEditable(false);
system->addItems(QString("IGS05,ETRF2000,NAD83,GDA94,SIRGAS95,SIRGAS2000,Custom").split(","));
system->setFrame(false);
system->setCurrentIndex(system->findText(hlp[iCol]));
_uploadTable->setCellWidget(iRow, iCol, system);
}
else if (iCol == 5) {
QCheckBox* com = new QCheckBox();
if (hlp[iCol].toInt() == Qt::Checked) {
com->setCheckState(Qt::Checked);
}
_uploadTable->setCellWidget(iRow, iCol, com);
}
else if (iCol == 8) {
bncTableItem* bncIt = new bncTableItem();
bncIt->setFlags(bncIt->flags() & ~Qt::ItemIsEditable);
_uploadTable->setItem(iRow, iCol, bncIt);
((bncApp*)qApp)->_uploadTableItems[iRow] = bncIt;
}
else {
_uploadTable->setItem(iRow, iCol, new QTableWidgetItem(hlp[iCol]));
}
}
}
}
//
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotSetUploadTrafo() {
bncCustomTrafo* dlg = new bncCustomTrafo(this);
dlg->exec();
delete dlg;
}
// Post-Processing
////////////////////////////////////////////////////////////////////////////
void bncWindow::slotPostProcessing() {
_actPostProcessing->setEnabled(false);
slotSaveOptions();
_postProgressBar->reset();
enableWidget(true, _postProgressLabel);
enableWidget(true, _postProgressBar);
//// beg test
for (int ii = 0; ii <= 10000; ++ii) {
_postProgressBar->setValue(ii/100);
}
//// end test
enableWidget(false, _postProgressLabel);
enableWidget(false, _postProgressBar);
_postProgressBar->reset();
_actPostProcessing->setEnabled(true);
}