2023-11-16 19:20:29 -05:00
# ifndef COINCIDENTANLAYZER_H
# define COINCIDENTANLAYZER_H
# include "Analyser.h"
//^===========================================
class CoincidentAnalyzer : public Analyzer {
Q_OBJECT
public :
CoincidentAnalyzer ( Digitizer * * digi , unsigned int nDigi , QMainWindow * parent = nullptr ) : Analyzer ( digi , nDigi , parent ) {
this - > digi = digi ;
this - > nDigi = nDigi ;
SetUpdateTimeInSec ( 1.0 ) ;
//RedefineEventBuilder({0}); // only build for the 0-th digitizer, otherwise, it will build event accross all digitizers
SetBackwardBuild ( false , 100 ) ; // using normal building (acceding in time) or backward building, int the case of backward building, default events to be build is 100.
evtbder = GetEventBuilder ( ) ;
evtbder - > SetTimeWindow ( 500 ) ;
//========== use the influx from the Analyzer
influx = new InfluxDB ( " https://fsunuc.physics.fsu.edu/influx/ " ) ;
dataBaseName = " testing " ;
2023-11-21 16:50:22 -05:00
allowSignalSlot = false ;
2023-11-16 19:20:29 -05:00
SetUpCanvas ( ) ;
2023-11-21 16:50:22 -05:00
LoadHistRange ( ) ;
}
~ CoincidentAnalyzer ( ) {
SaveHistRange ( ) ;
2023-11-16 19:20:29 -05:00
}
void SetUpCanvas ( ) ;
public slots :
void UpdateHistograms ( ) ;
private :
Digitizer * * digi ;
unsigned int nDigi ;
MultiBuilder * evtbder ;
2023-11-21 16:50:22 -05:00
bool allowSignalSlot ;
2023-11-16 19:20:29 -05:00
// declaie histograms
Histogram2D * h2D ;
Histogram1D * h1 ;
Histogram1D * h1g ;
Histogram1D * hMulti ;
2023-11-17 16:47:11 -05:00
QCheckBox * chkRunAnalyzer ;
2023-11-16 19:20:29 -05:00
RSpinBox * sbUpdateTime ;
2023-11-17 16:47:11 -05:00
QCheckBox * chkBackWardBuilding ;
2023-11-16 19:20:29 -05:00
RSpinBox * sbBackwardCount ;
RSpinBox * sbBuildWindow ;
// data source for the h2D
RComboBox * xDigi ;
RComboBox * xCh ;
RComboBox * yDigi ;
RComboBox * yCh ;
// data source for the h1
RComboBox * aDigi ;
RComboBox * aCh ;
2023-11-21 16:50:22 -05:00
void SaveHistRange ( ) ;
void LoadHistRange ( ) ;
2023-11-16 19:20:29 -05:00
} ;
inline void CoincidentAnalyzer : : SetUpCanvas ( ) {
setGeometry ( 0 , 0 , 1600 , 1000 ) ;
{ //^====== magnet and reaction setting
QGroupBox * box = new QGroupBox ( " Configuration " , this ) ;
layout - > addWidget ( box , 0 , 0 ) ;
QGridLayout * boxLayout = new QGridLayout ( box ) ;
boxLayout - > setAlignment ( Qt : : AlignTop | Qt : : AlignLeft ) ;
box - > setLayout ( boxLayout ) ;
2023-11-21 16:50:22 -05:00
{
chkRunAnalyzer = new QCheckBox ( " Run Analyzer " , this ) ;
boxLayout - > addWidget ( chkRunAnalyzer , 0 , 0 ) ;
QLabel * lbUpdateTime = new QLabel ( " Update Period [s] " , this ) ;
lbUpdateTime - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbUpdateTime , 0 , 1 ) ;
sbUpdateTime = new RSpinBox ( this , 1 ) ;
sbUpdateTime - > setMinimum ( 0.1 ) ;
sbUpdateTime - > setMaximum ( 5 ) ;
sbUpdateTime - > setValue ( 1 ) ;
boxLayout - > addWidget ( sbUpdateTime , 0 , 2 ) ;
2023-11-21 17:31:46 -05:00
connect ( sbUpdateTime , & RSpinBox : : valueChanged , this , [ = ] ( ) { sbUpdateTime - > setStyleSheet ( " color : blue " ) ; } ) ;
connect ( sbUpdateTime , & RSpinBox : : returnPressed , this , [ = ] ( ) {
sbUpdateTime - > setStyleSheet ( " " ) ;
SetUpdateTimeInSec ( sbUpdateTime - > value ( ) ) ;
} ) ;
2023-11-21 16:50:22 -05:00
chkBackWardBuilding = new QCheckBox ( " Use Backward builder " , this ) ;
boxLayout - > addWidget ( chkBackWardBuilding , 1 , 0 ) ;
QLabel * lbBKWindow = new QLabel ( " Max No. Backward Event " , this ) ;
lbBKWindow - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbBKWindow , 1 , 1 ) ;
sbBackwardCount = new RSpinBox ( this , 0 ) ;
sbBackwardCount - > setMinimum ( 1 ) ;
sbBackwardCount - > setMaximum ( 9999 ) ;
sbBackwardCount - > setValue ( 100 ) ;
boxLayout - > addWidget ( sbBackwardCount , 1 , 2 ) ;
chkBackWardBuilding - > setChecked ( false ) ;
sbBackwardCount - > setEnabled ( false ) ;
connect ( chkBackWardBuilding , & QCheckBox : : stateChanged , this , [ = ] ( int status ) {
SetBackwardBuild ( status , sbBackwardCount - > value ( ) ) ;
sbBackwardCount - > setEnabled ( status ) ;
2023-11-21 17:31:46 -05:00
SetBackwardBuild ( true , sbBackwardCount - > value ( ) ) ;
} ) ;
connect ( sbBackwardCount , & RSpinBox : : valueChanged , this , [ = ] ( ) {
sbBackwardCount - > setStyleSheet ( " color : blue; " ) ;
2023-11-21 16:50:22 -05:00
} ) ;
2023-11-21 17:31:46 -05:00
connect ( sbBackwardCount , & RSpinBox : : returnPressed , this , [ = ] ( ) {
sbBackwardCount - > setStyleSheet ( " " ) ;
SetBackwardBuild ( true , sbBackwardCount - > value ( ) ) ;
2023-11-21 16:50:22 -05:00
} ) ;
QLabel * lbBuildWindow = new QLabel ( " Event Window [tick] " , this ) ;
lbBuildWindow - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbBuildWindow , 2 , 1 ) ;
sbBuildWindow = new RSpinBox ( this , 0 ) ;
sbBuildWindow - > setMinimum ( 1 ) ;
sbBuildWindow - > setMaximum ( 9999999999 ) ;
boxLayout - > addWidget ( sbBuildWindow , 2 , 2 ) ;
2023-11-21 17:31:46 -05:00
connect ( sbBuildWindow , & RSpinBox : : valueChanged , this , [ = ] ( ) {
sbBuildWindow - > setStyleSheet ( " color : blue; " ) ;
} ) ;
connect ( sbBuildWindow , & RSpinBox : : returnPressed , this , [ = ] ( ) {
sbBuildWindow - > setStyleSheet ( " " ) ;
evtbder - > SetTimeWindow ( ( int ) sbBuildWindow - > value ( ) ) ;
2023-11-21 16:50:22 -05:00
} ) ;
2023-11-16 19:20:29 -05:00
}
2023-11-21 16:50:22 -05:00
{
QFrame * separator = new QFrame ( box ) ;
separator - > setFrameShape ( QFrame : : HLine ) ;
separator - > setFrameShadow ( QFrame : : Sunken ) ;
boxLayout - > addWidget ( separator , 3 , 0 , 1 , 4 ) ;
QLabel * lbXDigi = new QLabel ( " X-Digi " , this ) ;
lbXDigi - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbXDigi , 4 , 0 ) ;
xDigi = new RComboBox ( this ) ;
for ( unsigned int i = 0 ; i < nDigi ; i + + ) {
xDigi - > addItem ( " Digi- " + QString : : number ( digi [ i ] - > GetSerialNumber ( ) ) , i ) ;
}
boxLayout - > addWidget ( xDigi , 4 , 1 ) ;
QLabel * lbXCh = new QLabel ( " X-Ch " , this ) ;
lbXCh - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbXCh , 4 , 2 ) ;
xCh = new RComboBox ( this ) ;
for ( int i = 0 ; i < digi [ 0 ] - > GetNumInputCh ( ) ; i + + ) xCh - > addItem ( " Ch- " + QString : : number ( i ) , i ) ;
boxLayout - > addWidget ( xCh , 4 , 3 ) ;
QLabel * lbYDigi = new QLabel ( " Y-Digi " , this ) ;
lbYDigi - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbYDigi , 5 , 0 ) ;
yDigi = new RComboBox ( this ) ;
for ( unsigned int i = 0 ; i < nDigi ; i + + ) {
yDigi - > addItem ( " Digi- " + QString : : number ( digi [ i ] - > GetSerialNumber ( ) ) , i ) ;
}
boxLayout - > addWidget ( yDigi , 5 , 1 ) ;
QLabel * lbYCh = new QLabel ( " Y-Ch " , this ) ;
lbYCh - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbYCh , 5 , 2 ) ;
yCh = new RComboBox ( this ) ;
for ( int i = 0 ; i < digi [ 0 ] - > GetNumInputCh ( ) ; i + + ) yCh - > addItem ( " Ch- " + QString : : number ( i ) , i ) ;
boxLayout - > addWidget ( yCh , 5 , 3 ) ;
connect ( xDigi , & RComboBox : : currentIndexChanged , this , [ = ] ( ) {
allowSignalSlot = false ;
xCh - > clear ( ) ;
for ( int i = 0 ; i < digi [ 0 ] - > GetNumInputCh ( ) ; i + + ) xCh - > addItem ( " Ch- " + QString : : number ( i ) , i ) ;
allowSignalSlot = true ;
int bd = xDigi - > currentData ( ) . toInt ( ) ;
int ch = xCh - > currentData ( ) . toInt ( ) ;
h2D - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h2D - > UpdatePlot ( ) ;
} ) ;
connect ( xCh , & RComboBox : : currentIndexChanged , this , [ = ] ( ) {
if ( ! allowSignalSlot ) return ;
int bd = xDigi - > currentData ( ) . toInt ( ) ;
int ch = xCh - > currentData ( ) . toInt ( ) ;
h2D - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h2D - > UpdatePlot ( ) ;
} ) ;
connect ( yDigi , & RComboBox : : currentIndexChanged , this , [ = ] ( ) {
allowSignalSlot = false ;
yCh - > clear ( ) ;
for ( int i = 0 ; i < digi [ 0 ] - > GetNumInputCh ( ) ; i + + ) yCh - > addItem ( " Ch- " + QString : : number ( i ) , i ) ;
allowSignalSlot = true ;
int bd = yDigi - > currentData ( ) . toInt ( ) ;
int ch = yCh - > currentData ( ) . toInt ( ) ;
h2D - > SetYTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h2D - > UpdatePlot ( ) ;
} ) ;
connect ( yCh , & RComboBox : : currentIndexChanged , this , [ = ] ( ) {
if ( ! allowSignalSlot ) return ;
int bd = yDigi - > currentData ( ) . toInt ( ) ;
int ch = yCh - > currentData ( ) . toInt ( ) ;
h2D - > SetYTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h2D - > UpdatePlot ( ) ;
} ) ;
2023-11-16 19:20:29 -05:00
}
2023-11-21 16:50:22 -05:00
{
QFrame * separator1 = new QFrame ( box ) ;
separator1 - > setFrameShape ( QFrame : : HLine ) ;
separator1 - > setFrameShadow ( QFrame : : Sunken ) ;
boxLayout - > addWidget ( separator1 , 6 , 0 , 1 , 4 ) ;
QLabel * lbaDigi = new QLabel ( " ID-Digi " , this ) ;
lbaDigi - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbaDigi , 7 , 0 ) ;
aDigi = new RComboBox ( this ) ;
for ( unsigned int i = 0 ; i < nDigi ; i + + ) {
aDigi - > addItem ( " Digi- " + QString : : number ( digi [ i ] - > GetSerialNumber ( ) ) , i ) ;
}
boxLayout - > addWidget ( aDigi , 7 , 1 ) ;
QLabel * lbaCh = new QLabel ( " 1D-Ch " , this ) ;
lbaCh - > setAlignment ( Qt : : AlignRight | Qt : : AlignCenter ) ;
boxLayout - > addWidget ( lbaCh , 7 , 2 ) ;
aCh = new RComboBox ( this ) ;
for ( int i = 0 ; i < digi [ 0 ] - > GetNumInputCh ( ) ; i + + ) aCh - > addItem ( " Ch- " + QString : : number ( i ) , i ) ;
boxLayout - > addWidget ( aCh , 7 , 3 ) ;
connect ( aDigi , & RComboBox : : currentIndexChanged , this , [ = ] ( ) {
allowSignalSlot = false ;
aCh - > clear ( ) ;
for ( int i = 0 ; i < digi [ 0 ] - > GetNumInputCh ( ) ; i + + ) aCh - > addItem ( " Ch- " + QString : : number ( i ) , i ) ;
allowSignalSlot = true ;
int bd = aDigi - > currentData ( ) . toInt ( ) ;
int ch = aCh - > currentData ( ) . toInt ( ) ;
h1 - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h1 - > UpdatePlot ( ) ;
h1g - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h1g - > UpdatePlot ( ) ;
} ) ;
connect ( aCh , & RComboBox : : currentIndexChanged , this , [ = ] ( ) {
if ( ! allowSignalSlot ) return ;
int bd = aDigi - > currentData ( ) . toInt ( ) ;
int ch = aCh - > currentData ( ) . toInt ( ) ;
h1 - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h1 - > UpdatePlot ( ) ;
h1g - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h1g - > UpdatePlot ( ) ;
} ) ;
2023-11-16 19:20:29 -05:00
}
2023-11-21 16:50:22 -05:00
{
QFrame * separator1 = new QFrame ( box ) ;
separator1 - > setFrameShape ( QFrame : : HLine ) ;
separator1 - > setFrameShadow ( QFrame : : Sunken ) ;
boxLayout - > addWidget ( separator1 , 8 , 0 , 1 , 4 ) ;
QPushButton * bnClearHist = new QPushButton ( " Clear All Hist. " ) ;
boxLayout - > addWidget ( bnClearHist , 9 , 1 ) ;
connect ( bnClearHist , & QPushButton : : clicked , this , [ = ] ( ) {
h2D - > Clear ( ) ;
h1 - > Clear ( ) ;
h1g - > Clear ( ) ;
hMulti - > Clear ( ) ;
} ) ;
}
2023-11-16 19:20:29 -05:00
2023-11-17 16:47:11 -05:00
2023-11-16 19:20:29 -05:00
}
//============ histograms
hMulti = new Histogram1D ( " Multiplicity " , " " , 10 , 0 , 10 , this ) ;
layout - > addWidget ( hMulti , 0 , 1 ) ;
// the "this" make the histogram a child of the SplitPole class. When SplitPole destory, all childs destory as well.
h2D = new Histogram2D ( " Coincident Plot " , " XXX " , " YYY " , 100 , 0 , 5000 , 100 , 0 , 5000 , this ) ;
//layout is inheriatge from Analyzer
layout - > addWidget ( h2D , 1 , 0 , 2 , 1 ) ;
2023-11-21 16:50:22 -05:00
int bd = xDigi - > currentData ( ) . toInt ( ) ;
int ch = xCh - > currentData ( ) . toInt ( ) ;
h2D - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
bd = yDigi - > currentData ( ) . toInt ( ) ;
ch = yCh - > currentData ( ) . toInt ( ) ;
h2D - > SetYTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h2D - > UpdatePlot ( ) ;
h1 = new Histogram1D ( " 1D Plot " , " XXX " , 300 , 0 , 5000 , this ) ;
2023-11-16 19:20:29 -05:00
h1 - > SetColor ( Qt : : darkGreen ) ;
h1 - > AddDataList ( " Test " , Qt : : red ) ; // add another histogram in h1, Max Data List is 10
2023-11-21 16:50:22 -05:00
bd = aDigi - > currentData ( ) . toInt ( ) ;
ch = aCh - > currentData ( ) . toInt ( ) ;
h1 - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h1 - > UpdatePlot ( ) ;
2023-11-16 19:20:29 -05:00
layout - > addWidget ( h1 , 1 , 1 ) ;
2023-11-21 16:50:22 -05:00
h1g = new Histogram1D ( " 1D Plot (PID gated) " , " XXX " , 300 , 0 , 5000 , this ) ;
h1g - > SetXTitle ( " Digi- " + QString : : number ( digi [ bd ] - > GetSerialNumber ( ) ) + " , Ch- " + QString : : number ( ch ) ) ;
h1g - > UpdatePlot ( ) ;
2023-11-16 19:20:29 -05:00
layout - > addWidget ( h1g , 2 , 1 ) ;
layout - > setColumnStretch ( 0 , 1 ) ;
layout - > setColumnStretch ( 1 , 1 ) ;
}
inline void CoincidentAnalyzer : : UpdateHistograms ( ) {
if ( this - > isVisible ( ) = = false ) return ;
2023-11-17 16:47:11 -05:00
if ( chkRunAnalyzer - > isChecked ( ) = = false ) return ;
2023-11-16 19:20:29 -05:00
BuildEvents ( ) ; // call the event builder to build events
//============ Get events, and do analysis
long eventBuilt = evtbder - > eventBuilt ;
if ( eventBuilt = = 0 ) return ;
//============ Get the cut list, if any
QList < QPolygonF > cutList = h2D - > GetCutList ( ) ;
const int nCut = cutList . count ( ) ;
unsigned long long tMin [ nCut ] = { 0xFFFFFFFFFFFFFFFF } , tMax [ nCut ] = { 0 } ;
unsigned int count [ nCut ] = { 0 } ;
2023-11-21 16:50:22 -05:00
//============ Get the channel to plot
int a_bd = aDigi - > currentData ( ) . toInt ( ) ;
int a_ch = aCh - > currentData ( ) . toInt ( ) ;
int x_bd = xDigi - > currentData ( ) . toInt ( ) ;
int x_ch = xCh - > currentData ( ) . toInt ( ) ;
int y_bd = yDigi - > currentData ( ) . toInt ( ) ;
int y_ch = yCh - > currentData ( ) . toInt ( ) ;
2023-11-16 19:20:29 -05:00
//============ Processing data and fill histograms
long eventIndex = evtbder - > eventIndex ;
long eventStart = eventIndex - eventBuilt + 1 ;
if ( eventStart < 0 ) eventStart + = MaxNEvent ;
for ( long i = eventStart ; i < = eventIndex ; i + + ) {
std : : vector < Hit > event = evtbder - > events [ i ] ;
hMulti - > Fill ( ( int ) event . size ( ) ) ;
if ( event . size ( ) = = 0 ) return ;
2023-11-21 16:50:22 -05:00
int aE = - 1 ;
int xE = - 1 , yE = - 1 ;
unsigned long long xT = 0 ;
2023-11-16 19:20:29 -05:00
for ( int k = 0 ; k < ( int ) event . size ( ) ; k + + ) {
//event[k].Print();
2023-11-21 16:50:22 -05:00
if ( event [ k ] . bd = = a_bd & & event [ k ] . ch = = a_ch ) {
h1 - > Fill ( event [ k ] . energy ) ;
aE = event [ k ] . energy ;
}
if ( event [ k ] . bd = = x_bd & & event [ k ] . ch = = x_ch ) {
xE = event [ k ] . energy ;
xT = event [ k ] . timestamp ;
}
if ( event [ k ] . bd = = y_bd & & event [ k ] . ch = = y_ch ) yE = event [ k ] . energy ;
2023-11-16 19:20:29 -05:00
}
2023-11-21 16:50:22 -05:00
if ( xE > = 0 & & yE > = 0 ) h2D - > Fill ( xE , yE ) ;
//check events inside any Graphical cut and extract the rate
2023-11-16 19:20:29 -05:00
for ( int p = 0 ; p < cutList . count ( ) ; p + + ) {
if ( cutList [ p ] . isEmpty ( ) ) continue ;
2023-11-21 16:50:22 -05:00
if ( cutList [ p ] . containsPoint ( QPointF ( xE , yE ) , Qt : : OddEvenFill ) & & xE > = 0 & & yE > = 0 ) {
if ( xT < tMin [ p ] ) tMin [ p ] = xT ;
if ( xT > tMax [ p ] ) tMax [ p ] = xT ;
count [ p ] + + ;
//printf(".... %d \n", count[p]);
if ( p = = 0 & & aE > = 0 ) h1g - > Fill ( aE ) ; // only for the 1st gate
}
2023-11-16 19:20:29 -05:00
}
}
h2D - > UpdatePlot ( ) ;
h1 - > UpdatePlot ( ) ;
hMulti - > UpdatePlot ( ) ;
h1g - > UpdatePlot ( ) ;
2023-11-21 16:50:22 -05:00
// QList<QString> cutNameList = h2D->GetCutNameList();
// for( int p = 0; p < cutList.count(); p ++){
// if( cutList[p].isEmpty() ) continue;
2023-11-16 19:20:29 -05:00
// double dT = (tMax[p]-tMin[p]) * tick2ns / 1e9; // tick to sec
// double rate = count[p]*1.0/(dT);
//printf("%llu %llu, %f %d\n", tMin[p], tMax[p], dT, count[p]);
//printf("%10s | %d | %f Hz \n", cutNameList[p].toStdString().c_str(), count[p], rate);
// influx->AddDataPoint("Cut,name=" + cutNameList[p].toStdString()+ " value=" + std::to_string(rate));
// influx->WriteData(dataBaseName);
// influx->ClearDataPointsBuffer();
2023-11-21 16:50:22 -05:00
// }
}
inline void CoincidentAnalyzer : : SaveHistRange ( ) {
}
inline void CoincidentAnalyzer : : LoadHistRange ( ) {
2023-11-16 19:20:29 -05:00
}
# endif