8 #include "TContextMenu.h"
9 #include "TRootCanvas.h"
11 #include "KVKeyHandler.h"
12 #include <KeySymbols.h>
14 #include "KVHistoManipulator.h"
18 #include "TVirtualX.h"
19 #include <Riostream.h>
23 TObject* gCopyObject = 0x0;
61 KVCanvas::KVCanvas(
const char* name,
const char* title, Int_t ww, Int_t wh, Bool_t): TCanvas(name, title, ww, wh)
79 KVCanvas::KVCanvas(
const char* name, Int_t ww, Int_t wh, Int_t winid): TCanvas(name, ww, wh, winid)
129 if (!TestBit(kAutoExec))
return;
131 ((TPad*)gPad)->AutoExec();
192 const Int_t kTMAX = 256;
193 static char atext[kTMAX];
195 if (!TestBit(kShowEventStatus) || !selected)
return;
197 if (!fCanvasImp)
return;
199 TVirtualPad* savepad;
201 gPad = GetSelectedPad();
203 fCanvasImp->SetStatusText(selected->GetTitle(), 0);
204 fCanvasImp->SetStatusText(selected->GetName(), 1);
205 if (event == kKeyPress)
206 snprintf(atext, kTMAX,
"%c", (
char) px);
208 snprintf(atext, kTMAX,
"%d,%d", px, py);
209 fCanvasImp->SetStatusText(atext, 2);
210 fCanvasImp->SetStatusText(selected->GetObjectInfo(px, py), 3);
230 TPad* prevSelPad = (TPad*) fSelectedPad;
231 TObject* prevSelObj = fSelected;
233 fPadSave = (TPad*)gPad;
241 Bool_t sendOrder =
true;
255 pad = Pick(px, py, prevSelObj);
258 EnterLeave(prevSelPad, prevSelObj);
264 if (sendOrder) fSelected->ExecuteEvent(event, px, py);
276 Double_t ppx = AbsPixeltoX(px);
277 Double_t ppy = AbsPixeltoY(py);
279 TAxis* ax = TheHisto->GetXaxis();
280 Int_t
X0 = ax->GetFirst();
281 Int_t X1 = ax->GetLast();
282 Int_t NbinsX = ax->GetNbins();
283 px = ax->FindBin(ppx);
285 Double_t ddX = (X1 +
X0) * 0.5 - px;
286 Double_t distX = TMath::Abs(ddX) / (X1 -
X0);
287 if (distX >= 0.5)
return;
289 TAxis* ay = TheHisto->GetYaxis();
290 Int_t
Y0 = ay->GetFirst();
291 Int_t Y1 = ay->GetLast();
292 Int_t NbinsY = ay->GetNbins();
293 py = ay->FindBin(ppy);
295 Double_t ddY = (Y1 +
Y0) * 0.5 - py;
296 Double_t distY = TMath::Abs(ddY) / (Y1 -
Y0);
297 if (distY >= 0.5)
return;
299 if ((distX <= size) && (distY <= size))
return;
300 dX = TMath::Nint(ddX * (0.05 + 0.05 *
fVenerMode));
301 dY = TMath::Nint(ddY * (0.05 + 0.05 *
fVenerMode));
303 if (TMath::Abs(dX) < 1) dX = TMath::Sign(1., ddX);
304 if (TMath::Abs(dY) < 1) dY = TMath::Sign(1., ddY);
308 if ((
X0 - dX > 0) && (X1 - dX < NbinsX)) {
309 ax->SetRange(
X0 - dX, X1 - dX);
312 if ((
Y0 - dY > 0) && (Y1 - dY < NbinsY)) {
313 ay->SetRange(
Y0 - dY, Y1 - dY);
326 if (!fDoubleBuffer) FeedbackMode(kTRUE);
333 TObject* sobj = fSelected;
334 TPad* spad = fSelectedPad;
337 EnterLeave(prevSelPad, prevSelObj);
340 if (!fDoubleBuffer) FeedbackMode(kFALSE);
350 pad = Pick(px, py, prevSelObj);
358 fSelected->ExecuteEvent(event, px, py);
362 if (fSelected->InheritsFrom(
"TH2")) {
367 gVirtualX->DrawBox(XtoAbsPixel(
xmin), YtoAbsPixel(
ymin),
oldx,
oldy, TVirtualX::kHollow);
376 if (fSelected->InheritsFrom(
"TH2")) {
377 gVirtualX->DrawBox(XtoAbsPixel(
xmin), YtoAbsPixel(
ymin),
oldx,
oldy, TVirtualX::kHollow);
380 gVirtualX->DrawBox(XtoAbsPixel(
xmin), YtoAbsPixel(
ymin),
oldx,
oldy, TVirtualX::kHollow);
384 case kButton1ShiftMotion:
388 if (sendOrder) fSelected->ExecuteEvent(event, px, py);
391 if (!fSelected->InheritsFrom(TAxis::Class())) {
392 Bool_t resize = kFALSE;
393 if (fSelected->InheritsFrom(TBox::Class()))
394 resize = ((TBox*)fSelected)->IsBeingResized();
395 if (fSelected->InheritsFrom(TVirtualPad::Class()))
396 resize = ((TVirtualPad*)fSelected)->IsBeingResized();
398 if ((!resize && TestBit(kMoveOpaque)) || (resize && TestBit(kResizeOpaque))) {
415 if (sendOrder) fSelected->ExecuteEvent(event, px, py);
425 if (fSelected->InheritsFrom(
"TH2") &&
moved && !fSelected->InheritsFrom(
"TH3")) {
426 xmax = AbsPixeltoX(GetEventX());
427 ymax = AbsPixeltoY(GetEventY());
451 pad = Pick(px, py, prevSelObj);
459 if (!fSelected->InheritsFrom(
"TH1")) fSelected->Pop();
462 if (fSelected->InheritsFrom(
"TH2") && !fSelected->InheritsFrom(
"TH3")) {
475 Int_t pixelWidthX = gPad->XtoAbsPixel(gPad->GetUxmax()) - gPad->XtoAbsPixel(gPad->GetUxmin());
476 Int_t pixelWidthY = gPad->YtoAbsPixel(gPad->GetUymax()) - gPad->YtoAbsPixel(gPad->GetUymin());
488 printf(
"Current Pad: %s / %s\n", pad->GetName(), pad->GetTitle());
492 TIter next(gROOT->GetListOfCanvases());
494 while ((tc = (TCanvas*)next()))
505 if (fSelected && fSelected->InheritsFrom(
"TH2") && !fSelected->InheritsFrom(
"TH3")) {
511 Bool_t changed = kFALSE;
512 Int_t newXfirst =
Xfirst0 - dXbins;
519 newXlast =
Xlast0 - dXbins;
525 if (newXfirst !=
Xf1) {
531 Int_t newYfirst =
Yfirst0 - dYbins;
538 newYlast =
Ylast0 - dYbins;
544 if (newYfirst !=
Yf1) {
561 if (sendOrder) fSelected->ExecuteEvent(event, px, py);
573 pad = Pick(px, py, prevSelObj);
576 if (!fDoubleBuffer) FeedbackMode(kFALSE);
578 if (fContextMenu && !fSelected->TestBit(kNoContextMenu) && !pad->TestBit(kNoContextMenu) && !TestBit(kNoContextMenu)) {
579 if (sendOrder) fContextMenu->Popup(px, py, fSelected,
this, pad);
580 else fSelected->ExecuteEvent(event, px, py);
589 if (!fDoubleBuffer) FeedbackMode(kTRUE);
604 if (!fSelectedPad || !fSelected)
return;
607 fSelected->ExecuteEvent(event, px, py);
615 pad = Pick(px, py, prevSelObj);
619 EnterLeave(prevSelPad, prevSelObj);
624 fSelected->ExecuteEvent(event, px, py);
630 pad = Pick(px, py, prevSelObj);
633 sign = (
event == kWheelUp ? 1 : -1);
636 if (fSelected->InheritsFrom(
"TAxis")) fSelected->ExecuteEvent(event, px, py);
637 else if (fSelected->InheritsFrom(
"TH2"))
DynamicZoom(sign, px, py);
646 if (fPadSave && event != kButton2Down)
649 if (event != kMouseLeave) {
650 ProcessedEvent(event, px, py, fSelected);
661 if (!TheHisto)
return;
662 TAxis* ax = TheHisto->GetXaxis();
664 Double_t ratio1 = (
xmin - GetUxmin()) / (GetUxmax() - GetUxmin());
665 Double_t ratio2 = (
xmax - GetUxmin()) / (GetUxmax() - GetUxmin());
666 if ((ratio2 - ratio1 > 0.05)) ax->SetRangeUser(
xmin,
xmax);
668 ax = TheHisto->GetYaxis();
670 ratio1 = (
ymin - GetUymin()) / (GetUymax() - GetUymin());
671 ratio2 = (
ymax - GetUymin()) / (GetUymax() - GetUymin());
672 if ((ratio2 - ratio1 > 0.05)) ax->SetRangeUser(
ymin,
ymax);
687 if (!fSelected)
return;
690 Double_t percent = 0.15 - Sign * 0.05;
694 px = AbsPixeltoX(px);
696 TAxis* ax = TheHisto->GetXaxis();
697 Int_t NbinsXtmp = ax->GetNbins();
698 Int_t X0tmp = ax->GetFirst();
699 Int_t X1tmp = ax->GetLast();
700 Int_t step = TMath::Min(TMath::Max(1, (Int_t)(percent * (X1tmp - X0tmp))), NbinsXtmp / 2);
702 X0tmp = TMath::Min(TMath::Max(X0tmp + step, 1), X1tmp - step);
703 X1tmp = TMath::Max(TMath::Min(X1tmp - step, NbinsXtmp), X0tmp);
704 if (X0tmp >= X1tmp) X0tmp = X1tmp - 1;
705 if (Sign > 0) dX = (Int_t)(X0tmp + (X1tmp - X0tmp) * 0.5 - ax->FindBin(px));
706 if ((X0tmp - dX) < 0) ax->SetRange(0, X1tmp - X0tmp);
707 else if ((X1tmp - dX) > ax->GetNbins()) ax->SetRange(ax->GetNbins() - (X1tmp - X0tmp), ax->GetNbins());
708 else ax->SetRange(X0tmp - dX, X1tmp - dX);
728 if (!fSelected)
return;
731 Double_t percent = 0.15 - Sign * 0.05;
736 Double_t ppx = AbsPixeltoX(px);
737 Double_t ppy = AbsPixeltoY(py);
739 TAxis* ax = TheHisto->GetXaxis();
740 Int_t NbinsXtmp = ax->GetNbins();
741 Int_t X0tmp = ax->GetFirst();
742 Int_t X1tmp = ax->GetLast();
743 Int_t step = TMath::Min(TMath::Max(1, (Int_t)(percent * (X1tmp - X0tmp))), NbinsXtmp / 2);
745 X0tmp = TMath::Min(TMath::Max(X0tmp + step, 1), X1tmp - step);
746 X1tmp = TMath::Max(TMath::Min(X1tmp - step, NbinsXtmp), X0tmp);
747 if (X0tmp >= X1tmp) X0tmp = X1tmp - 1;
748 if (Sign > 0) dX = (Int_t)(X0tmp + (X1tmp - X0tmp) * 0.5 - ax->FindBin(ppx));
749 if ((X0tmp - dX) < 0) ax->SetRange(0, X1tmp - X0tmp);
750 else if ((X1tmp - dX) > ax->GetNbins()) ax->SetRange(ax->GetNbins() - (X1tmp - X0tmp), ax->GetNbins());
751 else ax->SetRange(X0tmp - dX, X1tmp - dX);
753 ax = TheHisto->GetYaxis();
754 Int_t NbinsYtmp = ax->GetNbins();
755 Int_t Y0tmp = ax->GetFirst();
756 Int_t Y1tmp = ax->GetLast();
757 step = TMath::Min(TMath::Max(1, (Int_t)(percent * (Y1tmp - Y0tmp))), NbinsYtmp / 2);
759 Y0tmp = TMath::Min(TMath::Max(Y0tmp + step, 1), Y1tmp - step);
760 Y1tmp = TMath::Max(TMath::Min(Y1tmp - step, NbinsYtmp), Y0tmp);
761 if (Y0tmp >= Y1tmp) Y0tmp = Y1tmp - 1;
762 if (Sign > 0) dY = (Int_t)(Y0tmp + (Y1tmp - Y0tmp) * 0.5 - ax->FindBin(ppy));
763 if ((Y0tmp - dY) < 0) ax->SetRange(0, Y1tmp - Y0tmp);
764 else if ((Y1tmp - dY) > ax->GetNbins()) ax->SetRange(ax->GetNbins() - (Y1tmp - Y0tmp), ax->GetNbins());
765 else ax->SetRange(Y0tmp - dY, Y1tmp - dY);
785 TIter next(GetListOfPrimitives());
789 if (fSelected->InheritsFrom(
"TFrame")) fSelected =
FindHisto();
790 if (fSelected->InheritsFrom(
"TH2")) fSelected =
FindHisto();
791 if (!fSelected)
return kTRUE;
793 switch ((EKeySym)py) {
837 if (fSelected->InheritsFrom(
"TH1")) {
839 ((TH1*)fSelected)->GetXaxis()->UnZoom();
840 ((TH1*)fSelected)->GetYaxis()->UnZoom();
844 else if (fSelected->InheritsFrom(
"TAxis")) {
845 ((TAxis*)fSelected)->UnZoom();
859 gCopyObject->Delete();
862 if (fSelected) gCopyObject = fSelected->Clone();
867 if (fSelected->InheritsFrom(
"TF1")) {
869 if ((hh =
FindHisto())) hh->GetListOfFunctions()->Remove(fSelected);
871 else GetListOfPrimitives()->Remove(fSelected);
877 GetCanvasImp()->ShowEditor(!GetCanvasImp()->HasEditor());
881 if (fSelected->InheritsFrom(
"TH1"))((TH1*)fSelected)->FitPanel();
885 if (GetGridx() && GetGridy()) {
887 while ((obj = next())) {
888 if (obj->InheritsFrom(TPad::Class())) {
889 ((TPad*)obj)->SetGrid(0, 0);
895 while ((obj = next())) {
896 if (obj->InheritsFrom(TPad::Class())) {
897 ((TPad*)obj)->SetGrid();
915 if (fSelected->InheritsFrom(
"TH2")) SetLogz(!fLogz);
916 else if (fSelected->InheritsFrom(
"TH1")) SetLogy(!fLogy);
922 if (fSelected->InheritsFrom(
"TH1")) {
923 ((TH1*)fSelected)->Sumw2();
924 ((TH1*)fSelected)->Scale(1. / ((TH1*)fSelected)->Integral());
966 gCopyObject->Draw(option.Data());
981 gCopyObject = fSelected;
982 GetListOfPrimitives()->Remove(gCopyObject);
994 if (fSelected->InheritsFrom(
"TAxis"))
MoveAxis((TAxis*)fSelected, -1);
999 if (fSelected->InheritsFrom(
"TAxis"))
MoveAxis((TAxis*)fSelected, -1);
1004 if (fSelected->InheritsFrom(
"TAxis"))
MoveAxis((TAxis*)fSelected, 1);
1009 if (fSelected->InheritsFrom(
"TAxis"))
MoveAxis((TAxis*)fSelected, 1);
1014 if (fSelected->InheritsFrom(
"TH2")) {
1015 ((TH2*)fSelected)->SetMinimum(((TH1*)fSelected)->GetMinimum() + 1);
1019 else if (fSelected->InheritsFrom(
"TF1")) {
1020 ((TF1*)fSelected)->SetNpx(((TF1*)fSelected)->GetNpx() + 50);
1024 else if (fSelected->InheritsFrom(
"TH1")) {
1026 TIter it(((TH1*)fSelected)->GetListOfFunctions());
1027 while ((obj = it())) {
1028 ((TF1*)obj)->SetNpx(((TF1*)obj)->GetNpx() + 50);
1036 if (fSelected->InheritsFrom(
"TH2")) {
1037 if (((TH1*)fSelected)->GetMinimum() > 0)((TH2*)fSelected)->SetMinimum(((TH1*)fSelected)->GetMinimum() - 1);
1041 else if (fSelected->InheritsFrom(
"TF1")) {
1042 ((TF1*)fSelected)->SetNpx(((TF1*)fSelected)->GetNpx() - 50);
1046 else if (fSelected->InheritsFrom(
"TH1")) {
1048 TIter it(((TH1*)fSelected)->GetListOfFunctions());
1049 while ((obj = it()))((TF1*)obj)->SetNpx(((TF1*)obj)->GetNpx() - 50);
1074 Int_t nBins = ax->GetNbins();
1075 Int_t first = ax->GetFirst();
1076 Int_t last = ax->GetLast();
1078 Int_t dX = (last - first) / 10;
1081 if ((last + 2 < nBins) && (sign > 0)) {
1082 ax->SetRange(first + dX, last + dX);
1086 if ((first - 2 >= 0) && (sign < 0)) {
1087 ax->SetRange(first - dX, last - dX);
1108 std::cout << std::endl << std::endl;
1111 while ((info = (TNamed*)it())) {
1112 std::cout << Form(
"%20s", info->GetName()) <<
" " << info->GetTitle() << std::endl;
1114 std::cout << std::endl;
1136 AddShortcutsInfo(
"<ctrl> d",
"undraw the object under cursor (object not deleted)");
1143 AddShortcutsInfo(
"<ctrl> l",
"set/unset log scale on Y axis (TH1) or Z axis (TH2)");
1168 if ((pfx = FindObject(Form(
"%s_pfx", hh->GetName())))) pfx->Delete();
1169 hh->ProfileX(
"_pfx", 1, -1,
"i,d,same");
1170 if ((pfx = FindObject(Form(
"%s_pfx", hh->GetName()))))((TProfile*)pfx)->SetLineColor(kBlack);
1182 if ((obj = gROOT->FindObject(Form(
"%s_pfy", hh->GetName())))) obj->Delete();
1183 TProfile* pfy = hh->ProfileY(
"_pfy", 1, -1,
"i");
1188 gg->SetName(Form(
"%s_pfy", hh->GetName()));
1189 gg->SetLineColor(kBlack);
1205 file.ReplaceAll(gSystem->DirName(
fSavedAs.Data()),
"");
1206 file.ReplaceAll(
"/",
"");
1207 new TGMsgBox(gClient->GetDefaultRoot(), gClient->GetDefaultRoot(),
"File name exist",
1208 Form(
"File name '%s' already exists, OK to owerwrite it?", file.Data()),
1209 kMBIconExclamation, kMBOk | kMBCancel, &ret_val);
1211 if (ret_val & kMBOk) {
1217 const char* SaveAsTypes[] = {
1219 "PostScript",
"*.ps",
1220 "Encapsulated PostScript",
"*.eps",
1224 "ROOT macros",
"*.C",
1225 "ROOT files",
"*.root",
1236 TString workdir = gSystem->WorkingDirectory();
1237 static TString dir(
".");
1238 static Int_t typeidx = 0;
1239 static Bool_t overwr = kFALSE;
1241 fi.fFileTypes = SaveAsTypes;
1242 fi.fIniDir = StrDup(dir);
1243 fi.fFileTypeIdx = typeidx;
1244 fi.fOverwrite = overwr;
1245 new TGFileDialog(gClient->GetDefaultRoot(), gClient->GetDefaultRoot(), kFDSave, &fi);
1246 gSystem->ChangeDirectory(workdir.Data());
1247 if (!fi.fFilename)
return;
1248 TString fn = fi.fFilename;
1250 typeidx = fi.fFileTypeIdx;
1251 overwr = fi.fOverwrite;
1252 fCanvas->SaveAs(fn);
1263 TIter it(GetListOfPrimitives());
1264 while ((hh = (TObject*)it())) {
1265 if (hh->InheritsFrom(
"TH1"))
return (TH1*) hh;
1280 TIter it(hh->GetListOfFunctions());
1281 while ((obj = it())) {
1282 ((TF1*)obj)->SetRange(hh->GetXaxis()->GetXmin(), hh->GetXaxis()->GetXmax());
1303 TString pname = Form(
"%s_px", hh->GetName());
1305 while (gROOT->FindObject(pname.Data())) {
1306 pname = Form(
"%s_px%d", hh->GetName(), ip);
1310 TH1* px = hh->ProjectionX(pname.Data());
1312 Double_t minY = (hh->GetYaxis()->GetXmin());
1313 Double_t maxY = (hh->GetYaxis()->GetXmax());
1314 Double_t dY = (maxY - minY) * 0.8;
1316 Double_t maxH = px->GetBinContent(px->GetMaximumBin());
1319 if ((gg = (TGraph*)gROOT->FindObject(Form(
"%s_gjx", hh->GetName())))) gg->Delete();
1322 for (
int i = 0; i < px->GetNbinsX(); i++) {
1323 gg->SetPoint(i, px->GetBinCenter(i), minY + px->GetBinContent(i)*dY / maxH);
1326 gg->SetName(Form(
"%s_gjx", hh->GetName()));
1327 gg->SetTitle(Form(
"%s_gjx", hh->GetName()));
1328 gg->SetLineColor(kBlack);
1329 gg->SetMarkerColor(kBlack);
1330 gg->SetMarkerStyle(8);
1343 TString pname = Form(
"%s_py", hh->GetName());
1346 while (gROOT->FindObject(pname.Data())) {
1347 pname = Form(
"%s_py%d", hh->GetName(), ip);
1351 TH1* py = hh->ProjectionY(pname.Data());
1353 Double_t minY = (hh->GetXaxis()->GetXmin());
1354 Double_t maxY = (hh->GetXaxis()->GetXmax());
1355 Double_t dY = (maxY - minY) * 0.8;
1357 Double_t maxH = py->GetBinContent(py->GetMaximumBin());
1360 if ((gg = (TGraph*)gROOT->FindObject(Form(
"%s_gjy", hh->GetName())))) gg->Delete();
1363 for (
int i = 0; i < py->GetNbinsX(); i++) {
1364 gg->SetPoint(i, minY + py->GetBinContent(i)*dY / maxH, py->GetBinCenter(i));
1367 gg->SetName(Form(
"%s_gjy", hh->GetName()));
1368 gg->SetTitle(Form(
"%s_gjy", hh->GetName()));
1369 gg->SetLineColor(kBlack);
1370 gg->SetMarkerColor(kBlack);
1371 gg->SetMarkerStyle(8);
TCanvas with mouse-controlled dynamic zoom and pan & scan.
void ShowShortcutsInfos()
void ProjectionY(TH2 *hh)
Int_t Xlast0
number of bins on x-axis, initial displayed bins
void AddShortcutsInfo(const char *cut, const char *desc)
void MoveAxis(TAxis *ax, Int_t sign)
friend class KVKeyHandler
Bool_t ExpandFunctionRange()
void DisableObject(TObject *obj)
void HandleInput(EEventType event, Int_t px, Int_t py)
Int_t X0
! variables for pan & scan
void DisableClass(const char *className)
Int_t NdisYbins
number of displayed bins on X & Y
Bool_t fHasDisabledClasses
void ResetDisabledObject()
TAxis * theYaxis
the axes of the histogram
void RunAutoExec()
Execute the list of TExecs in the current pad.
Int_t Ylast0
number of bins on y-axis, initial displayed bins
void SetEnabledShortcuts(Int_t value=1)
void DrawEventStatus(Int_t event, Int_t px, Int_t py, TObject *selected)
TGFrame * fKeyHandler
handler for keys
void ZoomSelected(TH2 *TheHisto)
void DynamicZoomTH1(Int_t Sign, Int_t px, Int_t py)
Zoom in or out of histogram with mouse wheel.
void SetVenerMode(Int_t value=1)
Double_t YbinPixel
size of bins in pixels
virtual Bool_t HandleKey(Int_t, Int_t py)
virtual Bool_t HandleKey(Event_t* /*event*/){return kTRUE;}
void DynamicZoom(Int_t Sign, Int_t px, Int_t py)
void ResetDisabledClass()
Int_t Y0
coordinates of initial click in pad pixels
Int_t Yl1
last modification to axis limits
Bool_t fHasDisabledObject
void ProjectionX(TH2 *hh)
TH2 * PermuteAxis(TH2 *hh)
TGraphErrors * MakeGraphFrom(TProfile *pf, Bool_t Error=kTRUE)