00001
00002 #include "Rivet/Rivet.hh"
00003 #include "Rivet/Projections/Thrust.hh"
00004 #include "Rivet/Tools/Logging.hh"
00005
00006
00007 namespace Rivet {
00008
00009
00010 void Thrust::calc(const FinalState& fs) {
00011 calc(fs.particles());
00012 }
00013
00014 void Thrust::calc(const vector<Particle>& fsparticles) {
00015 vector<Vector3> threeMomenta;
00016 threeMomenta.reserve(fsparticles.size());
00017 foreach (const Particle& p, fsparticles) {
00018 const Vector3 p3 = p.momentum().vector3();
00019 threeMomenta.push_back(p3);
00020 }
00021 _calcThrust(threeMomenta);
00022 }
00023
00024 void Thrust::calc(const vector<FourMomentum>& fsmomenta) {
00025 vector<Vector3> threeMomenta;
00026 threeMomenta.reserve(fsmomenta.size());
00027 foreach (const FourMomentum& v, fsmomenta) {
00028 threeMomenta.push_back(v.vector3());
00029 }
00030 _calcThrust(threeMomenta);
00031 }
00032
00033 void Thrust::calc(const vector<Vector3>& fsmomenta) {
00034 _calcThrust(fsmomenta);
00035 }
00036
00037
00038
00039
00040
00041
00042 inline bool mod2Cmp(const Vector3& a, const Vector3& b) {
00043 return a.mod2() > b.mod2();
00044 }
00045
00046
00047
00048 void _calcT(const vector<Vector3>& momenta, double& t, Vector3& taxis) {
00049
00050
00051
00052
00053
00054 vector<Vector3> p = momenta;
00055 assert(p.size() >= 3);
00056 unsigned int n = 3;
00057 if (p.size() == 3) n = 3;
00058 vector<Vector3> tvec;
00059 vector<double> tval;
00060 std::sort(p.begin(), p.end(), mod2Cmp);
00061 for (unsigned int i=0 ; i<pow(2,n-1) ; i++) {
00062
00063 Vector3 foo(0,0,0);
00064 int sign=i;
00065 for (unsigned int k=0 ; k<n ; k++) {
00066 (sign%2)==1 ? foo+=p[k] : foo-=p[k];
00067 sign/=2;
00068 }
00069 foo=foo.unit();
00070
00071
00072 double diff=999.;
00073 while (diff>1e-5) {
00074 Vector3 foobar(0,0,0);
00075 for (unsigned int k=0 ; k<p.size() ; k++)
00076 foo.dot(p[k])>0 ? foobar+=p[k] : foobar-=p[k];
00077 diff=(foo-foobar.unit()).mod();
00078 foo=foobar.unit();
00079 }
00080
00081
00082 t=0.;
00083 for (unsigned int k=0 ; k<p.size() ; k++)
00084 t+=fabs(foo.dot(p[k]));
00085
00086
00087 tval.push_back(t);
00088 tvec.push_back(foo);
00089 }
00090
00091
00092 t=0.;
00093 for (unsigned int i=0 ; i<tvec.size() ; i++)
00094 if (tval[i]>t){
00095 t=tval[i];
00096 taxis=tvec[i];
00097 }
00098 }
00099
00100
00101
00102
00103 void Thrust::_calcThrust(const vector<Vector3>& fsmomenta) {
00104
00105 double momentumSum(0.0);
00106 foreach (const Vector3& p3, fsmomenta) {
00107 momentumSum += mod(p3);
00108 }
00109 getLog() << Log::DEBUG << "Number of particles = " << fsmomenta.size() << endl;
00110
00111
00112
00113 _thrusts.clear();
00114 _thrustAxes.clear();
00115
00116
00117
00118 if (fsmomenta.size() < 2) {
00119 for (int i = 0; i < 3; ++i) {
00120 _thrusts.push_back(-1);
00121 _thrustAxes.push_back(Vector3(0,0,0));
00122 }
00123 return;
00124 }
00125
00126
00127
00128 if (fsmomenta.size() == 2) {
00129 Vector3 axis(0,0,0);
00130 _thrusts.push_back(1.0);
00131 _thrusts.push_back(0.0);
00132 _thrusts.push_back(0.0);
00133 axis = fsmomenta[0].unit();
00134 if (axis.z() < 0) axis = -axis;
00135 _thrustAxes.push_back(axis);
00136
00137
00138 if (axis.z() < 0.75)
00139 _thrustAxes.push_back( (axis.cross(Vector3(0,0,1))).unit() );
00140 else
00141 _thrustAxes.push_back( (axis.cross(Vector3(0,1,0))).unit() );
00142 _thrustAxes.push_back( _thrustAxes[0].cross(_thrustAxes[1]) );
00143 return;
00144 }
00145
00146
00147
00148
00149 Vector3 axis(0,0,0);
00150 double val = 0.;
00151
00152
00153 _calcT(fsmomenta, val, axis);
00154 getLog() << Log::DEBUG << "Mom sum = " << momentumSum << endl;
00155 _thrusts.push_back(val / momentumSum);
00156
00157 if (axis.z() < 0) axis = -axis;
00158 axis = axis.unit();
00159 getLog() << Log::DEBUG << "Axis = " << axis << endl;
00160 _thrustAxes.push_back(axis);
00161
00162
00163 vector<Vector3> threeMomenta;
00164 foreach (const Vector3& v, fsmomenta) {
00165
00166 const Vector3 vpar = dot(v, axis.unit()) * axis.unit();
00167 threeMomenta.push_back(v - vpar);
00168 }
00169 _calcT(threeMomenta, val, axis);
00170 _thrusts.push_back(val / momentumSum);
00171 if (axis.x() < 0) axis = -axis;
00172 axis = axis.unit();
00173 _thrustAxes.push_back(axis);
00174
00175
00176 if (_thrustAxes[0].dot(_thrustAxes[1]) < 1e-10) {
00177 axis = _thrustAxes[0].cross(_thrustAxes[1]);
00178 _thrustAxes.push_back(axis);
00179 val = 0.0;
00180 foreach (const Vector3& v, fsmomenta) {
00181 val += fabs(dot(axis, v));
00182 }
00183 _thrusts.push_back(val / momentumSum);
00184 } else {
00185 _thrusts.push_back(-1.0);
00186 _thrustAxes.push_back(Vector3(0,0,0));
00187 }
00188
00189 }
00190
00191
00192
00193 }