StRoot  1
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Groups Pages
ObjExporter.h
1 
2 #ifndef __OBJEXPORTER_H__
3 #define __OBJEXPORTER_H__
4 #include "GenFit/FitStatus.h"
5 #include "GenFit/GFRaveVertexFactory.h"
6 #include <math.h>
7 
8 #include "StEvent/StEvent.h"
9 #include "StEvent/StFttCollection.h"
10 #include "StEvent/StFttCluster.h"
11 
12 class StEvent;
13 
14 
15 class ObjExporter {
16 public:
17 
18  // extra output for info and debug
19  const static bool verbose = true;
20 
21  // write a single vertex
22  void vert( ofstream &of, float x, float y, float z ){
23  of << "v " << x << " " << y << " " << z << endl;
24  numVertices++;
25  }
26  void vert( ofstream &of, TVector3 v ){
27  of << "v " << v.X() << " " << v.Y() << " " << v.Z() << endl;
28  numVertices++;
29  }
30 
31  // write a sphere with given sub-divsions to output file
32  void sphere(TVector3 pt, float radius, int nLatitude, int nLongitude, ofstream &fout){
33  int p, s, i, j;
34  float x, y, z, out;
35  int nPitch = nLongitude + 1;
36  const float DEGS_TO_RAD = 3.14159f/180.0f;
37 
38  float pitchInc = (180. / (float)nPitch) * DEGS_TO_RAD;
39  float rotInc = (360. / (float)nLatitude) * DEGS_TO_RAD;
40 
41  //## PRINT VERTICES:
42  vert(fout, pt.X(), pt.Y()+radius, pt.Z()); // Top vertex.
43  vert(fout, pt.X(), pt.Y()-radius, pt.Z()); // Bottom vertex.
44 
45  int fVert = numVertices; // Record the first vertex index for intermediate vertices.
46  for(p=1; p<nPitch; p++) { // Generate all "intermediate vertices":
47  out = fabs(radius * sin((float)p * pitchInc));
48  y = radius * cos(p * pitchInc);
49  for(s=0; s<nLatitude; s++) {
50  x = out * cos(s * rotInc);
51  z = out * sin(s * rotInc);
52  fout << TString::Format("v %g %g %g\n", x+pt.X(), y+pt.Y(), z+pt.Z()).Data();
53  numVertices++;
54  }
55  }
56 
57  //## OUTPUT FACES BETWEEN INTERMEDIATE POINTS:
58 
59  for(p=1; p<nPitch-1; p++) {
60  for(s=0; s<nLatitude; s++) {
61  i = p*nLatitude + s;
62  j = (s==nLatitude-1) ? i-nLatitude : i;
63  fout << TString::Format("f %d %d %d %d\n",
64  (i+1-nLatitude)+fVert, (j+2-nLatitude)+fVert, (j+2)+fVert, (i+1)+fVert);
65  }
66  }
67 
68  //## OUTPUT TRIANGULAR FACES CONNECTING TO TOP AND BOTTOM VERTEX:
69 
70  int offLastVerts = fVert + (nLatitude * (nLongitude-1));
71  for(s=0; s<nLatitude; s++)
72  {
73  j = (s==nLatitude-1) ? -1 : s;
74  fout << TString::Format("f %d %d %d\n", fVert-1, (j+2)+fVert, (s+1)+fVert ).Data();
75  fout << TString::Format("f %d %d %d\n", fVert, (s+1)+offLastVerts, (j+2)+offLastVerts).Data();
76  }
77  } // sphere
78 
79  // output a single face
80  void face( ofstream &fout, int anchor, int v0, int v1, int v2 ){
81  fout << TString::Format("f %d %d %d\n", anchor+v0, anchor+v1, anchor+v2).Data();
82  }
83  void face( ofstream &fout, int anchor, int v0, int v1, int v2, int v3 ){
84  fout << TString::Format("f %d %d %d %d\n", anchor+v0, anchor+v1, anchor+v2, anchor+v3).Data();
85  }
86  // output a general prism
87  void prism( ofstream &fout, TVector3 po, TVector3 ds ){
88  int fVert = numVertices; // Record the first vertex index for intermediate vertices.
89  // x-y at z=0
90  vert( fout, po.X(), po.Y(), po.Z() ); // 1
91  vert( fout, po.X() + ds.X(), po.Y(), po.Z() ); // 2
92  vert( fout, po.X() + ds.X(), po.Y() + ds.Y(), po.Z() ); // 3
93  vert( fout, po.X(), po.Y() + ds.Y(), po.Z() ); // 4
94 
95  // x-y at z=+dz
96  vert( fout, po.X(), po.Y(), po.Z() + ds.Z() ); // 5
97  vert( fout, po.X() + ds.X(), po.Y(), po.Z() + ds.Z() ); // 6
98  vert( fout, po.X() + ds.X(), po.Y() + ds.Y(), po.Z() + ds.Z() ); // 7
99  vert( fout, po.X(), po.Y() + ds.Y(), po.Z() + ds.Z() ); // 8
100 
101  face( fout, fVert, 1, 2, 3, 4 ); // bottom face
102  face( fout, fVert, 5, 6, 7, 8 ); // top face
103  face( fout, fVert, 1, 2, 6, 5 ); // side 1
104  face( fout, fVert, 2, 3, 7, 6 ); // side 2
105  face( fout, fVert, 3, 4, 8, 7 ); // side 3
106  face( fout, fVert, 1, 4, 8, 5 ); // side 4
107  }
108 
109  // output a tri
110  void tri( ofstream &fout, TVector3 v0, float dz, float w, float h, float phi ) {
111 
112  int fVert = numVertices;
113  float dphi = 0.1/2;
114  float yp = v0.Y() + (sin( phi + dphi ) + cos( phi + dphi ) ) * h;
115  float yn = v0.Y() + (sin( phi - dphi ) + cos( phi - dphi ) ) * h;
116  float xp = v0.X() + (cos( phi + dphi ) - sin( phi + dphi)) * h;
117  float xn = v0.X() + (cos( phi - dphi ) - sin( phi - dphi)) * h;
118  // top face
119  vert( fout, v0.X(), v0.Y(), v0.Z() ); // 1
120  vert( fout, xp, yp, v0.Z() );
121  vert( fout, xn, yn, v0.Z() );
122 
123  // top face
124  vert( fout, v0.X(), v0.Y(), v0.Z() - dz ); // 4
125  vert( fout, xp, yp, v0.Z() - dz );
126  vert( fout, xn, yn, v0.Z() - dz );
127 
128  face( fout, fVert, 1, 2, 3 ); // top
129  face( fout, fVert, 4, 5, 6 ); // bottom
130  face( fout, fVert, 1, 2, 5, 4 ); // side 1
131  face( fout, fVert, 2, 3, 6, 5 ); // side 2
132  face( fout, fVert, 3, 1, 4, 6 ); // side 3
133  }
134 
135  // Compute a track projection as a linear (straight) extrapolation from two points from full projections
136  static TVector3 projectAsStraightLine( genfit::Track * t, float* xyz0, float* xyz1, float* xyzf, float* planenorm_xyz, float * cov, TVector3 &mom ) {
137  TVector3 tv3A = trackPosition( t, xyz0, planenorm_xyz, cov, mom );
138  TVector3 tv3B = trackPosition( t, xyz1, planenorm_xyz, cov, mom );
139 
140  if (verbose){
141  LOG_INFO << "Straight Line Projection using" << endm;
142  LOG_INFO << "A.x = " << tv3A.X() << endm;
143  LOG_INFO << "B.x = " << tv3B.X() << endm;
144  }
145  //Assuming tv3A is starting point of line and tv3B is ending point of line and we are projecting to plane located at xyzf with normal planenorm_xyz
146  TVector3 projlinedir = tv3B-tv3A; //particle straight line direction
147  //Solution of intersection of line and plane where line has direction {xdir,ydir,zdir}*t and starts at {xorigin,yorigin,zorigin} and a plane that has some normal with a point on the plane; "t" is the free parameter in the parametric equation of the line.
148  double tintersection =
149  (planenorm_xyz[0]*(xyzf[0]-tv3A[0])+planenorm_xyz[1]*(xyzf[1]-tv3A[1])+planenorm_xyz[2]*(xyzf[2]-tv3A[2])) /
150  (planenorm_xyz[0]*projlinedir[0]+planenorm_xyz[1]*projlinedir[1]+planenorm_xyz[2]*projlinedir[2]);
151 
152  return TVector3( projlinedir[0]*tintersection+tv3A[0], projlinedir[1]*tintersection+tv3A[1], projlinedir[2]*tintersection+tv3A[2] );
153  /*
154  double dxdz = ( tv3B.X() - tv3A.X() ) / ( tv3B.Z() - tv3A.Z() );
155  double dydz = ( tv3B.Y() - tv3A.Y() ) / ( tv3B.Z() - tv3A.Z() );
156 
157  double dx = dxdz * ( zf - z1 );
158  double dy = dydz * ( zf - z1 );
159  TVector3 r( tv3B.X() + dx, tv3B.Y() + dy, zf );
160  return r;
161  */
162  }
163 
164  // Project a track to a given xyz-plane with a normal and return its position, momentum, and cov matrix
165  static TVector3 trackPosition( genfit::Track * t, float* xyz, float* norm, float * cov, TVector3 &mom ){
166  if (verbose){
167  LOG_INFO << "trackPosition(" << t << ", " << xyz << ", " << norm << ", " << cov << " )" << endm;
168  LOG_INFO << "Projecting to: " << xyz[0] << " " << xyz[1] << " " << xyz[2] << endm;
169  LOG_INFO << "Normal: " << norm[0] << " " << norm[1] << " " << norm[2] << endm;
170  LOG_INFO << "track has n points: " << t->getNumPointsWithMeasurement() << endm;
171  }
172  const int iPoint = 0;
173  // do not attempt projection if no points with measurements
174  // for some reason we can still get here even though we require convergence earlier
175  if ( t->getNumPointsWithMeasurement() == 0 ){
176  LOG_WARN << "Track has no points with measurement, cannot project" << endm;
177  return TVector3( -990, -990, -990 );
178  }
179  try {
180  auto plane = genfit::SharedPlanePtr(
181  new genfit::DetPlane(TVector3(xyz), TVector3(norm) )
182  );
183 
184  genfit::MeasuredStateOnPlane tst = t->getFittedState(iPoint);
185  auto TCM = t->getCardinalRep()->get6DCov(tst);
186  // returns the track length if needed
187  t->getCardinalRep()->extrapolateToPlane(tst, plane, false, true);
188 
189  TCM = t->getCardinalRep()->get6DCov(tst);
190 
191  // can get the projected positions if needed
192  float x = tst.getPos().X();
193  float y = tst.getPos().Y();
194  float _z = tst.getPos().Z();
195 
196  mom.SetXYZ( tst.getMom().X(), tst.getMom().Y(), tst.getMom().Z() );
197  if (verbose){
198  LOG_INFO << "Projected to: " << x << " " << y << " " << _z << endm;
199  }
200  if ( cov ){
201  cov[0] = TCM(0,0); cov[1] = TCM(1,0); cov[2] = TCM(2,0);
202  cov[3] = TCM(0,1); cov[4] = TCM(1,1); cov[5] = TCM(2,1);
203  cov[6] = TCM(0,2); cov[7] = TCM(1,2); cov[8] = TCM(2,2);
204  }
205 
206 
207  return TVector3( x, y, _z );
208  } catch ( genfit::Exception &e ){
209  LOG_INFO << "Track projection Failed from trackPoint " << iPoint << " E: " << e.what() << endm;
210  return TVector3( -990, -990, -990 );
211  }
212 
213 
214  return TVector3( -990, -990, -990 );
215  }
216 
217  // Utility method
218  static TVector3 trackPosition( genfit::Track * t, float z ){
219  float cov[9] = {0}; //Force intialize array to 0
220  TVector3 mom;
221  float detpos[3] = {0,0,z};
222  float detnorm[3] = {0,0,1};
223  return trackPosition( t, detpos, detnorm, cov, mom);
224  }
225 
226  // Output the sTGC strips into a useful format for event display
227  void output_ftt_strips(
228  ofstream &fout,
229  StEvent * event ){
230  fout << "\n" << endl;
231  fout << "o fttStrips" << endl;
232  fout << "usemtl stgc_hits\n" << endl;
233  float pz[] = {280.90499, 303.70498, 326.60501, 349.40499};
234  TVector3 cp;
235  const float SCALE = 0.1;
236 
237 
238  for ( size_t i = 0; i < event->fttCollection()->numberOfClusters(); i++ ){
239  StFttCluster* c = event->fttCollection()->clusters()[i];
240  if ( c->nStrips() < 2 ) continue;
241  float dw = 0.05, dlh = 60.0, dlv = 60.0;
242  float mx = 0.0, my = 0.0;
243  float sx = 1.0, sy = 1.0;
244 
245 
246  if ( c->quadrant() == kFttQuadrantA ){
247  mx = 0; my = 0;
248  sx = 1.0; sy = 1.0;
249  } else if ( c->quadrant() == kFttQuadrantB ){
250  mx = 10.16*SCALE; my = 0.0*SCALE;
251  sy = -1;
252  dlv = -dlv;
253 
254  } else if ( c->quadrant() == kFttQuadrantC ){
255  mx = -10.16*SCALE ; my = -00.0*SCALE;
256  sx = -1.0; sy = -1.0;
257  dlh = -dlh; dlv = -dlv;
258 
259  } else if ( c->quadrant() == kFttQuadrantD ){
260  sx = -1;
261  dlh = -dlh;
262  }
263 
264  cp.SetZ( -pz[ c->plane() ] * SCALE );
265  if ( c->orientation() == kFttHorizontal ){
266  cp.SetY( my + sy * c->x()/10.0 * SCALE );
267  cp.SetX( mx );
268  prism( fout, cp, TVector3( dlh * SCALE, dw, dw ) );
269  } else if ( c->orientation() == kFttVertical ){
270  cp.SetX( mx + sx * c->x()/10.0 * SCALE );
271  cp.SetY( my );
272  prism( fout, cp, TVector3( dw, dlv * SCALE, dw ) );
273  }
274  }
275  } // ftt_strips
276 
277  // Output the event info into a useful format for event display
278  void output( std::string filename,
279  StEvent * event,
280  std::vector< Seed_t> seeds,
281  std::vector< genfit::Track *> tracks,
282  const std::vector< genfit::GFRaveVertex *> &vertices,
283  std::vector<TVector3> &fttHits,
284  std::vector<TVector3> &fstHits,
285  std::vector<TVector3> &fcsPreHits, // EPD = preshower
286  std::vector<TVector3> &fcsClusters,
287  std::vector<float> &fcsClusterEnergy
288  ){
289 
290  const float SCALE = 0.1;
291  LOG_INFO << "Writing: " << filename << endm;
292  numVertices = 0;
293  // OPEN output
294  ofstream ofile( (filename + ".obj" ).c_str() );
295 
296  ofile << "\nmtllib materials.mtl\n\n" << endl;
297 
298 
299  output_ftt_strips( ofile, event );
300 
301  // Write FWD vertices
302  TVector3 startPos;
303  if ( vertices.size() > 0 ){
304  ofile << "o FwdVertices" << endl;
305  for ( auto v : vertices ) {
306  startPos.SetXYZ( v->getPos().X(), v->getPos().Y(), v->getPos().Z() );
307  sphere( TVector3( v->getPos().X() * SCALE, v->getPos().Y() * SCALE, -v->getPos().Z() * SCALE ), 0.5, 10, 10, ofile );
308  }
309  }
310 
311  // Write FTT hits (points)
312  if ( verbose ){
313  LOG_INFO << "Viz has " << fttHits.size() << " FTT Hits" << endm;
314  }
315  if ( fttHits.size() > 0 ){
316  ofile << "\n" << endl;
317  ofile << "o fttHits" << endl;
318  ofile << "usemtl stgc_hits\n" << endl;
319  for ( auto p : fttHits ){
320  sphere( TVector3( p.X() * SCALE, p.Y() * SCALE, -p.Z() * SCALE ), 0.15, 12, 12, ofile );
321  }
322  }
323 
324  // write FST hits
325  if ( verbose ){
326  LOG_INFO << "Viz has " << fstHits.size() << " FST Hits" << endm;
327  }
328  if ( fstHits.size() > 0 ) {
329  ofile << "\n" << endl;
330  ofile << "o fstHits" << endl;
331  ofile << "usemtl fst_hits\n" << endl;
332  for ( auto p : fstHits ){
333 
334  // float fstphi = TMath::ATan2( p.Y(), p.X() );
335  // printf( "FST PHI: %f \n", fstphi );
336  // tri( ofile, TVector3( p.X() * SCALE, p.Y() * SCALE, -p.Z() * SCALE ), 0.1f, 0.1f, 3.0f, fstphi );
337  sphere( TVector3( p.X() * SCALE, p.Y() * SCALE, -p.Z() * SCALE ), 0.3, 10, 10, ofile );
338  }
339  }
340 
341  // Output the EPD hits
342  if (verbose){
343  LOG_INFO << "Viz has " << fcsPreHits.size() << " EPD Hits" << endm;
344  }
345  if ( fcsPreHits.size() > 0 ){
346  ofile << "\n" << endl;
347  ofile << "o epd" << endl;
348  ofile << "usemtl fcs_hits\n" << endl;
349  for ( auto p : fcsPreHits ){
350 
351  sphere( TVector3( p.X() * SCALE, p.Y() * SCALE, -p.Z() * SCALE ), 0.25, 10, 10, ofile );
352  }
353  }
354 
355  if (verbose){
356  LOG_INFO << "Viz has " << fcsClusters.size() << " FCS Hits" << endm;
357  }
358  if ( fcsClusters.size() > 0 ){
359  ofile << "\n" << endl;
360  ofile << "o fcs" << endl;
361  ofile << "usemtl fcs_hits\n" << endl;
362  int i = 0;
363  for ( auto p : fcsClusters ){
364  // sphere( TVector3( p.X() * SCALE, p.Y() * SCALE, -p.Z() * SCALE ), 0.75, 10, 10, ofile );
365  TVector3 ds = TVector3( 1, 1, fcsClusterEnergy[i] );
366  prism( ofile, TVector3( p.X() * SCALE, p.Y() * SCALE, -p.Z() * SCALE ), ds );
367  i++;
368  }
369  }
370 
371  // Write the track seeds
372  if (verbose){
373  LOG_INFO << "Viz has " << seeds.size() << " seeds" << endm;
374  }
375  if ( seeds.size() > 0 ){
376  ofile << "\n\no FwdSeeds\n" << endl;
377  ofile << "usemtl seeds\n" << endl;
378  // numVertices = 0;
379  for ( auto s : seeds ) {
380  size_t vStart = numVertices;
381  for ( auto h : s ){
382 
383  vert( ofile, h->getX() * SCALE, h->getY() * SCALE, -h->getZ() * SCALE );
384 
385  }
386 
387  ofile << "l ";
388  for ( size_t i = vStart; i < numVertices; i++){
389  ofile << i+1 << " ";
390  }
391  ofile << endl;
392  }
393  }
394 
395  // uncomment if you want a separate file for tracks
396  // ofile.open( (filename + "_tracks.obj" ).c_str() );
397  // numVertices = 0;
398  // EXPORT TRACKS
399 
400  if (verbose){
401  LOG_INFO << "Viz has " << tracks.size() << " tracks Hits" << endm;
402  }
403  if ( tracks.size() > 0 ){
404  ofile << "\n\no FwdTracks\n" << endl;
405  ofile << "usemtl tracks\n" << endl;
406  float zStep = 5.0; // cm
407  for ( auto t : tracks ) {
408  size_t vStart = numVertices;
409 
410 
411  TVector3 lpoint;
412  for ( float z = startPos.Z(); z < 875; z += zStep ){
413  TVector3 point = trackPosition( t, z );
414  if ( point.X() < -900 && point.Y() < -900 ) break;
415  if ( point.X() < -90 && point.Y() < -90 ) { z+= 50; continue;}
416 
417  vert( ofile, point.X() * SCALE, point.Y() * SCALE, -point.Z() * SCALE );
418  lpoint = point;
419  }
420 
421  ofile << "l ";
422  for ( size_t i = vStart; i < numVertices; i++){
423  ofile << i+1 << " ";
424  }
425  ofile << endl;
426  } // for t in tracks
427  } // if tracks.size() > 0
428 
429  ofile.close();
430  }
431 
432  size_t numVertices;
433 
434 };
435 
436 
437 #endif