source: trunk/MagicSoft/Mars/mhist/MH3.cc@ 2312

Last change on this file since 2312 was 2300, checked in by wittek, 22 years ago
*** empty log message ***
File size: 19.0 KB
Line 
1/* ======================================================================== *\
2!
3! *
4! * This file is part of MARS, the MAGIC Analysis and Reconstruction
5! * Software. It is distributed to you in the hope that it can be a useful
6! * and timesaving tool in analysing Data of imaging Cerenkov telescopes.
7! * It is distributed WITHOUT ANY WARRANTY.
8! *
9! * Permission to use, copy, modify and distribute this software and its
10! * documentation for any purpose is hereby granted without fee,
11! * provided that the above copyright notice appear in all copies and
12! * that both that copyright notice and this permission notice appear
13! * in supporting documentation. It is provided "as is" without express
14! * or implied warranty.
15! *
16!
17!
18! Author(s): Thomas Bretz 2002 <mailto:tbretz@astro.uni-wuerzburg.de>
19!
20! Copyright: MAGIC Software Development, 2000-2002
21!
22!
23\* ======================================================================== */
24
25/////////////////////////////////////////////////////////////////////////////
26//
27// MH3
28//
29// With this histogram you can fill a histogram with up to three
30// variables from Mars parameter containers in an eventloop.
31//
32// In the constructor you can give up to three variables which should be
33// filled in the histogram. Dependend on the number of given variables
34// (data members) a TH1F, TH2F or TH3F is created.
35// Specify the data mamber like the following:
36// "MHillas.fLength"
37// Where MHillas is the name of the parameter container in the parameter
38// list and fLength is the name of the data member which should be filled
39// in the histogram. Assuming that your MHillas container has a different
40// name (MyHillas) the name to give would be:
41// "MyHillas.fLength"
42//
43// The axis binning is retrieved from the parameter list, too. Create a
44// MBinning with the name "Binning" plus the name of your MH3 container
45// plus the axis name ("X", "Y" or "Z") and add it to the parameter list.
46//
47// If you want to use a different unit for histogramming use SetScaleX,
48// SetScaleY and SetScaleZ.
49//
50// For example:
51// MH3 myhist("MHillas.fLength");
52// myhist.SetName("MyHist");
53// myhist.SetScaleX(geomcam.GetConvMm2Deg()); //convert length to degree
54// MBinning bins("BinningMyHistX");
55// bins.SetEdges(10, 0, 150);
56// plist.AddToList(&myhist);
57// plist.AddToList(&bins);
58//
59/////////////////////////////////////////////////////////////////////////////
60#include "MH3.h"
61
62#include <ctype.h> // tolower
63#include <fstream>
64
65#include <TPad.h>
66#include <TStyle.h>
67#include <TCanvas.h>
68
69#include <TH2.h>
70#include <TH3.h>
71#include <TProfile.h>
72#include <TProfile2D.h>
73
74#include "MLog.h"
75#include "MLogManip.h"
76
77#include "MParList.h"
78#include "MBinning.h"
79#include "MDataChain.h"
80
81ClassImp(MH3);
82
83using namespace std;
84
85static const TString gsDefName = "MH3";
86static const TString gsDefTitle = "Container for a %dD Mars Histogram";
87
88// --------------------------------------------------------------------------
89//
90// Default constructor.
91//
92MH3::MH3(const unsigned int dim)
93 : fDimension(dim>3?3:dim), fHist(NULL)
94{
95 switch (fDimension)
96 {
97 case 1:
98 fHist = new TH1F;
99 fHist->SetYTitle("Counts");
100 break;
101 case 2:
102 fHist = new TH2F;
103 fHist->SetZTitle("Counts");
104 break;
105 case 3:
106 fHist = new TH3F;
107 break;
108 }
109
110 fData[0] = NULL;
111 fData[1] = NULL;
112 fData[2] = NULL;
113
114 fName = gsDefName;
115 fTitle = Form(gsDefTitle.Data(), 1);
116
117 if (fHist)
118 fHist->SetDirectory(NULL);
119
120 fScale[0] = 1;
121 fScale[1] = 1;
122 fScale[2] = 1;
123}
124
125// --------------------------------------------------------------------------
126//
127// Creates an TH1F. memberx is filled into the X-bins. For a more detailed
128// description see the class description above.
129//
130MH3::MH3(const char *memberx)
131 : fDimension(1)
132{
133 fHist = new TH1F;
134
135 fData[0] = new MDataChain(memberx);
136 fData[1] = NULL;
137 fData[2] = NULL;
138
139 fName = gsDefName;
140 fTitle = Form(gsDefTitle.Data(), 1);
141
142 fHist->SetDirectory(NULL);
143 fHist->SetYTitle("Counts");
144
145 fScale[0] = 1;
146 fScale[1] = 1;
147 fScale[2] = 1;
148}
149
150// --------------------------------------------------------------------------
151//
152// Creates an TH2F. memberx is filled into the X-bins. membery is filled
153// into the Y-bins. For a more detailed description see the class
154// description above.
155//
156MH3::MH3(const char *memberx, const char *membery)
157 : fDimension(2)
158{
159 fHist = new TH2F;
160
161 fData[0] = new MDataChain(memberx);
162 fData[1] = new MDataChain(membery);
163 fData[2] = NULL;
164
165 fName = gsDefName;
166 fTitle = Form(gsDefTitle.Data(), 2);
167
168 fHist->SetDirectory(NULL);
169 fHist->SetZTitle("Counts");
170
171 fScale[0] = 1;
172 fScale[1] = 1;
173 fScale[2] = 1;
174}
175
176// --------------------------------------------------------------------------
177//
178// Creates an TH3F. memberx is filled into the X-bins. membery is filled
179// into the Y-bins. membery is filled into the Z-bins. For a more detailed
180// description see the class description above.
181//
182MH3::MH3(const char *memberx, const char *membery, const char *memberz)
183 : fDimension(3)
184{
185 fHist = new TH3F;
186
187 fData[0] = new MDataChain(memberx);
188 fData[1] = new MDataChain(membery);
189 fData[2] = new MDataChain(memberz);
190
191 fName = gsDefName;
192 fTitle = Form(gsDefTitle.Data(), 3);
193
194 fHist->SetDirectory(NULL);
195
196 fScale[0] = 1;
197 fScale[1] = 1;
198 fScale[2] = 1;
199}
200
201// --------------------------------------------------------------------------
202//
203// Deletes the histogram
204//
205MH3::~MH3()
206{
207 delete fHist;
208
209 for (int i=0; i<3; i++)
210 if (fData[i])
211 delete fData[i];
212}
213
214// --------------------------------------------------------------------------
215//
216// Return the data members used by the data chain to be used in
217// MTask::AddBranchToList
218//
219TString MH3::GetDataMember() const
220{
221 TString str=fData[0]->GetDataMember();
222 if (fData[1])
223 {
224 str += ";";
225 str += fData[1]->GetDataMember();
226 }
227 if (fData[2])
228 {
229 str += ";";
230 str += fData[2]->GetDataMember();
231 }
232 return str;
233}
234
235// --------------------------------------------------------------------------
236//
237// Setup the Binning for the histograms automatically if the correct
238// instances of MBinning are found in the parameter list
239// For a more detailed description see class description above.
240//
241Bool_t MH3::SetupFill(const MParList *plist)
242{
243 // reset histogram (necessary if the same eventloop is run more than once)
244 fHist->Reset();
245
246 TString bname("Binning");
247 bname += fName;
248
249 MBinning *binsx = NULL;
250 MBinning *binsy = NULL;
251 MBinning *binsz = NULL;
252
253 switch (fDimension)
254 {
255 case 3:
256 binsz = (MBinning*)plist->FindObject(bname+"Z", "MBinning");
257 if (!binsz)
258 {
259 *fLog << err << dbginf << "MBinning '" << bname << "X' not found... aborting." << endl;
260 return kFALSE;
261 }
262 if (binsz->IsLogarithmic())
263 fHist->SetBit(kIsLogz);
264 if (fData[2]) fHist->SetZTitle(fData[2]->GetTitle());
265 if (fData[2] && !fData[2]->PreProcess(plist))
266 return kFALSE;
267 case 2:
268 binsy = (MBinning*)plist->FindObject(bname+"Y", "MBinning");
269 if (!binsy)
270 {
271 *fLog << err << dbginf << "MBinning '" << bname << "Y' not found... aborting." << endl;
272 return kFALSE;
273 }
274 if (binsy->IsLogarithmic())
275 fHist->SetBit(kIsLogy);
276 if (fData[1]) fHist->SetYTitle(fData[1]->GetTitle());
277 if (fData[1] && !fData[1]->PreProcess(plist))
278 return kFALSE;
279 case 1:
280 binsx = (MBinning*)plist->FindObject(bname+"X", "MBinning");
281 if (!binsx)
282 {
283 if (fDimension==1)
284 binsx = (MBinning*)plist->FindObject(bname, "MBinning");
285
286 if (!binsx)
287 {
288 *fLog << err << dbginf << "Neither MBinning '" << bname << "X' nor '" << bname << "' found... aborting." << endl;
289 return kFALSE;
290 }
291
292 }
293 if (binsx->IsLogarithmic())
294 fHist->SetBit(kIsLogx);
295
296 if (fData[0]!=NULL) fHist->SetXTitle(fData[0]->GetTitle());
297 if (fData[0] && !fData[0]->PreProcess(plist))
298 return kFALSE;
299 }
300
301 fHist->SetName(fName);
302
303 TString title("Histogram for ");
304 title += fName;
305
306 switch (fDimension)
307 {
308 case 1:
309 fHist->SetTitle(title+" (1D)");
310 SetBinning(fHist, binsx);
311 return kTRUE;
312 case 2:
313 fHist->SetTitle(title+" (2D)");
314 SetBinning((TH2*)fHist, binsx, binsy);
315 return kTRUE;
316 case 3:
317 fHist->SetTitle(title+" (3D)");
318 SetBinning((TH3*)fHist, binsx, binsy, binsz);
319 return kTRUE;
320 }
321 cout << "Still alive...?" << endl;
322 return kTRUE;
323}
324
325// --------------------------------------------------------------------------
326//
327// Set the name of the histogram ant the MH3 container
328//
329void MH3::SetName(const char *name)
330{
331 fHist->SetName(name);
332 MParContainer::SetName(name);
333}
334
335// --------------------------------------------------------------------------
336//
337// Set the title of the histogram ant the MH3 container
338//
339void MH3::SetTitle(const char *title)
340{
341 fHist->SetTitle(title);
342 MParContainer::SetTitle(title);
343}
344
345// --------------------------------------------------------------------------
346//
347// Fills the one, two or three data members into our histogram
348//
349Bool_t MH3::Fill(const MParContainer *par, const Stat_t w)
350{
351 Double_t x=0;
352 Double_t y=0;
353 Double_t z=0;
354
355 switch (fDimension)
356 {
357 case 3:
358 z = fData[2]->GetValue()*fScale[2];
359 case 2:
360 y = fData[1]->GetValue()*fScale[1];
361 case 1:
362 x = fData[0]->GetValue()*fScale[0];
363 }
364
365 switch (fDimension)
366 {
367 case 3:
368 ((TH3*)fHist)->Fill(x, y, z, w);
369 return kTRUE;
370 case 2:
371 ((TH2*)fHist)->Fill(x, y, w);
372 return kTRUE;
373 case 1:
374 fHist->Fill(x, w);
375 return kTRUE;
376 }
377
378 return kFALSE;
379}
380/*
381// --------------------------------------------------------------------------
382//
383// Set the palette you wanna use:
384// - you could set the root "Pretty Palette Violet->Red" by
385// gStyle->SetPalette(1, 0), but in some cases this may look
386// confusing
387// - The maximum colors root allowes us to set by ourself
388// is 50 (idx: 51-100). This colors are set to a grayscaled
389// palette
390// - the number of contours must be two less than the number
391// of palette entries
392//
393void MHStarMap::PrepareDrawing() const
394{
395 const Int_t numg = 32; // number of gray scaled colors
396 const Int_t numw = 32; // number of white
397
398 Int_t palette[numg+numw];
399
400 //
401 // The first half of the colors are white.
402 // This is some kind of optical background supression
403 //
404 gROOT->GetColor(51)->SetRGB(1, 1, 1);
405
406 Int_t i;
407 for (i=0; i<numw; i++)
408 palette[i] = 51;
409
410 //
411 // now the (gray) scaled part is coming
412 //
413 for (;i<numw+numg; i++)
414 {
415 const Float_t gray = 1.0-(float)(i-numw)/(numg-1.0);
416
417 gROOT->GetColor(52+i)->SetRGB(gray, gray, gray);
418 palette[i] = 52+i;
419 }
420
421 //
422 // Set the palette and the number of contour levels
423 //
424 gStyle->SetPalette(numg+numw, palette);
425 fStarMap->SetContour(numg+numw-2);
426}
427*/
428// --------------------------------------------------------------------------
429//
430// Setup a inversed deep blue sea palette for the fCenter histogram.
431//
432void MH3::SetColors() const
433{
434 // FIXME: This must be redone each time the canvas is repainted....
435 gStyle->SetPalette(51, NULL);
436 Int_t c[50];
437 for (int i=0; i<50; i++)
438 c[49-i] = gStyle->GetColorPalette(i);
439 gStyle->SetPalette(50, c);
440}
441
442// --------------------------------------------------------------------------
443//
444// Draw clone of histogram. So that the object can be deleted
445//
446// Possible options are:
447// PROFX: Draw a x-profile into the histogram (for 2D histograms only)
448// PROFY: Draw a y-profile into the histogram (for 2D histograms only)
449// ONLY: Draw the profile histogram only (for 2D histograms only)
450//
451// If the kIsLog?-Bit is set the axis is displayed lkogarithmically.
452// eg this is set when applying a logarithmic MBinning
453//
454// and the histogram is still visible in the canvas.
455// The cloned object are deleted together with the canvas if the canvas is
456// destroyed. If you want to handle destroying the canvas you can get a
457// pointer to it from this function
458//
459/*
460TObject *MH3::DrawClone(Option_t *opt) const
461{
462 TString str(opt);
463
464 TVirtualPad *c = gPad;
465 if (!str.Contains("nonew", TString::kIgnoreCase))
466 {
467 c = MH::MakeDefCanvas(fHist);
468
469 //
470 // This is necessary to get the expected bahviour of DrawClone
471 //
472 gROOT->SetSelectedPad(NULL);
473 }
474
475 if (str.Contains("COL", TString::kIgnoreCase))
476 SetColors();
477
478 Bool_t only = str.Contains("ONLY", TString::kIgnoreCase) && fDimension==2;
479 if (!only)
480 fHist->DrawCopy(opt);
481
482 if (str.Contains("PROFX", TString::kIgnoreCase) && fDimension==2)
483 {
484 TProfile *p = ((TH2*)fHist)->ProfileX("_pfx", -1, 9999, "s");
485 p->SetLineColor(kBlue);
486 p->Draw(only?"":"same");
487 p->SetBit(kCanDelete);
488 p->SetDirectory(NULL);
489 }
490 if (str.Contains("PROFY", TString::kIgnoreCase) && fDimension==2)
491 {
492 TProfile *p = ((TH2*)fHist)->ProfileY("_pfy", -1, 9999, "s");
493 p->SetLineColor(kBlue);
494 p->Draw(only?"":"same");
495 p->SetBit(kCanDelete);
496 p->SetDirectory(NULL);
497 }
498
499 if (fHist->TestBit(kIsLogx)) c->SetLogx();
500 if (fHist->TestBit(kIsLogy)) c->SetLogy();
501 if (fHist->TestBit(kIsLogz)) c->SetLogz();
502
503 c->Modified();
504 c->Update();
505
506 return c;
507}
508*/
509
510// --------------------------------------------------------------------------
511//
512// Creates a new canvas and draws the histogram into it.
513//
514// Possible options are:
515// PROFX: Draw a x-profile into the histogram (for 2D histograms only)
516// PROFY: Draw a y-profile into the histogram (for 2D histograms only)
517// ONLY: Draw the profile histogram only (for 2D histograms only)
518//
519// If the kIsLog?-Bit is set the axis is displayed lkogarithmically.
520// eg this is set when applying a logarithmic MBinning
521//
522// Be careful: The histogram belongs to this object and won't get deleted
523// together with the canvas.
524//
525void MH3::Draw(Option_t *opt)
526{
527 TVirtualPad *pad = gPad ? gPad : MakeDefCanvas(fHist);
528 pad->SetBorderMode(0);
529
530 AppendPad("");
531
532 TString str(opt);
533
534 // FIXME: Do it in Paint()
535 if (str.Contains("COL", TString::kIgnoreCase))
536 SetColors();
537
538 fHist->SetFillStyle(4000);
539
540 Bool_t only = str.Contains("ONLY", TString::kIgnoreCase) && fDimension==2;
541 if (!only)
542 fHist->Draw(opt);
543
544 if (str.Contains("PROFX", TString::kIgnoreCase) && fDimension==2)
545 {
546 TProfile *p = ((TH2*)fHist)->ProfileX("_pfx", -1, 9999, "s");
547 p->SetLineColor(kBlue);
548 p->Draw(only?"":"same");
549 p->SetBit(kCanDelete);
550 p->SetDirectory(NULL);
551 }
552 if (str.Contains("PROFY", TString::kIgnoreCase) && fDimension==2)
553 {
554 TProfile *p = ((TH2*)fHist)->ProfileY("_pfy", -1, 9999, "s");
555 p->SetLineColor(kBlue);
556 p->Draw(only?"":"same");
557 p->SetBit(kCanDelete);
558 p->SetDirectory(NULL);
559 }
560
561 if (fHist->TestBit(kIsLogx)) pad->SetLogx();
562 if (fHist->TestBit(kIsLogy)) pad->SetLogy();
563 if (fHist->TestBit(kIsLogz)) pad->SetLogz();
564
565 pad->Modified();
566 pad->Update();
567}
568
569// --------------------------------------------------------------------------
570//
571// Implementation of SavePrimitive. Used to write the call to a constructor
572// to a macro. In the original root implementation it is used to write
573// gui elements to a macro-file.
574//
575void MH3::StreamPrimitive(ofstream &out) const
576{
577 TString name = GetUniqueName();
578
579 out << " MH3 " << name << "(\"";
580 out << fData[0]->GetRule() << "\"";
581 if (fDimension>1)
582 out << ", \"" << fData[1]->GetRule() << "\"";
583 if (fDimension>2)
584 out << ", \"" << fData[2]->GetRule() << "\"";
585
586 out << ");" << endl;
587
588 if (fName!=gsDefName)
589 out << " " << name << ".SetName(\"" << fName << "\");" << endl;
590
591 if (fTitle!=Form(gsDefTitle.Data(), fDimension))
592 out << " " << name << ".SetTitle(\"" << fTitle << "\");" << endl;
593
594 switch (fDimension)
595 {
596 case 3:
597 if (fScale[2]!=1)
598 out << " " << name << ".SetScaleZ(" << fScale[2] << ");" << endl;
599 case 2:
600 if (fScale[1]!=1)
601 out << " " << name << ".SetScaleY(" << fScale[1] << ");" << endl;
602 case 1:
603 if (fScale[0]!=1)
604 out << " " << name << ".SetScaleX(" << fScale[0] << ");" << endl;
605 }
606}
607
608// --------------------------------------------------------------------------
609//
610// Used to rebuild a MH3 object of the same type (data members,
611// dimension, ...)
612//
613MParContainer *MH3::New() const
614{
615 MH3 *h = NULL;
616
617 if (fData[0] == NULL)
618 h=new MH3(fDimension);
619 else
620 switch (fDimension)
621 {
622 case 1:
623 h=new MH3(fData[0]->GetRule());
624 break;
625 case 2:
626 h=new MH3(fData[0]->GetRule(), fData[1]->GetRule());
627 break;
628 case 3:
629 h=new MH3(fData[0]->GetRule(), fData[1]->GetRule(), fData[2]->GetRule());
630 break;
631 }
632 switch (fDimension)
633 {
634 case 3:
635 h->SetScaleZ(fScale[2]);
636 case 2:
637 h->SetScaleY(fScale[1]);
638 case 1:
639 h->SetScaleX(fScale[0]);
640 }
641 return h;
642}
643
644TString MH3::GetRule(const Char_t axis) const
645{
646 switch (tolower(axis))
647 {
648 case 'x':
649 return fData[0] ? fData[0]->GetRule() : TString("");
650 case 'y':
651 return fData[1] ? fData[1]->GetRule() : TString("");
652 case 'z':
653 return fData[2] ? fData[2]->GetRule() : TString("");
654 default:
655 return "<n/a>";
656 }
657}
658
659// --------------------------------------------------------------------------
660//
661// Returns the total number of bins in a histogram (excluding under- and
662// overflow bins)
663//
664Int_t MH3::GetNbins() const
665{
666 Int_t num = 1;
667
668 switch (fDimension)
669 {
670 case 3:
671 num *= fHist->GetNbinsZ()+2;
672 case 2:
673 num *= fHist->GetNbinsY()+2;
674 case 1:
675 num *= fHist->GetNbinsX()+2;
676 }
677
678 return num;
679}
680
681// --------------------------------------------------------------------------
682//
683// Returns the total number of bins in a histogram (excluding under- and
684// overflow bins) Return -1 if bin is underflow or overflow
685//
686Int_t MH3::FindFixBin(Double_t x, Double_t y, Double_t z) const
687{
688 const TAxis &axex = *fHist->GetXaxis();
689 const TAxis &axey = *fHist->GetYaxis();
690 const TAxis &axez = *fHist->GetZaxis();
691
692 Int_t binz = 0;
693 Int_t biny = 0;
694 Int_t binx = 0;
695
696 switch (fDimension)
697 {
698 case 3:
699 binz = axez.FindFixBin(z);
700 if (binz>axez.GetLast() || binz<axez.GetFirst())
701 return -1;
702 case 2:
703 biny = axey.FindFixBin(y);
704 if (biny>axey.GetLast() || biny<axey.GetFirst())
705 return -1;
706 case 1:
707 binx = axex.FindFixBin(x);
708 if (binx<axex.GetFirst() || binx>axex.GetLast())
709 return -1;
710 }
711
712 const Int_t nx = fHist->GetNbinsX()+2;
713 const Int_t ny = fHist->GetNbinsY()+2;
714
715 return binx + nx*(biny +ny*binz);
716}
Note: See TracBrowser for help on using the repository browser.