mirror of
				https://github.com/saitohirga/WSJT-X.git
				synced 2025-10-31 13:10:19 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			2269 lines
		
	
	
		
			66 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			2269 lines
		
	
	
		
			66 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| //------------------------------------------------------------------ MainWindow
 | |
| #include "mainwindow.h"
 | |
| #include "ui_mainwindow.h"
 | |
| #include "devsetup.h"
 | |
| #include "plotter.h"
 | |
| #include "about.h"
 | |
| #include "astro.h"
 | |
| #include "widegraph.h"
 | |
| #include "messages.h"
 | |
| #include "bandmap.h"
 | |
| #include "txtune.h"
 | |
| #include "sleep.h"
 | |
| #include <portaudio.h>
 | |
| 
 | |
| #define NFFT 32768
 | |
| 
 | |
| short int iwave[2*60*11025];          //Wave file for Tx audio
 | |
| int nwave;                            //Length of Tx waveform
 | |
| bool btxok;                           //True if OK to transmit
 | |
| bool bTune;
 | |
| bool bIQxt;
 | |
| double outputLatency;                 //Latency in seconds
 | |
| int txPower;
 | |
| int iqAmp;
 | |
| int iqPhase;
 | |
| qint16 id[4*60*96000];
 | |
| 
 | |
| Astro*     g_pAstro = NULL;
 | |
| WideGraph* g_pWideGraph = NULL;
 | |
| Messages*  g_pMessages = NULL;
 | |
| BandMap*   g_pBandMap = NULL;
 | |
| TxTune*    g_pTxTune = NULL;
 | |
| QSharedMemory mem_m65("mem_m65");
 | |
| 
 | |
| QString Program_Title_Version="  MAP65 3.0.0-devel   by K1JT, G4WJS, K9AN, and IV3NWV";
 | |
| 
 | |
| extern const int RxDataFrequency = 96000;
 | |
| extern const int TxDataFrequency = 11025;
 | |
| 
 | |
| //-------------------------------------------------- MainWindow constructor
 | |
| MainWindow::MainWindow(QWidget *parent) :
 | |
|   QMainWindow(parent),
 | |
|   ui(new Ui::MainWindow)
 | |
| {
 | |
|   ui->setupUi(this);
 | |
| 
 | |
|   on_EraseButton_clicked();
 | |
|   ui->labUTC->setStyleSheet( \
 | |
|         "QLabel { background-color : black; color : yellow; }");
 | |
|   ui->labTol1->setStyleSheet( \
 | |
|         "QLabel { background-color : white; color : black; }");
 | |
|   ui->labTol1->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   ui->dxStationGroupBox->setStyleSheet("QFrame{border: 5px groove red}");
 | |
| 
 | |
|   QActionGroup* paletteGroup = new QActionGroup(this);
 | |
|   ui->actionCuteSDR->setActionGroup(paletteGroup);
 | |
|   ui->actionLinrad->setActionGroup(paletteGroup);
 | |
|   ui->actionAFMHot->setActionGroup(paletteGroup);
 | |
|   ui->actionBlue->setActionGroup(paletteGroup);
 | |
| 
 | |
|   QActionGroup* modeGroup = new QActionGroup(this);
 | |
|   ui->actionNoJT65->setActionGroup(modeGroup);
 | |
|   ui->actionJT65A->setActionGroup(modeGroup);
 | |
|   ui->actionJT65B->setActionGroup(modeGroup);
 | |
|   ui->actionJT65C->setActionGroup(modeGroup);
 | |
| 
 | |
|   QActionGroup* modeGroup2 = new QActionGroup(this);
 | |
|   ui->actionNoQ65->setActionGroup(modeGroup2);
 | |
|   ui->actionQ65A->setActionGroup(modeGroup2);
 | |
|   ui->actionQ65B->setActionGroup(modeGroup2);
 | |
|   ui->actionQ65C->setActionGroup(modeGroup2);
 | |
|   ui->actionQ65D->setActionGroup(modeGroup2);
 | |
|   ui->actionQ65E->setActionGroup(modeGroup2);
 | |
| 
 | |
|   QActionGroup* saveGroup = new QActionGroup(this);
 | |
|   ui->actionSave_all->setActionGroup(saveGroup);
 | |
|   ui->actionNone->setActionGroup(saveGroup);
 | |
| 
 | |
|   QActionGroup* DepthGroup = new QActionGroup(this);
 | |
|   ui->actionNo_Deep_Search->setActionGroup(DepthGroup);
 | |
|   ui->actionNormal_Deep_Search->setActionGroup(DepthGroup);
 | |
|   ui->actionAggressive_Deep_Search->setActionGroup(DepthGroup);
 | |
| 
 | |
|   QButtonGroup* txMsgButtonGroup = new QButtonGroup;
 | |
|   txMsgButtonGroup->addButton(ui->txrb1,1);
 | |
|   txMsgButtonGroup->addButton(ui->txrb2,2);
 | |
|   txMsgButtonGroup->addButton(ui->txrb3,3);
 | |
|   txMsgButtonGroup->addButton(ui->txrb4,4);
 | |
|   txMsgButtonGroup->addButton(ui->txrb5,5);
 | |
|   txMsgButtonGroup->addButton(ui->txrb6,6);
 | |
|   connect(txMsgButtonGroup,SIGNAL(buttonClicked(int)),SLOT(set_ntx(int)));
 | |
|   connect(ui->decodedTextBrowser,SIGNAL(selectCallsign(bool)),this,
 | |
|           SLOT(selectCall2(bool)));
 | |
| 
 | |
|   setWindowTitle(Program_Title_Version);
 | |
| 
 | |
|   connect(&soundInThread, SIGNAL(readyForFFT(int)),
 | |
|              this, SLOT(dataSink(int)));
 | |
|   connect(&soundInThread, SIGNAL(error(QString)), this,
 | |
|           SLOT(showSoundInError(QString)));
 | |
|   connect(&soundInThread, SIGNAL(status(QString)), this,
 | |
|           SLOT(showStatusMessage(QString)));
 | |
|   createStatusBar();
 | |
| 
 | |
|   connect(&proc_m65, SIGNAL(readyReadStandardOutput()),
 | |
|                     this, SLOT(readFromStdout()));
 | |
| 
 | |
|   connect(&proc_m65, SIGNAL(error(QProcess::ProcessError)),
 | |
|           this, SLOT(m65_error()));
 | |
| 
 | |
|   connect(&proc_m65, SIGNAL(readyReadStandardError()),
 | |
|           this, SLOT(readFromStderr()));
 | |
| 
 | |
|   connect(&proc_editor, SIGNAL(error(QProcess::ProcessError)),
 | |
|           this, SLOT(editor_error()));
 | |
| 
 | |
|   QTimer *guiTimer = new QTimer(this);
 | |
|   connect(guiTimer, SIGNAL(timeout()), this, SLOT(guiUpdate()));
 | |
|   guiTimer->start(100);                            //Don't change the 100 ms!
 | |
| 
 | |
|   m_auto=false;
 | |
|   m_waterfallAvg = 1;
 | |
|   m_network = true;
 | |
|   m_txFirst=false;
 | |
|   m_txMute=false;
 | |
|   btxok=false;
 | |
|   m_restart=false;
 | |
|   m_transmitting=false;
 | |
|   m_killAll=false;
 | |
|   m_widebandDecode=false;
 | |
|   m_ntx=1;
 | |
|   m_myCall="K1JT";
 | |
|   m_myGrid="FN20qi";
 | |
|   m_appDir = QApplication::applicationDirPath();
 | |
|   m_saveDir="/users/joe/map65/install/save";
 | |
|   m_azelDir="/users/joe/map65/install/";
 | |
|   m_editorCommand="notepad";
 | |
|   m_txFreq=125;
 | |
|   m_setftx=0;
 | |
|   m_loopall=false;
 | |
|   m_startAnother=false;
 | |
|   m_saveAll=false;
 | |
|   m_onlyEME=false;
 | |
|   m_sec0=-1;
 | |
|   m_hsym0=-1;
 | |
|   m_palette="CuteSDR";
 | |
|   m_map65RxLog=1;                     //Write Date and Time to all65.txt
 | |
|   m_nutc0=9999;
 | |
|   m_kb8rq=false;
 | |
|   m_NB=false;
 | |
|   m_mode="JT65B";
 | |
|   m_mode65=2;
 | |
|   m_fs96000=true;
 | |
|   m_udpPort=50004;
 | |
|   m_adjustIQ=0;
 | |
|   m_applyIQcal=0;
 | |
|   m_colors="000066ff0000ffff00969696646464";
 | |
|   m_nsave=0;
 | |
|   m_modeJT65=0;
 | |
|   m_modeQ65=0;
 | |
|   m_TRperiod=60;
 | |
|   m_modeTx="JT65";
 | |
|   bTune=false;
 | |
|   txPower=100;
 | |
|   iqAmp=0;
 | |
|   iqPhase=0;
 | |
| 
 | |
|   xSignalMeter = new SignalMeter(ui->xMeterFrame);
 | |
|   xSignalMeter->resize(50, 160);
 | |
|   ySignalMeter = new SignalMeter(ui->yMeterFrame);
 | |
|   ySignalMeter->resize(50, 160);
 | |
| 
 | |
| #ifdef WIN32
 | |
|   while(true) {
 | |
|       int iret=killbyname("m65.exe");
 | |
|       if(iret == 603) break;
 | |
|       if(iret != 0) msgBox("KillByName return code: " +
 | |
|                            QString::number(iret));
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   if(!mem_m65.attach()) {
 | |
|     if (!mem_m65.create(sizeof(datcom_))) {
 | |
|       msgBox("Unable to create shared memory segment.");
 | |
|     }
 | |
|   }
 | |
|   char *to = (char*)mem_m65.data();
 | |
|   int size=sizeof(datcom_);
 | |
|   if(datcom_.newdat==0) {
 | |
|     int noffset = 4*4*5760000 + 4*4*322*32768 + 4*4*32768;
 | |
|     to += noffset;
 | |
|     size -= noffset;
 | |
|   }
 | |
|   memset(to,0,size);         //Zero all decoding params in shared memory
 | |
| 
 | |
|   PaError paerr=Pa_Initialize();                    //Initialize Portaudio
 | |
|   if(paerr!=paNoError) {
 | |
|     msgBox("Unable to initialize PortAudio.");
 | |
|   }
 | |
|   readSettings();		             //Restore user's setup params
 | |
|   QFile lockFile(m_appDir + "/.lock"); //Create .lock so m65 will wait
 | |
|   lockFile.open(QIODevice::ReadWrite);
 | |
|   QFile quitFile(m_appDir + "/.lock");
 | |
|   quitFile.remove();
 | |
|   proc_m65.start(QDir::toNativeSeparators(m_appDir + "/m65"), {"-s", });
 | |
| 
 | |
|   m_pbdecoding_style1="QPushButton{background-color: cyan; \
 | |
|       border-style: outset; border-width: 1px; border-radius: 5px; \
 | |
|       border-color: black; min-width: 5em; padding: 3px;}";
 | |
|   m_pbmonitor_style="QPushButton{background-color: #00ff00; \
 | |
|       border-style: outset; border-width: 1px; border-radius: 5px; \
 | |
|       border-color: black; min-width: 5em; padding: 3px;}";
 | |
|   m_pbAutoOn_style="QPushButton{background-color: red; \
 | |
|       border-style: outset; border-width: 1px; border-radius: 5px; \
 | |
|       border-color: black; min-width: 5em; padding: 3px;}";
 | |
| 
 | |
|   genStdMsgs("");
 | |
| 
 | |
|   on_actionAstro_Data_triggered();           //Create the other windows
 | |
|   on_actionWide_Waterfall_triggered();
 | |
|   on_actionMessages_triggered();
 | |
|   on_actionBand_Map_triggered();
 | |
|   g_pMessages->setColors(m_colors);
 | |
|   g_pBandMap->setColors(m_colors);
 | |
|   g_pAstro->setFontSize(m_astroFont);
 | |
| 
 | |
|   if(m_modeQ65==0) on_actionNoQ65_triggered();
 | |
|   if(m_modeQ65==1) on_actionQ65A_triggered();
 | |
|   if(m_modeQ65==2) on_actionQ65B_triggered();
 | |
|   if(m_modeQ65==3) on_actionQ65C_triggered();
 | |
|   if(m_modeQ65==4) on_actionQ65D_triggered();
 | |
|   if(m_modeQ65==5) on_actionQ65E_triggered();
 | |
| 
 | |
|   if(m_modeJT65==0) on_actionNoJT65_triggered();
 | |
|   if(m_modeJT65==1) on_actionJT65A_triggered();
 | |
|   if(m_modeJT65==2) on_actionJT65B_triggered();
 | |
|   if(m_modeJT65==3) on_actionJT65C_triggered();
 | |
|   future1 = new QFuture<void>;
 | |
|   watcher1 = new QFutureWatcher<void>;
 | |
|   connect(watcher1, SIGNAL(finished()),this,SLOT(diskDat()));
 | |
| 
 | |
|   future2 = new QFuture<void>;
 | |
|   watcher2 = new QFutureWatcher<void>;
 | |
|   connect(watcher2, SIGNAL(finished()),this,SLOT(diskWriteFinished()));
 | |
| 
 | |
| // Assign input device and start input thread
 | |
|   soundInThread.setInputDevice(m_paInDevice);
 | |
|   if(m_fs96000) soundInThread.setRate(96000.0);
 | |
|   if(!m_fs96000) soundInThread.setRate(95238.1);
 | |
|   soundInThread.setBufSize(10*7056);
 | |
|   soundInThread.setNetwork(m_network);
 | |
|   soundInThread.setPort(m_udpPort);
 | |
|   if(!m_xpol) soundInThread.setNrx(1);
 | |
|   if(m_xpol) soundInThread.setNrx(2);
 | |
|   soundInThread.start(QThread::HighestPriority);
 | |
| 
 | |
|   // Assign output device and start output thread
 | |
|   soundOutThread.setOutputDevice(m_paOutDevice);
 | |
| //  soundOutThread.start(QThread::HighPriority);
 | |
| 
 | |
|   m_monitoring=true;                           // Start with Monitoring ON
 | |
|   soundInThread.setMonitoring(m_monitoring);
 | |
|   m_diskData=false;
 | |
|   m_tol=500;
 | |
|   g_pWideGraph->setTol(m_tol);
 | |
|   g_pWideGraph->setFcal(m_fCal);
 | |
|   if(m_fs96000) g_pWideGraph->setFsample(96000);
 | |
|   if(!m_fs96000) g_pWideGraph->setFsample(95238);
 | |
|   g_pWideGraph->m_mult570=m_mult570;
 | |
|   g_pWideGraph->m_mult570Tx=m_mult570Tx;
 | |
|   g_pWideGraph->m_cal570=m_cal570;
 | |
|   g_pWideGraph->m_TxOffset=m_TxOffset;
 | |
|   if(m_initIQplus) g_pWideGraph->initIQplus();
 | |
| 
 | |
| // Create "m_worked", a dictionary of all calls in wsjt.log
 | |
|   QFile f("wsjt.log");
 | |
|   f.open(QIODevice::ReadOnly);
 | |
|   if(f.isOpen()) {
 | |
|     QTextStream in(&f);
 | |
|     QString line,t,callsign;
 | |
|     for(int i=0; i<99999; i++) {
 | |
|       line=in.readLine();
 | |
|       if(line.length()<=0) break;
 | |
|       t=line.mid(18,12);
 | |
|       callsign=t.mid(0,t.indexOf(","));
 | |
|       m_worked[callsign]=true;
 | |
|     }
 | |
|     f.close();
 | |
|   }
 | |
| 
 | |
|   if(ui->actionLinrad->isChecked()) on_actionLinrad_triggered();
 | |
|   if(ui->actionCuteSDR->isChecked()) on_actionCuteSDR_triggered();
 | |
|   if(ui->actionAFMHot->isChecked()) on_actionAFMHot_triggered();
 | |
|   if(ui->actionBlue->isChecked()) on_actionBlue_triggered();
 | |
| 
 | |
|   m_dataDir = QStandardPaths::writableLocation (QStandardPaths::DataLocation);
 | |
|   m_tempDir = QStandardPaths::writableLocation (QStandardPaths::TempLocation) + "/map65";
 | |
| 
 | |
|                                              // End of MainWindow constructor
 | |
| }
 | |
| 
 | |
|   //--------------------------------------------------- MainWindow destructor
 | |
| MainWindow::~MainWindow()
 | |
| {
 | |
|   writeSettings();
 | |
|   if (soundInThread.isRunning()) {
 | |
|     soundInThread.quit();
 | |
|     soundInThread.wait(3000);
 | |
|   }
 | |
|   if (soundOutThread.isRunning()) {
 | |
|     soundOutThread.quitExecution=true;
 | |
|     soundOutThread.wait(3000);
 | |
|   }
 | |
|   Pa_Terminate();
 | |
|   if(!m_decoderBusy) {
 | |
|     QFile lockFile(m_appDir + "/.lock");
 | |
|     lockFile.remove();
 | |
|   }
 | |
|   delete ui;
 | |
| }
 | |
| 
 | |
| //-------------------------------------------------------- writeSettings()
 | |
| void MainWindow::writeSettings()
 | |
| {
 | |
|   QString inifile = m_appDir + "/map65.ini";
 | |
|   QSettings settings(inifile, QSettings::IniFormat);
 | |
| 
 | |
|   settings.beginGroup("MainWindow");
 | |
|   settings.setValue("geometry", saveGeometry());
 | |
|   settings.setValue("MRUdir", m_path);
 | |
|   settings.setValue("TxFirst",m_txFirst);
 | |
|   settings.setValue("DXcall",ui->dxCallEntry->text());
 | |
|   settings.setValue("DXgrid",ui->dxGridEntry->text());
 | |
| 
 | |
|   if(g_pAstro->isVisible()) {
 | |
|     m_astroGeom = g_pAstro->geometry();
 | |
|     settings.setValue("AstroGeom",m_astroGeom);
 | |
|   }
 | |
| 
 | |
|   if(g_pWideGraph->isVisible()) {
 | |
|     m_wideGraphGeom = g_pWideGraph->geometry();
 | |
|     settings.setValue("WideGraphGeom",m_wideGraphGeom);
 | |
|   }
 | |
|   if(g_pMessages->isVisible()) {
 | |
|     m_messagesGeom = g_pMessages->geometry();
 | |
|     settings.setValue("MessagesGeom",m_messagesGeom);
 | |
|   }
 | |
|   if(g_pBandMap->isVisible()) {
 | |
|     m_bandMapGeom = g_pBandMap->geometry();
 | |
|     settings.setValue("BandMapGeom",m_bandMapGeom);
 | |
|   }
 | |
|   settings.endGroup();
 | |
| 
 | |
|   settings.beginGroup("Common");
 | |
|   settings.setValue("MyCall",m_myCall);
 | |
|   settings.setValue("MyGrid",m_myGrid);
 | |
|   settings.setValue("IDint",m_idInt);
 | |
|   settings.setValue("PTTport",m_pttPort);
 | |
|   settings.setValue("AstroFont",m_astroFont);
 | |
|   settings.setValue("Xpol",m_xpol);
 | |
|   settings.setValue("XpolX",m_xpolx);
 | |
|   settings.setValue("SaveDir",m_saveDir);
 | |
|   settings.setValue("AzElDir",m_azelDir);
 | |
|   settings.setValue("Editor",m_editorCommand);
 | |
|   settings.setValue("DXCCpfx",m_dxccPfx);
 | |
|   settings.setValue("Timeout",m_timeout);
 | |
|   settings.setValue("TxPower",txPower);
 | |
|   settings.setValue("IQamp",iqAmp);
 | |
|   settings.setValue("IQphase",iqPhase);
 | |
|   settings.setValue("ApplyIQcal",m_applyIQcal);
 | |
|   settings.setValue("dPhi",m_dPhi);
 | |
|   settings.setValue("Fcal",m_fCal);
 | |
|   settings.setValue("Fadd",m_fAdd);
 | |
|   settings.setValue("NetworkInput", m_network);
 | |
|   settings.setValue("FSam96000", m_fs96000);
 | |
|   settings.setValue("SoundInIndex",m_nDevIn);
 | |
|   settings.setValue("paInDevice",m_paInDevice);
 | |
|   settings.setValue("SoundOutIndex",m_nDevOut);
 | |
|   settings.setValue("paOutDevice",m_paOutDevice);
 | |
|   settings.setValue("IQswap",m_IQswap);
 | |
|   settings.setValue("Plus10dB",m_10db);
 | |
|   settings.setValue("IQxt",m_bIQxt);
 | |
|   settings.setValue("InitIQplus",m_initIQplus);
 | |
|   settings.setValue("UDPport",m_udpPort);
 | |
|   settings.setValue("PaletteCuteSDR",ui->actionCuteSDR->isChecked());
 | |
|   settings.setValue("PaletteLinrad",ui->actionLinrad->isChecked());
 | |
|   settings.setValue("PaletteAFMHot",ui->actionAFMHot->isChecked());
 | |
|   settings.setValue("PaletteBlue",ui->actionBlue->isChecked());
 | |
|   settings.setValue("Mode",m_mode);
 | |
|   settings.setValue("nModeJT65",m_modeJT65);
 | |
|   settings.setValue("nModeQ65",m_modeQ65);
 | |
|   settings.setValue("TxMode",m_modeTx);
 | |
|   settings.setValue("SaveNone",ui->actionNone->isChecked());
 | |
|   settings.setValue("SaveAll",ui->actionSave_all->isChecked());
 | |
|   settings.setValue("NDepth",m_ndepth);
 | |
|   settings.setValue("NEME",m_onlyEME);
 | |
|   settings.setValue("KB8RQ",m_kb8rq);
 | |
|   settings.setValue("NB",m_NB);
 | |
|   settings.setValue("NBslider",m_NBslider);
 | |
|   settings.setValue("GainX",(double)m_gainx);
 | |
|   settings.setValue("GainY",(double)m_gainy);
 | |
|   settings.setValue("PhaseX",(double)m_phasex);
 | |
|   settings.setValue("PhaseY",(double)m_phasey);
 | |
|   settings.setValue("Mult570",m_mult570);
 | |
|   settings.setValue("Mult570Tx",m_mult570Tx);
 | |
|   settings.setValue("Cal570",m_cal570);
 | |
|   settings.setValue("TxOffset",m_TxOffset);
 | |
|   settings.setValue("Colors",m_colors);
 | |
|   settings.endGroup();
 | |
| }
 | |
| 
 | |
| //---------------------------------------------------------- readSettings()
 | |
| void MainWindow::readSettings()
 | |
| {
 | |
|   QString inifile = m_appDir + "/map65.ini";
 | |
|   QSettings settings(inifile, QSettings::IniFormat);
 | |
|   settings.beginGroup("MainWindow");
 | |
|   restoreGeometry(settings.value("geometry").toByteArray());
 | |
|   ui->dxCallEntry->setText(settings.value("DXcall","").toString());
 | |
|   ui->dxGridEntry->setText(settings.value("DXgrid","").toString());
 | |
| 
 | |
|   m_astroGeom = settings.value("AstroGeom", QRect(71,390,227,403)).toRect();
 | |
| 
 | |
|   m_wideGraphGeom = settings.value("WideGraphGeom", \
 | |
|                                    QRect(45,30,1023,340)).toRect();
 | |
|   m_messagesGeom = settings.value("MessagesGeom", \
 | |
|                                   QRect(800,400,381,400)).toRect();
 | |
|   m_bandMapGeom = settings.value("BandMapGeom", \
 | |
|                                   QRect(280,400,142,400)).toRect();
 | |
|   m_path = settings.value("MRUdir", m_appDir + "/save").toString();
 | |
|   m_txFirst = settings.value("TxFirst",false).toBool();
 | |
|   ui->txFirstCheckBox->setChecked(m_txFirst);
 | |
|   settings.endGroup();
 | |
| 
 | |
|   settings.beginGroup("Common");
 | |
|   m_myCall=settings.value("MyCall","").toString();
 | |
|   m_myGrid=settings.value("MyGrid","").toString();
 | |
|   m_idInt=settings.value("IDint",0).toInt();
 | |
|   m_pttPort=settings.value("PTTport",0).toInt();
 | |
|   m_astroFont=settings.value("AstroFont",20).toInt();
 | |
|   m_xpol=settings.value("Xpol",false).toBool();
 | |
|   ui->actionFind_Delta_Phi->setEnabled(m_xpol);
 | |
|   m_xpolx=settings.value("XpolX",false).toBool();
 | |
|   m_saveDir=settings.value("SaveDir",m_appDir + "/save").toString();
 | |
|   m_azelDir=settings.value("AzElDir",m_appDir).toString();
 | |
|   m_editorCommand=settings.value("Editor","notepad").toString();
 | |
|   m_dxccPfx=settings.value("DXCCpfx","").toString();
 | |
|   m_timeout=settings.value("Timeout",20).toInt();
 | |
|   txPower=settings.value("TxPower",100).toInt();
 | |
|   iqAmp=settings.value("IQamp",0).toInt();
 | |
|   iqPhase=settings.value("IQphase",0).toInt();
 | |
|   m_applyIQcal=settings.value("ApplyIQcal",0).toInt();
 | |
|   ui->actionApply_IQ_Calibration->setChecked(m_applyIQcal!=0);
 | |
|   m_dPhi=settings.value("dPhi",0).toInt();
 | |
|   m_fCal=settings.value("Fcal",0).toInt();
 | |
|   m_fAdd=settings.value("FAdd",0).toDouble();
 | |
|   soundInThread.setFadd(m_fAdd);
 | |
|   m_network = settings.value("NetworkInput",true).toBool();
 | |
|   m_fs96000 = settings.value("FSam96000",true).toBool();
 | |
|   m_nDevIn = settings.value("SoundInIndex", 0).toInt();
 | |
|   m_paInDevice = settings.value("paInDevice",0).toInt();
 | |
|   m_nDevOut = settings.value("SoundOutIndex", 0).toInt();
 | |
|   m_paOutDevice = settings.value("paOutDevice",0).toInt();
 | |
|   m_IQswap = settings.value("IQswap",false).toBool();
 | |
|   m_10db = settings.value("Plus10dB",false).toBool();
 | |
|   m_initIQplus = settings.value("InitIQplus",false).toBool();
 | |
|   m_bIQxt = settings.value("IQxt",false).toBool();
 | |
|   m_udpPort = settings.value("UDPport",50004).toInt();
 | |
|   soundInThread.setSwapIQ(m_IQswap);
 | |
|   soundInThread.set10db(m_10db);
 | |
|   soundInThread.setPort(m_udpPort);
 | |
|   ui->actionCuteSDR->setChecked(settings.value(
 | |
|                                   "PaletteCuteSDR",true).toBool());
 | |
|   ui->actionLinrad->setChecked(settings.value(
 | |
|                                  "PaletteLinrad",false).toBool());
 | |
|   m_mode=settings.value("Mode","JT65B").toString();
 | |
|   m_modeJT65=settings.value("nModeJT65",2).toInt();
 | |
|   if(m_modeJT65==0) ui->actionNoJT65->setChecked(true);
 | |
|   if(m_modeJT65==1) ui->actionJT65A->setChecked(true);
 | |
|   if(m_modeJT65==2) ui->actionJT65B->setChecked(true);
 | |
|   if(m_modeJT65==3) ui->actionJT65C->setChecked(true);
 | |
| 
 | |
|   m_modeQ65=settings.value("nModeQ65",2).toInt();
 | |
|   m_modeTx=settings.value("TxMode","JT65").toString();
 | |
|   if(m_modeQ65==0) ui->actionNoQ65->setChecked(true);
 | |
|   if(m_modeQ65==1) ui->actionQ65A->setChecked(true);
 | |
|   if(m_modeQ65==2) ui->actionQ65B->setChecked(true);
 | |
|   if(m_modeQ65==3) ui->actionQ65C->setChecked(true);
 | |
|   if(m_modeQ65==4) ui->actionQ65D->setChecked(true);
 | |
|   if(m_modeQ65==5) ui->actionQ65E->setChecked(true);
 | |
|   if(m_modeTx=="JT65")  ui->pbTxMode->setText("Tx JT65   #");
 | |
|   if(m_modeTx=="Q65") ui->pbTxMode->setText("Tx Q65  :");
 | |
| 
 | |
|   ui->actionNone->setChecked(settings.value("SaveNone",true).toBool());
 | |
|   ui->actionSave_all->setChecked(settings.value("SaveAll",false).toBool());
 | |
|   m_saveAll=ui->actionSave_all->isChecked();
 | |
|   m_ndepth=settings.value("NDepth",0).toInt();
 | |
|   m_onlyEME=settings.value("NEME",false).toBool();
 | |
|   ui->actionOnly_EME_calls->setChecked(m_onlyEME);
 | |
|   m_kb8rq=settings.value("KB8RQ",false).toBool();
 | |
|   ui->actionF4_sets_Tx6->setChecked(m_kb8rq);
 | |
|   m_NB=settings.value("NB",false).toBool();
 | |
|   ui->NBcheckBox->setChecked(m_NB);
 | |
|   m_NBslider=settings.value("NBslider",40).toInt();
 | |
|   ui->NBslider->setValue(m_NBslider);
 | |
|   m_gainx=settings.value("GainX",1.0).toFloat();
 | |
|   m_gainy=settings.value("GainY",1.0).toFloat();
 | |
|   m_phasex=settings.value("PhaseX",0.0).toFloat();
 | |
|   m_phasey=settings.value("PhaseY",0.0).toFloat();
 | |
|   m_mult570=settings.value("Mult570",2).toInt();
 | |
|   m_mult570Tx=settings.value("Mult570Tx",1).toInt();
 | |
|   m_cal570=settings.value("Cal570",0.0).toDouble();
 | |
|   m_TxOffset=settings.value("TxOffset",130.9).toDouble();
 | |
|   m_colors=settings.value("Colors","000066ff0000ffff00969696646464").toString();
 | |
|   settings.endGroup();
 | |
| 
 | |
|   if(!ui->actionLinrad->isChecked() && !ui->actionCuteSDR->isChecked() &&
 | |
|     !ui->actionAFMHot->isChecked() && !ui->actionBlue->isChecked()) {
 | |
|     on_actionLinrad_triggered();
 | |
|     ui->actionLinrad->setChecked(true);
 | |
|   }
 | |
|   if(m_ndepth==0) ui->actionNo_Deep_Search->setChecked(true);
 | |
|   if(m_ndepth==1) ui->actionNormal_Deep_Search->setChecked(true);
 | |
|   if(m_ndepth==2) ui->actionAggressive_Deep_Search->setChecked(true);
 | |
| }
 | |
| 
 | |
| //-------------------------------------------------------------- dataSink()
 | |
| void MainWindow::dataSink(int k)
 | |
| {
 | |
|   static float s[NFFT],splot[NFFT];
 | |
|   static int n=0;
 | |
|   static int ihsym=0;
 | |
|   static int nzap=0;
 | |
|   static int ntrz=0;
 | |
|   static int nkhz;
 | |
|   static int nfsample=96000;
 | |
|   static int nxpol=0;
 | |
|   static float fgreen;
 | |
|   static int ndiskdat;
 | |
|   static int nb;
 | |
|   static int nadj=0;
 | |
|   static float px=0.0,py=0.0;
 | |
|   static uchar lstrong[1024];
 | |
|   static float rejectx;
 | |
|   static float rejecty;
 | |
|   static float slimit;
 | |
| 
 | |
| 
 | |
|   if(m_diskData) {
 | |
|     ndiskdat=1;
 | |
|     datcom_.ndiskdat=1;
 | |
|   } else {
 | |
|     ndiskdat=0;
 | |
|     datcom_.ndiskdat=0;
 | |
|   }
 | |
| // Get x and y power, polarized spectrum, nkhz, and ihsym
 | |
|   nb=0;
 | |
|   if(m_NB) nb=1;
 | |
|   nfsample=96000;
 | |
|   if(!m_fs96000) nfsample=95238;
 | |
|   nxpol=0;
 | |
|   if(m_xpol) nxpol=1;
 | |
|   fgreen=(float)g_pWideGraph->fGreen();
 | |
|   nadj++;
 | |
|   if(m_adjustIQ==0) nadj=0;
 | |
|   symspec_(&k, &nxpol, &ndiskdat, &nb, &m_NBslider, &m_dPhi,
 | |
|            &nfsample, &fgreen, &m_adjustIQ, &m_applyIQcal,
 | |
|            &m_gainx, &m_gainy, &m_phasex, &m_phasey, &rejectx, &rejecty,
 | |
|            &px, &py, s, &nkhz, &ihsym, &nzap, &slimit, lstrong);
 | |
|   QString t;
 | |
|   m_pctZap=nzap/178.3;
 | |
|   if(m_xpol) {
 | |
|     lab4->setText (
 | |
|                   QString {" Rx noise: %1  %2 %3 %% "}
 | |
|                      .arg (px, 5, 'f', 1)
 | |
|                      .arg (py, 5, 'f', 1)
 | |
|                      .arg (m_pctZap, 5, 'f', 1)
 | |
|                   );
 | |
|   } else {
 | |
|     lab4->setText (
 | |
|                   QString {" Rx noise: %1  %2 %% "}
 | |
|                   .arg (px, 5, 'f', 1)
 | |
|                   .arg (m_pctZap, 5, 'f', 1)
 | |
|                   );
 | |
|   }
 | |
|   xSignalMeter->setValue(px);                   // Update the signal meters
 | |
|   ySignalMeter->setValue(py);
 | |
|   if(m_monitoring || m_diskData) {
 | |
|     g_pWideGraph->dataSink2(s,nkhz,ihsym,m_diskData,lstrong);
 | |
|   }
 | |
| 
 | |
|   if(nadj == 10) {
 | |
|     if(m_xpol) {
 | |
|       ui->decodedTextBrowser->append (
 | |
|                                       QString {"Amp: %1 %2   Phase: %3 %4"}
 | |
|                                          .arg (m_gainx, 6, 'f', 4).arg (m_gainy, 6, 'f', 4)
 | |
|                                          .arg (m_phasex, 6, 'f', 4)
 | |
|                                          .arg (m_phasey, 6, 'f', 4)
 | |
|                                       );
 | |
|     } else {
 | |
|       ui->decodedTextBrowser->append(
 | |
|                                      QString {"Amp: %1   Phase: %1"}
 | |
|                                         .arg (m_gainx, 6, 'f', 4)
 | |
|                                         .arg (m_phasex, 6, 'f', 4)
 | |
|                                      );
 | |
|     }
 | |
|     ui->decodedTextBrowser->append(t);
 | |
|     m_adjustIQ=0;
 | |
|   }
 | |
| 
 | |
|   //Average over specified number of spectra
 | |
|   if (n==0) {
 | |
|     for (int i=0; i<NFFT; i++)
 | |
|       splot[i]=s[i];
 | |
|   } else {
 | |
|     for (int i=0; i<NFFT; i++)
 | |
|       splot[i] += s[i];
 | |
|   }
 | |
|   n++;
 | |
| 
 | |
|   if (n>=m_waterfallAvg) {
 | |
|     for (int i=0; i<NFFT; i++) {
 | |
|         splot[i] /= n;                           //Normalize the average
 | |
|     }
 | |
| 
 | |
| // Time according to this computer
 | |
|     qint64 ms = QDateTime::currentMSecsSinceEpoch() % 86400000;
 | |
|     int ntr = (ms/1000) % m_TRperiod;
 | |
|     if((m_diskData && ihsym <= m_waterfallAvg) || (!m_diskData && ntr<ntrz)) {
 | |
|       for (int i=0; i<NFFT; i++) {
 | |
|         splot[i] = 1.e30;
 | |
|       }
 | |
|     }
 | |
|     ntrz=ntr;
 | |
|     n=0;
 | |
|   }
 | |
|   if(ihsym == 280) {
 | |
|     datcom_.newdat=1;
 | |
|     datcom_.nagain=0;
 | |
|     QDateTime t = QDateTime::currentDateTimeUtc();
 | |
|     m_dateTime=t.toString("yyyy-MMM-dd hh:mm");
 | |
|     decode();                                           //Start the decoder
 | |
|     if(m_saveAll and !m_diskData) {
 | |
|       QString fname=m_saveDir + "/" + t.date().toString("yyMMdd") + "_" +
 | |
|           t.time().toString("hhmm");
 | |
|       if(m_xpol) fname += ".tf2";
 | |
|       if(!m_xpol) fname += ".iq";
 | |
|       *future2 = QtConcurrent::run(savetf2, fname, m_xpol);
 | |
|       watcher2->setFuture(*future2);
 | |
|     }
 | |
|   }
 | |
|   soundInThread.m_dataSinkBusy=false;
 | |
| }
 | |
| 
 | |
| void MainWindow::showSoundInError(const QString& errorMsg)
 | |
|  {QMessageBox::critical(this, tr("Error in SoundIn"), errorMsg);}
 | |
| 
 | |
| void MainWindow::showStatusMessage(const QString& statusMsg)
 | |
|  {statusBar()->showMessage(statusMsg);}
 | |
| 
 | |
| void MainWindow::on_actionDeviceSetup_triggered()               //Setup Dialog
 | |
| {
 | |
|   DevSetup dlg(this);
 | |
|   dlg.m_myCall=m_myCall;
 | |
|   dlg.m_myGrid=m_myGrid;
 | |
|   dlg.m_idInt=m_idInt;
 | |
|   dlg.m_pttPort=m_pttPort;
 | |
|   dlg.m_astroFont=m_astroFont;
 | |
|   dlg.m_xpol=m_xpol;
 | |
|   dlg.m_xpolx=m_xpolx;
 | |
|   dlg.m_saveDir=m_saveDir;
 | |
|   dlg.m_azelDir=m_azelDir;
 | |
|   dlg.m_editorCommand=m_editorCommand;
 | |
|   dlg.m_dxccPfx=m_dxccPfx;
 | |
|   dlg.m_timeout=m_timeout;
 | |
|   dlg.m_dPhi=m_dPhi;
 | |
|   dlg.m_fCal=m_fCal;
 | |
|   dlg.m_fAdd=m_fAdd;
 | |
|   dlg.m_network=m_network;
 | |
|   dlg.m_fs96000=m_fs96000;
 | |
|   dlg.m_nDevIn=m_nDevIn;
 | |
|   dlg.m_nDevOut=m_nDevOut;
 | |
|   dlg.m_udpPort=m_udpPort;
 | |
|   dlg.m_IQswap=m_IQswap;
 | |
|   dlg.m_10db=m_10db;
 | |
|   dlg.m_initIQplus=m_initIQplus;
 | |
|   dlg.m_bIQxt=m_bIQxt;
 | |
|   dlg.m_cal570=m_cal570;
 | |
|   dlg.m_TxOffset=m_TxOffset;
 | |
|   dlg.m_mult570=m_mult570;
 | |
|   dlg.m_mult570Tx=m_mult570Tx;
 | |
|   dlg.m_colors=m_colors;
 | |
| 
 | |
|   dlg.initDlg();
 | |
|   if(dlg.exec() == QDialog::Accepted) {
 | |
|     m_myCall=dlg.m_myCall;
 | |
|     m_myGrid=dlg.m_myGrid;
 | |
|     m_idInt=dlg.m_idInt;
 | |
|     m_pttPort=dlg.m_pttPort;
 | |
|     m_astroFont=dlg.m_astroFont;
 | |
|     if(g_pAstro->isVisible()) g_pAstro->setFontSize(m_astroFont);
 | |
|     m_xpol=dlg.m_xpol;
 | |
|     ui->actionFind_Delta_Phi->setEnabled(m_xpol);
 | |
|     m_xpolx=dlg.m_xpolx;
 | |
|     m_saveDir=dlg.m_saveDir;
 | |
|     m_azelDir=dlg.m_azelDir;
 | |
|     m_editorCommand=dlg.m_editorCommand;
 | |
|     m_dxccPfx=dlg.m_dxccPfx;
 | |
|     m_timeout=dlg.m_timeout;
 | |
|     m_dPhi=dlg.m_dPhi;
 | |
|     m_fCal=dlg.m_fCal;
 | |
|     m_fAdd=dlg.m_fAdd;
 | |
|     g_pWideGraph->setFcal(m_fCal);
 | |
|     m_fs96000=dlg.m_fs96000;
 | |
|     m_network=dlg.m_network;
 | |
|     m_nDevIn=dlg.m_nDevIn;
 | |
|     m_paInDevice=dlg.m_paInDevice;
 | |
|     m_nDevOut=dlg.m_nDevOut;
 | |
|     m_paOutDevice=dlg.m_paOutDevice;
 | |
|     m_udpPort=dlg.m_udpPort;
 | |
|     m_IQswap=dlg.m_IQswap;
 | |
|     m_10db=dlg.m_10db;
 | |
|     m_initIQplus=dlg.m_initIQplus;
 | |
|     m_bIQxt=dlg.m_bIQxt;
 | |
|     m_colors=dlg.m_colors;
 | |
|     g_pMessages->setColors(m_colors);
 | |
|     g_pBandMap->setColors(m_colors);
 | |
|     m_cal570=dlg.m_cal570;
 | |
|     m_TxOffset=dlg.m_TxOffset;
 | |
|     m_mult570Tx=dlg.m_mult570Tx;
 | |
|     g_pWideGraph->m_mult570=m_mult570;
 | |
|     g_pWideGraph->m_mult570Tx=m_mult570Tx;
 | |
|     g_pWideGraph->m_cal570=m_cal570;
 | |
|     soundInThread.setSwapIQ(m_IQswap);
 | |
|     soundInThread.set10db(m_10db);
 | |
| 
 | |
|     if(dlg.m_restartSoundIn) {
 | |
|       soundInThread.quit();
 | |
|       soundInThread.wait(1000);
 | |
|       soundInThread.setNetwork(m_network);
 | |
|       if(m_fs96000) soundInThread.setRate(96000.0);
 | |
|       if(!m_fs96000) soundInThread.setRate(95238.1);
 | |
|       soundInThread.setFadd(m_fAdd);
 | |
|       if(!m_xpol) soundInThread.setNrx(1);
 | |
|       if(m_xpol) soundInThread.setNrx(2);
 | |
|       soundInThread.setInputDevice(m_paInDevice);
 | |
|       soundInThread.start(QThread::HighestPriority);
 | |
|     }
 | |
| 
 | |
|     if(dlg.m_restartSoundOut) {
 | |
|       soundOutThread.quitExecution=true;
 | |
|       soundOutThread.wait(1000);
 | |
|       soundOutThread.setOutputDevice(m_paOutDevice);
 | |
| //      soundOutThread.start(QThread::HighPriority);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_monitorButton_clicked()                  //Monitor
 | |
| {
 | |
|   m_monitoring=true;
 | |
|   soundInThread.setMonitoring(true);
 | |
|   m_diskData=false;
 | |
| }
 | |
| void MainWindow::on_actionLinrad_triggered()                 //Linrad palette
 | |
| {
 | |
|   if(g_pWideGraph != NULL) g_pWideGraph->setPalette("Linrad");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionCuteSDR_triggered()                //CuteSDR palette
 | |
| {
 | |
|   if(g_pWideGraph != NULL) g_pWideGraph->setPalette("CuteSDR");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionAFMHot_triggered()
 | |
| {
 | |
|   if(g_pWideGraph != NULL) g_pWideGraph->setPalette("AFMHot");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionBlue_triggered()
 | |
| {
 | |
|   if(g_pWideGraph != NULL) g_pWideGraph->setPalette("Blue");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionAbout_triggered()                  //Display "About"
 | |
| {
 | |
|   CAboutDlg dlg(this,Program_Title_Version);
 | |
|   dlg.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_autoButton_clicked()                     //Auto
 | |
| {
 | |
|   m_auto = !m_auto;
 | |
|   if(m_auto) {
 | |
|     ui->autoButton->setStyleSheet(m_pbAutoOn_style);
 | |
|     ui->autoButton->setText("Auto is ON");
 | |
|   } else {
 | |
|     btxok=false;
 | |
|     ui->autoButton->setStyleSheet("");
 | |
|     ui->autoButton->setText("Auto is OFF");
 | |
|     on_monitorButton_clicked();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_stopTxButton_clicked()                    //Stop Tx
 | |
| {
 | |
|   if(m_auto) on_autoButton_clicked();
 | |
|   btxok=false;
 | |
| }
 | |
| 
 | |
| void MainWindow::keyPressEvent( QKeyEvent *e )                //keyPressEvent
 | |
| {
 | |
|   switch(e->key())
 | |
|   {
 | |
|   case Qt::Key_F3:
 | |
|     m_txMute=!m_txMute;
 | |
|     break;
 | |
|   case Qt::Key_F4:
 | |
|     ui->dxCallEntry->setText("");
 | |
|     ui->dxGridEntry->setText("");
 | |
|     if(m_kb8rq) {
 | |
|       m_ntx=6;
 | |
|       ui->txrb6->setChecked(true);
 | |
|     }
 | |
|     break;
 | |
|   case Qt::Key_F6:
 | |
|     if(e->modifiers() & Qt::ShiftModifier) {
 | |
|       on_actionDecode_remaining_files_in_directory_triggered();
 | |
|     }
 | |
|     break;
 | |
|   case Qt::Key_F11:
 | |
|     if(e->modifiers() & Qt::ShiftModifier) {
 | |
|     } else {
 | |
|       int n0=g_pWideGraph->DF();
 | |
|       int n=(n0 + 10000) % 5;
 | |
|       if(n==0) n=5;
 | |
|       g_pWideGraph->setDF(n0-n);
 | |
|     }
 | |
|     break;
 | |
|   case Qt::Key_F12:
 | |
|     if(e->modifiers() & Qt::ShiftModifier) {
 | |
|     } else {
 | |
|       int n0=g_pWideGraph->DF();
 | |
|       int n=(n0 + 10000) % 5;
 | |
|       if(n==0) n=5;
 | |
|       g_pWideGraph->setDF(n0+n);
 | |
|     }
 | |
|     break;
 | |
|   case Qt::Key_G:
 | |
|     if(e->modifiers() & Qt::AltModifier) {
 | |
|       genStdMsgs("");
 | |
|     }
 | |
|     break;
 | |
|   case Qt::Key_L:
 | |
|     if(e->modifiers() & Qt::ControlModifier) {
 | |
|       lookup();
 | |
|       genStdMsgs("");
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::bumpDF(int n)                                  //bumpDF()
 | |
| {
 | |
|   if(n==11) {
 | |
|     int n0=g_pWideGraph->DF();
 | |
|     int n=(n0 + 10000) % 5;
 | |
|     if(n==0) n=5;
 | |
|     g_pWideGraph->setDF(n0-n);
 | |
|   }
 | |
|   if(n==12) {
 | |
|     int n0=g_pWideGraph->DF();
 | |
|     int n=(n0 + 10000) % 5;
 | |
|     if(n==0) n=5;
 | |
|     g_pWideGraph->setDF(n0+n);
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool MainWindow::eventFilter(QObject *object, QEvent *event)  //eventFilter()
 | |
| {
 | |
|   if (event->type() == QEvent::KeyPress) {
 | |
|     //Use the event in parent using its keyPressEvent()
 | |
|     QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
 | |
|     MainWindow::keyPressEvent(keyEvent);
 | |
|     return QObject::eventFilter(object, event);
 | |
|   }
 | |
|   return QObject::eventFilter(object, event);
 | |
| }
 | |
| 
 | |
| void MainWindow::createStatusBar()                           //createStatusBar
 | |
| {
 | |
|   lab1 = new QLabel("Receiving");
 | |
|   lab1->setAlignment(Qt::AlignHCenter);
 | |
|   lab1->setMinimumSize(QSize(80,10));
 | |
|   lab1->setStyleSheet("QLabel{background-color: #00ff00}");
 | |
|   lab1->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab1);
 | |
| 
 | |
|   lab2 = new QLabel("QSO freq:  125");
 | |
|   lab2->setAlignment(Qt::AlignHCenter);
 | |
|   lab2->setMinimumSize(QSize(90,10));
 | |
|   lab2->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab2);
 | |
| 
 | |
|   lab3 = new QLabel("QSO DF:   0");
 | |
|   lab3->setAlignment(Qt::AlignHCenter);
 | |
|   lab3->setMinimumSize(QSize(80,10));
 | |
|   lab3->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab3);
 | |
| 
 | |
|   lab4 = new QLabel("");
 | |
|   lab4->setAlignment(Qt::AlignHCenter);
 | |
|   lab4->setMinimumSize(QSize(80,10));
 | |
|   lab4->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab4);
 | |
| 
 | |
|   lab5 = new QLabel("");
 | |
|   lab5->setAlignment(Qt::AlignHCenter);
 | |
|   lab5->setMinimumSize(QSize(50,10));
 | |
|   lab5->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab5);
 | |
| 
 | |
|   lab6 = new QLabel("");
 | |
|   lab6->setAlignment(Qt::AlignHCenter);
 | |
|   lab6->setMinimumSize(QSize(50,10));
 | |
|   lab6->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab6);
 | |
| 
 | |
|   lab7 = new QLabel("Avg: 0");
 | |
|   lab7->setAlignment(Qt::AlignHCenter);
 | |
|   lab7->setMinimumSize(QSize(50,10));
 | |
|   lab7->setFrameStyle(QFrame::Panel | QFrame::Sunken);
 | |
|   statusBar()->addWidget(lab7);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tolSpinBox_valueChanged(int i)             //tolSpinBox
 | |
| {
 | |
|   static int ntol[] = {10,20,50,100,200,500,1000};
 | |
|   m_tol=ntol[i];
 | |
|   g_pWideGraph->setTol(m_tol);
 | |
|   ui->labTol1->setText(QString::number(ntol[i]));
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionExit_triggered()                     //Exit()
 | |
| {
 | |
|   OnExit();
 | |
| }
 | |
| 
 | |
| void MainWindow::closeEvent(QCloseEvent*)
 | |
| {
 | |
|   OnExit();
 | |
| }
 | |
| 
 | |
| void MainWindow::OnExit()
 | |
| {
 | |
|   g_pWideGraph->saveSettings();
 | |
|   m_killAll=true;
 | |
|   QFile quitFile(m_appDir + "/.quit");
 | |
|   quitFile.open(QIODevice::ReadWrite);
 | |
|   QFile lockFile(m_appDir + "/.lock");
 | |
|   lockFile.remove();                      // Allow m65 to terminate
 | |
|   bool b=proc_m65.waitForFinished(1000);
 | |
|   if(!b) proc_m65.kill();
 | |
|   quitFile.remove();
 | |
|   mem_m65.detach();
 | |
|   qApp->exit(0);                          // Exit the event loop
 | |
| }
 | |
| 
 | |
| void MainWindow::on_stopButton_clicked()                       //stopButton
 | |
| {
 | |
|   m_monitoring=false;
 | |
|   soundInThread.setMonitoring(m_monitoring);
 | |
|   m_loopall=false;  
 | |
| }
 | |
| 
 | |
| void MainWindow::msgBox(QString t)                             //msgBox
 | |
| {
 | |
|   msgBox0.setText(t);
 | |
|   msgBox0.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::stub()                                        //stub()
 | |
| {
 | |
|   msgBox("Not yet implemented.");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionOnline_Users_Guide_triggered()      //Display manual
 | |
| {
 | |
|   QDesktopServices::openUrl(QUrl(
 | |
|   "http://www.physics.princeton.edu/pulsar/K1JT/MAP65_Users_Guide.pdf",
 | |
|                               QUrl::TolerantMode));
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionAstro_Data_triggered()             //Display Astro
 | |
| {
 | |
|   if(g_pAstro==NULL) {
 | |
|     g_pAstro = new Astro(0);
 | |
|     g_pAstro->setWindowTitle("Astronomical Data");
 | |
|     Qt::WindowFlags flags = Qt::Dialog | Qt::WindowCloseButtonHint |
 | |
|         Qt::WindowMinimizeButtonHint;
 | |
|     g_pAstro->setWindowFlags(flags);
 | |
|     g_pAstro->setGeometry(m_astroGeom);
 | |
|   }
 | |
|   g_pAstro->show();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionWide_Waterfall_triggered()      //Display Waterfalls
 | |
| {
 | |
|   if(g_pWideGraph==NULL) {
 | |
|     g_pWideGraph = new WideGraph(0);
 | |
|     g_pWideGraph->setWindowTitle("Wide Graph");
 | |
|     g_pWideGraph->setGeometry(m_wideGraphGeom);
 | |
|     Qt::WindowFlags flags = Qt::WindowCloseButtonHint |
 | |
|         Qt::WindowMinimizeButtonHint;
 | |
|     g_pWideGraph->setWindowFlags(flags);
 | |
|     connect(g_pWideGraph, SIGNAL(freezeDecode2(int)),this,
 | |
|             SLOT(freezeDecode(int)));
 | |
|     connect(g_pWideGraph, SIGNAL(f11f12(int)),this,
 | |
|             SLOT(bumpDF(int)));
 | |
|   }
 | |
|   g_pWideGraph->show();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionBand_Map_triggered()              //Display BandMap
 | |
| {
 | |
|   if(g_pBandMap==NULL) {
 | |
|     g_pBandMap = new BandMap(0);
 | |
|     g_pBandMap->setWindowTitle("Band Map");
 | |
|     Qt::WindowFlags flags = Qt::Dialog | Qt::WindowCloseButtonHint |
 | |
|         Qt::WindowMinimizeButtonHint;
 | |
|     g_pBandMap->setWindowFlags(flags);
 | |
|     g_pBandMap->setGeometry(m_bandMapGeom);
 | |
|   }
 | |
|   g_pBandMap->show();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionMessages_triggered()              //Display Messages
 | |
| {
 | |
|   if(g_pMessages==NULL) {
 | |
|     g_pMessages = new Messages(0);
 | |
|     g_pMessages->setWindowTitle("Messages");
 | |
|     Qt::WindowFlags flags = Qt::Dialog | Qt::WindowCloseButtonHint |
 | |
|         Qt::WindowMinimizeButtonHint;
 | |
|     g_pMessages->setWindowFlags(flags);
 | |
|     g_pMessages->setGeometry(m_messagesGeom);
 | |
|     connect(g_pMessages, SIGNAL(click2OnCallsign(QString, QString)),this,
 | |
|             SLOT(doubleClickOnMessages(QString, QString)));
 | |
|   }
 | |
|   g_pMessages->show();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionOpen_triggered()                     //Open File
 | |
| {
 | |
|   m_monitoring=false;
 | |
|   soundInThread.setMonitoring(m_monitoring);
 | |
|   QString fname;
 | |
|   if(m_xpol) {
 | |
|     fname=QFileDialog::getOpenFileName(this, "Open File", m_path,
 | |
|                                        "MAP65 Files (*.tf2)");
 | |
|   } else {
 | |
|     fname=QFileDialog::getOpenFileName(this, "Open File", m_path,
 | |
|                                        "MAP65 Files (*.iq)");
 | |
|   }
 | |
|   if(fname != "") {
 | |
|     m_path=fname;
 | |
|     int i;
 | |
|     i=fname.indexOf(".iq") - 11;
 | |
|     if(m_xpol) i=fname.indexOf(".tf2") - 11;
 | |
|     if(i>=0) {
 | |
|       lab1->setStyleSheet("QLabel{background-color: #66ff66}");
 | |
|       lab1->setText(" " + fname.mid(i,15) + " ");
 | |
|     }
 | |
|     on_stopButton_clicked();
 | |
|     m_diskData=true;
 | |
|     int dbDgrd=0;
 | |
|     if(m_myCall=="K1JT" and m_idInt<0) dbDgrd=m_idInt;
 | |
|     *future1 = QtConcurrent::run(getfile, fname, m_xpol, dbDgrd);
 | |
|     watcher1->setFuture(*future1);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionOpen_next_in_directory_triggered()   //Open Next
 | |
| {
 | |
|   int i,len;
 | |
|   QFileInfo fi(m_path);
 | |
|   QStringList list;
 | |
|   if(m_xpol) {
 | |
|       list= fi.dir().entryList().filter(".tf2");
 | |
|   } else {
 | |
|       list= fi.dir().entryList().filter(".iq");
 | |
|   }
 | |
|   for (i = 0; i < list.size()-1; ++i) {
 | |
|     if(i==list.size()-2) m_loopall=false;
 | |
|     len=list.at(i).length();
 | |
|     if(list.at(i)==m_path.right(len)) {
 | |
|       int n=m_path.length();
 | |
|       QString fname=m_path.replace(n-len,len,list.at(i+1));
 | |
|       m_path=fname;
 | |
|       int i;
 | |
|       i=fname.indexOf(".iq") - 11;
 | |
|       if(m_xpol) i=fname.indexOf(".tf2") - 11;
 | |
|       if(i>=0) {
 | |
|         lab1->setStyleSheet("QLabel{background-color: #66ff66}");
 | |
|         lab1->setText(" " + fname.mid(i,len) + " ");
 | |
|       }
 | |
|       m_diskData=true;
 | |
|       int dbDgrd=0;
 | |
|       if(m_myCall=="K1JT" and m_idInt<0) dbDgrd=m_idInt;
 | |
|       *future1 = QtConcurrent::run(getfile, fname, m_xpol, dbDgrd);
 | |
|       watcher1->setFuture(*future1);
 | |
|       return;
 | |
|     }
 | |
|   }
 | |
| }
 | |
|                                                    //Open all remaining files
 | |
| void MainWindow::on_actionDecode_remaining_files_in_directory_triggered()
 | |
| {
 | |
|   m_loopall=true;
 | |
|   on_actionOpen_next_in_directory_triggered();
 | |
| }
 | |
| 
 | |
| void MainWindow::diskDat()                                   //diskDat()
 | |
| {
 | |
|   double hsym;
 | |
|   //These may be redundant??
 | |
|   m_diskData=true;
 | |
|   datcom_.newdat=1;
 | |
| 
 | |
|   if(m_fs96000) hsym=2048.0*96000.0/11025.0;   //Samples per JT65 half-symbol
 | |
|   if(!m_fs96000) hsym=2048.0*95238.1/11025.0;
 | |
|   for(int i=0; i<284; i++) {           // Do the half-symbol FFTs
 | |
|     int k = i*hsym + 2048.5;
 | |
|     dataSink(k);
 | |
|     if(i%10 == 0) qApp->processEvents();       //Keep the GUI responsive
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::diskWriteFinished()                      //diskWriteFinished
 | |
| {
 | |
| //  qDebug() << "diskWriteFinished";
 | |
| }
 | |
|                                                         //Delete ../save/*.tf2
 | |
| void MainWindow::on_actionDelete_all_tf2_files_in_SaveDir_triggered()
 | |
| {
 | |
|   int i;
 | |
|   QString fname;
 | |
|   int ret = QMessageBox::warning(this, "Confirm Delete",
 | |
|       "Are you sure you want to delete all *.tf2 and *.iq files in\n" +
 | |
|        QDir::toNativeSeparators(m_saveDir) + " ?",
 | |
|        QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
 | |
|   if(ret==QMessageBox::Yes) {
 | |
|     QDir dir(m_saveDir);
 | |
|     QStringList files=dir.entryList(QDir::Files);
 | |
|     QList<QString>::iterator f;
 | |
|     for(f=files.begin(); f!=files.end(); ++f) {
 | |
|       fname=*f;
 | |
|       i=(fname.indexOf(".tf2"));
 | |
|       if(i==11) dir.remove(fname);
 | |
|       i=(fname.indexOf(".iq"));
 | |
|       if(i==11) dir.remove(fname);
 | |
|     }
 | |
|   }
 | |
| }
 | |
|                                           //Clear BandMap and Messages windows
 | |
| void MainWindow::on_actionErase_Band_Map_and_Messages_triggered()
 | |
| {
 | |
|   g_pBandMap->setText("");
 | |
|   g_pMessages->setText("","");
 | |
|   m_map65RxLog |= 4;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionFind_Delta_Phi_triggered()              //Find dPhi
 | |
| {
 | |
|   m_map65RxLog |= 8;
 | |
|   on_DecodeButton_clicked();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionF4_sets_Tx6_triggered()                //F4 sets Tx6
 | |
| {
 | |
|   m_kb8rq = !m_kb8rq;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionOnly_EME_calls_triggered()          //EME calls only
 | |
| {
 | |
|   m_onlyEME = ui->actionOnly_EME_calls->isChecked();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionNo_shorthands_if_Tx1_triggered()
 | |
| {
 | |
|   stub();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionNo_Deep_Search_triggered()          //No Deep Search
 | |
| {
 | |
|   m_ndepth=0;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionNormal_Deep_Search_triggered()      //Normal DS
 | |
| {
 | |
|   m_ndepth=1;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionAggressive_Deep_Search_triggered()  //Aggressive DS
 | |
| {
 | |
|   m_ndepth=2;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionNone_triggered()                    //Save None
 | |
| {
 | |
|   m_saveAll=false;
 | |
| }
 | |
| 
 | |
| // ### Implement "Save Last" here? ###
 | |
| 
 | |
| void MainWindow::on_actionSave_all_triggered()                //Save All
 | |
| {
 | |
|   m_saveAll=true;
 | |
| }
 | |
|                                           //Display list of keyboard shortcuts
 | |
| void MainWindow::on_actionKeyboard_shortcuts_triggered()
 | |
| {
 | |
|   stub();
 | |
| }
 | |
|                                               //Display list of mouse commands
 | |
| void MainWindow::on_actionSpecial_mouse_commands_triggered()
 | |
| {
 | |
|   stub();
 | |
| }
 | |
|                                               //Diaplay list of Add-On pfx/sfx
 | |
| void MainWindow::on_actionAvailable_suffixes_and_add_on_prefixes_triggered()
 | |
| {
 | |
|   stub();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_DecodeButton_clicked()                    //Decode request
 | |
| {
 | |
|   int n=m_sec0%m_TRperiod;
 | |
|   if(m_monitoring and n>47 and (n<52 or m_decoderBusy)) return;
 | |
|   if(!m_decoderBusy) {
 | |
|     datcom_.newdat=0;
 | |
|     datcom_.nagain=1;
 | |
|     decode();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::freezeDecode(int n)                          //freezeDecode()
 | |
| {
 | |
|   if(n==2) {
 | |
|     ui->tolSpinBox->setValue(5);
 | |
|     datcom_.ntol=m_tol;
 | |
|     datcom_.mousedf=0;
 | |
|   } else {
 | |
|     ui->tolSpinBox->setValue(3);
 | |
|     datcom_.ntol=m_tol;
 | |
|   }
 | |
|   if(!m_decoderBusy) {
 | |
|     datcom_.nagain=1;
 | |
|     datcom_.newdat=0;
 | |
|     decode();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::decode()                                       //decode()
 | |
| {
 | |
|   ui->DecodeButton->setStyleSheet(m_pbdecoding_style1);
 | |
|   if(datcom_.nagain==0 && (!m_diskData)) {
 | |
|     qint64 ms = QDateTime::currentMSecsSinceEpoch() % 86400000;
 | |
|     int imin=ms/60000;
 | |
|     int ihr=imin/60;
 | |
|     imin=imin % 60;
 | |
|     datcom_.nutc=100*ihr + imin;
 | |
|   }
 | |
| 
 | |
|   datcom_.idphi=m_dPhi;
 | |
|   datcom_.mousedf=g_pWideGraph->DF();
 | |
|   datcom_.mousefqso=g_pWideGraph->QSOfreq();
 | |
|   datcom_.ndepth=m_ndepth;
 | |
|   datcom_.ndiskdat=0;
 | |
|   if(m_diskData) datcom_.ndiskdat=1;
 | |
|   datcom_.neme=0;
 | |
|   if(ui->actionOnly_EME_calls->isChecked()) datcom_.neme=1;
 | |
| 
 | |
|   int ispan=int(g_pWideGraph->fSpan());
 | |
|   if(ispan%2 == 1) ispan++;
 | |
|   int ifc=int(1000.0*(datcom_.fcenter - int(datcom_.fcenter))+0.5);
 | |
|   int nfa=g_pWideGraph->nStartFreq();
 | |
|   int nfb=nfa+ispan;
 | |
|   int nfshift=nfa + ispan/2 - ifc;
 | |
| 
 | |
|   datcom_.nfa=nfa;
 | |
|   datcom_.nfb=nfb;
 | |
|   datcom_.nfcal=m_fCal;
 | |
|   datcom_.nfshift=nfshift;
 | |
|   datcom_.mcall3=0;
 | |
|   if(m_call3Modified) datcom_.mcall3=1;
 | |
|   datcom_.ntimeout=m_timeout;
 | |
|   datcom_.ntol=m_tol;
 | |
|   datcom_.nxant=0;
 | |
|   if(m_xpolx) datcom_.nxant=1;
 | |
|   if(datcom_.nutc < m_nutc0) m_map65RxLog |= 1;  //Date and Time to all65.txt
 | |
|   m_nutc0=datcom_.nutc;
 | |
|   datcom_.map65RxLog=m_map65RxLog;
 | |
|   datcom_.nfsample=96000;
 | |
|   if(!m_fs96000) datcom_.nfsample=95238;
 | |
|   datcom_.nxpol=0;
 | |
|   if(m_xpol) datcom_.nxpol=1;
 | |
|   datcom_.nmode=10*m_modeQ65 + m_modeJT65;
 | |
|   datcom_.nfast=1;                               //No longer used
 | |
|   datcom_.nsave=m_nsave;
 | |
| 
 | |
|   QString mcall=(m_myCall+"            ").mid(0,12);
 | |
|   QString mgrid=(m_myGrid+"            ").mid(0,6);
 | |
|   QString hcall=(ui->dxCallEntry->text()+"            ").mid(0,12);
 | |
|   QString hgrid=(ui->dxGridEntry->text()+"      ").mid(0,6);
 | |
| 
 | |
|   memcpy(datcom_.mycall, mcall.toLatin1(), 12);
 | |
|   memcpy(datcom_.mygrid, mgrid.toLatin1(), 6);
 | |
|   memcpy(datcom_.hiscall, hcall.toLatin1(), 12);
 | |
|   memcpy(datcom_.hisgrid, hgrid.toLatin1(), 6);
 | |
|   memcpy(datcom_.datetime, m_dateTime.toLatin1(), 20);
 | |
|   memcpy(datcom_.datadir, m_dataDir.toLatin1(),m_dataDir.length());
 | |
|   memcpy(datcom_.tempdir, m_tempDir.toLatin1(),m_tempDir.length());
 | |
|   //newdat=1  ==> this is new data, must do the big FFT
 | |
|   //nagain=1  ==> decode only at fQSO +/- Tol
 | |
| 
 | |
|   char *to = (char*)mem_m65.data();
 | |
|   char *from = (char*) datcom_.d4;
 | |
|   int size=sizeof(datcom_);
 | |
|   if(datcom_.newdat==0) {
 | |
|     int noffset = 4*4*5760000 + 4*4*322*32768 + 4*4*32768;
 | |
|     to += noffset;
 | |
|     from += noffset;
 | |
|     size -= noffset;
 | |
|   }
 | |
| 
 | |
|   memcpy(to, from, qMin(mem_m65.size(), size));
 | |
|   datcom_.nagain=0;
 | |
|   datcom_.ndiskdat=0;
 | |
|   m_call3Modified=false;
 | |
| 
 | |
|   QFile lockFile(m_appDir + "/.lock");       // Allow m65 to start
 | |
|   lockFile.remove();
 | |
|   decodeBusy(true);
 | |
| }
 | |
| 
 | |
| void MainWindow::m65_error()                                     //m65_error
 | |
| {
 | |
|   if(!m_killAll) {
 | |
|     msgBox("Error starting or running\n" + m_appDir + "/m65 -s");
 | |
|     exit(1);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::editor_error()                                 //editor_error
 | |
| {
 | |
|   if(!m_killAll) {
 | |
|     msgBox("Error starting or running\n" + m_appDir + "/" + m_editorCommand);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::readFromStderr()                             //readFromStderr
 | |
| {
 | |
|   QByteArray t=proc_m65.readAllStandardError();
 | |
|   msgBox(t);
 | |
| }
 | |
| 
 | |
| void MainWindow::readFromStdout()                             //readFromStdout
 | |
| {
 | |
|   while(proc_m65.canReadLine())
 | |
|   {
 | |
|     QByteArray t=proc_m65.readLine();
 | |
|     if(t.indexOf("<QuickDecodeDone>") >= 0) {
 | |
|       m_nsum=t.mid(17,4).toInt();
 | |
|       m_nsave=t.mid(21,4).toInt();
 | |
|       lab7->setText (QString {"Avg: %1"}.arg (m_nsum));
 | |
|       if(m_modeQ65>0) g_pWideGraph->setDecodeFinished();
 | |
|     }
 | |
|     if(t.indexOf("<DecodeFinished>") >= 0) {
 | |
|       if(m_widebandDecode) {
 | |
|         g_pMessages->setText(m_messagesText,m_bandmapText);
 | |
|         g_pBandMap->setText(m_bandmapText);
 | |
|         m_widebandDecode=false;
 | |
|       }
 | |
|       QFile lockFile(m_appDir + "/.lock");
 | |
|       lockFile.open(QIODevice::ReadWrite);
 | |
|       ui->DecodeButton->setStyleSheet("");
 | |
|       decodeBusy(false);
 | |
|       m_map65RxLog=0;
 | |
|       m_startAnother=m_loopall;
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if(t.indexOf("!") >= 0) {
 | |
|       int n=t.length();
 | |
|       if(n>=30) ui->decodedTextBrowser->append(t.mid(1,n-3));
 | |
| //      if(n<30) ui->decodedTextBrowser->append(t.mid(1,n-3));  //Write a no-decode JT65 line
 | |
|       n=ui->decodedTextBrowser->verticalScrollBar()->maximum();
 | |
|       ui->decodedTextBrowser->verticalScrollBar()->setValue(n);
 | |
|       m_messagesText="";
 | |
|       m_bandmapText="";
 | |
|     }
 | |
| 
 | |
|     if(t.indexOf("@") >= 0) {
 | |
|       m_messagesText += t.mid(1);
 | |
|       m_widebandDecode=true;
 | |
|     }
 | |
| 
 | |
|     if(t.indexOf("&") >= 0) {
 | |
|       QString q(t);
 | |
|       QString callsign=q.mid(5);
 | |
|       callsign=callsign.mid(0,callsign.indexOf(" "));
 | |
|       if(callsign.length()>2) {
 | |
|         if(m_worked[callsign]) {
 | |
|           q=q.mid(1,4) + "  " + q.mid(5);
 | |
|         } else {
 | |
|           q=q.mid(1,4) + " *" + q.mid(5);
 | |
|         }
 | |
|         m_bandmapText += q;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_EraseButton_clicked()                          //Erase
 | |
| {
 | |
|   ui->decodedTextBrowser->clear();
 | |
| }
 | |
| 
 | |
| void MainWindow::decodeBusy(bool b)                             //decodeBusy()
 | |
| {
 | |
|   m_decoderBusy=b;
 | |
|   ui->DecodeButton->setEnabled(!b);
 | |
|   ui->actionOpen->setEnabled(!b);
 | |
|   ui->actionOpen_next_in_directory->setEnabled(!b);
 | |
|   ui->actionDecode_remaining_files_in_directory->setEnabled(!b);
 | |
| }
 | |
| 
 | |
| //------------------------------------------------------------- //guiUpdate()
 | |
| void MainWindow::guiUpdate()
 | |
| {
 | |
|   static int iptt0=0;
 | |
|   static int iptt=0;
 | |
|   static bool btxok0=false;
 | |
|   static bool bTune0=false;
 | |
|   static bool bMonitoring0=false;
 | |
|   static int nc0=1;
 | |
|   static int nc1=1;
 | |
|   static char msgsent[23];
 | |
|   static int nsendingsh=0;
 | |
|   int khsym=0;
 | |
| 
 | |
|   double tx1=0.0;
 | |
|   double tx2=126.0*4096.0/11025.0 + 1.8;
 | |
|   if(m_modeTx=="Q65") tx2=84.0*6912.0/12000.0 + 1.8;
 | |
| 
 | |
|   if(!m_txFirst) {
 | |
|     tx1 += m_TRperiod;
 | |
|     tx2 += m_TRperiod;
 | |
|   }
 | |
|   qint64 ms = QDateTime::currentMSecsSinceEpoch() % 86400000;
 | |
|   int nsec=ms/1000;
 | |
|   double tsec=0.001*ms;
 | |
|   double t2p=fmod(tsec,120.0);
 | |
|   bool bTxTime = (t2p >= tx1) and (t2p < tx2);
 | |
| 
 | |
|   if(bTune0 and !bTune) {
 | |
|     btxok=false;
 | |
|     m_monitoring=bMonitoring0;
 | |
|     soundInThread.setMonitoring(m_monitoring);
 | |
|   }
 | |
|   if(bTune and !bTune0) bMonitoring0=m_monitoring;
 | |
|   bTune0=bTune;
 | |
| 
 | |
|   if(m_auto or bTune) {
 | |
|     if((bTxTime or bTune) and iptt==0 and !m_txMute) {
 | |
|       int itx=1;
 | |
|       int ierr = ptt_(&m_pttPort,&itx,&iptt);       // Raise PTT
 | |
|       if(ierr != 0) {
 | |
|         on_stopTxButton_clicked();
 | |
|         char s[18];
 | |
|         sprintf(s,"Cannot open COM%d",m_pttPort);
 | |
|         msgBox(s);
 | |
|       }
 | |
| 
 | |
|       if(m_bIQxt) g_pWideGraph->tx570();     // Set Si570 to Tx Freq
 | |
| 
 | |
|       if(!soundOutThread.isRunning()) {
 | |
|         soundOutThread.start(QThread::HighPriority);
 | |
|       }
 | |
|     }
 | |
|     if((!bTxTime and !bTune) or m_txMute) {
 | |
|       btxok=false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
| // Calculate Tx waveform when needed
 | |
|   if((iptt==1 && iptt0==0) || m_restart) {
 | |
|     char message[23];
 | |
|     QByteArray ba;
 | |
|     if(m_ntx == 1) ba=ui->tx1->text().toLocal8Bit();
 | |
|     if(m_ntx == 2) ba=ui->tx2->text().toLocal8Bit();
 | |
|     if(m_ntx == 3) ba=ui->tx3->text().toLocal8Bit();
 | |
|     if(m_ntx == 4) ba=ui->tx4->text().toLocal8Bit();
 | |
|     if(m_ntx == 5) ba=ui->tx5->text().toLocal8Bit();
 | |
|     if(m_ntx == 6) ba=ui->tx6->text().toLocal8Bit();
 | |
| 
 | |
|     ba2msg(ba,message);
 | |
|     int len1=22;
 | |
|     int mode65=m_mode65;
 | |
|     int ichk=0;
 | |
|     int ntxFreq=1000;
 | |
|     int itype=0;
 | |
|     double samfac=1.0;
 | |
| 
 | |
|     if(m_modeTx=="JT65") {
 | |
|       gen65_(message,&mode65,&samfac,&nsendingsh,msgsent,iwave,
 | |
|              &nwave,len1,len1);
 | |
|     } else {
 | |
|       if(m_modeQ65==5) ntxFreq=700;
 | |
|       gen_q65_wave_(message,&ichk,&ntxFreq,&m_modeQ65,&itype,msgsent,iwave,
 | |
|                  &nwave,len1,len1);
 | |
|     }
 | |
|     msgsent[22]=0;
 | |
| 
 | |
|     if(m_restart) {
 | |
|       QFile f("map65_tx.log");
 | |
|       f.open(QIODevice::WriteOnly | QIODevice::Text | QIODevice::Append);
 | |
|       QTextStream out(&f);
 | |
|       out << QDateTime::currentDateTimeUtc().toString("yyyy-MMM-dd hh:mm")
 | |
|           << "  Tx message:  " << QString::fromLatin1(msgsent)
 | |
| #if QT_VERSION >= QT_VERSION_CHECK (5, 15, 0)
 | |
|           << Qt::endl
 | |
| #else
 | |
|           << endl
 | |
| #endif
 | |
|         ;
 | |
|       f.close();
 | |
|     }
 | |
| 
 | |
|     m_restart=false;
 | |
|   }
 | |
| 
 | |
| // If PTT was just raised, start a countdown for raising TxOK:
 | |
|   if(iptt==1 && iptt0==0) nc1=-9;    // TxDelay = 0.8 s
 | |
|   if(nc1 <= 0) nc1++;
 | |
|   if(nc1 == 0) {
 | |
|     xSignalMeter->setValue(0);
 | |
|     ySignalMeter->setValue(0);
 | |
|     m_monitoring=false;
 | |
|     soundInThread.setMonitoring(false);
 | |
|     btxok=true;
 | |
|     m_transmitting=true;
 | |
|     g_pWideGraph->enableSetRxHardware(false);
 | |
| 
 | |
|     QFile f("map65_tx.log");
 | |
|     f.open(QIODevice::WriteOnly | QIODevice::Text | QIODevice::Append);
 | |
|     QTextStream out(&f);
 | |
|     out << QDateTime::currentDateTimeUtc().toString("yyyy-MMM-dd hh:mm")
 | |
|         << "  Tx message:  " << QString::fromLatin1(msgsent)
 | |
| #if QT_VERSION >= QT_VERSION_CHECK (5, 15, 0)
 | |
|         << Qt::endl
 | |
| #else
 | |
|         << endl
 | |
| #endif
 | |
|       ;
 | |
|     f.close();
 | |
|   }
 | |
| 
 | |
| // If btxok was just lowered, start a countdown for lowering PTT
 | |
|   if(!btxok && btxok0 && iptt==1) nc0=-11;  //RxDelay = 1.0 s
 | |
|   btxok0=btxok;
 | |
|   if(nc0 <= 0) nc0++;
 | |
|   if(nc0 == 0) {
 | |
|     if(m_bIQxt) g_pWideGraph->rx570();     // Set Si570 back to Rx Freq
 | |
|     int itx=0;
 | |
|     ptt_(&m_pttPort,&itx,&iptt);       // Lower PTT
 | |
|     if(!m_txMute) {
 | |
|       soundOutThread.quitExecution=true;\
 | |
|     }
 | |
|     m_transmitting=false;
 | |
|     g_pWideGraph->enableSetRxHardware(true);
 | |
|     if(m_auto) {
 | |
|       m_monitoring=true;
 | |
|       soundInThread.setMonitoring(m_monitoring);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if(iptt == 0 && !btxok) {
 | |
|     // sending=""
 | |
|     // nsendingsh=0
 | |
|   }
 | |
| 
 | |
|   if(m_monitoring) {
 | |
|     ui->monitorButton->setStyleSheet(m_pbmonitor_style);
 | |
|   } else {
 | |
|     ui->monitorButton->setStyleSheet("");
 | |
|   }
 | |
| 
 | |
|   lab2->setText("QSO Freq:  " + QString::number(g_pWideGraph->QSOfreq()));
 | |
|   lab3->setText("QSO DF:  " + QString::number(g_pWideGraph->DF()));
 | |
| 
 | |
|   g_pWideGraph->updateFreqLabel();
 | |
| 
 | |
|   if(m_startAnother) {
 | |
|     m_startAnother=false;
 | |
|     on_actionOpen_next_in_directory_triggered();
 | |
|   }
 | |
|   if(m_modeQ65==0 and m_modeTx=="Q65") on_pbTxMode_clicked();
 | |
|   if(m_modeJT65==0  and m_modeTx=="JT65")  on_pbTxMode_clicked();
 | |
| 
 | |
|   if(nsec != m_sec0) {                                     //Once per second
 | |
| //    qDebug() << "A" << nsec%60 << m_mode65 << m_modeQ65 << m_modeTx;
 | |
|     soundInThread.setForceCenterFreqMHz(g_pWideGraph->m_dForceCenterFreq);
 | |
|     soundInThread.setForceCenterFreqBool(g_pWideGraph->m_bForceCenterFreq);
 | |
| 
 | |
|     if(m_pctZap>30.0 and !m_transmitting) {
 | |
|       lab4->setStyleSheet("QLabel{background-color: #ff0000}");
 | |
|     } else {
 | |
|       lab4->setStyleSheet("");
 | |
|     }
 | |
| 
 | |
|     if(m_transmitting) {
 | |
|       if(nsendingsh==1) {
 | |
|         lab1->setStyleSheet("QLabel{background-color: #66ffff}");
 | |
|       } else if(nsendingsh==-1) {
 | |
|         lab1->setStyleSheet("QLabel{background-color: #ffccff}");
 | |
|       } else {
 | |
|         lab1->setStyleSheet("QLabel{background-color: #ffff33}");
 | |
|       }
 | |
|       char s[37];
 | |
|       sprintf(s,"Tx: %s",msgsent);
 | |
|       lab1->setText(s);
 | |
|     } else if(m_monitoring) {
 | |
|       lab1->setStyleSheet("QLabel{background-color: #00ff00}");
 | |
|       m_nrx=soundInThread.nrx();
 | |
|       khsym=soundInThread.mhsym();
 | |
|       QString t;
 | |
|       if(m_network) {
 | |
|         if(m_nrx==-1) t="F1";
 | |
|         if(m_nrx==1) t="I1";
 | |
|         if(m_nrx==-2) t="F2";
 | |
|         if(m_nrx==+2) t="I2";
 | |
|       } else {
 | |
|         if(m_nrx==1) t="S1";
 | |
|         if(m_nrx==2) t="S2";
 | |
|       }
 | |
|       if((abs(m_nrx)==1 and m_xpol) or (abs(m_nrx)==2 and !m_xpol))
 | |
|         lab1->setStyleSheet("QLabel{background-color: #ff1493}");
 | |
|       if(khsym==m_hsym0) {
 | |
|         t="Nil";
 | |
|         lab1->setStyleSheet("QLabel{background-color: #ffc0cb}");
 | |
|       }
 | |
|       lab1->setText("Receiving " + t);
 | |
|     } else if (!m_diskData) {
 | |
|       lab1->setStyleSheet("");
 | |
|       lab1->setText("");
 | |
|     }
 | |
| 
 | |
|     QDateTime t = QDateTime::currentDateTimeUtc();
 | |
|     int fQSO=g_pWideGraph->QSOfreq();
 | |
|     g_pAstro->astroUpdate(t, m_myGrid, m_hisGrid, fQSO, m_setftx,
 | |
|                           m_txFreq, m_azelDir);
 | |
|     m_setftx=0;
 | |
|     QString utc = t.date().toString(" yyyy MMM dd \n") + t.time().toString();
 | |
|     ui->labUTC->setText(utc);
 | |
|     if((!m_monitoring and !m_diskData) or (khsym==m_hsym0)) {
 | |
|       xSignalMeter->setValue(0);
 | |
|       ySignalMeter->setValue(0);
 | |
|       lab4->setText(" Rx noise:    0.0     0.0  0.0% ");
 | |
|     }
 | |
|     m_hsym0=khsym;
 | |
|     m_sec0=nsec;
 | |
|   }
 | |
|   iptt0=iptt;
 | |
|   bIQxt=m_bIQxt;
 | |
| }
 | |
| 
 | |
| void MainWindow::ba2msg(QByteArray ba, char message[])             //ba2msg()
 | |
| {
 | |
|   bool eom;
 | |
|   eom=false;
 | |
|   for(int i=0;i<22; i++) {
 | |
|     if (i >= ba.size () || !ba[i]) eom=true;
 | |
|     if(eom) {
 | |
|       message[i] = ' ';
 | |
|     } else {
 | |
|       message[i]=ba[i];
 | |
|     }
 | |
|   }
 | |
|   message[22] = '\0';
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txFirstCheckBox_stateChanged(int nstate)        //TxFirst
 | |
| {
 | |
|   m_txFirst = (nstate==2);
 | |
| }
 | |
| 
 | |
| void MainWindow::set_ntx(int n)                                   //set_ntx()
 | |
| {
 | |
|   m_ntx=n;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txb1_clicked()                                //txb1
 | |
| {
 | |
|   m_ntx=1;
 | |
|   ui->txrb1->setChecked(true);
 | |
|   m_restart=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txb2_clicked()                                //txb2
 | |
| {
 | |
|   m_ntx=2;
 | |
|   ui->txrb2->setChecked(true);
 | |
|   m_restart=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txb3_clicked()                                //txb3
 | |
| {
 | |
|   m_ntx=3;
 | |
|   ui->txrb3->setChecked(true);
 | |
|   m_restart=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txb4_clicked()                                //txb4
 | |
| {
 | |
|   m_ntx=4;
 | |
|   ui->txrb4->setChecked(true);
 | |
|   m_restart=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txb5_clicked()                                //txb5
 | |
| {
 | |
|   m_ntx=5;
 | |
|   ui->txrb5->setChecked(true);
 | |
|   m_restart=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_txb6_clicked()                                //txb6
 | |
| {
 | |
|   m_ntx=6;
 | |
|   ui->txrb6->setChecked(true);
 | |
|   m_restart=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::selectCall2(bool ctrl)                         //selectCall2
 | |
| {
 | |
|   QString t = ui->decodedTextBrowser->toPlainText();   //Full contents
 | |
|   int i=ui->decodedTextBrowser->textCursor().position();
 | |
|   int i0=t.lastIndexOf(" ",i);
 | |
|   int i1=t.indexOf(" ",i);
 | |
|   QString hiscall=t.mid(i0+1,i1-i0-1);
 | |
|   if(hiscall!="") {
 | |
|     int n=hiscall.length();
 | |
|     if( n>2 and n<13 and hiscall.toDouble()==0.0 and \
 | |
|         hiscall.mid(2,-1).toInt()==0) doubleClickOnCall(hiscall, ctrl);
 | |
|   }
 | |
| }
 | |
|                                                           //doubleClickOnCall
 | |
| void MainWindow::doubleClickOnCall(QString hiscall, bool ctrl)
 | |
| {
 | |
|   if(m_worked[hiscall]) {
 | |
|     msgBox("Possible dupe: " + hiscall + " already in log.");
 | |
|   }
 | |
|   ui->dxCallEntry->setText(hiscall);
 | |
|   QString t = ui->decodedTextBrowser->toPlainText();   //Full contents
 | |
|   int i2=ui->decodedTextBrowser->textCursor().position();
 | |
|   QString t1 = t.mid(0,i2);              //contents up to text cursor
 | |
|   int i1=t1.lastIndexOf("\n") + 1;
 | |
|   QString t2 = t1.mid(i1,i2-i1);         //selected line
 | |
|   int n = 60*t2.mid(14,2).toInt() + t2.mid(16,2).toInt();
 | |
|   m_txFirst = ((n%2) == 1);
 | |
|   ui->txFirstCheckBox->setChecked(m_txFirst);
 | |
|   if((t2.indexOf("#")>0) and m_modeTx!="JT65") on_pbTxMode_clicked();
 | |
|   if((t2.indexOf(":")>0) and m_modeTx!="Q65") on_pbTxMode_clicked();
 | |
|   QString rpt="";
 | |
|   if(ctrl or m_modeTx=="Q65") rpt=t2.mid(25,3);
 | |
|   lookup();
 | |
|   genStdMsgs(rpt);
 | |
|   if(t2.indexOf(m_myCall)>0) {
 | |
|     m_ntx=2;
 | |
|     ui->txrb2->setChecked(true);
 | |
|   } else {
 | |
|     m_ntx=1;
 | |
|     ui->txrb1->setChecked(true);
 | |
|   }
 | |
| }
 | |
|                                                       //doubleClickOnMessages
 | |
| void MainWindow::doubleClickOnMessages(QString hiscall, QString t2)
 | |
| {
 | |
|   if(m_worked[hiscall]) {
 | |
|     msgBox("Possible dupe: " + hiscall + " already in log.");
 | |
|   }
 | |
|   ui->dxCallEntry->setText(hiscall);
 | |
|   int n = 60*t2.mid(13,2).toInt() + t2.mid(15,2).toInt();
 | |
|   m_txFirst = ((n%2) == 1);
 | |
|   ui->txFirstCheckBox->setChecked(m_txFirst);
 | |
|   lookup();
 | |
|   genStdMsgs("");
 | |
|   if(t2.indexOf(m_myCall)>0) {
 | |
|     m_ntx=2;
 | |
|     ui->txrb2->setChecked(true);
 | |
|   } else {
 | |
|     m_ntx=1;
 | |
|     ui->txrb1->setChecked(true);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::genStdMsgs(QString rpt)                       //genStdMsgs()
 | |
| {
 | |
|   QString hiscall=ui->dxCallEntry->text().toUpper().trimmed();
 | |
|   ui->dxCallEntry->setText(hiscall);
 | |
|   QString t0=hiscall + " " + m_myCall + " ";
 | |
|   QString t=t0;
 | |
|   if(t0.indexOf("/")<0) t=t0 + m_myGrid.mid(0,4);
 | |
|   msgtype(t, ui->tx1);
 | |
|   if(rpt == "" and m_modeTx=="Q65") rpt="-24";
 | |
|   if(rpt == "" and m_modeTx=="JT65") {
 | |
|     t=t+" OOO";
 | |
|     msgtype(t, ui->tx2);
 | |
|     msgtype("RO", ui->tx3);
 | |
|     msgtype("RRR", ui->tx4);
 | |
|     msgtype("73", ui->tx5);
 | |
|   } else {
 | |
|     t=t0 + rpt;
 | |
|     msgtype(t, ui->tx2);
 | |
|     t=t0 + "R" + rpt;
 | |
|     msgtype(t, ui->tx3);
 | |
|     t=t0 + "RRR";
 | |
|     msgtype(t, ui->tx4);
 | |
|     t=t0 + "73";
 | |
|     msgtype(t, ui->tx5);
 | |
|   }
 | |
|   t="CQ " + m_myCall + " " + m_myGrid.mid(0,4);
 | |
|   msgtype(t, ui->tx6);
 | |
|   m_ntx=1;
 | |
|   ui->txrb1->setChecked(true);
 | |
| }
 | |
| 
 | |
| void MainWindow::lookup()                                       //lookup()
 | |
| {
 | |
|   QString hiscall=ui->dxCallEntry->text().toUpper().trimmed();
 | |
|   ui->dxCallEntry->setText(hiscall);
 | |
|   QString call3File = m_appDir + "/CALL3.TXT";
 | |
|   QFile f(call3File);
 | |
|   if(!f.open(QIODevice::ReadOnly | QIODevice::Text)) {
 | |
|     msgBox("Cannot open " + call3File);
 | |
|     return;
 | |
|   }
 | |
|   char c[132];
 | |
|   qint64 n=0;
 | |
|   for(int i=0; i<999999; i++) {
 | |
|     n=f.readLine(c,sizeof(c));
 | |
|     if(n <= 0) {
 | |
|       ui->dxGridEntry->setText("");
 | |
|       break;
 | |
|      }
 | |
|     QString t=QString(c);
 | |
|     if(t.indexOf(hiscall)==0) {
 | |
|       int i1=t.indexOf(",");
 | |
|       QString hisgrid=t.mid(i1+1,6);
 | |
|       i1=hisgrid.indexOf(",");
 | |
|       if(i1>0) {
 | |
|         hisgrid=hisgrid.mid(0,4);
 | |
|       } else {
 | |
|         hisgrid=hisgrid.mid(0,4) + hisgrid.mid(4,2).toLower();
 | |
|       }
 | |
|       ui->dxGridEntry->setText(hisgrid);
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
|   f.close();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_lookupButton_clicked()                    //Lookup button
 | |
| {
 | |
|   lookup();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_addButton_clicked()                       //Add button
 | |
| {
 | |
|   if(ui->dxGridEntry->text()=="") {
 | |
|     msgBox("Please enter a valid grid locator.");
 | |
|     return;
 | |
|   }
 | |
|   m_call3Modified=false;
 | |
|   QString hiscall=ui->dxCallEntry->text().toUpper().trimmed();
 | |
|   QString hisgrid=ui->dxGridEntry->text().trimmed();
 | |
|   QString newEntry=hiscall + "," + hisgrid;
 | |
| 
 | |
|   int ret = QMessageBox::warning(this, "Add",
 | |
|        newEntry + "\n" + "Is this station known to be active on EME?",
 | |
|        QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
 | |
|   if(ret==QMessageBox::Yes) {
 | |
|     newEntry += ",EME,,";
 | |
|   } else {
 | |
|     newEntry += ",,,";
 | |
|   }
 | |
|   QString call3File = m_appDir + "/CALL3.TXT";
 | |
|   QFile f1(call3File);
 | |
|   if(!f1.open(QIODevice::ReadWrite | QIODevice::Text)) {
 | |
|     msgBox("Cannot open " + call3File);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   if(f1.size()==0) {
 | |
|     QTextStream out(&f1);
 | |
|     out << "ZZZZZZ"
 | |
| #if QT_VERSION >= QT_VERSION_CHECK (5, 15, 0)
 | |
|         << Qt::endl
 | |
| #else
 | |
|         << endl
 | |
| #endif
 | |
|       ;
 | |
|     f1.close();
 | |
|     f1.open(QIODevice::ReadOnly | QIODevice::Text);
 | |
|   }
 | |
| 
 | |
|   QString tmpFile = m_appDir + "/CALL3.TMP";
 | |
|   QFile f2(tmpFile);
 | |
|   if(!f2.open(QIODevice::WriteOnly | QIODevice::Text)) {
 | |
|     msgBox("Cannot open " + tmpFile);
 | |
|     return;
 | |
|   }
 | |
|   QTextStream in(&f1);
 | |
|   QTextStream out(&f2);
 | |
|   QString hc=hiscall;
 | |
|   QString hc1="";
 | |
|   QString hc2="000000";
 | |
|   QString s;
 | |
|   do {
 | |
|     s=in.readLine();
 | |
|     hc1=hc2;
 | |
|     if(s.mid(0,2)=="//") {
 | |
|       out << s + "\n";
 | |
|     } else {
 | |
|       int i1=s.indexOf(",");
 | |
|       hc2=s.mid(0,i1);
 | |
|       if(hc>hc1 && hc<hc2) {
 | |
|         out << newEntry + "\n";
 | |
|         out << s + "\n";
 | |
|         m_call3Modified=true;
 | |
|       } else if(hc==hc2) {
 | |
|         QString t=s + "\n\n is already in CALL3.TXT\n" +
 | |
|             "Do you wish to replace it?";
 | |
|         int ret = QMessageBox::warning(this, "Add",t,
 | |
|              QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
 | |
|         if(ret==QMessageBox::Yes) {
 | |
|           out << newEntry + "\n";
 | |
|           m_call3Modified=true;
 | |
|         }
 | |
|       } else {
 | |
|         if(s!="") out << s + "\n";
 | |
|       }
 | |
|     }
 | |
|   } while(!s.isNull());
 | |
| 
 | |
|   f1.close();
 | |
|   if(hc>hc1 && !m_call3Modified) out << newEntry + "\n";
 | |
|   if(m_call3Modified) {
 | |
|     QFile f0(m_appDir + "/CALL3.OLD");
 | |
|     if(f0.exists()) f0.remove();
 | |
|     QFile f1(m_appDir + "/CALL3.TXT");
 | |
|     f1.rename(m_appDir + "/CALL3.OLD");
 | |
|     f2.rename(m_appDir + "/CALL3.TXT");
 | |
|     f2.close();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::msgtype(QString t, QLineEdit* tx)                //msgtype()
 | |
| {
 | |
| //  if(t.length()<1) return 0;
 | |
|   char message[23];
 | |
|   char msgsent[23];
 | |
|   int len1=22;
 | |
|   int mode65=0;            //mode65 ==> check message but don't make wave()
 | |
|   double samfac=1.0;
 | |
|   int nsendingsh=0;
 | |
|   int mwave;
 | |
|   t=t.toUpper();
 | |
|   int i1=t.indexOf(" OOO");
 | |
|   QByteArray s=t.toUpper().toLocal8Bit();
 | |
|   ba2msg(s,message);
 | |
|   gen65_(message,&mode65,&samfac,&nsendingsh,msgsent,iwave,
 | |
|          &mwave,len1,len1);
 | |
| 
 | |
|   QPalette p(tx->palette());
 | |
|   if(nsendingsh==1) {
 | |
|     p.setColor(QPalette::Base,"#66ffff");
 | |
|   } else if(nsendingsh==-1) {
 | |
|     p.setColor(QPalette::Base,"#ffccff");
 | |
|   } else {
 | |
|     p.setColor(QPalette::Base,Qt::white);
 | |
|   }
 | |
|   tx->setPalette(p);
 | |
|   int len=t.length();
 | |
|   if(nsendingsh==-1) {
 | |
|     len=qMin(len,13);
 | |
|     if(i1>10) {
 | |
|       tx->setText(t.mid(0,len).toUpper() + " OOO");
 | |
|     } else {
 | |
|       tx->setText(t.mid(0,len).toUpper());
 | |
|     }
 | |
|   } else {
 | |
|     tx->setText(t);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tx1_editingFinished()                       //tx1 edited
 | |
| {
 | |
|   QString t=ui->tx1->text();
 | |
|   msgtype(t, ui->tx1);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tx2_editingFinished()                       //tx2 edited
 | |
| {
 | |
|   QString t=ui->tx2->text();
 | |
|   msgtype(t, ui->tx2);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tx3_editingFinished()                       //tx3 edited
 | |
| {
 | |
|   QString t=ui->tx3->text();
 | |
|   msgtype(t, ui->tx3);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tx4_editingFinished()                       //tx4 edited
 | |
| {
 | |
|   QString t=ui->tx4->text();
 | |
|   msgtype(t, ui->tx4);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tx5_editingFinished()                       //tx5 edited
 | |
| {
 | |
|   QString t=ui->tx5->text();
 | |
|   msgtype(t, ui->tx5);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_tx6_editingFinished()                       //tx6 edited
 | |
| {
 | |
|   QString t=ui->tx6->text();
 | |
|   msgtype(t, ui->tx6);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_setTxFreqButton_clicked()                  //Set Tx Freq
 | |
| {
 | |
|   m_setftx=1;
 | |
|   m_txFreq=g_pWideGraph->QSOfreq();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_dxCallEntry_textChanged(const QString &t) //dxCall changed
 | |
| {
 | |
|   m_hisCall=t.toUpper().trimmed();
 | |
|   ui->dxCallEntry->setText(m_hisCall);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_dxGridEntry_textChanged(const QString &t) //dxGrid changed
 | |
| {
 | |
|   int n=t.length();
 | |
|   if(n!=4 and n!=6) return;
 | |
|   if(!t[0].isLetter() or !t[1].isLetter()) return;
 | |
|   if(!t[2].isDigit() or !t[3].isDigit()) return;
 | |
|   if(n==4) m_hisGrid=t.mid(0,2).toUpper() + t.mid(2,2);
 | |
|   if(n==6) m_hisGrid=t.mid(0,2).toUpper() + t.mid(2,2) +
 | |
|       t.mid(4,2).toLower();
 | |
|   ui->dxGridEntry->setText(m_hisGrid);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_genStdMsgsPushButton_clicked()         //genStdMsgs button
 | |
| {
 | |
|   genStdMsgs("");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_logQSOButton_clicked()                 //Log QSO button
 | |
| {
 | |
|   int nMHz=int(datcom_.fcenter);
 | |
|   QDateTime t = QDateTime::currentDateTimeUtc();
 | |
|   QString logEntry=t.date().toString("yyyy-MMM-dd,") +
 | |
|       t.time().toString("hh:mm,") + m_hisCall + "," + m_hisGrid + "," +
 | |
|           QString::number(nMHz) + "," + m_mode + "\r\n";
 | |
| 
 | |
|   int ret = QMessageBox::warning(this, "Log Entry",
 | |
|        "Please confirm log entry:\n\n" + logEntry + "\n",
 | |
|        QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
 | |
|   if(ret==QMessageBox::No) return;
 | |
|   QFile f("wsjt.log");
 | |
|   if(!f.open(QFile::Append)) {
 | |
|     msgBox("Cannot open file \"wsjt.log\".");
 | |
|     return;
 | |
|   }
 | |
|   QTextStream out(&f);
 | |
|   out << logEntry;
 | |
|   f.close();
 | |
|   m_worked[m_hisCall]=true;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionErase_map65_rx_log_triggered()     //Erase Rx log
 | |
| {
 | |
|   int ret = QMessageBox::warning(this, "Confirm Erase",
 | |
|       "Are you sure you want to erase file map65_rx.log ?",
 | |
|        QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
 | |
|   if(ret==QMessageBox::Yes) {
 | |
|     m_map65RxLog |= 2;                      // Rewind map65_rx.log
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionErase_map65_tx_log_triggered()     //Erase Tx log
 | |
| {
 | |
|   int ret = QMessageBox::warning(this, "Confirm Erase",
 | |
|       "Are you sure you want to erase file map65_tx.log ?",
 | |
|        QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
 | |
|   if(ret==QMessageBox::Yes) {
 | |
|     QFile f("map65_tx.log");
 | |
|     f.remove();
 | |
|   }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionNoJT65_triggered()
 | |
| {
 | |
|   m_mode65=0;
 | |
|   m_modeJT65=0;
 | |
|   lab5->setStyleSheet("");
 | |
|   lab5->setText("");
 | |
| }
 | |
| void MainWindow::on_actionJT65A_triggered()
 | |
| {
 | |
|   m_mode="JT65A";
 | |
|   m_modeJT65=1;
 | |
|   m_mode65=1;
 | |
|   m_modeJT65=1;
 | |
|   m_TRperiod=60;
 | |
|   soundInThread.setPeriod(m_TRperiod);
 | |
|   soundOutThread.setPeriod(m_TRperiod);
 | |
|   g_pWideGraph->setMode65(m_mode65);
 | |
|   g_pWideGraph->setPeriod(m_TRperiod);
 | |
|   lab5->setStyleSheet("QLabel{background-color: #ff6666}");
 | |
|   lab5->setText("JT65A");
 | |
|   ui->actionJT65A->setChecked(true);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionJT65B_triggered()
 | |
| {
 | |
|   m_mode="JT65B";
 | |
|   m_modeJT65=2;
 | |
|   m_mode65=2;
 | |
|   m_modeJT65=2;
 | |
|   m_TRperiod=60;
 | |
|   soundInThread.setPeriod(m_TRperiod);
 | |
|   soundOutThread.setPeriod(m_TRperiod);
 | |
|   g_pWideGraph->setMode65(m_mode65);
 | |
|   g_pWideGraph->setPeriod(m_TRperiod);
 | |
|   lab5->setStyleSheet("QLabel{background-color: #ffff66}");
 | |
|   lab5->setText("JT65B");
 | |
|   ui->actionJT65B->setChecked(true);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionJT65C_triggered()
 | |
| {
 | |
|   m_mode="JT65C";
 | |
|   m_modeJT65=3;
 | |
|   m_mode65=4;
 | |
|   m_TRperiod=60;
 | |
|   soundInThread.setPeriod(m_TRperiod);
 | |
|   soundOutThread.setPeriod(m_TRperiod);
 | |
|   g_pWideGraph->setMode65(m_mode65);
 | |
|   g_pWideGraph->setPeriod(m_TRperiod);
 | |
|   lab5->setStyleSheet("QLabel{background-color: #66ffb2}");
 | |
|   lab5->setText("JT65C");
 | |
|   ui->actionJT65C->setChecked(true);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionNoQ65_triggered()
 | |
| {
 | |
|   m_modeQ65=0;
 | |
|   lab6->setStyleSheet("");
 | |
|   lab6->setText("");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionQ65A_triggered()
 | |
| {
 | |
|   m_modeQ65=1;
 | |
|   lab6->setStyleSheet("QLabel{background-color: #ffb266}");
 | |
|   lab6->setText("Q65A");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionQ65B_triggered()
 | |
| {
 | |
|   m_modeQ65=2;
 | |
|   lab6->setStyleSheet("QLabel{background-color: #b2ff66}");
 | |
|   lab6->setText("Q65B");
 | |
| }
 | |
| 
 | |
| 
 | |
| void MainWindow::on_actionQ65C_triggered()
 | |
| {
 | |
|   m_modeQ65=3;
 | |
|   lab6->setStyleSheet("QLabel{background-color: #66ffff}");
 | |
|   lab6->setText("Q65C");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionQ65D_triggered()
 | |
| {
 | |
|   m_modeQ65=4;
 | |
|   lab6->setStyleSheet("QLabel{background-color: #b266ff}");
 | |
|   lab6->setText("Q65D");
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionQ65E_triggered()
 | |
| {
 | |
|   m_modeQ65=5;
 | |
|   lab6->setStyleSheet("QLabel{background-color: #ff66ff}");
 | |
|   lab6->setText("Q65E");
 | |
| }
 | |
| 
 | |
| 
 | |
| void MainWindow::on_NBcheckBox_toggled(bool checked)
 | |
| {
 | |
|   m_NB=checked;
 | |
|   ui->NBslider->setEnabled(m_NB);
 | |
| }
 | |
| 
 | |
| void MainWindow::on_NBslider_valueChanged(int n)
 | |
| {
 | |
|   m_NBslider=n;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionAdjust_IQ_Calibration_triggered()
 | |
| {
 | |
|   m_adjustIQ=1;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionApply_IQ_Calibration_triggered()
 | |
| {
 | |
|   m_applyIQcal= 1-m_applyIQcal;
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionFUNcube_Dongle_triggered()
 | |
| {
 | |
|   proc_qthid.start (QDir::toNativeSeparators(m_appDir + "/qthid"), QStringList {});
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionEdit_wsjt_log_triggered()
 | |
| {
 | |
|   proc_editor.start (QDir::toNativeSeparators (m_editorCommand), {QDir::toNativeSeparators (m_appDir + "/wsjt.log"), });
 | |
| }
 | |
| 
 | |
| void MainWindow::on_actionTx_Tune_triggered()
 | |
| {
 | |
|   if(g_pTxTune==NULL) {
 | |
|     g_pTxTune = new TxTune(0);
 | |
|   }
 | |
|   g_pTxTune->set_iqAmp(iqAmp);
 | |
|   g_pTxTune->set_iqPhase(iqPhase);
 | |
|   g_pTxTune->set_txPower(txPower);
 | |
|   g_pTxTune->show();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_pbTxMode_clicked()
 | |
| {
 | |
|   if(m_modeTx=="Q65") {
 | |
|     m_modeTx="JT65";
 | |
|     ui->pbTxMode->setText("Tx JT65   #");
 | |
|   } else {
 | |
|     m_modeTx="Q65";
 | |
|     ui->pbTxMode->setText("Tx Q65  :");
 | |
|   }
 | |
| //  m_wideGraph->setModeTx(m_modeTx);
 | |
| //  statusChanged();
 | |
| }
 |