#include "TROOT.h"
#include "TClassRef.h"
#include "THStack.h"
#include "TVirtualPad.h"
#include "TVirtualHistPainter.h"
#include "THashList.h"
#include "TH2.h"
#include "TH3.h"
#include "TList.h"
#include "TStyle.h"
#include "Riostream.h"
#include "TBrowser.h"
#include "TMath.h"
#include "TObjString.h"
ClassImp(THStack)
THStack::THStack(): TNamed()
{
fHists = 0;
fStack = 0;
fHistogram = 0;
fMaximum = -1111;
fMinimum = -1111;
}
THStack::THStack(const char *name, const char *title)
: TNamed(name,title)
{
fHists = 0;
fStack = 0;
fHistogram = 0;
fMaximum = -1111;
fMinimum = -1111;
gROOT->GetListOfCleanups()->Add(this);
}
THStack::THStack(const TH1* hist, Option_t *axis ,
const char *name , const char *title ,
Int_t firstbin , Int_t lastbin ,
Int_t firstbin2 , Int_t lastbin2 ,
Option_t* proj_option , Option_t* draw_option ): TNamed(name, title) {
fHists = 0;
fStack = 0;
fHistogram = 0;
fMaximum = -1111;
fMinimum = -1111;
gROOT->GetListOfCleanups()->Add(this);
if (!axis) {
Warning("THStack", "Need an axis.");
return;
}
if (!hist) {
Warning("THStack", "Need a histogram.");
return;
}
Bool_t isTH2=hist->IsA()->InheritsFrom(TH2::Class());
Bool_t isTH3=hist->IsA()->InheritsFrom(TH3::Class());
if (!isTH2 && !isTH3) {
Warning("THStack", "Need a histogram deriving from TH2 or TH3.");
return;
}
if (!fName.Length())
fName=Form("%s_stack%s", hist->GetName(), axis);
if (!fTitle.Length()) {
if (hist->GetTitle() && strlen(hist->GetTitle()))
fTitle=Form("%s, stack of %s projections", hist->GetTitle(), axis);
else
fTitle=Form("stack of %s projections", axis);
}
if (isTH2) {
TH2* hist2=(TH2*) hist;
Bool_t useX=(strchr(axis,'x')) || (strchr(axis,'X'));
Bool_t useY=(strchr(axis,'y')) || (strchr(axis,'Y'));
if ((!useX && !useY) || (useX && useY)) {
Warning("THStack", "Need parameter axis=\"x\" or \"y\" for a TH2, not none or both.");
return;
}
TAxis* haxis= useX ? hist->GetYaxis() : hist->GetXaxis();
if (!haxis) {
Warning("HStack","Histogram axis is NULL");
return;
}
Int_t nbins = haxis->GetNbins();
if (firstbin < 0) firstbin = 1;
if (lastbin < 0) lastbin = nbins;
if (lastbin > nbins+1) lastbin = nbins;
for (Int_t iBin=firstbin; iBin<=lastbin; iBin++) {
TH1* hProj=0;
if (useX) hProj=hist2->ProjectionX(Form("%s_px%d",hist2->GetName(), iBin),
iBin, iBin, proj_option);
else hProj=hist2->ProjectionY(Form("%s_py%d",hist2->GetName(), iBin),
iBin, iBin, proj_option);
Add(hProj, draw_option);
}
} else {
TH3* hist3=(TH3*) hist;
TString sAxis(axis);
sAxis.ToLower();
Int_t dim=3-sAxis.Length();
if (dim<1 || dim>2) {
Warning("THStack", "Invalid length for parameter axis.");
return;
}
if (dim==1) {
TAxis* haxis = 0;
if (sAxis.First('x')==kNPOS)
haxis=hist->GetXaxis();
else if (sAxis.First('y')==kNPOS)
haxis=hist->GetYaxis();
else if (sAxis.First('z')==kNPOS)
haxis=hist->GetZaxis();
if (!haxis) {
Warning("HStack","Histogram axis is NULL");
return;
}
Int_t nbins = haxis->GetNbins();
if (firstbin < 0) firstbin = 1;
if (lastbin < 0) lastbin = nbins;
if (lastbin > nbins+1) lastbin = nbins;
Int_t iFirstOld=haxis->GetFirst();
Int_t iLastOld=haxis->GetLast();
for (Int_t iBin=firstbin; iBin<=lastbin; iBin++) {
haxis->SetRange(iBin, iBin);
TH1* hProj=hist3->Project3D(Form("%s_%s%s_%d", hist3->GetName(),
axis, proj_option, iBin));
Add(hProj, draw_option);
}
haxis->SetRange(iFirstOld, iLastOld);
} else {
TAxis* haxis1 = 0;
TAxis* haxis2 = 0;
if (sAxis.First('x')!=kNPOS) {
haxis1=hist->GetYaxis();
haxis2=hist->GetZaxis();
} else if (sAxis.First('y')!=kNPOS) {
haxis1=hist->GetXaxis();
haxis2=hist->GetZaxis();
} else if (sAxis.First('z')!=kNPOS) {
haxis1=hist->GetXaxis();
haxis2=hist->GetYaxis();
}
if (!haxis1 || !haxis2) {
Warning("HStack","Histogram axis is NULL");
return;
}
Int_t nbins1 = haxis1->GetNbins();
Int_t nbins2 = haxis2->GetNbins();
if (firstbin < 0) firstbin = 1;
if (lastbin < 0) lastbin = nbins1;
if (lastbin > nbins1+1) lastbin = nbins1;
if (firstbin2 < 0) firstbin2 = 1;
if (lastbin2 < 0) lastbin2 = nbins2;
if (lastbin2 > nbins2+1) lastbin2 = nbins2;
Int_t iFirstOld1=haxis1->GetFirst();
Int_t iLastOld1=haxis1->GetLast();
Int_t iFirstOld2=haxis2->GetFirst();
Int_t iLastOld2=haxis2->GetLast();
for (Int_t iBin=firstbin; iBin<=lastbin; iBin++) {
haxis1->SetRange(iBin, iBin);
for (Int_t jBin=firstbin2; jBin<=lastbin2; jBin++) {
haxis2->SetRange(jBin, jBin);
TH1* hProj=hist3->Project3D(Form("%s_%s%s_%d", hist3->GetName(),
axis, proj_option, iBin));
Add(hProj, draw_option);
}
}
haxis1->SetRange(iFirstOld1, iLastOld1);
haxis2->SetRange(iFirstOld2, iLastOld2);
}
}
}
THStack::~THStack()
{
gROOT->GetListOfCleanups()->Remove(this);
if (!fHists) return;
fHists->Clear("nodelete");
delete fHists;
fHists = 0;
if (fStack) {fStack->Delete(); delete fStack;}
delete fHistogram;
fHistogram = 0;
}
THStack::THStack(const THStack &hstack) :
TNamed(hstack),
fHists(0),
fStack(0),
fHistogram(0),
fMaximum(hstack.fMaximum),
fMinimum(hstack.fMinimum)
{
if (hstack.GetHists()) {
TIter next(hstack.GetHists());
TH1 *h;
while ((h=(TH1*)next())) Add(h);
}
}
void THStack::Add(TH1 *h1, Option_t *option)
{
if (!h1) return;
if (h1->GetDimension() > 2) {
Error("Add","THStack supports only 1-d and 2-d histograms");
return;
}
if (!fHists) fHists = new TList();
fHists->Add(h1,option);
Modified();
}
void THStack::Browse(TBrowser *b)
{
Draw(b ? b->GetDrawOption() : "");
gPad->Update();
}
void THStack::BuildStack()
{
if (fStack) return;
if (!fHists) return;
Int_t nhists = fHists->GetSize();
if (!nhists) return;
fStack = new TObjArray(nhists);
Bool_t add = TH1::AddDirectoryStatus();
TH1::AddDirectory(kFALSE);
TH1 *h = (TH1*)fHists->At(0)->Clone();
fStack->Add(h);
for (Int_t i=1;i<nhists;i++) {
h = (TH1*)fHists->At(i)->Clone();
h->Add((TH1*)fStack->At(i-1));
fStack->AddAt(h,i);
}
TH1::AddDirectory(add);
}
Int_t THStack::DistancetoPrimitive(Int_t px, Int_t py)
{
const Int_t kMaxDiff = 10;
Int_t distance = 9999;
if (fHistogram) {
distance = fHistogram->DistancetoPrimitive(px,py);
if (distance <= 0) {return distance;}
if (distance <= 1) {gPad->SetSelected(fHistogram);return distance;}
}
if (!fHists) return distance;
TH1 *h = 0;
const char *doption = GetDrawOption();
Int_t nhists = fHists->GetSize();
for (Int_t i=0;i<nhists;i++) {
h = (TH1*)fHists->At(i);
if (fStack && !strstr(doption,"nostack")) h = (TH1*)fStack->At(i);
Int_t dist = h->DistancetoPrimitive(px,py);
if (dist <= 0) return 0;
if (dist < kMaxDiff) {
gPad->SetSelected(fHists->At(i));
gPad->SetCursor(kPointer);
return dist;
}
}
return distance;
}
void THStack::Draw(Option_t *option)
{
TString opt = option;
opt.ToLower();
if (gPad) {
if (!gPad->IsEditable()) gROOT->MakeDefCanvas();
if (!opt.Contains("same")) {
if (TestBit(kCanDelete)) gPad->GetListOfPrimitives()->Remove(this);
gPad->Clear();
}
}
AppendPad(opt.Data());
}
TH1 *THStack::GetHistogram() const
{
if (fHistogram) return fHistogram;
if (!gPad) return 0;
gPad->Modified();
gPad->Update();
if (fHistogram) return fHistogram;
TH1 *h1 = (TH1*)gPad->FindObject("hframe");
return h1;
}
Double_t THStack::GetMaximum(Option_t *option)
{
TString opt = option;
opt.ToLower();
Bool_t lerr = kFALSE;
if (opt.Contains("e")) lerr = kTRUE;
Double_t them=0, themax = -1e300, c1, e1;
if (!fHists) return 0;
Int_t nhists = fHists->GetSize();
TH1 *h;
Int_t first,last;
if (!opt.Contains("nostack")) {
BuildStack();
h = (TH1*)fStack->At(nhists-1);
themax = h->GetMaximum();
} else {
for (Int_t i=0;i<nhists;i++) {
h = (TH1*)fHists->At(i);
them = h->GetMaximum();
if (them > themax) themax = them;
}
}
if (lerr) {
for (Int_t i=0;i<nhists;i++) {
h = (TH1*)fHists->At(i);
first = h->GetXaxis()->GetFirst();
last = h->GetXaxis()->GetLast();
for (Int_t j=first; j<=last;j++) {
e1 = h->GetBinError(j);
c1 = h->GetBinContent(j);
themax = TMath::Max(themax,c1+e1);
}
}
}
return themax;
}
Double_t THStack::GetMinimum(Option_t *option)
{
TString opt = option;
opt.ToLower();
Bool_t lerr = kFALSE;
if (opt.Contains("e")) lerr = kTRUE;
Double_t them=0, themin = 1e300, c1, e1;
if (!fHists) return 0;
Int_t nhists = fHists->GetSize();
Int_t first,last;
TH1 *h;
if (!opt.Contains("nostack")) {
BuildStack();
h = (TH1*)fStack->At(nhists-1);
themin = h->GetMinimum();
} else {
for (Int_t i=0;i<nhists;i++) {
h = (TH1*)fHists->At(i);
them = h->GetMinimum();
if (them <= 0 && gPad && gPad->GetLogy()) them = h->GetMinimum(0);
if (them < themin) themin = them;
}
}
if (lerr) {
for (Int_t i=0;i<nhists;i++) {
h = (TH1*)fHists->At(i);
first = h->GetXaxis()->GetFirst();
last = h->GetXaxis()->GetLast();
for (Int_t j=first; j<=last;j++) {
e1 = h->GetBinError(j);
c1 = h->GetBinContent(j);
themin = TMath::Min(themin,c1-e1);
}
}
}
return themin;
}
Int_t THStack::GetNhists() const
{
if (fHists) return fHists->GetSize();
return 0;
}
TObjArray *THStack::GetStack()
{
BuildStack();
return fStack;
}
TAxis *THStack::GetXaxis() const
{
if (!gPad) return 0;
TH1 *h = GetHistogram();
if (!h) return 0;
return h->GetXaxis();
}
TAxis *THStack::GetYaxis() const
{
if (!gPad) return 0;
TH1 *h = GetHistogram();
if (!h) return 0;
return h->GetYaxis();
}
void THStack::ls(Option_t *option) const
{
TROOT::IndentLevel();
cout <<IsA()->GetName()
<<" Name= "<<GetName()<<" Title= "<<GetTitle()<<" Option="<<option<<endl;
TROOT::IncreaseDirLevel();
if (fHists) fHists->ls(option);
TROOT::DecreaseDirLevel();
}
Long64_t THStack::Merge(TCollection* li, TFileMergeInfo * )
{
if (li==0 || li->GetEntries()==0) {
return fHists->GetEntries();
}
TIter next(li);
TList histLists;
while (TObject* o = next()) {
THStack *stack = dynamic_cast<THStack*> (o);
if (!stack) {
Error("Merge",
"Cannot merge - an object which doesn't inherit from THStack found in the list");
return -1;
}
histLists.Add(stack->GetHists());
}
fHists->Merge(&histLists);
return fHists->GetEntries();
}
void THStack::Modified()
{
if (!fStack) return;
fStack->Delete();
delete fStack;
fStack = 0;
delete fHistogram;
fHistogram = 0;
}
void THStack::Paint(Option_t *option)
{
if (!fHists) return;
if (!fHists->GetSize()) return;
TString opt = option;
opt.ToLower();
Bool_t lsame = kFALSE;
if (opt.Contains("same")) {
lsame = kTRUE;
opt.ReplaceAll("same","");
}
Bool_t lclear = kTRUE;
if (opt.Contains("noclear")) {
lclear = kFALSE;
opt.ReplaceAll("noclear","");
}
if (opt.Contains("pads")) {
Int_t npads = fHists->GetSize();
TVirtualPad *padsav = gPad;
Int_t nps = 0;
TObject *obj;
TIter nextp(padsav->GetListOfPrimitives());
while ((obj = nextp())) {
if (obj->InheritsFrom(TVirtualPad::Class())) nps++;
}
if (nps < npads) {
padsav->Clear();
Int_t nx = (Int_t)TMath::Sqrt((Double_t)npads);
if (nx*nx < npads) nx++;
Int_t ny = nx;
if (((nx*ny)-nx) >= npads) ny--;
padsav->Divide(nx,ny);
}
TH1 *h;
Int_t i = 0;
TObjOptLink *lnk = (TObjOptLink*)fHists->FirstLink();
while (lnk) {
i++;
padsav->cd(i);
h = (TH1*)lnk->GetObject();
h->Draw(lnk->GetOption());
lnk = (TObjOptLink*)lnk->Next();
}
padsav->cd();
return;
}
TH1 *h;
TIter next(fHists);
Double_t xmin = 1e100;
Double_t xmax = -xmin;
Double_t ymin = 1e100;
Double_t ymax = -xmin;
while ((h=(TH1*)next())) {
if (h->GetBuffer()) h->BufferEmpty(-1);
if (h->GetXaxis()->GetXmin() < xmin) xmin = h->GetXaxis()->GetXmin();
if (h->GetXaxis()->GetXmax() > xmax) xmax = h->GetXaxis()->GetXmax();
if (h->GetYaxis()->GetXmin() < ymin) ymin = h->GetYaxis()->GetXmin();
if (h->GetYaxis()->GetXmax() > ymax) ymax = h->GetYaxis()->GetXmax();
}
char loption[32];
snprintf(loption,31,"%s",opt.Data());
char *nostack = strstr(loption,"nostack");
if (!opt.Contains("nostack")) BuildStack();
Double_t themax,themin;
if (fMaximum == -1111) themax = GetMaximum(option);
else themax = fMaximum;
if (fMinimum == -1111) {
themin = GetMinimum(option);
if (gPad->GetLogy()){
if (themin>0) themin *= .9;
else themin = themax*1.e-3;
}
else if (themin > 0)
themin = 0;
}
else themin = fMinimum;
if (!fHistogram) {
Bool_t add = TH1::AddDirectoryStatus();
TH1::AddDirectory(kFALSE);
h = (TH1*)fHists->At(0);
TAxis *xaxis = h->GetXaxis();
TAxis *yaxis = h->GetYaxis();
const TArrayD *xbins = xaxis->GetXbins();
if (h->GetDimension() > 1) {
if (strlen(option) == 0) strlcpy(loption,"lego1",32);
const TArrayD *ybins = yaxis->GetXbins();
if (xbins->fN != 0 && ybins->fN != 0) {
fHistogram = new TH2F(GetName(),GetTitle(),
xaxis->GetNbins(), xbins->GetArray(),
yaxis->GetNbins(), ybins->GetArray());
} else if (xbins->fN != 0 && ybins->fN == 0) {
fHistogram = new TH2F(GetName(),GetTitle(),
xaxis->GetNbins(), xbins->GetArray(),
yaxis->GetNbins(), ymin, ymax);
} else if (xbins->fN == 0 && ybins->fN != 0) {
fHistogram = new TH2F(GetName(),GetTitle(),
xaxis->GetNbins(), xmin, xmax,
yaxis->GetNbins(), ybins->GetArray());
} else {
fHistogram = new TH2F(GetName(),GetTitle(),
xaxis->GetNbins(), xmin, xmax,
yaxis->GetNbins(), ymin, ymax);
}
} else {
if (xbins->fN != 0) {
fHistogram = new TH1F(GetName(),GetTitle(),
xaxis->GetNbins(), xbins->GetArray());
} else {
fHistogram = new TH1F(GetName(),GetTitle(),xaxis->GetNbins(),xmin, xmax);
}
}
fHistogram->SetStats(0);
TH1::AddDirectory(add);
} else {
fHistogram->SetTitle(GetTitle());
}
if (nostack) {*nostack = 0; strncat(nostack,nostack+7,7);}
else fHistogram->GetPainter()->SetStack(fHists);
if (!fHistogram->TestBit(TH1::kIsZoomed)) {
if (nostack && fMaximum != -1111) fHistogram->SetMaximum(fMaximum);
else {
if (gPad->GetLogy()) fHistogram->SetMaximum(themax*(1+0.2*TMath::Log10(themax/themin)));
else fHistogram->SetMaximum((1+gStyle->GetHistTopMargin())*themax);
}
if (nostack && fMinimum != -1111) fHistogram->SetMinimum(fMinimum);
else {
if (gPad->GetLogy()) fHistogram->SetMinimum(themin/(1+0.5*TMath::Log10(themax/themin)));
else fHistogram->SetMinimum(themin);
}
}
TH1 *hfirst;
TObjOptLink *lnk = (TObjOptLink*)fHists->FirstLink();
hfirst = (TH1*)lnk->GetObject();
THashList* labels = hfirst->GetXaxis()->GetLabels();
if (labels) {
TIter iL(labels);
TObjString* lb;
Int_t ilab = 1;
while ((lb=(TObjString*)iL())) {
fHistogram->GetXaxis()->SetBinLabel(ilab,lb->String().Data());
ilab++;
}
}
if (!lsame) fHistogram->Paint(loption);
if (fHistogram->GetDimension() > 1) SetDrawOption(loption);
if (strstr(loption,"lego")) return;
char noption[32];
strlcpy(noption,loption,32);
Int_t nhists = fHists->GetSize();
if (nostack) {
lnk = (TObjOptLink*)fHists->FirstLink();
for (Int_t i=0;i<nhists;i++) {
if (strstr(lnk->GetOption(),"same")) {
snprintf(loption,31,"%s%s",noption,lnk->GetOption());
} else {
snprintf(loption,31,"%ssame%s",noption,lnk->GetOption());
}
fHists->At(i)->Paint(loption);
lnk = (TObjOptLink*)lnk->Next();
}
} else {
lnk = (TObjOptLink*)fHists->LastLink();
TH1 *h1;
Int_t h1col, h1fill;
for (Int_t i=0;i<nhists;i++) {
if (strstr(lnk->GetOption(),"same")) {
snprintf(loption,31,"%s%s",noption,lnk->GetOption());
} else {
snprintf(loption,31,"%ssame%s",noption,lnk->GetOption());
}
h1 = (TH1*)fStack->At(nhists-i-1);
if (i>0 && lclear) {
h1col = h1->GetFillColor();
h1fill = h1->GetFillStyle();
h1->SetFillColor(10);
h1->SetFillStyle(1001);
h1->Paint(loption);
static TClassRef clTFrame = TClass::GetClass("TFrame",kFALSE);
TAttFill *frameFill = (TAttFill*)clTFrame->DynamicCast(TAttFill::Class(),gPad->GetFrame());
if (frameFill) {
h1->SetFillColor(frameFill->GetFillColor());
h1->SetFillStyle(frameFill->GetFillStyle());
}
h1->Paint(loption);
h1->SetFillColor(h1col);
h1->SetFillStyle(h1fill);
}
h1->Paint(loption);
lnk = (TObjOptLink*)lnk->Prev();
}
}
if (!lsame) fHistogram->Paint("axissame");
}
void THStack::Print(Option_t *option) const
{
TH1 *h;
if (fHists) {
TIter next(fHists);
while ((h = (TH1*) next())) {
h->Print(option);
}
}
}
void THStack::RecursiveRemove(TObject *obj)
{
if (!fHists) return;
fHists->RecursiveRemove(obj);
while (fHists->IndexOf(obj) >= 0) fHists->Remove(obj);
}
void THStack::SavePrimitive(ostream &out, Option_t *option )
{
char quote = '"';
out<<" "<<endl;
if (gROOT->ClassSaved(THStack::Class())) {
out<<" ";
} else {
out<<" THStack *";
}
out<<GetName()<<" = new THStack();"<<endl;
out<<" "<<GetName()<<"->SetName("<<quote<<GetName()<<quote<<");"<<endl;
out<<" "<<GetName()<<"->SetTitle("<<quote<<GetTitle()<<quote<<");"<<endl;
if (fMinimum != -1111) {
out<<" "<<GetName()<<"->SetMinimum("<<fMinimum<<");"<<endl;
}
if (fMaximum != -1111) {
out<<" "<<GetName()<<"->SetMaximum("<<fMaximum<<");"<<endl;
}
static Int_t frameNumber = 0;
if (fHistogram) {
frameNumber++;
TString hname = fHistogram->GetName();
hname += "_stack_";
hname += frameNumber;
fHistogram->SetName(hname.Data());
fHistogram->SavePrimitive(out,"nodraw");
out<<" "<<GetName()<<"->SetHistogram("<<fHistogram->GetName()<<");"<<endl;
out<<" "<<endl;
}
TH1 *h;
if (fHists) {
TObjOptLink *lnk = (TObjOptLink*)fHists->FirstLink();
while (lnk) {
h = (TH1*)lnk->GetObject();
h->SavePrimitive(out,"nodraw");
out<<" "<<GetName()<<"->Add("<<h->GetName()<<","<<quote<<lnk->GetOption()<<quote<<");"<<endl;
lnk = (TObjOptLink*)lnk->Next();
}
}
out<<" "<<GetName()<<"->Draw("
<<quote<<option<<quote<<");"<<endl;
}
void THStack::SetMaximum(Double_t maximum)
{
fMaximum = maximum;
if (fHistogram) fHistogram->SetMaximum(maximum);
}
void THStack::SetMinimum(Double_t minimum)
{
fMinimum = minimum;
if (fHistogram) fHistogram->SetMinimum(minimum);
}