// @(#)root/graf:$Id$
// Author: Rene Brun 16/05/97
/*************************************************************************
* Copyright (C) 1995-2000, Rene Brun and Fons Rademakers. *
* All rights reserved. *
* *
* For the licensing terms see $ROOTSYS/LICENSE. *
* For the list of contributors see $ROOTSYS/README/CREDITS. *
*************************************************************************/
//______________________________________________________________________________
/* Begin_Html
Graphical cut class
A Graphical cut.
A TCutG object is a closed polygon defining a closed region in a x,y plot.
It can be created via the graphics editor option "CutG" or directly by
invoking its constructor. The first and last points should be the same.
To create a TCutG via the graphics editor, use the left button to select the
points building the contour of the cut. Click on the right button to close the
TCutG. When it is created via the graphics editor, the TCutG object is named
"CUTG". It is recommended to immediatly change the name by using the context
menu item "SetName". When the graphics editor is used, the names of the
variables X,Y are automatically taken from the current pad title.
Example:
Assume a TTree object T and:
Root > T.Draw("abs(fMomemtum)%fEtot")
the TCutG members fVarX, fVary will be set to:
fVarx = fEtot
fVary = abs(fMomemtum)
A graphical cut can be used in a TTree selection expression:
Root > T.Draw("fEtot","cutg1")
where "cutg1" is the name of an existing graphical cut.
Note that, as shown in the example above, a graphical cut may be used in a
selection expression when drawing TTrees expressions of 1-d, 2-d or
3-dimensions. The expressions used in TTree::Draw can reference the variables in
the fVarX, fVarY of the graphical cut plus other variables.
When the TCutG object is created by TTree::Draw, it is added to the list of special objects in
the main TROOT object pointed by gROOT. To retrieve a pointer to this object
from the code or command line, do:
TCutG *mycutg;
mycutg = (TCutG*)gROOT->GetListOfSpecials()->FindObject("CUTG")
mycutg->SetName("mycutg");
When the TCutG is not created via TTree::Draw, one must set the variable names
corresponding to x,y if one wants to use the cut as input to TTree::Draw,eg
TCutG *cutg = new TCutG("mycut",5);
cutg->SetVarX("y");
cutg->SetVarY("x");
cutg->SetPoint(0,-0.3586207,1.509534);
cutg->SetPoint(1,-1.894181,-0.529661);
cutg->SetPoint(2,0.07780173,-1.21822);
cutg->SetPoint(3,-1.0375,-0.07944915);
cutg->SetPoint(4,0.756681,0.1853814);
cutg->SetPoint(5,-0.3586207,1.509534);
Example of use of a TCutG in TTree::Draw:
tree.Draw("x:y","mycutg && z>0 %% sqrt(x)>1")
A Graphical cut may be drawn via TGraph::Draw. It can be edited like a normal
TGraph.
A Graphical cut may be saved to a file via TCutG::Write.
End_Html */
#include
#include "Riostream.h"
#include "TROOT.h"
#include "TCutG.h"
#include "TVirtualPad.h"
#include "TPaveText.h"
#include "TH2.h"
#include "TClass.h"
#include "TMath.h"
ClassImp(TCutG)
//______________________________________________________________________________
TCutG::TCutG() : TGraph()
{
// TCutG default constructor.
fObjectX = 0;
fObjectY = 0;
}
//______________________________________________________________________________
TCutG::TCutG(const TCutG &cutg)
:TGraph(cutg)
{
// TCutG copy constructor
fVarX = cutg.fVarX;
fVarY = cutg.fVarY;
fObjectX = cutg.fObjectX;
fObjectY = cutg.fObjectY;
}
//______________________________________________________________________________
TCutG::TCutG(const char *name, Int_t n)
:TGraph(n)
{
// TCutG normal constructor.
fObjectX = 0;
fObjectY = 0;
SetName(name);
delete gROOT->GetListOfSpecials()->FindObject(name);
gROOT->GetListOfSpecials()->Add(this);
// Take name of cut variables from pad title if title contains ":"
if (gPad) {
TPaveText *ptitle = (TPaveText*)gPad->FindObject("title");
if (!ptitle) return;
TText *ttitle = ptitle->GetLineWith(":");
if (!ttitle) ttitle = ptitle->GetLineWith("{");
if (!ttitle) ttitle = ptitle->GetLine(0);
if (!ttitle) return;
const char *title = ttitle->GetTitle();
Int_t nch = strlen(title);
char *vars = new char[nch+1];
strlcpy(vars,title,nch+1);
char *col = strstr(vars,":");
if (col) {
*col = 0;
col++;
char *brak = strstr(col," {");
if (brak) *brak = 0;
fVarY = vars;
fVarX = col;
} else {
char *brak = strstr(vars," {");
if (brak) *brak = 0;
fVarX = vars;
}
delete [] vars;
}
}
//______________________________________________________________________________
TCutG::TCutG(const char *name, Int_t n, const Float_t *x, const Float_t *y)
:TGraph(n,x,y)
{
// TCutG normal constructor.
fObjectX = 0;
fObjectY = 0;
SetName(name);
delete gROOT->GetListOfSpecials()->FindObject(name);
gROOT->GetListOfSpecials()->Add(this);
// Take name of cut variables from pad title if title contains ":"
if (gPad) {
TPaveText *ptitle = (TPaveText*)gPad->FindObject("title");
if (!ptitle) return;
TText *ttitle = ptitle->GetLineWith(":");
if (!ttitle) ttitle = ptitle->GetLineWith("{");
if (!ttitle) ttitle = ptitle->GetLine(0);
if (!ttitle) return;
const char *title = ttitle->GetTitle();
Int_t nch = strlen(title);
char *vars = new char[nch+1];
strlcpy(vars,title,nch+1);
char *col = strstr(vars,":");
if (col) {
*col = 0;
col++;
char *brak = strstr(col," {");
if (brak) *brak = 0;
fVarY = vars;
fVarX = col;
} else {
char *brak = strstr(vars," {");
if (brak) *brak = 0;
fVarX = vars;
}
delete [] vars;
}
}
//______________________________________________________________________________
TCutG::TCutG(const char *name, Int_t n, const Double_t *x, const Double_t *y)
:TGraph(n,x,y)
{
// TCutG normal constructor.
fObjectX = 0;
fObjectY = 0;
SetName(name);
delete gROOT->GetListOfSpecials()->FindObject(name);
gROOT->GetListOfSpecials()->Add(this);
// Take name of cut variables from pad title if title contains ":"
if (gPad) {
TPaveText *ptitle = (TPaveText*)gPad->FindObject("title");
if (!ptitle) return;
TText *ttitle = ptitle->GetLineWith(":");
if (!ttitle) ttitle = ptitle->GetLineWith("{");
if (!ttitle) ttitle = ptitle->GetLine(0);
if (!ttitle) return;
const char *title = ttitle->GetTitle();
Int_t nch = strlen(title);
char *vars = new char[nch+1];
strlcpy(vars,title,nch+1);
char *col = strstr(vars,":");
if (col) {
*col = 0;
col++;
char *brak = strstr(col," {");
if (brak) *brak = 0;
fVarY = vars;
fVarX = col;
} else {
char *brak = strstr(vars," {");
if (brak) *brak = 0;
fVarX = vars;
}
delete [] vars;
}
}
//______________________________________________________________________________
TCutG::~TCutG()
{
// TCutG destructor.
delete fObjectX;
delete fObjectY;
gROOT->GetListOfSpecials()->Remove(this);
}
//______________________________________________________________________________
TCutG &TCutG::operator=(const TCutG &rhs)
{
// Assignment operator.
if (this != &rhs) {
TGraph::operator=(rhs);
delete fObjectX;
delete fObjectY;
fObjectX = rhs.fObjectX->Clone();
fObjectY = rhs.fObjectY->Clone();
}
return *this;
}
//______________________________________________________________________________
Double_t TCutG::Area() const
{
// Compute the area inside this TCutG
// The algorithm uses Stoke's theorem over the border of the closed polygon.
// Just as a reminder: Stoke's theorem reduces a surface integral
// to a line integral over the border of the surface integral.
Double_t a = 0;
Int_t n = GetN();
for (Int_t i=0;i xmax) xmax = fX[i];
if (fY[i] < ymin) ymin = fY[i];
if (fY[i] > ymax) ymax = fY[i];
}
TAxis *xaxis = h->GetXaxis();
TAxis *yaxis = h->GetYaxis();
Int_t binx1 = xaxis->FindBin(xmin);
Int_t binx2 = xaxis->FindBin(xmax);
Int_t biny1 = yaxis->FindBin(ymin);
Int_t biny2 = yaxis->FindBin(ymax);
Int_t nbinsx = h->GetNbinsX();
Stat_t integral = 0;
// Loop on bins for which the bin center is in the cut
TString opt = option;
opt.ToLower();
Bool_t width = kFALSE;
if (opt.Contains("width")) width = kTRUE;
Int_t bin, binx, biny;
for (biny=biny1;biny<=biny2;biny++) {
Double_t y = yaxis->GetBinCenter(biny);
for (binx=binx1;binx<=binx2;binx++) {
Double_t x = xaxis->GetBinCenter(binx);
if (!IsInside(x,y)) continue;
bin = binx +(nbinsx+2)*biny;
if (width) integral += h->GetBinContent(bin)*xaxis->GetBinWidth(binx)*yaxis->GetBinWidth(biny);
else integral += h->GetBinContent(bin);
}
}
return integral;
}
//______________________________________________________________________________
void TCutG::SavePrimitive(ostream &out, Option_t *option /*= ""*/)
{
// Save primitive as a C++ statement(s) on output stream out.
char quote = '"';
out<<" "<ClassSaved(TCutG::Class())) {
out<<" ";
} else {
out<<" TCutG *";
}
out<<"cutg = new TCutG("<SetVarX("<SetVarY("<SetTitle("<SetPoint("<Draw("
<GetListOfSpecials()->Add(this);
} else {
R__b.WriteClassBuffer(TCutG::Class(), this);
}
}