#include "FSUDAQ.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "analyzers/CoincidentAnalyzer.h" #include "analyzers/SplitPoleAnalyzer.h" #include "analyzers/EncoreAnalyzer.h" #include "analyzers/MUSICAnalyzer.h" #include "analyzers/NeutronGamma.h" #include "analyzers/Cross.h" std::vector onlineAnalyzerList = {"Coincident","Splie-Pole", "Encore", "MUSICS", "Neutron-Gamma", "Cross"}; FSUDAQ::FSUDAQ(QWidget *parent) : QMainWindow(parent){ DebugPrint("%s", "FSUDAQ"); setWindowTitle("FSU DAQ"); setGeometry(500, 100, 1100, 600); digi = nullptr; nDigi = 0; isACQStarted= false; scalar = nullptr; scope = nullptr; digiSettings = nullptr; singleHistograms = nullptr; onlineAnalyzer = nullptr; runTimer = new QTimer(); breakAutoRepeat = true; needManualComment = true; runRecord = nullptr; model = nullptr; influx = nullptr; scalarCount = 0; QWidget * mainLayoutWidget = new QWidget(this); setCentralWidget(mainLayoutWidget); QVBoxLayout * layoutMain = new QVBoxLayout(mainLayoutWidget); mainLayoutWidget->setLayout(layoutMain); {//^======================= QGroupBox * box = new QGroupBox("Digitizer(s)", mainLayoutWidget); layoutMain->addWidget(box); QGridLayout * layout = new QGridLayout(box); cbOpenDigitizers = new RComboBox(this); cbOpenDigitizers->addItem("Open Digitizers ... ", 0); cbOpenDigitizers->addItem("Open Digitizers via Optical/USB", 1); // cbOpenDigitizers->addItem("Open Digitizers (default program)", 2); // cbOpenDigitizers->addItem("Open Digitizers + load Settings", 3); //cbOpenDigitizers->addItem("Open Digitizers via USB", 3); cbOpenDigitizers->addItem("Open Digitizers via A4818(s)", 4); layout->addWidget(cbOpenDigitizers, 0, 0); connect(cbOpenDigitizers, &RComboBox::currentIndexChanged, this, &FSUDAQ::OpenDigitizers); cbOpenMethod = new RComboBox(this); cbOpenMethod->addItem("w/o settings", 0); cbOpenMethod->addItem("w/ settings", 1); cbOpenMethod->addItem("default Program", 2); layout->addWidget(cbOpenMethod, 1, 0); bnCloseDigitizers = new QPushButton("Close Digitizers", this); layout->addWidget(bnCloseDigitizers, 2, 0); connect(bnCloseDigitizers, &QPushButton::clicked, this, &FSUDAQ::CloseDigitizers); bnDigiSettings = new QPushButton("Digitizers Settings", this); layout->addWidget(bnDigiSettings, 0, 1); connect(bnDigiSettings, &QPushButton::clicked, this, &FSUDAQ::OpenDigiSettings); bnOpenScope = new QPushButton("Open Scope", this); layout->addWidget(bnOpenScope, 1, 1); connect(bnOpenScope, &QPushButton::clicked, this, &FSUDAQ::OpenScope); cbAnalyzer = new RComboBox(this); layout->addWidget(cbAnalyzer, 0, 2); cbAnalyzer->addItem("Choose Online Analyzer", -1); for( int i = 0; i < (int) onlineAnalyzerList.size() ; i++) cbAnalyzer->addItem(onlineAnalyzerList[i].c_str(), i); connect(cbAnalyzer, &RComboBox::currentIndexChanged, this, &FSUDAQ::OpenAnalyzer); bnCanvas = new QPushButton("Online Histograms", this); layout->addWidget(bnCanvas, 1, 2); connect(bnCanvas, &QPushButton::clicked, this, &FSUDAQ::OpenSingleHistograms); bnSync = new QPushButton("Sync Boards", this); layout->addWidget(bnSync, 2, 1); connect(bnSync, &QPushButton::clicked, this, &FSUDAQ::SetSyncMode); } {//^====================== influx and Elog QGroupBox * otherBox = new QGroupBox("Database and Elog", mainLayoutWidget); layoutMain->addWidget(otherBox); QGridLayout * layout = new QGridLayout(otherBox); layout->setVerticalSpacing(1); int rowID = 0; bnLock = new QPushButton("Unlock", this); bnLock->setChecked(true); layout->addWidget(bnLock, rowID, 0); QLabel * lbInfluxIP = new QLabel("Influx IP : ", this); lbInfluxIP->setAlignment(Qt::AlignRight | Qt::AlignCenter); layout->addWidget(lbInfluxIP, rowID, 1); leInfluxIP = new QLineEdit(this); leInfluxIP->setReadOnly(true); layout->addWidget(leInfluxIP, rowID, 2); QLabel * lbDatabaseName = new QLabel("Database Name : ", this); lbDatabaseName->setAlignment(Qt::AlignRight | Qt::AlignCenter); layout->addWidget(lbDatabaseName, rowID, 3); leDatabaseName = new QLineEdit(this); leDatabaseName->setReadOnly(true); layout->addWidget(leDatabaseName, rowID, 4); chkInflux = new QCheckBox("Enable", this); chkInflux->setChecked(true); layout->addWidget(chkInflux, rowID, 5); rowID ++; QLabel * lbElogIP = new QLabel("Elog IP : ", this); lbElogIP->setAlignment(Qt::AlignRight | Qt::AlignCenter); layout->addWidget(lbElogIP, rowID, 1); leElogIP = new QLineEdit(this); leElogIP->setReadOnly(true); layout->addWidget(leElogIP, rowID, 2); QLabel * lbElogName = new QLabel("Elog Name : ", this); lbElogName->setAlignment(Qt::AlignRight | Qt::AlignCenter); layout->addWidget(lbElogName, rowID, 3); leElogName = new QLineEdit(this); leElogName->setReadOnly(true); layout->addWidget(leElogName, rowID, 4); chkElog = new QCheckBox("Enable", this); chkElog->setChecked(true); layout->addWidget(chkElog, rowID, 5); connect(bnLock, &QPushButton::clicked, this, &FSUDAQ::SetAndLockInfluxElog); } {//^====================== ACQ control QGroupBox * box = new QGroupBox("ACQ Control", mainLayoutWidget); layoutMain->addWidget(box); QGridLayout * layout = new QGridLayout(box); int rowID = 0; //------------------------------------------ QLabel * lbDataPath = new QLabel("Data Path : ", this); lbDataPath->setAlignment(Qt::AlignRight | Qt::AlignCenter); leDataPath = new QLineEdit(this); leDataPath->setReadOnly(true); QPushButton * bnSetDataPath = new QPushButton("Set Path", this); connect(bnSetDataPath, &QPushButton::clicked, this, &FSUDAQ::OpenDataPath); QPushButton * bnOpenRecord = new QPushButton("Open Record", this); connect(bnOpenRecord, &QPushButton::clicked, this, &FSUDAQ::OpenRecord); layout->addWidget(lbDataPath, rowID, 0); layout->addWidget(leDataPath, rowID, 1, 1, 5); layout->addWidget(bnSetDataPath, rowID, 6); layout->addWidget(bnOpenRecord, rowID, 7); //------------------------------------------ rowID ++; QLabel * lbPrefix = new QLabel("Prefix : ", this); lbPrefix->setAlignment(Qt::AlignRight | Qt::AlignCenter); lePrefix = new QLineEdit(this); lePrefix->setAlignment(Qt::AlignHCenter); connect(lePrefix, &QLineEdit::textChanged, this, [=](){ lePrefix->setStyleSheet("color:blue;"); }); connect(lePrefix, &QLineEdit::returnPressed, this, &FSUDAQ::SaveLastRunFile); QLabel * lbRunID = new QLabel("Run No. :", this); lbRunID->setAlignment(Qt::AlignRight | Qt::AlignCenter); leRunID = new QLineEdit(this); leRunID->setReadOnly(true); leRunID->setAlignment(Qt::AlignHCenter); chkSaveData = new QCheckBox("Save Data", this); connect( chkSaveData, &QCheckBox::stateChanged, this, [=](int state){ cbAutoRun->setEnabled(state); if( state == 0 ) cbAutoRun->setCurrentIndex(0); }); cbAutoRun = new RComboBox(this); cbAutoRun->addItem("Single Infinite", 0); cbAutoRun->addItem("Single 1 min", 1); cbAutoRun->addItem("Single 30 mins", 30); cbAutoRun->addItem("Single 60 mins", 60); cbAutoRun->addItem("Single 120 mins", 120); cbAutoRun->addItem("Repeat 1 mins", -1); cbAutoRun->addItem("Repeat 60 mins", -60); cbAutoRun->addItem("Repeat 120 mins", -120); cbAutoRun->setEnabled(false); bnStartACQ = new QPushButton("Start ACQ", this); connect( bnStartACQ, &QPushButton::clicked, this, &FSUDAQ::AutoRun); bnStopACQ = new QPushButton("Stop ACQ", this); connect( bnStopACQ, &QPushButton::clicked, this, [=](){ if( runTimer->isActive() ){ runTimer->stop(); runTimer->disconnect(runTimerConnection); }else{ breakAutoRepeat = true; runTimer->disconnect(runTimerConnection); } needManualComment = true; StopACQ(); }); layout->addWidget(lbPrefix, rowID, 0); layout->addWidget(lePrefix, rowID, 1); layout->addWidget(lbRunID, rowID, 2); layout->addWidget(leRunID, rowID, 3); layout->addWidget(chkSaveData, rowID, 4); layout->addWidget(cbAutoRun, rowID, 5); layout->addWidget(bnStartACQ, rowID, 6); layout->addWidget(bnStopACQ, rowID, 7); //------------------------------------------ rowID ++; QLabel * lbComment = new QLabel("Run Comment : ", this); lbComment->setAlignment(Qt::AlignRight | Qt::AlignCenter); leComment = new QLineEdit(this); leComment->setReadOnly(true); bnOpenScaler = new QPushButton("Scalar", this); connect(bnOpenScaler, &QPushButton::clicked, this, &FSUDAQ::OpenScalar); layout->addWidget(lbComment, rowID, 0); layout->addWidget(leComment, rowID, 1, 1, 6); layout->addWidget(bnOpenScaler, rowID, 7); layout->setColumnStretch(0, 1); layout->setColumnStretch(1, 2); layout->setColumnStretch(2, 1); layout->setColumnStretch(3, 1); layout->setColumnStretch(4, 1); layout->setColumnStretch(5, 1); layout->setColumnStretch(6, 3); layout->setColumnStretch(7, 3); } {//^===================== Log Msg logMsgHTMLMode = true; QGroupBox * box3 = new QGroupBox("Log Message", mainLayoutWidget); layoutMain->addWidget(box3); layoutMain->setStretchFactor(box3, 1); QVBoxLayout * layout3 = new QVBoxLayout(box3); logInfo = new QPlainTextEdit(this); logInfo->setReadOnly(true); QFont font; font.setFamily("Courier New"); logInfo->setFont(font); layout3->addWidget(logInfo); } LogMsg("Welcome to FSU DAQ."); rawDataPath = ""; prefix = "temp"; runID = 0; elogID = 0; elogName = ""; elogUser = ""; elogPWD = ""; influxIP = ""; dataBaseName = ""; influxToken = ""; programSettingsFilePath = QDir::current().absolutePath() + "/programSettings.txt"; LoadProgramSettings(); //=========== disable widget WaitForDigitizersOpen(true); SetUpInflux(); CheckElog(); LogMsg("====== FSU DAQ is ready. ======"); } FSUDAQ::~FSUDAQ(){ DebugPrint("%s", "FSUDAQ"); if( scalar ) { scalarTimingThread->Stop(); scalarTimingThread->quit(); scalarTimingThread->exit(); // scalarTimer->stop(); // if( scalarThread->isRunning() ){ // scalarThread->quit(); // scalarThread->exit(); // } CleanUpScalar(); //don't need to delete scalar, it is managed by this } if( digi ) CloseDigitizers(); SaveProgramSettings(); if( scope ) delete scope; if( singleHistograms ) delete singleHistograms; if( onlineAnalyzer ) delete onlineAnalyzer; if( digiSettings ) delete digiSettings; delete influx; printf("-------- remove %s\n", DAQLockFile); remove(DAQLockFile); } //*************************************************************** //*************************************************************** void FSUDAQ::OpenDataPath(){ DebugPrint("%s", "FSUDAQ"); QFileDialog fileDialog(this); fileDialog.setFileMode(QFileDialog::Directory); int result = fileDialog.exec(); //qDebug() << fileDialog.selectedFiles(); if( result > 0 ) { leDataPath->setText(fileDialog.selectedFiles().at(0)); rawDataPath = leDataPath->text(); }else{ leDataPath->clear(); rawDataPath = ""; } if( !rawDataPath.isEmpty() ) chkSaveData->setEnabled(true); SaveProgramSettings(); LoadLastRunFile(); } void FSUDAQ::OpenRecord(){ DebugPrint("%s", "FSUDAQ"); QString filePath = leDataPath->text() + "/RunTimeStamp.dat"; if( runRecord == nullptr ){ runRecord = new QMainWindow(this); runRecord->setGeometry(0,0, 500, 500); runRecord->setWindowTitle("Run Record"); QWidget * widget = new QWidget(runRecord); runRecord->setCentralWidget(widget); QVBoxLayout * layout = new QVBoxLayout(widget); widget->setLayout(layout); QLabel * lbFilePath = new QLabel(widget); lbFilePath->setText(filePath); layout->addWidget(lbFilePath); tableView = new QTableView(widget); layout->addWidget(tableView); model = new QStandardItemModel(runRecord); tableView->setModel(model); } UpdateRecord(); runRecord->show(); } void FSUDAQ::UpdateRecord(){ DebugPrint("%s", "FSUDAQ"); if( !runRecord ) return; QString filePath = leDataPath->text() + "/RunTimeStamp.dat"; model->clear(); if (!filePath.isEmpty()) { QFile file(filePath); if (file.open(QIODevice::ReadOnly | QIODevice::Text)) { QTextStream stream(&file); while (!stream.atEnd()) { QString line = stream.readLine(); QStringList fields = line.split('|'); QList items; for (const QString& field : fields) { items.append(new QStandardItem(field)); } model->appendRow(items); } file.close(); tableView->resizeColumnsToContents(); } } tableView->scrollToBottom(); } void FSUDAQ::LoadProgramSettings(){ DebugPrint("%s", "FSUDAQ"); LogMsg("Loading " + programSettingsFilePath + " for Program Settings."); QFile file(programSettingsFilePath); if( !file.open(QIODevice::Text | QIODevice::ReadOnly) ) { LogMsg("" + programSettingsFilePath + " not found."); }else{ QTextStream in(&file); QString line = in.readLine(); int count = 0; while( !line.isNull()){ if( line.left(6) == "//----") break; if( count == 0 ) rawDataPath = line; if( count == 1 ) influxIP = line; if( count == 2 ) dataBaseName = line; if( count == 3 ) influxToken = line; if( count == 4 ) elogIP = line; if( count == 5 ) elogName = line; if( count == 6 ) elogUser = line; if( count == 7 ) elogPWD = line; count ++; line = in.readLine(); } //looking for the lastRun.sh for leDataPath->setText(rawDataPath); leInfluxIP->setText(influxIP); leDatabaseName->setText(dataBaseName); leElogIP->setText(elogIP); leElogName->setText(elogName); logMsgHTMLMode = false; LogMsg(" Raw Data Path : " + rawDataPath); LogMsg(" Influx IP : " + influxIP); LogMsg(" Database Name : " + dataBaseName); LogMsg("Database Token : " + maskText(influxToken)); LogMsg(" Elog IP : " + elogIP); LogMsg(" Elog Name : " + elogName); LogMsg(" Elog User : " + maskText(elogUser)); LogMsg(" Elog PWD : " + maskText(elogPWD)); logMsgHTMLMode = true; //check is rawDataPath exist, if not, create one QDir rawDataDir; if( !rawDataDir.exists(rawDataPath ) ) { if( rawDataDir.mkdir(rawDataPath) ){ LogMsg("Created folder " + rawDataPath + " for storing root files."); }else{ LogMsg("" + rawDataPath + " Raw data folder cannot be created. Access right problem? " ); } }else{ LogMsg("" + rawDataPath + " already exist." ); } LoadLastRunFile(); } } void FSUDAQ::SaveProgramSettings(){ DebugPrint("%s", "FSUDAQ"); rawDataPath = leDataPath->text(); QFile file(programSettingsFilePath); file.open(QIODevice::Text | QIODevice::WriteOnly); file.write((rawDataPath+"\n").toStdString().c_str()); file.write((influxIP+"\n").toStdString().c_str()); file.write((dataBaseName+"\n").toStdString().c_str()); file.write((influxToken+"\n").toStdString().c_str()); file.write((elogIP+"\n").toStdString().c_str()); file.write((elogName+"\n").toStdString().c_str()); file.write((elogUser+"\n").toStdString().c_str()); file.write((elogPWD+"\n").toStdString().c_str()); file.write("//------------end of file.\n"); file.close(); LogMsg("Saved program settings to "+ programSettingsFilePath + "."); } void FSUDAQ::LoadLastRunFile(){ DebugPrint("%s", "FSUDAQ"); QFile file(rawDataPath + "/lastRun.sh"); if( !file.open(QIODevice::Text | QIODevice::ReadOnly) ) { LogMsg("" + rawDataPath + "/lastRun.sh not found."); runID = 0; prefix = "temp"; leRunID->setText(QString::number(runID)); lePrefix->setText(prefix); }else{ QTextStream in(&file); QString line = in.readLine(); int count = 0; while( !line.isNull()){ int index = line.indexOf("="); QString haha = line.mid(index+1).remove(" "); //qDebug() << haha; switch (count){ case 0 : prefix = haha; break; case 1 : runID = haha.toInt(); break; case 2 : elogID = haha.toInt(); break; } count ++; line = in.readLine(); } lePrefix->setText(prefix); lePrefix->setStyleSheet(""); leRunID->setText(QString::number(runID)); } } void FSUDAQ::SaveLastRunFile(){ DebugPrint("%s", "FSUDAQ"); QFile file(rawDataPath + "/lastRun.sh"); prefix = lePrefix->text(); lePrefix->setStyleSheet(""); file.open(QIODevice::Text | QIODevice::WriteOnly); file.write(("prefix=" + prefix + "\n").toStdString().c_str()); file.write(("runID=" + QString::number(runID) + "\n").toStdString().c_str()); file.write(("elogID=" + QString::number(elogID) + "\n").toStdString().c_str()); file.write("//------------end of file."); file.close(); LogMsg("Saved program settings to "+ rawDataPath + "/lastRun.sh."); } //*************************************************************** //*************************************************************** void FSUDAQ::OpenDigitizers(){ DebugPrint("%s", "FSUDAQ"); if( cbOpenDigitizers->currentIndex() == 0 ) return; // placeholder for USB // if( cbOpenDigitizers->currentData().toInt() == 3 ) { // return; // } QStringList a4818PIDs; if( cbOpenDigitizers->currentData().toInt() == 4 ) { QString a4818Path = QDir::current().absolutePath() + "/a4818_list.txt"; LogMsg("Looking " + a4818Path + ""); QFile file(a4818Path); if( !file.open(QIODevice::Text | QIODevice::ReadOnly) ) { LogMsg("" + a4818Path + " not found."); LogMsg("Please create such file and put the a4818 PIDs inseperate lines."); cbOpenDigitizers->setCurrentIndex(0); return; }else{ QTextStream in(&file); QString line = in.readLine(); while( !line.isNull()){ a4818PIDs.push_back(line); line = in.readLine(); } } if( a4818PIDs.isEmpty()){ LogMsg("" + a4818Path + " is empty."); cbOpenDigitizers->setCurrentIndex(0); return; }else{ if( a4818PIDs.size() > MaxNPorts){ LogMsg("There are more than "+ QString::number(MaxNPorts) + " a4818, please edit the MaxNPorts in macro.h and recompile."); } } } if( cbOpenDigitizers->currentData().toInt() == 4 ) { LogMsg("Searching digitizers via A4818 .....Please wait"); }else{ LogMsg("Searching digitizers via optical link or USB .....Please wait"); } logMsgHTMLMode = false; nDigi = 0; std::vector> portList; //boardID, portID if( cbOpenDigitizers->currentData().toInt() == 4 ) { //for A4818 for( int i = 0; i < std::min((int)a4818PIDs.size(), MaxNPorts); i++){ int port = a4818PIDs.at(i).toInt(); for( int board = 0; board < MaxNBoards; board ++){ /// max number of diasy chain Digitizer dig; dig.OpenDigitizer(board, port); if( dig.IsConnected() ){ nDigi++; portList.push_back(std::pair(board, port)); LogMsg(QString("... Found at port: %1, board: %2. SN: %3 %4").arg(port).arg(board).arg(dig.GetSerialNumber(), 3, 10, QChar(' ')).arg(dig.GetDPPString().c_str())); } dig.CloseDigitizer(); QCoreApplication::processEvents(); //to prevent Qt said application not responding. } } }else{ // optical fiber for(int port = 0; port < MaxNPorts; port++){ for( int board = 0; board < MaxNBoards; board ++){ /// max number of diasy chain Digitizer dig; dig.OpenDigitizer(board, port); if( dig.IsConnected() ){ nDigi++; portList.push_back(std::pair(board, port)); LogMsg(QString("... Found at port: %1, board: %2. SN: %3 %4").arg(port).arg(board).arg(dig.GetSerialNumber(), 3, 10, QChar(' ')).arg(dig.GetDPPString().c_str())); } dig.CloseDigitizer(); QCoreApplication::processEvents(); //to prevent Qt said application not responding. } } } logMsgHTMLMode = true; if( nDigi == 0 ) { LogMsg(QString("Done seraching. No digitizer found from port 0 to ") + QString::number(MaxNPorts) + " and board 0 to " + QString::number(MaxNBoards) + "."); cbOpenDigitizers->setCurrentIndex(0); return; }else{ if( cbOpenMethod->currentData().toInt() == 0 ) LogMsg(QString("Done seraching. Found %1 digitizer(s). Opening digitizer(s)....").arg(nDigi)); if( cbOpenMethod->currentData().toInt() == 1 ) LogMsg(QString("Done seraching. Found %1 digitizer(s). Opening digitizer(s) and program default....").arg(nDigi)); if( cbOpenMethod->currentData().toInt() == 2 ) LogMsg(QString("Done seraching. Found %1 digitizer(s). Opening digitizer(s) and load settings....").arg(nDigi)); } digi = new Digitizer * [nDigi]; readDataThread = new ReadDataThread * [nDigi]; for( unsigned int i = 0; i < nDigi; i++){ digi[i] = new Digitizer(portList[i].first, portList[i].second); //digi[i]->Reset(); if( cbOpenMethod->currentData().toInt() == 2 ) { digi[i]->ProgramBoard(); } ///============== load settings if( cbOpenMethod->currentData().toInt() <= 1 ){ QString fileName = rawDataPath + "/Digi-" + QString::number(digi[i]->GetSerialNumber()) + "_" + QString::fromStdString(digi[i]->GetData()->DPPTypeStr) + ".bin"; QFile file(fileName); if( !file.open(QIODevice::Text | QIODevice::ReadOnly) ) { if( digi[i]->GetDPPType() == V1730_DPP_PHA_CODE ) { //digi[i]->ProgramBoard_PHA(); //LogMsg("" + fileName + " not found. Program predefined PHA settings."); LogMsg("" + fileName + " not found."); } if( digi[i]->GetDPPType() == V1730_DPP_PSD_CODE ){ //digi[i]->ProgramBoard_PSD(); //LogMsg("" + fileName + " not found. Program predefined PSD settings."); LogMsg("" + fileName + " not found."); } if( digi[i]->GetDPPType() == V1740_DPP_QDC_CODE ){ //digi[i]->ProgramBoard_QDC(); //LogMsg("" + fileName + " not found. Program predefined PSD settings."); LogMsg("" + fileName + " not found."); } }else{ LogMsg("Found " + fileName + " for digitizer settings."); if( cbOpenMethod->currentData().toInt() == 1 ){ if( digi[i]->LoadSettingBinaryToMemory(fileName.toStdString().c_str()) == 0 ){ LogMsg("Loaded settings file " + fileName + " for Digi-" + QString::number(digi[i]->GetSerialNumber())); digi[i]->ProgramSettingsToBoard(); }else{ LogMsg("Fail to Loaded settings file " + fileName + " for Digi-" + QString::number(digi[i]->GetSerialNumber())); } }else{ LogMsg("Save the setting file path, but not load."); digi[i]->SetSettingBinaryPath(fileName.toStdString()); } } } digi[i]->ReadAllSettingsFromBoard(true); //===== set no trace, even when FSQDAQ segfault at scope, the digitizer will save no trace digi[i]->SetTrace(false); // if( digi[i]->GetDPPType() == V1730_DPP_PHA_CODE) digi[i]->WriteRegister(DPP::BoardConfiguration, 0xE8915); readDataThread[i] = new ReadDataThread(digi[i], i); connect(readDataThread[i], &ReadDataThread::sendMsg, this, &FSUDAQ::LogMsg); QCoreApplication::processEvents(); //to prevent Qt said application not responding. } singleHistograms = new SingleSpectra(digi, nDigi, rawDataPath); LogMsg("====== " + QString("Done. Opened %1 digitizer(s).").arg(nDigi) + " ====="); WaitForDigitizersOpen(false); bnStartACQ->setStyleSheet("background-color: green;"); bnStopACQ->setEnabled(false); bnStopACQ->setStyleSheet(""); bnSync->setEnabled( nDigi >= 1 ); if( rawDataPath == "" ) { chkSaveData->setChecked(false); chkSaveData->setEnabled(false); } SetupScalar(); cbOpenDigitizers->setCurrentIndex(0); } void FSUDAQ::CloseDigitizers(){ LogMsg("FSUDAQ::Closing Digitizer(s)...."); if( scope ) { scope->close(); delete scope; scope = nullptr; } // scalarTimer->stop(); // if( scalarThread->isRunning() ){ // scalarThread->quit(); // scalarThread->exit(); // } scalarTimingThread->Stop(); if( scalarTimingThread->isRunning() ){ scalarTimingThread->quit(); scalarTimingThread->exit(); } if( scalar ) CleanUpScalar(); if( onlineAnalyzer ){ onlineAnalyzer->close(); delete onlineAnalyzer; onlineAnalyzer = nullptr; } if( singleHistograms ){ singleHistograms->close(); delete singleHistograms; singleHistograms = nullptr; } if( digiSettings ){ digiSettings->close(); delete digiSettings; digiSettings = nullptr; } if( digi == nullptr ) return; for(unsigned int i = 0; i < nDigi; i ++){ readDataThread[i]->Stop(); readDataThread[i]->quit(); readDataThread[i]->wait(); delete readDataThread[i]; printf(" readDataThread[%d] is deleted.\n", i); } delete [] readDataThread; readDataThread = nullptr; for(unsigned int i = 0; i < nDigi; i ++){ digi[i]->StopACQ(); digi[i]->CloseDigitizer(); delete digi[i]; } delete [] digi; digi = nullptr; LogMsg("Done. Closed " + QString::number(nDigi) + " Digitizer(s)."); nDigi = 0; WaitForDigitizersOpen(true); bnStartACQ->setStyleSheet(""); bnStopACQ->setStyleSheet(""); printf("End of FSUDAQ::%s\n", __func__); } void FSUDAQ::WaitForDigitizersOpen(bool onOff){ DebugPrint("%s", "FSUDAQ"); // bnOpenDigitizers->setEnabled(onOff); cbOpenDigitizers->setEnabled(onOff); cbOpenMethod->setEnabled(onOff); bnCloseDigitizers->setEnabled(!onOff); bnOpenScope->setEnabled(!onOff); bnDigiSettings->setEnabled(!onOff); bnOpenScaler->setEnabled(!onOff); bnStartACQ->setEnabled(!onOff); bnStopACQ->setEnabled(!onOff); bnStopACQ->setStyleSheet(""); chkSaveData->setEnabled(!onOff); bnCanvas->setEnabled(!onOff); cbAnalyzer->setEnabled(!onOff); cbAutoRun->setEnabled(chkSaveData->isChecked()); bnSync->setEnabled(false); } //*************************************************************** //*************************************************************** void FSUDAQ::SetupScalar(){ DebugPrint("%s", "FSUDAQ"); // printf("%s\n", __func__); scalar = new QMainWindow(this); scalar->setWindowTitle("Scalar"); QScrollArea * scopeScroll = new QScrollArea(scalar); scalar->setCentralWidget(scopeScroll); scopeScroll->setWidgetResizable(true); scopeScroll->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); QWidget * layoutWidget = new QWidget(scalar); scopeScroll->setWidget(layoutWidget); scalarLayout = new QGridLayout(layoutWidget); scalarLayout->setSpacing(0); scalarLayout->setAlignment(Qt::AlignTop); leTrigger = nullptr; leAccept = nullptr; lbLastUpdateTime = nullptr; lbScalarACQStatus = nullptr; lbTotalFileSize = nullptr; scalarTimingThread = new TimingThread(scalar); scalarTimingThread->SetWaitTimeinSec(ScalarUpdateinMiliSec / 1000.); connect(scalarTimingThread, &TimingThread::timeUp, this, &FSUDAQ::UpdateScalar); // scalarThread = new QThread(this); // scalarWorker = new ScalarWorker(this); // scalarWorker->moveToThread(scalarThread); // scalarTimer = new QTimer(this); // connect( scalarTimer, &QTimer::timeout, scalarWorker, &ScalarWorker::UpdateScalar); // scalarThread->start(); unsigned short maxNChannel = 0; for( unsigned int k = 0; k < nDigi; k ++ ){ if( digi[k]->GetNumInputCh() > maxNChannel ) maxNChannel = digi[k]->GetNumInputCh(); } scalar->setGeometry(0, 0, 50 + nDigi * 240, 160 + maxNChannel * 25); if( lbLastUpdateTime == nullptr ){ lbLastUpdateTime = new QLabel("Last update : NA", scalar); lbScalarACQStatus = new QLabel("ACQ status", scalar); lbTotalFileSize = new QLabel("Total File Size", scalar); } lbLastUpdateTime->setAlignment(Qt::AlignRight); scalarLayout->removeWidget(lbLastUpdateTime); scalarLayout->addWidget(lbLastUpdateTime, 0, 0, 1, 1 + nDigi); lbScalarACQStatus->setAlignment(Qt::AlignCenter); scalarLayout->removeWidget(lbScalarACQStatus); scalarLayout->addWidget(lbScalarACQStatus, 0, 1 + nDigi); lbTotalFileSize->setAlignment(Qt::AlignCenter); scalarLayout->removeWidget(lbTotalFileSize); scalarLayout->addWidget(lbTotalFileSize, 1, 0, 1, 1 + 2*nDigi); ///==== create the header row int rowID = 4; for( int ch = 0; ch < maxNChannel; ch++){ if( ch == 0 ){ QLabel * lbCH_H = new QLabel("Ch", scalar); lbCH_H->setAlignment(Qt::AlignCenter); scalarLayout->addWidget(lbCH_H, rowID, 0); } rowID ++; QLabel * lbCH = new QLabel(QString::number(ch), scalar); lbCH->setAlignment(Qt::AlignCenter); scalarLayout->addWidget(lbCH, rowID, 0); } ///===== create the trigger and accept leTrigger = new QLineEdit**[nDigi]; leAccept = new QLineEdit**[nDigi]; for( unsigned int iDigi = 0; iDigi < nDigi; iDigi++){ rowID = 2; uint32_t chMask = digi[iDigi]->GetRegChannelMask(); QWidget * hBox = new QWidget(scalar); QHBoxLayout * hBoxLayout = new QHBoxLayout(hBox); scalarLayout->addWidget(hBox, rowID, 2*iDigi+1, 1, 2); QLabel * lbDigi = new QLabel("Digi-" + QString::number(digi[iDigi]->GetSerialNumber()), scalar); // QLabel * lbDigi = new QLabel(QString::number(digi[iDigi]->GetSerialNumber()), scalar); // lbDigi->setAlignment(Qt::AlignRight | Qt::AlignCenter); lbDigi->setAlignment(Qt::AlignCenter); hBoxLayout->addWidget(lbDigi); runStatus[iDigi] = new QPushButton("", scalar); runStatus[iDigi]->setEnabled(false); runStatus[iDigi]->setFixedSize(QSize(20,20)); runStatus[iDigi]->setToolTip("ACQ RUN On/OFF"); runStatus[iDigi]->setToolTipDuration(-1); hBoxLayout->addWidget(runStatus[iDigi]); rowID = 3; QWidget * hBox2 = new QWidget(scalar); QHBoxLayout * hBoxLayout2 = new QHBoxLayout(hBox2); scalarLayout->addWidget(hBox2, rowID, 2*iDigi+1, 1, 2); lbAggCount[iDigi] = new QLabel("AggCount/ReadCount", scalar); lbAggCount[iDigi]->setAlignment(Qt::AlignLeft | Qt::AlignCenter); hBoxLayout2->addWidget(lbAggCount[iDigi]); lbFileSize[iDigi] = new QLabel("File Size", scalar); lbFileSize[iDigi]->setAlignment(Qt::AlignLeft | Qt::AlignCenter); hBoxLayout2->addWidget(lbFileSize[iDigi]); rowID = 4; QLabel * lbA = new QLabel("Trig. [Hz]", scalar); lbA->setAlignment(Qt::AlignCenter); scalarLayout->addWidget(lbA, rowID, 2*iDigi+1); QLabel * lbB = new QLabel("Accp. [Hz]", scalar); lbB->setAlignment(Qt::AlignCenter); scalarLayout->addWidget(lbB, rowID, 2*iDigi+2); leTrigger[iDigi] = new QLineEdit *[digi[iDigi]->GetNumInputCh()]; leAccept[iDigi] = new QLineEdit *[digi[iDigi]->GetNumInputCh()]; for( int ch = 0; ch < digi[iDigi]->GetNumInputCh(); ch++){ rowID ++; leTrigger[iDigi][ch] = new QLineEdit(scalar); leTrigger[iDigi][ch]->setReadOnly(true); leTrigger[iDigi][ch]->setFixedSize(120, 25); leTrigger[iDigi][ch]->setAlignment(Qt::AlignRight); scalarLayout->addWidget(leTrigger[iDigi][ch], rowID, 2*iDigi+1); leAccept[iDigi][ch] = new QLineEdit(scalar); leAccept[iDigi][ch]->setReadOnly(true); leAccept[iDigi][ch]->setFixedSize(120, 25); leAccept[iDigi][ch]->setAlignment(Qt::AlignRight); leAccept[iDigi][ch]->setStyleSheet("background-color: #F0F0F0;"); if( digi[iDigi]->IsInputChEqRegCh() ){ leTrigger[iDigi][ch]->setEnabled( (chMask >> ch) & 0x1 ); leAccept[iDigi][ch]->setEnabled( (chMask >> ch) & 0x1 ); }else{ int grpID = ch/digi[iDigi]->GetNumRegChannels(); leTrigger[iDigi][ch]->setEnabled( (chMask >> grpID) & 0x1 ); leAccept[iDigi][ch]->setEnabled( (chMask >> grpID) & 0x1 ); } scalarLayout->addWidget(leAccept[iDigi][ch], rowID, 2*iDigi+2); } } } void FSUDAQ::UpdateScalar(){ DebugPrint("%s", "FSUDAQ"); // printf("================== FSUDAQ::%s\n", __func__); if( digi == nullptr ) return; if( scalar == nullptr ) return; //if( !scalar->isVisible() ) return; // digi[0]->GetData()->PrintAllData(); // lbLastUpdateTime->setText("Last update: " + QDateTime::currentDateTime().toString("MM.dd hh:mm:ss")); lbLastUpdateTime->setText(QDateTime::currentDateTime().toString("MM/dd hh:mm:ss")); scalarCount ++; uint64_t totalFileSize = 0; for( unsigned int iDigi = 0; iDigi < nDigi; iDigi++){ // printf("======== digi-%d\n", iDigi); if( digi[iDigi]->IsBoardDisabled() ) continue; uint32_t acqStatus = digi[iDigi]->GetACQStatusFromMemory(); //printf("Digi-%d : acq on/off ? : %d \n", digi[iDigi]->GetSerialNumber(), (acqStatus >> 2) & 0x1 ); if( ( acqStatus >> 2 ) & 0x1 ){ if( runStatus[iDigi]->styleSheet() == "") runStatus[iDigi]->setStyleSheet("background-color : green;"); }else{ if( runStatus[iDigi]->styleSheet() != "") runStatus[iDigi]->setStyleSheet(""); } if(digiSettings && digiSettings->isVisible() && digiSettings->GetTabID() == iDigi) digiSettings->UpdateACQStatus(acqStatus); // digiMTX[iDigi].lock(); QString blockCountStr = QString::number(digi[iDigi]->GetData()->AggCount); blockCountStr += "/" + QString::number(readDataThread[iDigi]->GetReadCount()); readDataThread[iDigi]->SetReadCountZero(); lbAggCount[iDigi]->setText(blockCountStr); lbFileSize[iDigi]->setText(QString::number(digi[iDigi]->GetData()->GetTotalFileSize()/1024./1024., 'f', 3) + " MB"); digi[iDigi]->GetData()->CalTriggerRate(); //this will reset NumEventDecode & AggCount if( chkSaveData->isChecked() ) totalFileSize += digi[iDigi]->GetData()->GetTotalFileSize(); for( int i = 0; i < digi[iDigi]->GetNumInputCh(); i++){ QString a = ""; QString b = ""; if( digi[iDigi]->GetInputChannelOnOff(i) == true ) { // printf(" %3d %2d | %7.2f %7.2f \n", digi[iDigi]->GetSerialNumber(), i, digi[iDigi]->GetData()->TriggerRate[i], digi[iDigi]->GetData()->NonPileUpRate[i]); QString a = QString::number(digi[iDigi]->GetData()->TriggerRate[i], 'f', 2); QString b = QString::number(digi[iDigi]->GetData()->NonPileUpRate[i], 'f', 2); leTrigger[iDigi][i]->setText(a); leAccept[iDigi][i]->setText(b); if( influx && chkInflux->isChecked() && a != "inf" ){ influx->AddDataPoint("TrigRate,Bd="+std::to_string(digi[iDigi]->GetSerialNumber()) + ",Ch=" + QString::number(i).rightJustified(2, '0').toStdString() + " value=" + a.toStdString()); } } } // digiMTX[iDigi].unlock(); // printf("============= end of FSUDAQ::%s\n", __func__); } lbTotalFileSize->setText("Total Data Size : " + QString::number(totalFileSize/1024./1024., 'f', 3) + " MB"); // repaint(); // scalar->repaint(); if( influx && chkInflux->isChecked() && scalarCount >= 3){ if( chkSaveData->isChecked() ) { influx->AddDataPoint("RunID value=" + std::to_string(runID)); influx->AddDataPoint("FileSize value=" + std::to_string(totalFileSize)); } //nflux->PrintDataPoints(); influx->WriteData(dataBaseName.toStdString()); influx->ClearDataPointsBuffer(); scalarCount = 0; } // printf("end of %s\n", __func__); } void FSUDAQ::CleanUpScalar(){ DebugPrint("%s", "FSUDAQ"); if( scalar == nullptr) return; scalar->close(); if( leTrigger == nullptr ) return; for( unsigned int i = 0; i < nDigi; i++){ for( int ch = 0; ch < digi[i]->GetNumInputCh(); ch ++){ delete leTrigger[i][ch]; delete leAccept[i][ch]; } delete [] leTrigger[i]; delete [] leAccept[i]; } delete [] leTrigger; leTrigger = nullptr; leAccept = nullptr; //Clean up QLabel QList labelChildren = scalar->findChildren(); for( int i = 0; i < labelChildren.size(); i++) delete labelChildren[i]; printf("---- end of %s \n", __func__); } void FSUDAQ::OpenScalar(){ DebugPrint("%s", "FSUDAQ"); scalar->show(); } //*************************************************************** //*************************************************************** void FSUDAQ::StartACQ(){ DebugPrint("%s", "FSUDAQ"); if( digi == nullptr ) return; bool commentResult = true; if( chkSaveData->isChecked()) commentResult = CommentDialog(true); if( commentResult == false) return; if( chkSaveData->isChecked() ) { LogMsg("===================== Start a new Run-" + QString::number(runID) + ""); WriteRunTimestamp(true); }else{ LogMsg("===================== Start a non-save Run"); } //assume master board is the 0-th board for( int i = (int) nDigi-1; i >= 0 ; i--){ if( digi[i]->IsBoardDisabled() ) continue; if( chkSaveData->isChecked() ) { std::string runSettingName = (rawDataPath + "/" + prefix + "_" + QString::number(runID).rightJustified(3, '0') + "_" + QString::number(digi[i]->GetSerialNumber())).toStdString(); runSettingName += "_" + digi[i]->GetData()->DPPTypeStr + ".bin"; digi[i]->SaveAllSettingsAsTextForRun(runSettingName); if( digi[i]->GetData()->OpenSaveFile((rawDataPath + "/" + prefix + "_" + QString::number(runID).rightJustified(3, '0')).toStdString()) == false ) { LogMsg("Cannot open save file : " + QString::fromStdString(digi[i]->GetData()->GetOutFileName() ) + ". Probably read-only?"); continue; }; } readDataThread[i]->SetSaveData(chkSaveData->isChecked()); LogMsg("Digi-" + QString::number(digi[i]->GetSerialNumber()) + " is starting ACQ." ); digi[i]->WriteRegister(DPP::SoftwareClear_W, 1); digi[i]->StartACQ(); readDataThread[i]->start(); } if( chkSaveData->isChecked() ) SaveLastRunFile(); // printf("------------ wait for 2 sec \n"); // usleep(1000*1000); // printf("------------ Go! \n"); // for( unsigned int i = 0; i < nDigi; i++) readDataThread[i]->go(); // if( scalar ) scalarTimer->start(ScalarUpdateinMiliSec); if( scalar ) scalarTimingThread->start(); if( !scalar->isVisible() ) { scalar->show(); }else{ scalar->activateWindow(); } lbScalarACQStatus->setText("ACQ On"); if( singleHistograms ) singleHistograms->startTimer(); if( onlineAnalyzer ) onlineAnalyzer->startTimer(); bnStartACQ->setEnabled(false); bnStartACQ->setStyleSheet(""); bnStopACQ->setEnabled(true); bnStopACQ->setStyleSheet("background-color: red;"); bnOpenScope->setEnabled(false); cbAutoRun->setEnabled(false); bnSync->setEnabled(false); if( digiSettings ) digiSettings->EnableButtons(false); {//^=== elog and database if( influx && chkInflux->isChecked() ){ influx->AddDataPoint("RunID value=" + std::to_string(runID)); if( !elogName.isEmpty() ) influx->AddDataPoint("SavingData,ExpName=" + elogName.toStdString() + " value=1"); influx->WriteData(dataBaseName.toStdString()); influx->ClearDataPointsBuffer(); } if( elogID > 0 && chkElog->isChecked() && chkSaveData->isChecked() ){ QString msg = "================================= Run-" + QString::number(runID).rightJustified(3, '0') + "

" + QDateTime::currentDateTime().toString("MM.dd hh:mm:ss") + "

" + startComment + "

" "---------------------------------

"; WriteElog(msg, "Run Log", "Run", runID); } } isACQStarted = true; chkSaveData->setEnabled(false); // bnDigiSettings->setEnabled(false); } void FSUDAQ::StopACQ(){ DebugPrint("%s", "FSUDAQ"); QCoreApplication::processEvents(); if( digi == nullptr ) return; bool commentResult = true; if( chkSaveData->isChecked() ) commentResult = CommentDialog(false); if( commentResult == false) return; if( chkSaveData->isChecked() ) { LogMsg("===================== Stop Run-" + QString::number(runID)); WriteRunTimestamp(false); }else{ LogMsg("===================== Stop a non-save Run"); } for( unsigned int i = 0; i < nDigi; i++){ if( digi[i]->IsBoardDisabled() ) continue; readDataThread[i]->Stop(); readDataThread[i]->quit(); readDataThread[i]->wait(); digiMTX[i].lock(); digi[i]->StopACQ(); digiMTX[i].unlock(); if( chkSaveData->isChecked() ) digi[i]->GetData()->CloseSaveFile(); LogMsg("Digi-" + QString::number(digi[i]->GetSerialNumber()) + " ACQ is stopped." ); QCoreApplication::processEvents(); } if( scalarTimingThread->isRunning()){ scalarTimingThread->Stop(); scalarTimingThread->quit(); scalarTimingThread->wait(); } // if( scalar ) scalarTimer->stop(); if( singleHistograms ) singleHistograms->stopTimer(); if( onlineAnalyzer ) onlineAnalyzer->stopTimer(); lbScalarACQStatus->setText("ACQ Off"); bnStartACQ->setEnabled(true); bnStartACQ->setStyleSheet("background-color: green;"); bnStopACQ->setEnabled(false); bnStopACQ->setStyleSheet(""); bnOpenScope->setEnabled(true); cbAutoRun->setEnabled(true); bnSync->setEnabled(true); if( scalar ){ for( unsigned int iDigi = 0; iDigi < nDigi; iDigi ++){ uint32_t acqStatus = digi[iDigi]->ReadRegister(DPP::AcquisitionStatus_R); if( ( acqStatus >> 2 ) & 0x1 ){ runStatus[iDigi]->setStyleSheet("background-color : green;"); }else{ runStatus[iDigi]->setStyleSheet(""); } QCoreApplication::processEvents(); } } if( digiSettings ) { digiSettings->EnableButtons(true); digiSettings->ReadSettingsFromBoard(); } {//^=== elog and database if( influx && chkInflux->isChecked() && elogName != "" ) { if( !elogName.isEmpty() ) influx->AddDataPoint("SavingData,ExpName=" + elogName.toStdString() + " value=0"); influx->WriteData(dataBaseName.toStdString()); influx->ClearDataPointsBuffer(); } if( elogID > 0 && chkElog->isChecked() && chkSaveData->isChecked()){ QString msg = QDateTime::currentDateTime().toString("MM.dd hh:mm:ss") + "

" + stopComment + "

"; uint64_t totalFileSize = 0; for(unsigned int i = 0 ; i < nDigi; i++){ uint64_t fileSize = digi[i]->GetData()->GetTotalFileSize(); totalFileSize += fileSize; msg += "Digi-" + QString::number(digi[i]->GetSerialNumber()) + " Size : " + QString::number(fileSize/1024./1024., 'f', 2) + " MB

"; } msg += "..... Total File Size : " + QString::number(totalFileSize/1024./1024., 'f', 2) + "MB

" + "=================================

"; AppendElog(msg); } } chkSaveData->setEnabled(true); // bnDigiSettings->setEnabled(true); isACQStarted = false; repaint(); // printf("================ end of %s \n", __func__); } void FSUDAQ::AutoRun(){ DebugPrint("%s", "FSUDAQ"); runTimer->disconnect(runTimerConnection); if( chkSaveData->isChecked() == false){ StartACQ(); return; } if( cbAutoRun->currentData().toInt() == 0 ){ StartACQ(); //disconnect(runTimer, runTimerConnection); //runTimer->disconnect(runTimerConnection); return; }else{ // auto run needManualComment = true; StartACQ(); runTimerConnection = connect( runTimer, &QTimer::timeout, this, [=](){ needManualComment = false; LogMsg("Time Up, Stopping ACQ..."); StopACQ(); if( cbAutoRun->currentData().toInt() < 0 ){ bnStartACQ->setEnabled(false); bnStartACQ->setStyleSheet(""); bnStopACQ->setEnabled(true); bnStopACQ->setStyleSheet("background-color : red;"); LogMsg("Wait for 10 sec for next Run ...." ); QElapsedTimer elapsedTimer; elapsedTimer.invalidate(); elapsedTimer.start(); while( elapsedTimer.elapsed() < 10000) { if( breakAutoRepeat ) { LogMsg("Break Auto repeat."); bnStartACQ->setEnabled(true); bnStartACQ->setStyleSheet("background-color : green"); bnStopACQ->setEnabled(false); bnStopACQ->setStyleSheet(""); return; } QCoreApplication::processEvents(); } needManualComment = false; StartACQ(); runTimer->setSingleShot(true); runTimer->start(qAbs( cbAutoRun->currentData().toInt() * 60 * 1000)); } }); } int timeMiliSec = cbAutoRun->currentData().toInt() * 60 * 1000; runTimer->setSingleShot(true); runTimer->start(qAbs(timeMiliSec)); if( timeMiliSec ) breakAutoRepeat = false; // ///=========== single run // if ( timeMiliSec > 0 ){ // runTimer->setSingleShot(true); // runTimer->start(timeMiliSec); // } // ///=========== infinite repeat run // if ( timeMiliSec < 0 ){ // runTimer->setSingleShot(false); // runTimer->start(qAbs(timeMiliSec)); // } } void FSUDAQ::SetSyncMode(){ DebugPrint("%s", "FSUDAQ"); QDialog dialog; dialog.setWindowTitle("Board Synchronization"); QVBoxLayout * layout = new QVBoxLayout(&dialog); QLabel * lbInfo1 = new QLabel("This will reset 0x8100 and 0x811C \nMaster must be the 1st board.\n (could be 100 ticks offset)", &dialog); lbInfo1->setStyleSheet("color : red;"); QPushButton * bnNoSync = new QPushButton("No Sync"); QPushButton * bnMethod1 = new QPushButton("Software TRG-OUT --> TRG-IN "); QPushButton * bnMethod2 = new QPushButton("Software TRG-OUT --> S-IN "); QPushButton * bnMethod3 = new QPushButton("External --> 1st S-IN,\nTRG-OUT --> S-IN "); QPushButton * bnMethod4 = new QPushButton("External All S-IN "); layout->addWidget(lbInfo1, 0); layout->addWidget( bnNoSync, 2); layout->addWidget(bnMethod1, 3); layout->addWidget(bnMethod2, 4); layout->addWidget(bnMethod3, 5); layout->addWidget(bnMethod4, 6); bnNoSync->setFixedHeight(40); bnMethod1->setFixedHeight(40); bnMethod2->setFixedHeight(40); bnMethod3->setFixedHeight(40); bnMethod4->setFixedHeight(40); connect(bnNoSync, &QPushButton::clicked, [&](){ /// No Sync LogMsg("Set No Sync across digitizers."); LogMsg("Software start ACQ, internal clock."); for(unsigned int i = 0; i < nDigi; i++){ digi[i]->WriteRegister(DPP::AcquisitionControl, 0); digi[i]->WriteRegister(DPP::FrontPanelIOControl, 0); } if( digiSettings && digiSettings->isVisible() ) digiSettings->UpdatePanelFromMemory(); dialog.accept(); }); connect(bnMethod1, &QPushButton::clicked, [&](){ /// Software TRG-OUT --> TRG-IN LogMsg("Set Software TRG-OUT -> TRG-IN"); LogMsg("Set master saftware ACQ, internal clock."); LogMsg("Set slaves TRG-IN, external clock"); digi[0]->WriteRegister(DPP::AcquisitionControl, 0); digi[0]->WriteRegister(DPP::FrontPanelIOControl, 0x10000); //RUN for(unsigned int i = 1; i < nDigi; i++){ digi[i]->WriteRegister(DPP::AcquisitionControl, 0x42); digi[i]->WriteRegister(DPP::FrontPanelIOControl, 0x10000); // S-IN } if( digiSettings && digiSettings->isVisible() ) digiSettings->UpdatePanelFromMemory(); dialog.accept(); }); connect(bnMethod2, &QPushButton::clicked, [&](){ /// Software TRG-OUT --> S-IN LogMsg("Set Software TRG-OUT -> S-IN"); LogMsg("Set master saftware ACQ, internal clock."); LogMsg("Set slaves S-IN, external clock"); digi[0]->WriteRegister(DPP::AcquisitionControl, 0); digi[0]->WriteRegister(DPP::FrontPanelIOControl, 0x10000); //RUN for(unsigned int i = 1; i < nDigi; i++){ digi[i]->WriteRegister(DPP::AcquisitionControl, 0x41); digi[i]->WriteRegister(DPP::FrontPanelIOControl, 0x30000); // S-IN } if( digiSettings && digiSettings->isVisible() ) digiSettings->UpdatePanelFromMemory(); dialog.accept(); }); connect(bnMethod3, &QPushButton::clicked, [&](){ ///External TRG-OUT --> S-IN LogMsg("Set master External -> S-IN, slave TRG-OUT -> S-IN"); LogMsg("Set master external S-IN, internal clock."); LogMsg("Set slaves S-IN, external clock"); digi[0]->WriteRegister(DPP::AcquisitionControl, 0x01); for(unsigned int i = 0; i < nDigi; i++){ digi[i]->WriteRegister(DPP::AcquisitionControl, 0x41); digi[i]->WriteRegister(DPP::FrontPanelIOControl, 0x30000); // S-IN } if( digiSettings && digiSettings->isVisible() ) digiSettings->UpdatePanelFromMemory(); dialog.accept(); }); connect(bnMethod4, &QPushButton::clicked, [&](){ /// External All S-IN LogMsg("Set all External -> S-IN"); LogMsg("Set master internal clock, slaves external clock"); digi[0]->WriteRegister(DPP::AcquisitionControl, 0x01); for(unsigned int i = 1; i < nDigi; i++){ digi[i]->WriteRegister(DPP::AcquisitionControl, 0x41); } if( digiSettings && digiSettings->isVisible() ) digiSettings->UpdatePanelFromMemory(); dialog.accept(); }); dialog.exec(); } void FSUDAQ::SetAndLockInfluxElog(){ DebugPrint("%s", "FSUDAQ"); if( leInfluxIP->isReadOnly() ){ bnLock->setText("Lock and Set"); leInfluxIP->setReadOnly(false); leDatabaseName->setReadOnly(false); leElogIP->setReadOnly(false); leElogName->setReadOnly(false); leInfluxIP->setEnabled(true); leDatabaseName->setEnabled(true); leElogIP->setEnabled(true); leElogName->setEnabled(true); leInfluxIP->setStyleSheet("color : blue;"); leDatabaseName->setStyleSheet("color : blue;"); leElogIP->setStyleSheet("color : blue;"); leElogName->setStyleSheet("color : blue;"); }else{ bnLock->setText("Unlock"); leInfluxIP->setReadOnly(true); leDatabaseName->setReadOnly(true); leElogIP->setReadOnly(true); leElogName->setReadOnly(true); leInfluxIP->setStyleSheet(""); leDatabaseName->setStyleSheet(""); leElogIP->setStyleSheet(""); leElogName->setStyleSheet(""); influxIP = leInfluxIP->text(); dataBaseName = leDatabaseName->text(); elogIP = leElogIP->text(); elogName = leElogName->text(); if( !influxIP.isEmpty() && !dataBaseName.isEmpty() ){ QDialog dialog; dialog.setWindowTitle("Database Token"); QVBoxLayout layout(&dialog); QLineEdit tokenLineEdit; tokenLineEdit.setFixedSize(1000, 20); tokenLineEdit.setText(influxToken); layout.addWidget(new QLabel("Only for version 2+, version 1+ can be skipped.")); layout.addWidget(&tokenLineEdit); // Buttons for OK and Cancel QDialogButtonBox buttonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel); layout.addWidget(&buttonBox); QObject::connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept); QObject::connect(&buttonBox, &QDialogButtonBox::rejected, &dialog, &QDialog::reject); dialog.resize(400, dialog.sizeHint().height()); // Set the width to 400 pixels // Show the dialog and get the result if (dialog.exec() == QDialog::Accepted) { influxToken = tokenLineEdit.text(); } } if( !elogIP.isEmpty() && !elogName.isEmpty() ){ QDialog dialog; dialog.setWindowTitle("ELog Login info."); QVBoxLayout layout(&dialog); QFormLayout formLayout; QLineEdit usernameLineEdit; QLineEdit passwordLineEdit; //passwordLineEdit.setEchoMode(QLineEdit::Password); formLayout.addRow("Username:", &usernameLineEdit); formLayout.addRow("Password:", &passwordLineEdit); usernameLineEdit.setText(elogUser); passwordLineEdit.setText(elogPWD); layout.addLayout(&formLayout); // Buttons for OK and Cancel QDialogButtonBox buttonBox(QDialogButtonBox::Ok | QDialogButtonBox::Cancel); layout.addWidget(&buttonBox); QObject::connect(&buttonBox, &QDialogButtonBox::accepted, &dialog, &QDialog::accept); QObject::connect(&buttonBox, &QDialogButtonBox::rejected, &dialog, &QDialog::reject); dialog.resize(400, dialog.sizeHint().height()); // Set the width to 400 pixels // Show the dialog and get the result if (dialog.exec() == QDialog::Accepted) { QString username = usernameLineEdit.text(); QString password = passwordLineEdit.text(); // Check if username and password are not empty if (!username.isEmpty() && !password.isEmpty()) { elogUser = username; elogPWD = password; } else { qDebug() << "Please enter both username and password."; } } } SaveProgramSettings(); SetUpInflux(); CheckElog(); } } bool FSUDAQ::CommentDialog(bool isStartRun){ DebugPrint("%s", "FSUDAQ"); if( isStartRun ) runID ++; QString runIDStr = QString::number(runID).rightJustified(3, '0'); int result = QDialog::Rejected ; QLineEdit *lineEdit = new QLineEdit(this); if( needManualComment ) { QDialog * dOpen = new QDialog(this); if( isStartRun ) { dOpen->setWindowTitle("Start Run Comment"); }else{ dOpen->setWindowTitle("Stop Run Comment"); } dOpen->setWindowFlags(Qt::Dialog | Qt::WindowTitleHint | Qt::CustomizeWindowHint); dOpen->setMinimumWidth(600); connect(dOpen, &QDialog::finished, dOpen, &QDialog::deleteLater); QGridLayout * vlayout = new QGridLayout(dOpen); QLabel *label = new QLabel("Enter Run comment for Run-" + runIDStr + " : ", dOpen); QPushButton *button1 = new QPushButton("OK", dOpen); QPushButton *button2 = new QPushButton("Cancel", dOpen); vlayout->addWidget(label, 0, 0, 1, 2); vlayout->addWidget(lineEdit, 1, 0, 1, 2); vlayout->addWidget(button1, 2, 0); vlayout->addWidget(button2, 2, 1); connect(button1, &QPushButton::clicked, dOpen, &QDialog::accept); connect(button2, &QPushButton::clicked, dOpen, &QDialog::reject); result = dOpen->exec(); }else{ if( isStartRun ){ lineEdit->setText("Auto Start, repeat every " + QString::number(qAbs(cbAutoRun->currentData().toInt())) + " mins."); }else{ lineEdit->setText("Auto Stop, after " + QString::number(qAbs(cbAutoRun->currentData().toInt())) + " mins."); } result = QDialog::Accepted; } if(result == QDialog::Accepted ){ if( isStartRun ){ startComment = lineEdit->text(); if( startComment == "") startComment = "No commet was typed."; if( needManualComment ){ int minute = cbAutoRun->currentData().toInt(); if( minute > 0 ) { startComment += ", single run of " + QString::number(minute) + " mins."; }else{ startComment += ", repeat run of " + QString::number(qAbs(minute)) + " mins."; } } startComment = "Start Comment: " + startComment; leComment->setText(startComment); leRunID->setText(QString::number(runID)); }else{ stopComment = lineEdit->text(); if( stopComment == "") stopComment = "No commet was typed."; stopComment = "Stop Comment: " + stopComment; leComment->setText(stopComment); } }else{ if( isStartRun ){ LogMsg("Start Run aborted. "); runID --; leRunID->setText(QString::number(runID)); }else{ LogMsg("Stop Run cancelled. "); } return false; } return true; } void FSUDAQ::WriteRunTimestamp(bool isStartRun){ DebugPrint("%s", "FSUDAQ"); QFile file(rawDataPath + "/RunTimeStamp.dat"); QString dateTime = QDateTime::currentDateTime().toString("yyyy.MM.dd hh:mm:ss"); if( file.open(QIODevice::Text | QIODevice::WriteOnly | QIODevice::Append) ){ if( isStartRun ){ file.write(("Start Run | " + QString::number(runID) + " | " + dateTime + " | " + lePrefix->text() + " | " + startComment + "\n").toStdString().c_str()); }else{ file.write((" Stop Run | " + QString::number(runID) + " | " + dateTime + " | " + lePrefix->text() + " | " + stopComment + "\n").toStdString().c_str()); } file.close(); } QFile fileCSV(rawDataPath + "/RunTimeStamp.csv"); if( fileCSV.open(QIODevice::Text | QIODevice::WriteOnly | QIODevice::Append) ){ QTextStream out(&fileCSV); if( isStartRun){ out << QString::number(runID) + "," + dateTime + "," + startComment; }else{ out << "," + dateTime + "," + stopComment + "\n"; } fileCSV.close(); } UpdateRecord(); } //*************************************************************** //*************************************************************** void FSUDAQ::OpenScope(){ DebugPrint("%s", "FSUDAQ"); QCoreApplication::processEvents(); if( scope == nullptr ) { scope = new Scope(digi, nDigi, readDataThread); connect(scope, &Scope::SendLogMsg, this, &FSUDAQ::LogMsg); connect(scope, &Scope::CloseWindow, this, [=](){ bnStartACQ->setEnabled(true); bnStartACQ->setStyleSheet("background-color: green;"); bnStopACQ->setEnabled(false); bnStopACQ->setStyleSheet(""); }); connect(scope, &Scope::TellACQOnOff, this, [=](bool onOff){ isACQStarted = onOff; if( onOff ){ if( influx && chkInflux->isChecked() && !elogName.isEmpty()) influx->AddDataPoint("SavingData,ExpName=" + elogName.toStdString() + " value=1"); if( scalar ){ lbScalarACQStatus->setText("ACQ On"); // scalarTimer->start(ScalarUpdateinMiliSec); scalarTimingThread->start(); } if( singleHistograms ) singleHistograms->startTimer(); if( onlineAnalyzer ) onlineAnalyzer->startTimer(); }else{ if( influx && chkInflux->isChecked() && !elogName.isEmpty()) influx->AddDataPoint("SavingData,ExpName=" + elogName.toStdString() + " value=0"); if( scalar ){ lbScalarACQStatus->setText("ACQ Off"); // scalarTimer->stop(); scalarTimingThread->Stop(); scalarTimingThread->quit(); scalarTimingThread->wait(); } if( singleHistograms ) singleHistograms->stopTimer(); if( onlineAnalyzer ) onlineAnalyzer->stopTimer(); } if( digiSettings ) digiSettings->EnableButtons(!onOff); }); connect(scope, &Scope::UpdateOtherPanels, this, [=](){ UpdateAllPanels(1); }); scope->show(); }else{ scope->show(); scope->activateWindow(); } bnStartACQ->setEnabled(false); bnStartACQ->setStyleSheet(""); bnStopACQ->setEnabled(false); bnStopACQ->setStyleSheet(""); chkSaveData->setChecked(false); } //*************************************************************** //*************************************************************** void FSUDAQ::OpenDigiSettings(){ DebugPrint("%s", "FSUDAQ"); if( digiSettings == nullptr ) { digiSettings = new DigiSettingsPanel(digi, nDigi, rawDataPath); //connect(scope, &Scope::SendLogMsg, this, &FSUDAQ::LogMsg); connect(digiSettings, &DigiSettingsPanel::UpdateOtherPanels, this, [=](){ UpdateAllPanels(2); }); connect(digiSettings, &DigiSettingsPanel::SendLogMsg, this, &FSUDAQ::LogMsg); digiSettings->show(); }else{ digiSettings->show(); digiSettings->activateWindow(); } } //*************************************************************** //*************************************************************** void FSUDAQ::OpenSingleHistograms(){ DebugPrint("%s", "FSUDAQ"); if( singleHistograms == nullptr ) { singleHistograms = new SingleSpectra(digi, nDigi, rawDataPath); singleHistograms->show(); }else{ singleHistograms->show(); singleHistograms->activateWindow(); singleHistograms->LoadSetting(); } } //*************************************************************** //*************************************************************** void FSUDAQ::OpenAnalyzer(){ DebugPrint("%s", "FSUDAQ"); int id = cbAnalyzer->currentData().toInt(); if( id < 0 ) return; if( onlineAnalyzer == nullptr ) { if( id == 0 ) onlineAnalyzer = new CoincidentAnalyzer(digi, nDigi, rawDataPath); if( id == 1 ) onlineAnalyzer = new SplitPole(digi, nDigi); if( id == 2 ) onlineAnalyzer = new Encore(digi, nDigi); if( id == 3 ) onlineAnalyzer = new MUSIC(digi, nDigi); if( id == 4 ) onlineAnalyzer = new NeutronGamma(digi, nDigi, rawDataPath); if( id == 5 ) onlineAnalyzer = new Cross(digi, nDigi); if( id >= 0 ) onlineAnalyzer->show(); if( isACQStarted ) onlineAnalyzer->startTimer(); }else{ delete onlineAnalyzer; if( id == 0 ) onlineAnalyzer = new CoincidentAnalyzer(digi, nDigi, rawDataPath); if( id == 1 ) onlineAnalyzer = new SplitPole(digi, nDigi); if( id == 2 ) onlineAnalyzer = new Encore(digi, nDigi); if( id == 3 ) onlineAnalyzer = new MUSIC(digi, nDigi); if( id == 4 ) onlineAnalyzer = new NeutronGamma(digi, nDigi, rawDataPath); if( id == 5 ) onlineAnalyzer = new Cross(digi, nDigi); if( id >= 0 ){ onlineAnalyzer->show(); onlineAnalyzer->activateWindow(); if( isACQStarted ) onlineAnalyzer->stopTimer(); } } cbAnalyzer->setCurrentIndex(0); } //*************************************************************** //*************************************************************** void FSUDAQ::UpdateAllPanels(int panelID){ DebugPrint("%s", "FSUDAQ"); //panelID is the source panel that call // scope = 1; // digiSetting = 2; if( panelID == 1 ){ // from scope if( digiSettings && digiSettings->isVisible() ) digiSettings->UpdatePanelFromMemory(); if( scalar ) { for( unsigned int iDigi = 0; iDigi < nDigi; iDigi++){ if( digi[iDigi]->IsBoardDisabled() ) continue; uint32_t acqStatus = digi[iDigi]->GetACQStatusFromMemory(); if( ( acqStatus >> 2 ) & 0x1 ){ runStatus[iDigi]->setStyleSheet("background-color : green;"); }else{ runStatus[iDigi]->setStyleSheet(""); } } } } if( panelID == 2 ){ if(scope && scope->isVisible() ) scope->UpdatePanelFromMomeory(); if(scalar) { for( unsigned int iDigi = 0; iDigi < nDigi; iDigi++){ uint32_t chMask = digi[iDigi]->GetRegChannelMask(); uint32_t subChMask = 0; for( int ch = 0; ch < digi[iDigi]->GetNumInputCh(); ch++){ // leTrigger[iDigi][i]->setEnabled( (chMask >> i) & 0x1 ); // leAccept[iDigi][i]->setEnabled( (chMask >> i) & 0x1 ); if( digi[iDigi]->IsInputChEqRegCh() ){ leTrigger[iDigi][ch]->setEnabled( (chMask >> ch) & 0x1 ); leAccept[iDigi][ch]->setEnabled( (chMask >> ch) & 0x1 ); }else{ int grpID = ch/digi[iDigi]->GetNumRegChannels(); leTrigger[iDigi][ch]->setEnabled( (chMask >> grpID) & 0x1 ); leAccept[iDigi][ch]->setEnabled( (chMask >> grpID) & 0x1 ); if( (chMask >> grpID ) & 0x1 ){ int subCh = ch%digi[iDigi]->GetNumRegChannels(); if( subCh == 0 ) subChMask = digi[iDigi]->GetSettingFromMemory(DPP::QDC::SubChannelMask, grpID); leTrigger[iDigi][ch]->setEnabled( (subChMask >> subCh) & 0x1 ); leAccept[iDigi][ch]->setEnabled( (subChMask >> subCh) & 0x1 ); } } } } } } } //*************************************************************** //*************************************************************** void FSUDAQ::SetUpInflux(){ DebugPrint("%s", "FSUDAQ"); if( influxIP == "" ) { LogMsg("Influx missing inputs. skip."); leInfluxIP->setEnabled(false); leDatabaseName->setEnabled(false); return; } if( influx ) { delete influx; influx = nullptr; } influx = new InfluxDB(influxIP.toStdString(), false); if( influx->TestingConnection() ){ LogMsg(" InfluxDB URL ("+ influxIP + ") is Valid. Version : " + QString::fromStdString(influx->GetVersionString())+ " "); if( influx->GetVersionNo() > 1 && influxToken.isEmpty() ) { LogMsg("A Token is required for accessing the database."); delete influx; influx = nullptr; return; } influx->SetToken(influxToken.toStdString()); //==== chck database exist influx->CheckDatabases(); std::vector databaseList = influx->GetDatabaseList(); bool foundDatabase = false; for( int i = 0; i < (int) databaseList.size(); i++){ if( databaseList[i] == dataBaseName.toStdString() ) foundDatabase = true; // LogMsg(QString::number(i) + "|" + QString::fromStdString(databaseList[i])); } if( foundDatabase ){ LogMsg(" Database " + dataBaseName + " found."); influx->AddDataPoint("ProgramStart value=1"); influx->WriteData(dataBaseName.toStdString()); influx->ClearDataPointsBuffer(); if( influx->IsWriteOK() ){ LogMsg("test write database OK."); }else{ LogMsg("test write database FAIL."); } }else{ LogMsg(" Database " + dataBaseName + " NOT found."); delete influx; influx = nullptr; } }else{ LogMsg(" InfluxDB URL ("+ influxIP + ") is NOT Valid "); delete influx; influx = nullptr; } if( influx == nullptr ){ leInfluxIP->setEnabled(false); leDatabaseName->setEnabled(false); } } void FSUDAQ::CheckElog(){ elogID = -1; DebugPrint("%s", "FSUDAQ"); if( !chkElog->isChecked() ) { LogMsg("Elog is disabled. Please chick the checkbox and lock again to check elog connectivity."); leElogIP->setEnabled(false); leElogName->setEnabled(false); return; } LogMsg("---- Checking elog... please wait...."); // printf("---- Checking elog... please wait....\n"); if( elogIP != "" && elogName != "" && elogUser != "" && elogPWD != "" ){ WriteElog("Testing communication.", "Testing communication.", "Other", 0); AppendElog("test append elog."); }else{ LogMsg("Elog missing inputs. skip."); leElogIP->setEnabled(false); leElogName->setEnabled(false); return; } if( elogID >= 0 ) { LogMsg("Elog testing OK."); // printf("Elog testing OK.\n"); return; } //QMessageBox::information(nullptr, "Information", "Elog write Fail.\nPlease set the elog User and PWD in the programSettings.txt.\nline 6 = user.\nline 7 = pwd."); LogMsg("Elog testing Fail"); // printf("Elog testing Fail\n"); if( elogIP == "" ) LogMsg("no elog IP"); if( elogName == "" ) LogMsg("no elog Name"); if( elogUser == "" ) LogMsg("no elog User name. Please set it in the programSettings.txt, line 6."); if( elogPWD == "" ) LogMsg("no elog User pwd. Please set it in the programSettings.txt, line 7."); if( elogID < 0 ) LogMsg("Possible elog IP, Name, User name, pwd incorrect, or elog not installed."); leElogIP->setEnabled(false); leElogName->setEnabled(false); } void FSUDAQ::WriteElog(QString htmlText, QString subject, QString category, int runNumber){ DebugPrint("%s", "FSUDAQ"); //if( elogID < 0 ) return; if( !chkElog->isChecked() ) return; if( elogName == "" ) return; if( elogUser == "" ) return; if( elogPWD == "" ) return; QStringList arg; arg << "-h" << elogIP << "-p" << "8080" << "-l" << elogName << "-u" << elogUser << elogPWD << "-a" << "Author=FSUDAQ"; if( runNumber > 0 ) arg << "-a" << "RunNo=" + QString::number(runNumber); if( category != "" ) arg << "-a" << "Category=" + category; arg << "-a" << "Subject=" + subject << "-n " << "2" << htmlText ; QProcess elogBash(this); elogBash.start("elog", arg); elogBash.waitForFinished(); QString output = QString::fromUtf8(elogBash.readAllStandardOutput()); QRegularExpression regex("ID=(\\d+)"); QRegularExpressionMatch match = regex.match(output); if (match.hasMatch()) { QString id = match.captured(1); elogID = id.toInt(); } else { elogID = -1; } } void FSUDAQ::AppendElog(QString appendHtmlText){ DebugPrint("%s", "FSUDAQ"); if( !chkElog->isChecked() )return; if( elogID < 1 ) return; if( elogName == "" ) return; QProcess elogBash(this); QStringList arg; arg << "-h" << elogIP << "-p" << "8080" << "-l" << elogName << "-u" << elogUser << elogPWD << "-w" << QString::number(elogID); //retrevie the elog elogBash.start("elog", arg); elogBash.waitForFinished(); QString output = QString::fromUtf8(elogBash.readAllStandardOutput()); //qDebug() << output; QString separator = "========================================"; int index = output.indexOf(separator); if( index != -1){ QString originalHtml = output.mid(index + separator.length()); arg.clear(); arg << "-h" << elogIP << "-p" << "8080" << "-l" << elogName << "-u" << elogUser << elogPWD << "-e" << QString::number(elogID) << "-n" << "2" << originalHtml + "
" + appendHtmlText; elogBash.start("elog", arg); elogBash.waitForFinished(); output = QString::fromUtf8(elogBash.readAllStandardOutput()); index = output.indexOf("ID="); if( index != -1 ){ elogID = output.mid(index+3).toInt(); }else{ elogID = -1; } }else{ elogID = -1; } } //*************************************************************** //*************************************************************** void FSUDAQ::LogMsg(QString msg){ DebugPrint("%s", "FSUDAQ"); QString outputStr = QStringLiteral("[%1] %2").arg(QDateTime::currentDateTime().toString("MM.dd hh:mm:ss"), msg); if( logMsgHTMLMode ){ logInfo->appendHtml(outputStr); }else{ logInfo->appendPlainText(outputStr); } QScrollBar *v = logInfo->verticalScrollBar(); v->setValue(v->maximum()); //qDebug() << outputStr; logInfo->repaint(); }