53 if ( wrappedProcessParticleChange->GetNumberOfSecondaries() == 0 )
return wrappedProcessParticleChange;
54 if ( wrappedProcessParticleChange->GetTrackStatus() ==
fKillTrackAndSecondaries )
return wrappedProcessParticleChange;
56 std::vector < G4Track* > secondariesAndPrimary;
57 for (
auto i = 0 ; i < wrappedProcessParticleChange->GetNumberOfSecondaries() ; i++ ) secondariesAndPrimary.push_back( wrappedProcessParticleChange->GetSecondary( i ) );
68 G4Track* finalStatePrimary (
nullptr );
69 if ( ( wrappedProcessParticleChange->GetTrackStatus() !=
fStopAndKill ) )
75 castParticleChange =
dynamic_cast< G4ParticleChange*
>( wrappedProcessParticleChange );
76 if ( castParticleChange ==
nullptr )
78 G4cout <<
" **** G4BOptnLeadingParticle::ApplyFinalStateBiasing(...) : can not bias for " << callingProcess->
GetProcessName() <<
", this is just a warning." <<
G4endl;
79 return wrappedProcessParticleChange;
81 finalStatePrimary =
new G4Track( *track );
86 secondariesAndPrimary.push_back( finalStatePrimary );
92 if ( finalStatePrimary ) primaryWeight = finalStatePrimary->
GetWeight();
95 for (
auto i = 0; i < wrappedProcessParticleChange->GetNumberOfSecondaries(); ++i)
96 secondariesAndPrimary[ i ]->SetWeight( primaryWeight );
99 std::size_t leadingIDX = 0;
101 std::map< G4Track*, G4bool > survivingMap;
102 for ( std::size_t idx = 0; idx < secondariesAndPrimary.size(); ++idx )
104 survivingMap[ secondariesAndPrimary[idx] ] =
false;
105 if ( secondariesAndPrimary[idx]->GetKineticEnergy() > leadingEnergy )
107 leadingEnergy = secondariesAndPrimary[idx]->GetKineticEnergy();
111 survivingMap[ secondariesAndPrimary[leadingIDX] ] =
true;
114 std::map < G4int, std::vector< G4Track* > > typesAndTracks;
115 for ( std::size_t idx = 0; idx < secondariesAndPrimary.size(); ++idx )
117 if ( idx == leadingIDX )
continue;
118 auto currentTrack = secondariesAndPrimary[idx];
119 auto GROUP = std::abs( currentTrack->GetDefinition()->GetPDGEncoding() );
120 if ( currentTrack->GetDefinition()->GetBaryonNumber() >= 2 )
123 if ( typesAndTracks.find( GROUP ) == typesAndTracks.cend() )
125 std::vector< G4Track* > v;
126 v.push_back( currentTrack );
127 typesAndTracks[ GROUP ] = std::move(v);
131 typesAndTracks[ GROUP ].push_back( currentTrack );
137 G4int nSecondaries = 0;
138 for (
auto& typeAndTrack : typesAndTracks )
140 std::size_t nTracks = (typeAndTrack.second).size();
145 auto keptTrackIDX = G4RandFlat::shootInt( nTracks );
146 keptTrack = (typeAndTrack.second)[keptTrackIDX];
151 keptTrack = (typeAndTrack.second)[0];
155 if ( fRussianRouletteKillingProbability > 0.0 )
159 keptTrack->
SetWeight( keptTrack->
GetWeight() / (1. - fRussianRouletteKillingProbability) );
163 else keepTrack =
true;
166 survivingMap[ keptTrack ] =
true;
167 if ( keptTrack != finalStatePrimary ) ++nSecondaries;
171 if ( secondariesAndPrimary[leadingIDX] != finalStatePrimary ) ++nSecondaries;
174 G4bool primarySurvived =
false;
175 if ( finalStatePrimary )
176 primarySurvived = survivingMap[ finalStatePrimary ];
181 fParticleChange.Initialize(*track);
182 if ( primarySurvived )
184 fParticleChange.ProposeTrackStatus ( wrappedProcessParticleChange->GetTrackStatus() );
185 fParticleChange.ProposeParentWeight ( finalStatePrimary->
GetWeight() );
194 fParticleChange.ProposeParentWeight( 0.0 );
195 fParticleChange.ProposeEnergy ( 0.0 );
198 fParticleChange.SetSecondaryWeightByProcess(
true);
199 fParticleChange.SetNumberOfSecondaries(nSecondaries);
202 for (
auto idx = 0 ; idx < wrappedProcessParticleChange->GetNumberOfSecondaries() ; ++idx )
204 G4Track* secondary = secondariesAndPrimary[idx];
212 if ( survivingMap[ secondary ] )
213 fParticleChange.AddSecondary( secondary );
220 wrappedProcessParticleChange->Clear();
222 if ( finalStatePrimary )
delete finalStatePrimary;
225 return &fParticleChange;
virtual G4VParticleChange * ApplyFinalStateBiasing(const G4BiasingProcessInterface *, const G4Track *, const G4Step *, G4bool &)
G4double GetEnergy() const
const G4ThreeVector * GetMomentumDirection() const
G4double GetWeight() const