#include "ANALYSIS/DReaction.h" DReaction::DReaction(string locReactionName) : dReactionName(locReactionName) { dKinFitType = d_NoFit; dChargedTrackFactoryTag = ""; dNeutralShowerFactoryTag = ""; dMinCombinedPIDFOM.first = false; dMinCombinedPIDFOM.second = 0.0; dMinCombinedTrackingFOM.first = false; dMinCombinedTrackingFOM.second = 0.0; dMinChargedPIDFOM.first = false; dMinChargedPIDFOM.second = 0.0; dMinPhotonPIDFOM.first = false; dMinPhotonPIDFOM.second = 5.73303E-7; dMinTrackingFOM.first = false; dMinTrackingFOM.second = 0.0027; dMaxPhotonRFDeltaT.first = false; dMaxPhotonRFDeltaT.second = 10.0*2.004; //10 RF buckets dMinProtonMomentum.first = true; dMinProtonMomentum.second = 0.25; dHasDetectorMatchFlag.first = false; dHasDetectorMatchFlag.second = false; dTTreeOutputFileName = ""; dEnableTTreeOutputFlag = false; dMinThrownMatchFOMForROOT = -1.0; //always } const DReactionStep* DReaction::Get_ReactionStep(size_t locStepIndex) const { if(locStepIndex >= dReactionSteps.size()) return NULL; return dReactionSteps[locStepIndex]; } string DReaction::Get_InitialParticlesROOTName(void) const { if(dReactionSteps.empty()) return (string()); return dReactionSteps[0]->Get_InitialParticlesROOTName(); } string DReaction::Get_DetectedParticlesROOTName(void) const { string locDetectedParticlesROOTName; Particle_t locPID; for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < dReactionSteps[loc_i]->Get_NumFinalParticleIDs(); ++loc_j) { locPID = dReactionSteps[loc_i]->Get_FinalParticleID(loc_j); if(dReactionSteps[loc_i]->Get_MissingParticleIndex() == int(loc_j)) continue; //missing particle //see if this pid is a parent in a future step if(Check_IsDecayingParticle(locPID, loc_i + 1)) continue; locDetectedParticlesROOTName += ParticleName_ROOT(locPID); } } return locDetectedParticlesROOTName; } string DReaction::Get_DecayChainFinalParticlesROOTNames(Particle_t locInitialPID, bool locKinFitResultsFlag) const { //if multiple decay steps have locInitialPID as the parent, only the first listed is used return Get_DecayChainFinalParticlesROOTNames(locInitialPID, -1, deque(), locKinFitResultsFlag); } string DReaction::Get_DecayChainFinalParticlesROOTNames(Particle_t locInitialPID, int locUpToStepIndex, deque locUpThroughPIDs, bool locKinFitResultsFlag) const { //if multiple decay steps have locInitialPID as the parent, only the first listed is used deque locPIDs; string locName = ""; for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { if(dReactionSteps[loc_i]->Get_InitialParticleID() != locInitialPID) continue; return Get_DecayChainFinalParticlesROOTNames(loc_i, locUpToStepIndex, locUpThroughPIDs, locKinFitResultsFlag, false); } return string(""); } string DReaction::Get_DecayChainFinalParticlesROOTNames(size_t locStepIndex, int locUpToStepIndex, deque locUpThroughPIDs, bool locKinFitResultsFlag, bool locExpandDecayingParticlesFlag) const { //if locKinFitResultsFlag = true: don't expand decaying particles (through decay chain) that were included in the kinfit (still expand resonances) string locName = ""; deque locPIDs; dReactionSteps[locStepIndex]->Get_FinalParticleIDs(locPIDs); int locMissingParticleIndex = dReactionSteps[locStepIndex]->Get_MissingParticleIndex(); for(size_t loc_j = 0; loc_j < locPIDs.size(); ++loc_j) { if(int(loc_j) == locMissingParticleIndex) continue; //exclude missing! if(int(locStepIndex) == locUpToStepIndex) { bool locPIDFoundFlag = false; for(deque::iterator locIterator = locUpThroughPIDs.begin(); locIterator != locUpThroughPIDs.end(); ++locIterator) { if((*locIterator) != locPIDs[loc_j]) continue; locUpThroughPIDs.erase(locIterator); locPIDFoundFlag = true; break; } if(!locPIDFoundFlag) continue; //skip it: don't want to include it } //find all future steps in which this pid is a parent int locDecayingStepIndex = -1; //expand if: not-kinfitting, or not a fixed mass if((!locKinFitResultsFlag) || (!IsFixedMass(locPIDs[loc_j])) || locExpandDecayingParticlesFlag) { for(size_t loc_k = locStepIndex + 1; loc_k < dReactionSteps.size(); ++loc_k) { if(dReactionSteps[loc_k]->Get_InitialParticleID() != locPIDs[loc_j]) continue; locDecayingStepIndex = loc_k; break; } } if(locDecayingStepIndex == -1) locName += ParticleName_ROOT(locPIDs[loc_j]); else locName += Get_DecayChainFinalParticlesROOTNames(locDecayingStepIndex, locUpToStepIndex, locUpThroughPIDs, locKinFitResultsFlag, locExpandDecayingParticlesFlag); } return locName; } bool DReaction::Check_IsDecayingParticle(Particle_t locPID, size_t locSearchStartIndex) const { //see if this pid is a parent in a future step for(size_t loc_k = locSearchStartIndex; loc_k < dReactionSteps.size(); ++loc_k) { if(dReactionSteps[loc_k]->Get_InitialParticleID() == locPID) return true; } return false; } void DReaction::Get_ReactionSteps(Particle_t locInitialPID, deque& locReactionSteps) const { locReactionSteps.clear(); for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { if(dReactionSteps[loc_i]->Get_InitialParticleID() == locInitialPID) locReactionSteps.push_back(dReactionSteps[loc_i]); } } void DReaction::Get_DetectedFinalPIDs(deque& locDetectedPIDs, bool locIncludeDuplicatesFlag) const //independent of step { Particle_t locPID; locDetectedPIDs.clear(); for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < dReactionSteps[loc_i]->Get_NumFinalParticleIDs(); ++loc_j) { locPID = dReactionSteps[loc_i]->Get_FinalParticleID(loc_j); if(dReactionSteps[loc_i]->Get_MissingParticleIndex() == int(loc_j)) continue; //missing particle //see if this pid is a parent in a future step if(Check_IsDecayingParticle(locPID, loc_i + 1)) continue; if(!locIncludeDuplicatesFlag) { //see if this PID is already stored bool locAlreadyHavePIDFlag = false; for(size_t loc_k = 0; loc_k < locDetectedPIDs.size(); ++loc_k) { if(locDetectedPIDs[loc_k] == locPID) { locAlreadyHavePIDFlag = true; break; } } if(locAlreadyHavePIDFlag) continue; } locDetectedPIDs.push_back(locPID); } } } void DReaction::Get_DetectedFinalChargedPIDs(deque& locDetectedChargedPIDs, bool locIncludeDuplicatesFlag) const //independent of step { Particle_t locPID; locDetectedChargedPIDs.clear(); for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < dReactionSteps[loc_i]->Get_NumFinalParticleIDs(); ++loc_j) { locPID = dReactionSteps[loc_i]->Get_FinalParticleID(loc_j); if(ParticleCharge(locPID) == 0) continue; //neutral if(dReactionSteps[loc_i]->Get_MissingParticleIndex() == int(loc_j)) continue; //missing particle //see if this pid is a parent in a future step if(Check_IsDecayingParticle(locPID, loc_i + 1)) continue; if(!locIncludeDuplicatesFlag) { //see if this PID is already stored bool locAlreadyHavePIDFlag = false; for(size_t loc_k = 0; loc_k < locDetectedChargedPIDs.size(); ++loc_k) { if(locDetectedChargedPIDs[loc_k] == locPID) { locAlreadyHavePIDFlag = true; break; } } if(locAlreadyHavePIDFlag) continue; } locDetectedChargedPIDs.push_back(locPID); } } } void DReaction::Get_FinalStatePIDs(deque& locFinalStatePIDs, bool locIncludeDuplicatesFlag) const { Particle_t locPID; locFinalStatePIDs.resize(0); for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < dReactionSteps[loc_i]->Get_NumFinalParticleIDs(); ++loc_j) { locPID = dReactionSteps[loc_i]->Get_FinalParticleID(loc_j); //see if this pid is a parent in a future step if(Check_IsDecayingParticle(locPID, loc_i + 1)) continue; //see if this PID is already stored if(!locIncludeDuplicatesFlag) { bool locAlreadyHavePIDFlag = false; for(size_t loc_k = 0; loc_k < locFinalStatePIDs.size(); ++loc_k) { if(locFinalStatePIDs[loc_k] == locPID) { locAlreadyHavePIDFlag = true; break; } } if(locAlreadyHavePIDFlag) continue; } locFinalStatePIDs.push_back(locPID); } } } void DReaction::Get_DetectedFinalPIDs(deque >& locDetectedPIDs, bool locIncludeDuplicatesFlag) const { Particle_t locPID; locDetectedPIDs.clear(); locDetectedPIDs.resize(dReactionSteps.size()); for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < dReactionSteps[loc_i]->Get_NumFinalParticleIDs(); ++loc_j) { locPID = dReactionSteps[loc_i]->Get_FinalParticleID(loc_j); if(dReactionSteps[loc_i]->Get_MissingParticleIndex() == int(loc_j)) continue; //missing particle //see if this pid is a parent in a future step if(Check_IsDecayingParticle(locPID, loc_i + 1)) continue; if(!locIncludeDuplicatesFlag) { //see if this PID is already stored bool locAlreadyHavePIDFlag = false; for(size_t loc_k = 0; loc_k < locDetectedPIDs[loc_i].size(); ++loc_k) { if(locDetectedPIDs[loc_i][loc_k] == locPID) { locAlreadyHavePIDFlag = true; break; } } if(locAlreadyHavePIDFlag) continue; } locDetectedPIDs[loc_i].push_back(locPID); } } } void DReaction::Get_DetectedFinalChargedPIDs(deque >& locDetectedChargedPIDs, bool locIncludeDuplicatesFlag) const { Particle_t locPID; locDetectedChargedPIDs.clear(); locDetectedChargedPIDs.resize(dReactionSteps.size()); for(size_t loc_i = 0; loc_i < dReactionSteps.size(); ++loc_i) { for(size_t loc_j = 0; loc_j < dReactionSteps[loc_i]->Get_NumFinalParticleIDs(); ++loc_j) { locPID = dReactionSteps[loc_i]->Get_FinalParticleID(loc_j); if(ParticleCharge(locPID) == 0) continue; //neutral if(dReactionSteps[loc_i]->Get_MissingParticleIndex() == int(loc_j)) continue; //missing particle //see if this pid is a parent in a future step if(Check_IsDecayingParticle(locPID, loc_i + 1)) continue; if(!locIncludeDuplicatesFlag) { //see if this PID is already stored bool locAlreadyHavePIDFlag = false; for(size_t loc_k = 0; loc_k < locDetectedChargedPIDs[loc_i].size(); ++loc_k) { if(locDetectedChargedPIDs[loc_i][loc_k] == locPID) { locAlreadyHavePIDFlag = true; break; } } if(locAlreadyHavePIDFlag) continue; } locDetectedChargedPIDs[loc_i].push_back(locPID); } } } DAnalysisAction* DReaction::Get_AnalysisAction(size_t locActionIndex) const { if(locActionIndex >= dAnalysisActions.size()) return NULL; return dAnalysisActions[locActionIndex]; } bool DReaction::Get_MissingPID(Particle_t& locPID) const { for(size_t loc_i = 0; loc_i < Get_NumReactionSteps(); ++loc_i) { if(Get_ReactionStep(loc_i)->Get_MissingPID(locPID)) return true; } return false; } bool DReaction::Check_AreStepsIdentical(const DReaction* locReaction) const { if(locReaction->Get_NumReactionSteps() != dReactionSteps.size()) return false; for(size_t loc_i = 0; loc_i < Get_NumReactionSteps(); ++loc_i) { if(locReaction->Get_ReactionStep(loc_i) != dReactionSteps[loc_i]) return false; } return true; }