Logo ROOT  
Reference Guide
 
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
Loading...
Searching...
No Matches
TPad.cxx
Go to the documentation of this file.
1// @(#)root/gpad:$Id$
2// Author: Rene Brun 12/12/94
3
4/*************************************************************************
5 * Copyright (C) 1995-2000, Rene Brun and Fons Rademakers. *
6 * All rights reserved. *
7 * *
8 * For the licensing terms see $ROOTSYS/LICENSE. *
9 * For the list of contributors see $ROOTSYS/README/CREDITS. *
10 *************************************************************************/
11
12#include <cstring>
13#include <cstdlib>
14#include <iostream>
15#include <locale>
16#include <memory>
17
18#include "TROOT.h"
19#include "TBuffer.h"
20#include "TError.h"
21#include "TMath.h"
22#include "TSystem.h"
23#include "TStyle.h"
24#include "TFile.h"
25#include "TH1.h"
26#include "TH2.h"
27#include "TH3.h"
28#include "TClass.h"
29#include "TBaseClass.h"
30#include "TClassTable.h"
31#include "TVirtualPS.h"
32#include "TVirtualX.h"
33#include "TVirtualViewer3D.h"
34#include "TView.h"
35#include "TPoint.h"
36#include "TGraph.h"
37#include "TMultiGraph.h"
38#include "THStack.h"
39#include "TPaveText.h"
40#include "TPaveStats.h"
41#include "TGroupButton.h"
42#include "TBrowser.h"
43#include "TVirtualGL.h"
44#include "TString.h"
45#include "TDataMember.h"
46#include "TMethod.h"
47#include "TDataType.h"
48#include "TFrame.h"
49#include "TExec.h"
50#include "TDatime.h"
51#include "TColor.h"
52#include "TCanvas.h"
53#include "TPluginManager.h"
54#include "TEnv.h"
55#include "TImage.h"
56#include "TViewer3DPad.h"
57#include "TCreatePrimitives.h"
58#include "TLegend.h"
59#include "TAtt3D.h"
60#include "TVirtualPadPainter.h"
61#include "strlcpy.h"
62#include "snprintf.h"
63
64#include "TVirtualMutex.h"
65
66static Int_t gReadLevel = 0;
67
69
71
72/** \class TPad
73\ingroup gpad
74
75The most important graphics class in the ROOT system.
76
77A Pad is contained in a Canvas.
78
79A Pad may contain other pads (unlimited pad hierarchy).
80
81A pad is a linked list of primitives of any type (graphics objects,
82histograms, detectors, tracks, etc.).
83
84Adding a new element into a pad is in general performed by the Draw
85member function of the object classes.
86
87It is important to realize that the pad is a linked list of references
88to the original object.
89For example, in case of a histogram, the histogram.Draw() operation
90only stores a reference to the histogram object and not a graphical
91representation of this histogram.
92When the mouse is used to change (say the bin content), the bin content
93of the original histogram is changed.
94
95The convention used in ROOT is that a Draw operation only adds
96a reference to the object. The effective drawing is performed
97when the canvas receives a signal to be painted.
98
99\image html gpad_pad1.png
100
101This signal is generally sent when typing carriage return in the
102command input or when a graphical operation has been performed on one
103of the pads of this canvas.
104When a Canvas/Pad is repainted, the member function Paint for all
105objects in the Pad linked list is invoked.
106
107\image html gpad_pad2.png
108
109When the mouse is moved on the Pad, The member function DistancetoPrimitive
110is called for all the elements in the pad. DistancetoPrimitive returns
111the distance in pixels to this object.
112
113When the object is within the distance window, the member function
114ExecuteEvent is called for this object.
115
116In ExecuteEvent, move, changes can be performed on the object.
117
118For examples of DistancetoPrimitive and ExecuteEvent functions,
119see classes
120~~~ {.cpp}
121 TLine::DistancetoPrimitive, TLine::ExecuteEvent
122 TBox::DistancetoPrimitive, TBox::ExecuteEvent
123 TH1::DistancetoPrimitive, TH1::ExecuteEvent
124~~~
125A Pad supports linear and log scales coordinate systems.
126The transformation coefficients are explained in TPad::ResizePad.
127*/
128
129////////////////////////////////////////////////////////////////////////////////
130/// Pad default constructor.
131
133{
134 fModified = kTRUE;
135 fTip = nullptr;
136 fPadPointer = nullptr;
137 fPrimitives = nullptr;
138 fExecs = nullptr;
139 fCanvas = nullptr;
140 fPadPaint = 0;
141 fPixmapID = -1;
142 fGLDevice = -1;
143 fCopyGLDevice = kFALSE;
144 fEmbeddedGL = kFALSE;
145 fTheta = 30;
146 fPhi = 30;
147 fNumber = 0;
148 fAbsCoord = kFALSE;
149 fEditable = kTRUE;
150 fCrosshair = 0;
151 fCrosshairPos = 0;
152 fPadView3D = nullptr;
153 fMother = (TPad*)gPad;
154
155 fAbsHNDC = 0.;
156 fAbsPixeltoXk = 0.;
157 fAbsPixeltoYk = 0.;
158 fAbsWNDC = 0.;
159 fAbsXlowNDC = 0.;
160 fAbsYlowNDC = 0.;
161 fBorderMode = 0;
162 fBorderSize = 0;
163 fPixeltoX = 0;
164 fPixeltoXk = 0.;
165 fPixeltoY = 0.;
166 fPixeltoYk = 0.;
167 fUtoAbsPixelk = 0.;
168 fUtoPixel = 0.;
169 fUtoPixelk = 0.;
170 fVtoAbsPixelk = 0.;
171 fVtoPixel = 0.;
172 fVtoPixelk = 0.;
173 fXtoAbsPixelk = 0.;
174 fXtoPixel = 0.;
175 fXtoPixelk = 0.;
176 fYtoAbsPixelk = 0.;
177 fYtoPixel = 0.;
178 fYtoPixelk = 0.;
179 fXUpNDC = 0.;
180 fYUpNDC = 0.;
181
182 fFixedAspectRatio = kFALSE;
183 fAspectRatio = 0.;
184
185 fNumPaletteColor = 0;
186 fNextPaletteColor = 0;
187 fCGnx = 0;
188 fCGny = 0;
189
190 fLogx = 0;
191 fLogy = 0;
192 fLogz = 0;
193 fGridx = false;
194 fGridy = false;
195 fTickx = 0;
196 fTicky = 0;
197 fFrame = nullptr;
198 fView = nullptr;
199
200 fUxmin = fUymin = fUxmax = fUymax = 0;
201
202 // Set default world coordinates to NDC [0,1]
203 fX1 = 0;
204 fX2 = 1;
205 fY1 = 0;
206 fY2 = 1;
207
208 // Set default pad range
209 fXlowNDC = 0;
210 fYlowNDC = 0;
211 fWNDC = 1;
212 fHNDC = 1;
213
214 fViewer3D = nullptr;
215 SetBit(kMustCleanup);
216
217 // the following line is temporarily disabled. It has side effects
218 // when the pad is a TDrawPanelHist or a TFitPanel.
219 // the line was supposed to fix a problem with DrawClonePad
220 // gROOT->SetSelectedPad(this);
221}
222
223////////////////////////////////////////////////////////////////////////////////
224/// Pad constructor.
225///
226/// A pad is a linked list of primitives.
227/// A pad is contained in a canvas. It may contain other pads.
228/// A pad has attributes. When a pad is created, the attributes
229/// defined in the current style are copied to the pad attributes.
230///
231/// \param[in] name pad name
232/// \param[in] title pad title
233/// \param[in] xlow [0,1] is the position of the bottom left point of the pad
234/// expressed in the mother pad reference system
235/// \param[in] ylow [0,1] is the Y position of this point.
236/// \param[in] xup [0,1] is the x position of the top right point of the pad
237/// expressed in the mother pad reference system
238/// \param[in] yup [0,1] is the Y position of this point.
239/// \param[in] color pad color
240/// \param[in] bordersize border size in pixels
241/// \param[in] bordermode border mode
242/// - bordermode = -1 box looks as it is behind the screen
243/// - bordermode = 0 no special effects
244/// - bordermode = 1 box looks as it is in front of the screen
245
246TPad::TPad(const char *name, const char *title, Double_t xlow,
247 Double_t ylow, Double_t xup, Double_t yup,
249 : TVirtualPad(name,title,xlow,ylow,xup,yup,color,bordersize,bordermode)
250{
252 fTip = nullptr;
255 if (gPad) fCanvas = gPad->GetCanvas();
256 else fCanvas = (TCanvas*)this;
257 fMother = (TPad*)gPad;
258 fPrimitives = new TList;
259 fExecs = new TList;
260 fPadPointer = nullptr;
261 fTheta = 30;
262 fPhi = 30;
267 fFrame = nullptr;
268 fView = nullptr;
269 fPadPaint = 0;
270 fPadView3D = nullptr;
271 fPixmapID = -1; // -1 means pixmap will be created by ResizePad()
274 fNumber = 0;
277 fCrosshair = 0;
278 fCrosshairPos = 0;
279
280 fVtoAbsPixelk = 0.;
281 fVtoPixelk = 0.;
282 fVtoPixel = 0.;
283 fAbsPixeltoXk = 0.;
284 fPixeltoXk = 0.;
285 fPixeltoX = 0;
286 fAbsPixeltoYk = 0.;
287 fPixeltoYk = 0.;
288 fPixeltoY = 0.;
289 fXlowNDC = 0;
290 fYlowNDC = 0;
291 fWNDC = 1;
292 fHNDC = 1;
293 fXUpNDC = 0.;
294 fYUpNDC = 0.;
295 fAbsXlowNDC = 0.;
296 fAbsYlowNDC = 0.;
297 fAbsWNDC = 0.;
298 fAbsHNDC = 0.;
299 fXtoAbsPixelk = 0.;
300 fXtoPixelk = 0.;
301 fXtoPixel = 0.;
302 fYtoAbsPixelk = 0.;
303 fYtoPixelk = 0.;
304 fYtoPixel = 0.;
305 fUtoAbsPixelk = 0.;
306 fUtoPixelk = 0.;
307 fUtoPixel = 0.;
308
309 fUxmin = fUymin = fUxmax = fUymax = 0;
313
315 fAspectRatio = 0.;
316
319 fCGnx = 0;
320 fCGny = 0;
321
322 fViewer3D = nullptr;
323
325 // Set default world coordinates to NDC [0,1]
326 fX1 = 0;
327 fX2 = 1;
328 fY1 = 0;
329 fY2 = 1;
330
331 if (!gPad) {
332 Error("TPad", "You must create a TCanvas before creating a TPad");
333 MakeZombie();
334 return;
335 }
336
338
340
341 if ((xlow < 0) || (xlow > 1) || (ylow < 0) || (ylow > 1)) {
342 Error("TPad", "illegal bottom left position: x=%f, y=%f", xlow, ylow);
343 zombie = kTRUE;
344 } else if ((xup < 0) || (xup > 1) || (yup < 0) || (yup > 1)) {
345 Error("TPad", "illegal top right position: x=%f, y=%f", xup, yup);
346 zombie = kTRUE;
347 } else if (xup-xlow <= 0) {
348 Error("TPad", "illegal width: %f", xup-xlow);
349 zombie = kTRUE;
350 } else if (yup-ylow <= 0) {
351 Error("TPad", "illegal height: %f", yup-ylow);
352 zombie = kTRUE;
353 }
354
355 if (zombie) {
356 // error in creating pad occurred, make this pad a zombie
357 MakeZombie();
358 return;
359 }
360
361
365
366 fUxmin = fUymin = fUxmax = fUymax = 0;
367
368 // Set pad parameters and Compute conversion coefficients
369 SetPad(name, title, xlow, ylow, xup, yup, color, bordersize, bordermode);
370 Range(0, 0, 1, 1);
373}
374
375
376////////////////////////////////////////////////////////////////////////////////
377/// Pad destructor.
378
380{
381 if (ROOT::Detail::HasBeenDeleted(this)) return;
382 Close();
385 auto primitives = fPrimitives;
386 // In some cases, fPrimitives has the kMustCleanup bit set which will lead
387 // its destructor to call RecursiveRemove and since this pad is still
388 // likely to be (indirectly) in the list of cleanups, we must set
389 // fPrimitives to nullptr to avoid TPad::RecursiveRemove from calling
390 // a member function of a partially destructed object.
391 fPrimitives = nullptr;
392 delete primitives;
394 delete fViewer3D;
395
396 // Required since we overload TObject::Hash.
398 if (this == gPad)
399 gPad = nullptr;
400}
401
402////////////////////////////////////////////////////////////////////////////////
403/// Add an object to list of primitives with speicified draw option
404/// When \par modified set to kTRUE (default) pad will be marked as modified
405/// Let avoid usage of gPad when drawing object(s) in canvas or in subpads.
406///
407/// ~~~{.cpp}
408/// auto c1 = new TCanvas("c1","Canvas with subpoads", 600, 600);
409/// c1->Divide(2,2);
410///
411/// for (Int_t n = 1; n <= 4; ++n) {
412/// auto h1 = new TH1I(TString::Format("hist_%d",n), "Random hist", 100, -5, 5);
413/// h1->FillRandom("gaus", 2000 + n*1000);
414/// c1->GetPad(n)->Add(h1);
415/// }
416/// ~~~
417
419{
420 if (!obj)
421 return;
422
423 if (!fPrimitives)
424 fPrimitives = new TList;
425
426 obj->SetBit(kMustCleanup);
427
428 fPrimitives->Add(obj, opt);
429
430 if (modified)
431 Modified();
432}
433
434////////////////////////////////////////////////////////////////////////////////
435/// Add an object as first in list of primitives with speicified draw option
436/// When \par modified set to kTRUE (default) pad will be marked as modified
437/// Let avoid usage of gPad when drawing object(s) in canvas or in subpads.
438
440{
441 if (!obj)
442 return;
443
444 if (!fPrimitives)
445 fPrimitives = new TList;
446
447 obj->SetBit(kMustCleanup);
448
449 fPrimitives->AddFirst(obj, opt);
450
451 if (modified)
452 Modified();
453}
454
455////////////////////////////////////////////////////////////////////////////////
456/// Add a new TExec object to the list of Execs.
457///
458/// When an event occurs in the pad (mouse click, etc) the list of C++ commands
459/// in the list of Execs are executed via TPad::AutoExec.
460///
461/// When a pad event occurs (mouse move, click, etc) all the commands
462/// contained in the fExecs list are executed in the order found in the list.
463///
464/// This facility is activated by default. It can be deactivated by using
465/// the canvas "Option" menu.
466///
467/// The following examples of TExec commands are provided in the tutorials:
468/// macros exec1.C and exec2.C.
469///
470/// ### Example1 of use of exec1.C
471///
472/// ~~~ {.cpp}
473/// Root > TFile f("hsimple.root")
474/// Root > hpx.Draw()
475/// Root > c1.AddExec("ex1",".x exec1.C")
476/// ~~~
477///
478/// At this point you can use the mouse to click on the contour of
479/// the histogram hpx. When the mouse is clicked, the bin number and its
480/// contents are printed.
481///
482/// ### Example2 of use of exec1.C
483///
484/// ~~~ {.cpp}
485/// Root > TFile f("hsimple.root")
486/// Root > hpxpy.Draw()
487/// Root > c1.AddExec("ex2",".x exec2.C")
488/// ~~~
489///
490/// When moving the mouse in the canvas, a second canvas shows the
491/// projection along X of the bin corresponding to the Y position
492/// of the mouse. The resulting histogram is fitted with a gaussian.
493/// A "dynamic" line shows the current bin position in Y.
494/// This more elaborated example can be used as a starting point
495/// to develop more powerful interactive applications exploiting the C++
496/// interpreter as a development engine.
497
498void TPad::AddExec(const char *name, const char *command)
499{
500 if (!fExecs) fExecs = new TList;
501 TExec *ex = new TExec(name,command);
502 fExecs->Add(ex);
503}
504
505////////////////////////////////////////////////////////////////////////////////
506/// Execute the list of Execs when a pad event occurs.
507
509{
510 if (GetCrosshair())
512
513 if (!fExecs)
514 return;
515 TIter next(fExecs);
516 while (auto exec = (TExec*)next())
517 exec->Exec();
518}
519
520////////////////////////////////////////////////////////////////////////////////
521/// Browse pad.
522
524{
525 cd();
527}
528
529////////////////////////////////////////////////////////////////////////////////
530/// Build a legend from the graphical objects in the pad.
531///
532/// A simple method to build automatically a TLegend from the primitives in a TPad.
533///
534/// Only those deriving from TAttLine, TAttMarker and TAttFill are added, excluding
535/// TPave and TFrame derived classes.
536///
537/// \return The built TLegend
538///
539/// \param[in] x1, y1, x2, y2 The TLegend coordinates
540/// \param[in] title The legend title. By default it is " "
541/// \param[in] option The TLegend option
542///
543/// The caller program owns the returned TLegend.
544///
545/// If the pad contains some TMultiGraph or THStack the individual
546/// graphs or histograms in them are added to the TLegend.
547///
548/// ### Automatic placement of the legend
549/// If `x1` is equal to `x2` and `y1` is equal to `y2` the legend will be automatically
550/// placed to avoid overlapping with the existing primitives already displayed.
551/// `x1` is considered as the width of the legend and `y1` the height. By default
552/// the legend is automatically placed with width = `x1`= `x2` = 0.3 and
553/// height = `y1`= `y2` = 0.21.
554
556 const char* title, Option_t *option)
557{
559 if (!lop) return nullptr;
560 TList *lof = nullptr;
561 TLegend *leg = nullptr;
562 TObject *obj = nullptr;
563 TIter next(lop);
564 TString mes;
565 TString opt;
566
567 auto AddEntryFromListOfFunctions = [&]() {
569 while ((obj = nextobj())) {
570 if (obj->InheritsFrom(TNamed::Class())) {
571 if (strlen(obj->GetTitle()))
572 mes = obj->GetTitle();
573 else
574 mes = obj->GetName();
575 } else {
576 mes = obj->ClassName();
577 }
578 leg->AddEntry(obj, mes.Data(), "lpf");
579 }
580 };
581
582 while(auto o = next()) {
583 if ((o->InheritsFrom(TAttLine::Class()) || o->InheritsFrom(TAttMarker::Class()) ||
584 o->InheritsFrom(TAttFill::Class())) &&
585 ( !(o->InheritsFrom(TFrame::Class())) && !(o->InheritsFrom(TPave::Class())) )) {
586 if (!leg)
587 leg = new TLegend(x1, y1, x2, y2, title);
588 if (o->InheritsFrom(TNamed::Class()) && strlen(o->GetTitle()))
589 mes = o->GetTitle();
590 else if (strlen(o->GetName()))
591 mes = o->GetName();
592 else
593 mes = o->ClassName();
594 if (option && strlen(option)) {
595 opt = option;
596 } else {
597 if (o->InheritsFrom(TAttLine::Class()))
598 opt += "l";
599 if (o->InheritsFrom(TAttMarker::Class()))
600 opt += "p";
601 if (o->InheritsFrom(TAttFill::Class()))
602 opt += "f";
603 }
604 leg->AddEntry(o,mes.Data(), opt.Data());
605 if (o->InheritsFrom(TH1::Class())) {
606 lof = ((TH1 *)o)->GetListOfFunctions();
608 }
609 if (o->InheritsFrom(TGraph::Class())) {
610 lof = ((TGraph *)o)->GetListOfFunctions();
612 }
613 } else if (o->InheritsFrom(TMultiGraph::Class())) {
614 if (!leg)
615 leg = new TLegend(x1, y1, x2, y2, title);
616 TList * grlist = ((TMultiGraph *)o)->GetListOfGraphs();
618 TGraph *gr = nullptr;
619 while ((obj = nextgraph())) {
620 gr = (TGraph*) obj;
621 if (strlen(gr->GetTitle()))
622 mes = gr->GetTitle();
623 else if (strlen(gr->GetName()))
624 mes = gr->GetName();
625 else
626 mes = gr->ClassName();
627 if (option && strlen(option))
628 opt = option;
629 else
630 opt = "lpf";
631 leg->AddEntry(obj, mes.Data(), opt);
632 }
633 lof = ((TMultiGraph *)o)->GetListOfFunctions();
635 } else if (o->InheritsFrom(THStack::Class())) {
636 if (!leg)
637 leg = new TLegend(x1, y1, x2, y2, title);
638 TList * hlist = ((THStack *)o)->GetHists();
640 while ((obj = nexthist())) {
641 TH1 *hist = (TH1*) obj;
642 if (strlen(hist->GetTitle()))
643 mes = hist->GetTitle();
644 else if (strlen(hist->GetName()))
645 mes = hist->GetName();
646 else
647 mes = hist->ClassName();
648 if (option && strlen(option))
649 opt = option;
650 else
651 opt = "lpf";
652 leg->AddEntry( obj, mes.Data(), opt );
653 }
654 }
655 opt = "";
656 }
657 if (leg) {
658 TContext ctxt(this, kTRUE);
659 leg->Draw();
660 } else {
661 Info("BuildLegend", "No object(s) to build a TLegend.");
662 }
663 return leg;
664}
665
666////////////////////////////////////////////////////////////////////////////////
667/// Set Current pad.
668///
669/// When a canvas/pad is divided via TPad::Divide, one can directly
670/// set the current path to one of the subdivisions.
671/// See TPad::Divide for the convention to number sub-pads.
672///
673/// Returns the new current pad, or 0 in case of failure.
674///
675/// For example:
676/// ~~~ {.cpp}
677/// c1.Divide(2,3); // create 6 pads (2 divisions along x, 3 along y).
678/// ~~~
679/// To set the current pad to the bottom right pad, do
680/// ~~~ {.cpp}
681/// c1.cd(6);
682/// ~~~
683/// Note1: c1.cd() is equivalent to c1.cd(0) and sets the current pad
684/// to c1 itself.
685///
686/// Note2: after a statement like c1.cd(6), the global variable gPad
687/// points to the current pad. One can use gPad to set attributes
688/// of the current pad.
689///
690/// Note3: One can get a pointer to one of the sub-pads of pad with:
691/// TPad *subpad = (TPad*)pad->GetPad(subpadnumber);
692
694{
695 if (!subpadnumber) {
696 gPad = this;
697 if (!gPad->IsBatch() && GetPainter()) GetPainter()->SelectDrawable(fPixmapID);
698 if (!fPrimitives) fPrimitives = new TList;
699 return gPad;
700 }
701
702 if (!fPrimitives) fPrimitives = new TList;
703 TIter next(fPrimitives);
704 while (auto obj = next()) {
705 if (obj->InheritsFrom(TPad::Class())) {
706 Int_t n = ((TPad*)obj)->GetNumber();
707 if (n == subpadnumber) {
708 return ((TPad*)obj)->cd();
709 }
710 }
711 }
712 return nullptr;
713}
714
715////////////////////////////////////////////////////////////////////////////////
716/// Delete all pad primitives.
717///
718/// If the bit kClearAfterCR has been set for this pad, the Clear function
719/// will execute only after having pressed a CarriageReturn
720/// Set the bit with `mypad->SetBit(TPad::kClearAfterCR)`
721
723{
724 if (!IsEditable()) return;
725
727
728 if (!fPadPaint) {
731 if (fFrame) {
733 fFrame = nullptr;
734 }
735 }
736 if (fCanvas) fCanvas->Cleared(this);
737
738 cd();
739
740 if (TestBit(kClearAfterCR)) {
741 // Intentional do not use the return value of getchar,
742 // we just want to get it and forget it
743 getchar();
744 }
745
746 if (!gPad->IsBatch() && GetPainter()) GetPainter()->ClearDrawable();
747 if (gVirtualPS && gPad == gPad->GetCanvas()) gVirtualPS->NewPage();
748
750 fCrosshairPos = 0;
752 fCollideGrid.clear();
753 fCGnx = 0;
754 fCGny = 0;
756}
757
758////////////////////////////////////////////////////////////////////////////////
759/// Clipping routine: Cohen Sutherland algorithm.
760///
761/// - If Clip ==2 the segment is outside the boundary.
762/// - If Clip ==1 the segment has one point outside the boundary.
763/// - If Clip ==0 the segment is inside the boundary.
764///
765/// \param[inout] x[],y[] Segment coordinates (2 points)
766/// \param[in] xclipl,yclipb,xclipr,yclipt Clipping boundary
767
769{
770 const Float_t kP=10000;
771 Int_t clip = 0;
772
773 for (Int_t i=0;i<2;i++) {
774 if (TMath::Abs(xclipl-x[i]) <= TMath::Abs(xclipr-xclipl)/kP) x[i] = xclipl;
775 if (TMath::Abs(xclipr-x[i]) <= TMath::Abs(xclipr-xclipl)/kP) x[i] = xclipr;
776 if (TMath::Abs(yclipb-y[i]) <= TMath::Abs(yclipt-yclipb)/kP) y[i] = yclipb;
777 if (TMath::Abs(yclipt-y[i]) <= TMath::Abs(yclipt-yclipb)/kP) y[i] = yclipt;
778 }
779
780 // Compute the first endpoint codes.
783
784 Double_t xt=0, yt=0;
785 Int_t clipped = 0; //this variable could be used in a future version
786 while(code1 + code2) {
787 clipped = 1;
788
789 // The line lies entirely outside the clipping boundary
790 if (code1&code2) {
791 clip = 2;
792 return clip;
793 }
794
795 // The line is subdivided into several parts
796 Int_t ic = code1;
797 if (ic == 0) ic = code2;
798 if (ic & 0x1) {
799 yt = y[0] + (y[1]-y[0])*(xclipl-x[0])/(x[1]-x[0]);
800 xt = xclipl;
801 }
802 if (ic & 0x2) {
803 yt = y[0] + (y[1]-y[0])*(xclipr-x[0])/(x[1]-x[0]);
804 xt = xclipr;
805 }
806 if (ic & 0x4) {
807 xt = x[0] + (x[1]-x[0])*(yclipb-y[0])/(y[1]-y[0]);
808 yt = yclipb;
809 }
810 if (ic & 0x8) {
811 xt = x[0] + (x[1]-x[0])*(yclipt-y[0])/(y[1]-y[0]);
812 yt = yclipt;
813 }
814 if (ic == code1) {
815 x[0] = xt;
816 y[0] = yt;
818 } else {
819 x[1] = xt;
820 y[1] = yt;
822 }
823 }
824 clip = clipped;
825 return clip;
826}
827
828/// @copydoc TPad::Clip(Float_t*,Float_t*,Float_t,Float_t,Float_t,Float_t)
829
831{
832 const Double_t kP = 10000;
833 Int_t clip = 0;
834
835 for (Int_t i=0;i<2;i++) {
836 if (TMath::Abs(xclipl-x[i]) <= TMath::Abs(xclipr-xclipl)/kP) x[i] = xclipl;
837 if (TMath::Abs(xclipr-x[i]) <= TMath::Abs(xclipr-xclipl)/kP) x[i] = xclipr;
838 if (TMath::Abs(yclipb-y[i]) <= TMath::Abs(yclipt-yclipb)/kP) y[i] = yclipb;
839 if (TMath::Abs(yclipt-y[i]) <= TMath::Abs(yclipt-yclipb)/kP) y[i] = yclipt;
840 }
841
842 // Compute the first endpoint codes.
843 Int_t code1 = 0;
844 if (x[0] < xclipl) code1 = code1 | 0x1;
845 if (x[0] > xclipr) code1 = code1 | 0x2;
846 if (y[0] < yclipb) code1 = code1 | 0x4;
847 if (y[0] > yclipt) code1 = code1 | 0x8;
848 Int_t code2 = 0;
849 if (x[1] < xclipl) code2 = code2 | 0x1;
850 if (x[1] > xclipr) code2 = code2 | 0x2;
851 if (y[1] < yclipb) code2 = code2 | 0x4;
852 if (y[1] > yclipt) code2 = code2 | 0x8;
853
854 Double_t xt=0, yt=0;
855 Int_t clipped = 0; //this variable could be used in a future version
856 while(code1 + code2) {
857 clipped = 1;
858
859 // The line lies entirely outside the clipping boundary
860 if (code1&code2) {
861 clip = 2;
862 return clip;
863 }
864
865 // The line is subdivided into several parts
866 Int_t ic = code1;
867 if (ic == 0) ic = code2;
868 if (ic & 0x1) {
869 yt = y[0] + (y[1]-y[0])*(xclipl-x[0])/(x[1]-x[0]);
870 xt = xclipl;
871 }
872 if (ic & 0x2) {
873 yt = y[0] + (y[1]-y[0])*(xclipr-x[0])/(x[1]-x[0]);
874 xt = xclipr;
875 }
876 if (ic & 0x4) {
877 xt = x[0] + (x[1]-x[0])*(yclipb-y[0])/(y[1]-y[0]);
878 yt = yclipb;
879 }
880 if (ic & 0x8) {
881 xt = x[0] + (x[1]-x[0])*(yclipt-y[0])/(y[1]-y[0]);
882 yt = yclipt;
883 }
884 if (ic == code1) {
885 x[0] = xt;
886 y[0] = yt;
888 } else {
889 x[1] = xt;
890 y[1] = yt;
892 }
893 }
894 clip = clipped;
895 return clip;
896}
897
898////////////////////////////////////////////////////////////////////////////////
899/// Compute the endpoint codes for TPad::Clip.
900
902{
903 Int_t code = 0;
904 if (x < xcl1) code = code | 0x1;
905 if (x > xcl2) code = code | 0x2;
906 if (y < ycl1) code = code | 0x4;
907 if (y > ycl2) code = code | 0x8;
908 return code;
909}
910
911////////////////////////////////////////////////////////////////////////////////
912/// Clip polygon using the Sutherland-Hodgman algorithm.
913///
914/// \param[in] n Number of points in the polygon to
915/// be clipped
916/// \param[in] x,y Polygon x[n], y[n] do be clipped vertices
917/// \param[in] xclipl,yclipb,xclipr,yclipt Clipping boundary
918/// \param[out] nn Number of points in xc and yc
919/// \param[out] xc,yc Clipped polygon vertices. The Int_t
920/// returned by this function is
921/// the number of points in the clipped
922/// polygon. These vectors must
923/// be allocated by the calling function.
924/// A size of 2*n for each is
925/// enough.
926///
927/// Sutherland and Hodgman's polygon-clipping algorithm uses a divide-and-conquer
928/// strategy: It solves a series of simple and identical problems that, when
929/// combined, solve the overall problem. The simple problem is to clip a polygon
930/// against a single infinite clip edge. Four clip edges, each defining one boundary
931/// of the clip rectangle, successively clip a polygon against a clip rectangle.
932///
933/// Steps of Sutherland-Hodgman's polygon-clipping algorithm:
934///
935/// * Polygons can be clipped against each edge of the window one at a time.
936/// Windows/edge intersections, if any, are easy to find since the X or Y coordinates
937/// are already known.
938/// * Vertices which are kept after clipping against one window edge are saved for
939/// clipping against the remaining edges.
940/// * Note that the number of vertices usually changes and will often increases.
941///
942/// The clip boundary determines a visible and invisible region. The edges from
943/// vertex i to vertex i+1 can be one of four types:
944///
945/// * Case 1 : Wholly inside visible region - save endpoint
946/// * Case 2 : Exit visible region - save the intersection
947/// * Case 3 : Wholly outside visible region - save nothing
948/// * Case 4 : Enter visible region - save intersection and endpoint
949
951{
952 Int_t nc, nc2;
953 Double_t x1, y1, x2, y2, slope; // Segment to be clipped
954
955 std::vector<Double_t> xc2(nn), yc2(nn);
956
957 // Clip against the left boundary
958 x1 = x[n-1]; y1 = y[n-1];
959 nc2 = 0;
960 Int_t i;
961 for (i=0; i<n; i++) {
962 x2 = x[i]; y2 = y[i];
963 if (x1 == x2) {
964 slope = 0;
965 } else {
966 slope = (y2-y1)/(x2-x1);
967 }
968 if (x1 >= xclipl) {
969 if (x2 < xclipl) {
970 xc2[nc2] = xclipl; yc2[nc2++] = slope*(xclipl-x1)+y1;
971 } else {
972 xc2[nc2] = x2; yc2[nc2++] = y2;
973 }
974 } else {
975 if (x2 >= xclipl) {
976 xc2[nc2] = xclipl; yc2[nc2++] = slope*(xclipl-x1)+y1;
977 xc2[nc2] = x2; yc2[nc2++] = y2;
978 }
979 }
980 x1 = x2; y1 = y2;
981 }
982
983 // Clip against the top boundary
984 x1 = xc2[nc2-1]; y1 = yc2[nc2-1];
985 nc = 0;
986 for (i=0; i<nc2; i++) {
987 x2 = xc2[i]; y2 = yc2[i];
988 if (y1 == y2) {
989 slope = 0;
990 } else {
991 slope = (x2-x1)/(y2-y1);
992 }
993 if (y1 <= yclipt) {
994 if (y2 > yclipt) {
995 xc[nc] = x1+(yclipt-y1)*slope; yc[nc++] = yclipt;
996 } else {
997 xc[nc] = x2; yc[nc++] = y2;
998 }
999 } else {
1000 if (y2 <= yclipt) {
1001 xc[nc] = x1+(yclipt-y1)*slope; yc[nc++] = yclipt;
1002 xc[nc] = x2; yc[nc++] = y2;
1003 }
1004 }
1005 x1 = x2; y1 = y2;
1006 }
1007
1008 if (nc>0) {
1009
1010 // Clip against the right boundary
1011 x1 = xc[nc-1]; y1 = yc[nc-1];
1012 nc2 = 0;
1013 for (i=0; i<nc; i++) {
1014 x2 = xc[i]; y2 = yc[i];
1015 if (x1 == x2) {
1016 slope = 0;
1017 } else {
1018 slope = (y2-y1)/(x2-x1);
1019 }
1020 if (x1 <= xclipr) {
1021 if (x2 > xclipr) {
1022 xc2[nc2] = xclipr; yc2[nc2++] = slope*(xclipr-x1)+y1;
1023 } else {
1024 xc2[nc2] = x2; yc2[nc2++] = y2;
1025 }
1026 } else {
1027 if (x2 <= xclipr) {
1028 xc2[nc2] = xclipr; yc2[nc2++] = slope*(xclipr-x1)+y1;
1029 xc2[nc2] = x2; yc2[nc2++] = y2;
1030 }
1031 }
1032 x1 = x2; y1 = y2;
1033 }
1034
1035 // Clip against the bottom boundary
1036 x1 = xc2[nc2-1]; y1 = yc2[nc2-1];
1037 nc = 0;
1038 for (i=0; i<nc2; i++) {
1039 x2 = xc2[i]; y2 = yc2[i];
1040 if (y1 == y2) {
1041 slope = 0;
1042 } else {
1043 slope = (x2-x1)/(y2-y1);
1044 }
1045 if (y1 >= yclipb) {
1046 if (y2 < yclipb) {
1047 xc[nc] = x1+(yclipb-y1)*slope; yc[nc++] = yclipb;
1048 } else {
1049 xc[nc] = x2; yc[nc++] = y2;
1050 }
1051 } else {
1052 if (y2 >= yclipb) {
1053 xc[nc] = x1+(yclipb-y1)*slope; yc[nc++] = yclipb;
1054 xc[nc] = x2; yc[nc++] = y2;
1055 }
1056 }
1057 x1 = x2; y1 = y2;
1058 }
1059 }
1060
1061 if (nc < 3) nc =0;
1062 return nc;
1063}
1064
1065////////////////////////////////////////////////////////////////////////////////
1066/// Delete all primitives in pad and pad itself.
1067/// Pad cannot be used anymore after this call.
1068/// Emits signal "Closed()".
1069
1071{
1072 if (ROOT::Detail::HasBeenDeleted(this)) return;
1073 if (!fMother) return;
1075
1076 if (fPrimitives)
1077 fPrimitives->Clear();
1078 if (fView) {
1080 fView = nullptr;
1081 }
1082 if (fFrame) {
1084 fFrame = nullptr;
1085 }
1086
1087 // emit signal
1088 if (IsA() != TCanvas::Class())
1089 Closed();
1090
1091 if (fPixmapID != -1) {
1092 if (gPad) {
1093 if (!gPad->IsBatch() && GetPainter())
1095 }
1096 fPixmapID = -1;
1097
1098 if (!gROOT->GetListOfCanvases()) return;
1099 if (fMother == this) {
1100 gROOT->GetListOfCanvases()->Remove(this);
1101 return; // in case of TCanvas
1102 }
1103
1104 // remove from the mother's list of primitives
1105 if (fMother) {
1106 fMother->Remove(this, kFALSE); // do not produce modified
1107
1108 if (gPad == this)
1109 fMother->cd();
1110 }
1111 if (fCanvas) {
1112 if (fCanvas->GetPadSave() == this)
1114 if (fCanvas->GetSelectedPad() == this)
1115 fCanvas->SetSelectedPad(nullptr);
1116 if (fCanvas->GetClickSelectedPad() == this)
1117 fCanvas->SetClickSelectedPad(nullptr);
1118 }
1119 }
1120
1121 fMother = nullptr;
1122 if (gROOT->GetSelectedPad() == this)
1123 gROOT->SetSelectedPad(nullptr);
1124}
1125
1126////////////////////////////////////////////////////////////////////////////////
1127/// Copy the pixmap of the pad to the canvas.
1128
1130{
1131 int px, py;
1132 XYtoAbsPixel(fX1, fY2, px, py);
1133
1134 if (fPixmapID != -1 && GetPainter())
1135 GetPainter()->CopyDrawable(fPixmapID, px, py);
1136
1137 if (this == gPad) HighLight(gPad->GetHighLightColor());
1138}
1139
1140////////////////////////////////////////////////////////////////////////////////
1141/// Copy the sub-pixmaps of the pad to the canvas.
1142
1144{
1145 if (!fPrimitives) fPrimitives = new TList;
1146 TIter next(GetListOfPrimitives());
1147 while (auto obj = next()) {
1148 if (obj->InheritsFrom(TPad::Class())) {
1149 ((TPad*)obj)->CopyPixmap();
1150 ((TPad*)obj)->CopyPixmaps();
1151 }
1152 }
1153}
1154
1155////////////////////////////////////////////////////////////////////////////////
1156/// Remove TExec name from the list of Execs.
1157
1158void TPad::DeleteExec(const char *name)
1159{
1160 if (!fExecs) fExecs = new TList;
1162 if (!ex) return;
1163 fExecs->Remove(ex);
1164 delete ex;
1165}
1166
1167////////////////////////////////////////////////////////////////////////////////
1168/// Compute distance from point px,py to a box.
1169///
1170/// Compute the closest distance of approach from point px,py to the
1171/// edges of this pad.
1172/// The distance is computed in pixels units.
1173
1175{
1176 Int_t pxl, pyl, pxt, pyt;
1177 Int_t px1 = gPad->XtoAbsPixel(fX1);
1178 Int_t py1 = gPad->YtoAbsPixel(fY1);
1179 Int_t px2 = gPad->XtoAbsPixel(fX2);
1180 Int_t py2 = gPad->YtoAbsPixel(fY2);
1181 if (px1 < px2) {pxl = px1; pxt = px2;}
1182 else {pxl = px2; pxt = px1;}
1183 if (py1 < py2) {pyl = py1; pyt = py2;}
1184 else {pyl = py2; pyt = py1;}
1185
1186 // Are we inside the box?
1187 // ======================
1188 if ( (px > pxl && px < pxt) && (py > pyl && py < pyt) ) {
1189 if (GetFillStyle()) return 0; //*-* if pad is filled
1190 }
1191
1192 // Are we on the edges?
1193 // ====================
1194 Int_t dxl = TMath::Abs(px - pxl);
1195 if (py < pyl) dxl += pyl - py;
1196 if (py > pyt) dxl += py - pyt;
1197 Int_t dxt = TMath::Abs(px - pxt);
1198 if (py < pyl) dxt += pyl - py;
1199 if (py > pyt) dxt += py - pyt;
1200 Int_t dyl = TMath::Abs(py - pyl);
1201 if (px < pxl) dyl += pxl - px;
1202 if (px > pxt) dyl += px - pxt;
1203 Int_t dyt = TMath::Abs(py - pyt);
1204 if (px < pxl) dyt += pxl - px;
1205 if (px > pxt) dyt += px - pxt;
1206
1207 Int_t distance = dxl;
1208 if (dxt < distance) distance = dxt;
1209 if (dyl < distance) distance = dyl;
1210 if (dyt < distance) distance = dyt;
1211
1212 return distance - Int_t(0.5*fLineWidth);
1213}
1214
1215////////////////////////////////////////////////////////////////////////////////
1216/// Automatic pad generation by division.
1217///
1218/// - The current canvas is divided in nx by ny equal divisions (pads).
1219/// - xmargin is the space along x between pads in percent of canvas.
1220/// - ymargin is the space along y between pads in percent of canvas.
1221/// - color is the color of the new pads. If 0, color is the canvas color.
1222///
1223/// Pads are automatically named `canvasname_n` where `n` is the division number
1224/// starting from top left pad.
1225///
1226/// Example if canvasname=c1 , nx=2, ny=3:
1227///
1228/// \image html gpad_pad3.png
1229///
1230/// Once a pad is divided into sub-pads, one can set the current pad
1231/// to a subpad with a given division number as illustrated above
1232/// with TPad::cd(subpad_number).
1233///
1234/// For example, to set the current pad to c1_4, one can do:
1235/// ~~~ {.cpp}
1236/// c1->cd(4)
1237/// ~~~
1238/// __Note1:__ c1.cd() is equivalent to c1.cd(0) and sets the current pad
1239/// to c1 itself.
1240///
1241/// __Note2:__ after a statement like c1.cd(6), the global variable gPad
1242/// points to the current pad. One can use gPad to set attributes
1243/// of the current pad.
1244///
1245/// __Note3:__ in case xmargin <=0 and ymargin <= 0, there is no space
1246/// between pads. The current pad margins are recomputed to
1247/// optimize the layout.
1248
1250{
1251 if (!IsEditable()) return;
1252
1253 if (gThreadXAR) {
1254 void *arr[7];
1255 arr[1] = this; arr[2] = (void*)&nx;arr[3] = (void*)& ny;
1256 arr[4] = (void*)&xmargin; arr[5] = (void *)& ymargin; arr[6] = (void *)&color;
1257 if ((*gThreadXAR)("PDCD", 7, arr, nullptr)) return;
1258 }
1259
1261
1262 cd();
1263 if (nx <= 0) nx = 1;
1264 if (ny <= 0) ny = 1;
1265 Int_t ix, iy;
1266 Double_t x1, y1, x2, y2, dx, dy;
1267 TPad *pad;
1268 TString name, title;
1269 Int_t n = 0;
1270 if (color == 0) color = GetFillColor();
1271 if (xmargin > 0 && ymargin > 0) {
1272 //general case
1273 dy = 1/Double_t(ny);
1274 dx = 1/Double_t(nx);
1275 for (iy=0;iy<ny;iy++) {
1276 y2 = 1 - iy*dy - ymargin;
1277 y1 = y2 - dy + 2*ymargin;
1278 if (y1 < 0) y1 = 0;
1279 if (y1 > y2) continue;
1280 for (ix=0;ix<nx;ix++) {
1281 x1 = ix*dx + xmargin;
1282 x2 = x1 +dx -2*xmargin;
1283 if (x1 > x2) continue;
1284 n++;
1285 name.Form("%s_%d", GetName(), n);
1286 pad = new TPad(name.Data(), name.Data(), x1, y1, x2, y2, color);
1287 pad->SetNumber(n);
1288 pad->Draw();
1289 }
1290 }
1291 } else {
1292 // special case when xmargin <= 0 && ymargin <= 0
1297 xl /= (1-xl+xr)*nx;
1298 xr /= (1-xl+xr)*nx;
1299 yb /= (1-yb+yt)*ny;
1300 yt /= (1-yb+yt)*ny;
1305 dx = (1-xl-xr)/nx;
1306 dy = (1-yb-yt)/ny;
1307 Int_t number = 0;
1308 for (Int_t i=0;i<nx;i++) {
1309 x1 = i*dx+xl;
1310 x2 = x1 + dx;
1311 if (i == 0) x1 = 0;
1312 if (i == nx-1) x2 = 1-xr;
1313 for (Int_t j=0;j<ny;j++) {
1314 number = j*nx + i +1;
1315 y2 = 1 -j*dy -yt;
1316 y1 = y2 - dy;
1317 if (j == 0) y2 = 1-yt;
1318 if (j == ny-1) y1 = 0;
1319 name.Form("%s_%d", GetName(), number);
1320 title.Form("%s_%d", GetTitle(), number);
1321 pad = new TPad(name.Data(), title.Data(), x1, y1, x2, y2);
1322 pad->SetNumber(number);
1323 pad->SetBorderMode(0);
1324 if (i == 0) pad->SetLeftMargin(xl*nx);
1325 else pad->SetLeftMargin(0);
1326 pad->SetRightMargin(0);
1327 pad->SetTopMargin(0);
1328 if (j == ny-1) pad->SetBottomMargin(yb*ny);
1329 else pad->SetBottomMargin(0);
1330 pad->Draw();
1331 }
1332 }
1333 }
1334 Modified();
1335}
1336
1337////////////////////////////////////////////////////////////////////////////////
1338/// "n" is the total number of sub-pads. The number of sub-pads along the X
1339/// and Y axis are computed according to the square root of n.
1340
1342{
1343 Int_t w = 1, h = 1;
1344 if (!fCanvas) {
1345 Error("DivideSquare", "No canvas associated with this pad.");
1346 return;
1347 }
1351 if (w*h < n) w++;
1352 } else {
1355 if (w*h < n) h++;
1356 }
1357
1358 Divide( w, h, xmargin, ymargin, color);
1359}
1360
1361////////////////////////////////////////////////////////////////////////////////
1362/// Draw Pad in Current pad (re-parent pad if necessary).
1363
1365{
1366 // if no canvas opened yet create a default canvas
1367 if (!gPad) {
1368 gROOT->MakeDefCanvas();
1369 }
1370
1371 // pad cannot be in itself and it can only be in one other pad at a time
1372 if (!fPrimitives) fPrimitives = new TList;
1373 if (gPad != this) {
1375 fMother->Remove(this, kFALSE);
1377 fCanvas = gPad->GetCanvas();
1378 //
1379 fMother = (TPad*)gPad;
1380 if (oldMother != fMother || fPixmapID == -1) ResizePad();
1381 }
1382
1383 if (fCanvas && fCanvas->IsWeb()) {
1384 Modified();
1386 } else {
1387 Paint();
1388 }
1389
1390 if (gPad->IsRetained() && gPad != this && fMother)
1391 fMother->Add(this, option);
1392}
1393
1394////////////////////////////////////////////////////////////////////////////////
1395/// Draw class inheritance tree of the class to which obj belongs.
1396///
1397/// If a class B inherits from a class A, description of B is drawn
1398/// on the right side of description of A.
1399///
1400/// Member functions overridden by B are shown in class A with a blue line
1401/// crossing-out the corresponding member function.
1402
1404{
1405 if (!classobj) return;
1406 char dname[256];
1407 const Int_t kMAXLEVELS = 10;
1408 TClass *clevel[kMAXLEVELS], *cl, *cll;
1409 TBaseClass *base, *cinherit;
1410 TText *ptext = nullptr;
1411 TString opt=option;
1412 Double_t x,y,dy,y1,v1,v2,dv;
1413 Int_t nd,nf,nc,nkd,nkf,i,j;
1414 TPaveText *pt;
1415 Int_t maxlev = 4;
1416 if (opt.Contains("2")) maxlev = 2;
1417 if (opt.Contains("3")) maxlev = 3;
1418 if (opt.Contains("5")) maxlev = 5;
1419 if (opt.Contains("6")) maxlev = 6;
1420 if (opt.Contains("7")) maxlev = 7;
1421
1422 // Clear and Set Pad range
1423 Double_t xpad = 20.5;
1424 Double_t ypad = 27.5;
1425 Clear();
1426 Range(0,0,xpad,ypad);
1427
1428 // Find number of levels
1429 Int_t nlevel = 0;
1430 TClass *obj = (TClass*)classobj;
1431 clevel[nlevel] = obj;
1432 TList *lbase = obj->GetListOfBases();
1433 while(lbase) {
1434 base = (TBaseClass*)lbase->First();
1435 if (!base) break;
1436 if (!base->GetClassPointer()) break;
1437 nlevel++;
1438 clevel[nlevel] = base->GetClassPointer();
1439 lbase = clevel[nlevel]->GetListOfBases();
1440 if (nlevel >= maxlev-1) break;
1441 }
1442 Int_t maxelem = 0;
1443 Int_t ncdraw = 0;
1445 for (ilevel=nlevel;ilevel>=0;ilevel--) {
1446 cl = clevel[ilevel];
1447 nelem = cl->GetNdata() + cl->GetNmethods();
1448 if (nelem > maxelem) maxelem = nelem;
1449 nc = (nelem/50) + 1;
1450 ncdraw += nc;
1451 }
1452
1453 Double_t tsizcm = 0.40;
1454 Double_t x1 = 0.25;
1455 Double_t x2 = 0;
1456 Double_t dx = 3.5;
1457 if (ncdraw > 4) {
1458 dx = dx - 0.42*Double_t(ncdraw-5);
1459 if (dx < 1.3) dx = 1.3;
1460 tsizcm = tsizcm - 0.03*Double_t(ncdraw-5);
1461 if (tsizcm < 0.27) tsizcm = 0.27;
1462 }
1463 Double_t tsiz = 1.2*tsizcm/ypad;
1464
1465 // Now loop on levels
1466 for (ilevel=nlevel;ilevel>=0;ilevel--) {
1467 cl = clevel[ilevel];
1468 nelem = cl->GetNdata() + cl->GetNmethods();
1469 if (nelem > maxelem) maxelem = nelem;
1470 nc = (nelem/50) + 1;
1471 dy = 0.45;
1472 if (ilevel < nlevel) x1 = x2 + 0.5;
1473 x2 = x1 + nc*dx;
1474 v2 = ypad - 0.5;
1475 lbase = cl->GetListOfBases();
1476 cinherit = nullptr;
1477 if (lbase) cinherit = (TBaseClass*)lbase->First();
1478
1479 do {
1480 nd = cl->GetNdata();
1481 nf = cl->GetNmethods() - 2; //do not show default constructor and destructor
1482 if (cl->GetListOfMethods()->FindObject("Dictionary")) {
1483 nf -= 6; // do not count the Dictionary/ClassDef functions
1484 }
1485 nkf= nf/nc +1;
1486 nkd= nd/nc +1;
1487 if (nd == 0) nkd=0;
1488 if (nf == 0) nkf=0;
1489 y1 = v2 - 0.7;
1490 v1 = y1 - Double_t(nkf+nkd+nc-1)*dy;
1491 dv = v2 - v1;
1492
1493 // Create a new PaveText
1494 pt = new TPaveText(x1,v1,x2,v2);
1496 pt->SetFillColor(19);
1497 pt->Draw();
1498 pt->SetTextColor(4);
1499 pt->SetTextFont(61);
1500 pt->SetTextAlign(12);
1502 TBox *box = pt->AddBox(0,(y1+0.01-v1)/dv,0,(v2-0.01-v1)/dv);
1503 if (box) box->SetFillColor(17);
1504 pt->AddLine(0,(y1-v1)/dv,0,(y1-v1)/dv);
1505 TText *title = pt->AddText(0.5,(0.5*(y1+v2)-v1)/dv,(char*)cl->GetName());
1506 title->SetTextAlign(22);
1507 title->SetTextSize(0.6*(v2-y1)/ypad);
1508
1509 // Draw data Members
1510 i = 0;
1511 x = 0.03;
1512 y = y1 + 0.5*dy;
1513 TDataMember *d;
1515 while ((d = (TDataMember *) nextd())) {
1516 if (i >= nkd) { i = 1; y = y1 - 0.5*dy; x += 1/Double_t(nc); }
1517 else { i++; y -= dy; }
1518
1519 // Take in account the room the array index will occupy
1520
1521 Int_t dim = d->GetArrayDim();
1522 Int_t indx = 0;
1523 snprintf(dname,256,"%s",d->GetName());
1524 Int_t ldname = 0;
1525 while (indx < dim ){
1526 ldname = strlen(dname);
1527 snprintf(&dname[ldname],256-ldname,"[%d]",d->GetMaxIndex(indx));
1528 indx++;
1529 }
1530 pt->AddText(x,(y-v1)/dv,dname);
1531 }
1532
1533 // Draw a separator line
1534 Double_t ysep;
1535 if (nd) {
1536 ysep = y1 - Double_t(nkd)*dy;
1537 pt->AddLine(0,(ysep-v1)/dv,0,(ysep-v1)/dv);
1538 ysep -= 0.5*dy;
1539 } else ysep = y1;
1540
1541 // Draw Member Functions
1542 Int_t fcount = 0;
1543 i = 0;
1544 x = 0.03;
1545 y = ysep + 0.5*dy;
1546 TMethod *m;
1548 while ((m = (TMethod *) nextm())) {
1549 if (
1550 !strcmp( m->GetName(), "Dictionary" ) ||
1551 !strcmp( m->GetName(), "Class_Version" ) ||
1552 !strcmp( m->GetName(), "DeclFileName" ) ||
1553 !strcmp( m->GetName(), "DeclFileLine" ) ||
1554 !strcmp( m->GetName(), "ImplFileName" ) ||
1555 !strcmp( m->GetName(), "ImplFileLine" )
1556 ) continue;
1557 fcount++;
1558 if (fcount > nf) break;
1559 if (i >= nkf) { i = 1; y = ysep - 0.5*dy; x += 1/Double_t(nc); }
1560 else { i++; y -= dy; }
1561
1562 ptext = pt->AddText(x,(y-v1)/dv,m->GetName());
1563 // Check if method is overloaded in a derived class
1564 // If yes, Change the color of the text to blue
1565 for (j=ilevel-1;j>=0;j--) {
1566 if (cl == clevel[ilevel]) {
1567 if (clevel[j]->GetMethodAny((char*)m->GetName())) {
1568 ptext->SetTextColor(15);
1569 break;
1570 }
1571 }
1572 }
1573 }
1574
1575 // Draw second inheritance classes for this class
1576 cll = nullptr;
1577 if (cinherit) {
1578 cinherit = (TBaseClass*)lbase->After(cinherit);
1579 if (cinherit) {
1580 cl = cinherit->GetClassPointer();
1581 cll = cl;
1582 v2 = v1 -0.4;
1583 dy = 0.35;
1584 }
1585 }
1586 } while (cll);
1587 }
1588 Update();
1589}
1590
1591////////////////////////////////////////////////////////////////////////////////
1592/// Function called to draw a crosshair in the canvas
1593///
1594/// Example:
1595/// ~~~ {.cpp}
1596/// Root > TFile f("hsimple.root");
1597/// Root > hpxpy.Draw();
1598/// Root > c1.SetCrosshair();
1599/// ~~~
1600/// When moving the mouse in the canvas, a crosshair is drawn
1601///
1602/// - if the canvas fCrosshair = 1 , the crosshair spans the full canvas
1603/// - if the canvas fCrosshair > 1 , the crosshair spans only the pad
1604
1606{
1607 if (!gPad || (gPad->GetEvent() == kMouseEnter)) return;
1608
1609 TPad *cpad = (TPad*)gPad;
1610 TCanvas *canvas = cpad->GetCanvas();
1611 canvas->FeedbackMode(kTRUE);
1612
1613 //erase old position and draw a line at current position
1614 Int_t pxmin,pxmax,pymin,pymax,px,py;
1615#ifndef R__HAS_COCOA
1616 Int_t pxold = fCrosshairPos%10000;
1617 Int_t pyold = fCrosshairPos/10000;
1618#endif // R__HAS_COCOA
1619 px = cpad->GetEventX();
1620 py = cpad->GetEventY()+1;
1621 if (canvas->GetCrosshair() > 1) { //crosshair only in the current pad
1622 pxmin = cpad->XtoAbsPixel(fX1);
1623 pxmax = cpad->XtoAbsPixel(fX2);
1624 pymin = cpad->YtoAbsPixel(fY1);
1625 pymax = cpad->YtoAbsPixel(fY2);
1626 } else { //default; crosshair spans the full canvas
1627 pxmin = 0;
1628 pxmax = canvas->GetWw();
1629 pymin = 0;
1630 pymax = cpad->GetWh();
1631 }
1632#ifndef R__HAS_COCOA
1633 // Not needed, no XOR with Cocoa.
1634 if(pxold) gVirtualX->DrawLine(pxold,pymin,pxold,pymax);
1635 if(pyold) gVirtualX->DrawLine(pxmin,pyold,pxmax,pyold);
1636#endif // R__HAS_COCOA
1637 if (cpad->GetEvent() == kButton1Down ||
1638 cpad->GetEvent() == kButton1Up ||
1639 cpad->GetEvent() == kMouseLeave) {
1640 fCrosshairPos = 0;
1641 return;
1642 }
1643 gVirtualX->DrawLine(px,pymin,px,pymax);
1644 gVirtualX->DrawLine(pxmin,py,pxmax,py);
1645 fCrosshairPos = px + 10000*py;
1646}
1647
1648////////////////////////////////////////////////////////////////////////////////
1649/// Draw an empty pad frame with X and Y axis.
1650///
1651/// \return The pointer to the histogram used to draw the frame.
1652///
1653/// \param[in] xmin X axis lower limit
1654/// \param[in] xmax X axis upper limit
1655/// \param[in] ymin Y axis lower limit
1656/// \param[in] ymax Y axis upper limit
1657/// \param[in] title Pad title.If title is of the form "stringt;stringx;stringy"
1658/// the pad title is set to stringt, the x axis title to
1659/// stringx, the y axis title to stringy.
1660///
1661/// #### Example:
1662///
1663/// Begin_Macro(source)
1664/// {
1665/// auto c = new TCanvas("c","c",200,10,500,300);
1666///
1667/// const Int_t n = 50;
1668/// auto g = new TGraph();
1669/// for (Int_t i=0;i<n;i++) g->SetPoint(i,i*0.1,100*sin(i*0.1+0.2));
1670///
1671/// auto frame = c->DrawFrame(0, -110, 2, 110);
1672/// frame->GetXaxis()->SetTitle("X axis");
1673///
1674/// g->Draw("L*");
1675/// }
1676/// End_Macro
1677
1679{
1680 if (!IsEditable())
1681 return nullptr;
1682
1683 if (this != gPad) {
1684 Warning("DrawFrame", "Must be called for the current pad only");
1685 if (gPad) return gPad->DrawFrame(xmin,ymin,xmax,ymax,title);
1686 }
1687
1688 cd();
1689
1690 TH1F *hframe = (TH1F*)FindObject("hframe");
1691 if (hframe) delete hframe;
1692 Int_t nbins = 1000;
1693 //if log scale in X, use variable bin size linear with log(x)
1694 //this gives a better precision when zooming on the axis
1695 if (fLogx && xmin > 0 && xmax > xmin) {
1698 Double_t dx = (xmaxl-xminl)/nbins;
1699 std::vector<Double_t> xbins(nbins+1);
1700 xbins[0] = xmin;
1701 for (Int_t i=1;i<=nbins;i++) {
1702 xbins[i] = TMath::Exp(xminl+i*dx);
1703 }
1704 hframe = new TH1F("hframe",title,nbins,xbins.data());
1705 } else {
1706 hframe = new TH1F("hframe",title,nbins,xmin,xmax);
1707 }
1708 hframe->SetBit(TH1::kNoStats);
1709 hframe->SetBit(kCanDelete);
1710 hframe->SetMinimum(ymin);
1711 hframe->SetMaximum(ymax);
1712 hframe->GetYaxis()->SetLimits(ymin,ymax);
1713 hframe->SetDirectory(nullptr);
1714 hframe->Draw(" ");
1715 Update();
1716 cd();
1717 return hframe;
1718}
1719
1720////////////////////////////////////////////////////////////////////////////////
1721/// Static function to Display Color Table in a pad.
1722
1724{
1725 Int_t i, j;
1726 Int_t color;
1727 Double_t xlow, ylow, xup, yup, hs, ws;
1728 Double_t x1, y1, x2, y2;
1729 x1 = y1 = 0;
1730 x2 = y2 = 20;
1731
1732 gPad->SetFillColor(0);
1733 gPad->Clear();
1734 gPad->Range(x1,y1,x2,y2);
1735
1736 TText text(0,0,"");
1737 text.SetTextFont(61);
1738 text.SetTextSize(0.07);
1739 text.SetTextAlign(22);
1740
1741 TBox box;
1742
1743 // Draw color table boxes.
1744 hs = (y2-y1)/Double_t(5);
1745 ws = (x2-x1)/Double_t(10);
1746 for (i=0;i<10;i++) {
1747 xlow = x1 + ws*(Double_t(i)+0.1);
1748 xup = x1 + ws*(Double_t(i)+0.9);
1749 for (j=0;j<5;j++) {
1750 ylow = y1 + hs*(Double_t(j)+0.1);
1751 yup = y1 + hs*(Double_t(j)+0.9);
1752 color = 10*j + i;
1753 box.SetFillStyle(1001);
1754 box.SetFillColor(color);
1755 box.DrawBox(xlow, ylow, xup, yup);
1756 box.SetFillStyle(0);
1757 box.SetLineColor(1);
1758 box.DrawBox(xlow, ylow, xup, yup);
1759 if (color == 1) text.SetTextColor(0);
1760 else text.SetTextColor(1);
1761 text.DrawText(0.5*(xlow+xup), 0.5*(ylow+yup), TString::Format("%d",color).Data());
1762 }
1763 }
1764}
1765
1766////////////////////////////////////////////////////////////////////////////////
1767/// Execute action corresponding to one event.
1768///
1769/// This member function is called when a TPad object is clicked.
1770///
1771/// If the mouse is clicked in one of the 4 corners of the pad (pA,pB,pC,pD)
1772/// the pad is resized with the rubber rectangle.
1773///
1774/// If the mouse is clicked inside the pad, the pad is moved.
1775///
1776/// If the mouse is clicked on the 4 edges (pL,pR,pTop,pBot), the pad is scaled
1777/// parallel to this edge.
1778///
1779/// \image html gpad_pad4.png
1780///
1781/// Note that this function duplicates on purpose the functionality
1782/// already implemented in TBox::ExecuteEvent.
1783/// If somebody modifies this function, may be similar changes should also
1784/// be applied to TBox::ExecuteEvent.
1785
1787{
1788 const Int_t kMaxDiff = 5;
1789 const Int_t kMinSize = 20;
1790 static Int_t pxorg, pyorg;
1791 static Int_t px1, px2, py1, py2, pxl, pyl, pxt, pyt, pxold, pyold;
1792 static Int_t px1p, px2p, py1p, py2p, pxlp, pylp, pxtp, pytp;
1793 static Bool_t pA, pB, pC, pD, pTop, pL, pR, pBot, pINSIDE;
1794 Int_t wx, wy;
1798
1799 if (!IsEditable() && event != kMouseEnter) return;
1800 TVirtualPad *parent = GetMother();
1801 if (!parent->IsEditable()) return;
1802
1803 HideToolTip(event);
1804
1805 if (fXlowNDC < 0 && event != kButton1Down) return;
1806 if (fYlowNDC < 0 && event != kButton1Down) return;
1807
1808 // keep old mouse position
1809 if (event == kButton1Down) {
1810 pxorg = px;
1811 pyorg = py;
1812 }
1813
1814 Int_t newcode = gROOT->GetEditorMode();
1815 if (newcode)
1816 pA = pB = pC = pD = pTop = pL = pR = pBot = pINSIDE = kFALSE;
1817 switch (newcode) {
1818 case kPad:
1819 TCreatePrimitives::Pad(event,px,py,0);
1820 break;
1821 case kMarker:
1822 case kText:
1823 TCreatePrimitives::Text(event,px,py,newcode);
1824 break;
1825 case kLine:
1826 TCreatePrimitives::Line(event,px,py,kLine);
1827 break;
1828 case kArrow:
1829 TCreatePrimitives::Line(event,px,py,kArrow);
1830 break;
1831 case kCurlyLine:
1833 break;
1834 case kCurlyArc:
1836 break;
1837 case kPolyLine:
1839 break;
1840 case kCutG:
1842 break;
1843 case kArc:
1844 TCreatePrimitives::Ellipse(event,px,py,kArc);
1845 break;
1846 case kEllipse:
1848 break;
1849 case kButton:
1850 case kPave:
1851 case kPaveLabel:
1852 case kPaveText:
1853 case kPavesText:
1854 case kDiamond:
1855 TCreatePrimitives::Pave(event,px,py,newcode);
1856 return;
1857 default:
1858 break;
1859 }
1860 if (newcode) return;
1861
1862 switch (event) {
1863
1864 case kMouseEnter:
1865 if (fTip)
1867 break;
1868
1869 case kArrowKeyPress:
1870 case kButton1Down:
1871
1874
1875 GetPainter()->SetLineColor(-1);
1876 TAttLine::Modify(); //Change line attributes only if necessary
1877 if (GetFillColor())
1879 else
1882
1883 // No break !!!
1884
1885 case kMouseMotion:
1886
1887 px1 = XtoAbsPixel(fX1);
1888 py1 = YtoAbsPixel(fY1);
1889 px2 = XtoAbsPixel(fX2);
1890 py2 = YtoAbsPixel(fY2);
1891
1892 if (px1 < px2) {
1893 pxl = px1;
1894 pxt = px2;
1895 } else {
1896 pxl = px2;
1897 pxt = px1;
1898 }
1899 if (py1 < py2) {
1900 pyl = py1;
1901 pyt = py2;
1902 } else {
1903 pyl = py2;
1904 pyt = py1;
1905 }
1906
1907 px1p = parent->XtoAbsPixel(parent->GetX1()) + parent->GetBorderSize();
1908 py1p = parent->YtoAbsPixel(parent->GetY1()) - parent->GetBorderSize();
1909 px2p = parent->XtoAbsPixel(parent->GetX2()) - parent->GetBorderSize();
1910 py2p = parent->YtoAbsPixel(parent->GetY2()) + parent->GetBorderSize();
1911
1912 if (px1p < px2p) {
1913 pxlp = px1p;
1914 pxtp = px2p;
1915 } else {
1916 pxlp = px2p;
1917 pxtp = px1p;
1918 }
1919 if (py1p < py2p) {
1920 pylp = py1p;
1921 pytp = py2p;
1922 } else {
1923 pylp = py2p;
1924 pytp = py1p;
1925 }
1926
1927 pA = pB = pC = pD = pTop = pL = pR = pBot = pINSIDE = kFALSE;
1928
1929 // case pA
1930 if (TMath::Abs(px - pxl) <= kMaxDiff && TMath::Abs(py - pyl) <= kMaxDiff) {
1931 pxold = pxl; pyold = pyl; pA = kTRUE;
1933 }
1934 // case pB
1935 if (TMath::Abs(px - pxt) <= kMaxDiff && TMath::Abs(py - pyl) <= kMaxDiff) {
1936 pxold = pxt; pyold = pyl; pB = kTRUE;
1938 }
1939 // case pC
1940 if (TMath::Abs(px - pxt) <= kMaxDiff && TMath::Abs(py - pyt) <= kMaxDiff) {
1941 pxold = pxt; pyold = pyt; pC = kTRUE;
1943 }
1944 // case pD
1945 if (TMath::Abs(px - pxl) <= kMaxDiff && TMath::Abs(py - pyt) <= kMaxDiff) {
1946 pxold = pxl; pyold = pyt; pD = kTRUE;
1948 }
1949
1950 if ((px > pxl+kMaxDiff && px < pxt-kMaxDiff) &&
1951 TMath::Abs(py - pyl) < kMaxDiff) { // top edge
1952 pxold = pxl; pyold = pyl; pTop = kTRUE;
1954 }
1955
1956 if ((px > pxl+kMaxDiff && px < pxt-kMaxDiff) &&
1957 TMath::Abs(py - pyt) < kMaxDiff) { // bottom edge
1958 pxold = pxt; pyold = pyt; pBot = kTRUE;
1960 }
1961
1962 if ((py > pyl+kMaxDiff && py < pyt-kMaxDiff) &&
1963 TMath::Abs(px - pxl) < kMaxDiff) { // left edge
1964 pxold = pxl; pyold = pyl; pL = kTRUE;
1966 }
1967
1968 if ((py > pyl+kMaxDiff && py < pyt-kMaxDiff) &&
1969 TMath::Abs(px - pxt) < kMaxDiff) { // right edge
1970 pxold = pxt; pyold = pyt; pR = kTRUE;
1972 }
1973
1974 if ((px > pxl+kMaxDiff && px < pxt-kMaxDiff) &&
1975 (py > pyl+kMaxDiff && py < pyt-kMaxDiff)) { // inside box
1976 pxold = px; pyold = py; pINSIDE = kTRUE;
1977 if (event == kButton1Down)
1979 else
1981 }
1982
1983 fResizing = kFALSE;
1984 if (pA || pB || pC || pD || pTop || pL || pR || pBot)
1985 fResizing = kTRUE;
1986
1987 if (!pA && !pB && !pC && !pD && !pTop && !pL && !pR && !pBot && !pINSIDE)
1989
1990 break;
1991
1992 case kArrowKeyRelease:
1993 case kButton1Motion:
1994
1995 if (TestBit(kCannotMove)) break;
1996 wx = wy = 0;
1997
1998 if (pA) {
2000 if (px > pxt-kMinSize) { px = pxt-kMinSize; wx = px; }
2001 if (py > pyt-kMinSize) { py = pyt-kMinSize; wy = py; }
2002 if (px < pxlp) { px = pxlp; wx = px; }
2003 if (py < pylp) { py = pylp; wy = py; }
2004 if (fixedr) {
2005 Double_t dy = Double_t(TMath::Abs(pxt-px))/parent->UtoPixel(1.) /
2007 Int_t npy2 = pyt - TMath::Abs(parent->VtoAbsPixel(dy) -
2008 parent->VtoAbsPixel(0));
2009 if (npy2 < pylp) {
2010 px = pxold;
2011 py = pyold;
2012 } else
2013 py = npy2;
2014
2015 wx = wy = 0;
2016 }
2017 if (!ropaque) gVirtualX->DrawBox(px, pyt, pxt, py, TVirtualX::kHollow);
2018 }
2019 if (pB) {
2020 if (!ropaque) gVirtualX->DrawBox(pxl , pyt, pxold, pyold, TVirtualX::kHollow);
2021 if (px < pxl+kMinSize) { px = pxl+kMinSize; wx = px; }
2022 if (py > pyt-kMinSize) { py = pyt-kMinSize; wy = py; }
2023 if (px > pxtp) { px = pxtp; wx = px; }
2024 if (py < pylp) { py = pylp; wy = py; }
2025 if (fixedr) {
2026 Double_t dy = Double_t(TMath::Abs(pxl-px))/parent->UtoPixel(1.) /
2028 Int_t npy2 = pyt - TMath::Abs(parent->VtoAbsPixel(dy) -
2029 parent->VtoAbsPixel(0));
2030 if (npy2 < pylp) {
2031 px = pxold;
2032 py = pyold;
2033 } else
2034 py = npy2;
2035
2036 wx = wy = 0;
2037 }
2038 if (!ropaque) gVirtualX->DrawBox(pxl , pyt, px , py, TVirtualX::kHollow);
2039 }
2040 if (pC) {
2041 if (!ropaque) gVirtualX->DrawBox(pxl , pyl, pxold, pyold, TVirtualX::kHollow);
2042 if (px < pxl+kMinSize) { px = pxl+kMinSize; wx = px; }
2043 if (py < pyl+kMinSize) { py = pyl+kMinSize; wy = py; }
2044 if (px > pxtp) { px = pxtp; wx = px; }
2045 if (py > pytp) { py = pytp; wy = py; }
2046 if (fixedr) {
2047 Double_t dy = Double_t(TMath::Abs(pxl-px))/parent->UtoPixel(1.) /
2049 Int_t npy2 = pyl + TMath::Abs(parent->VtoAbsPixel(dy) -
2050 parent->VtoAbsPixel(0));
2051 if (npy2 > pytp) {
2052 px = pxold;
2053 py = pyold;
2054 } else
2055 py = npy2;
2056
2057 wx = wy = 0;
2058 }
2059 if (!ropaque) gVirtualX->DrawBox(pxl, pyl, px, py, TVirtualX::kHollow);
2060 }
2061 if (pD) {
2063 if (px > pxt-kMinSize) { px = pxt-kMinSize; wx = px; }
2064 if (py < pyl+kMinSize) { py = pyl+kMinSize; wy = py; }
2065 if (px < pxlp) { px = pxlp; wx = px; }
2066 if (py > pytp) { py = pytp; wy = py; }
2067 if (fixedr) {
2068 Double_t dy = Double_t(TMath::Abs(pxt-px))/parent->UtoPixel(1.) /
2070 Int_t npy2 = pyl + TMath::Abs(parent->VtoAbsPixel(dy) -
2071 parent->VtoAbsPixel(0));
2072 if (npy2 > pytp) {
2073 px = pxold;
2074 py = pyold;
2075 } else
2076 py = npy2;
2077
2078 wx = wy = 0;
2079 }
2080 if (!ropaque) gVirtualX->DrawBox(px, py, pxt, pyl, TVirtualX::kHollow);
2081 }
2082 if (pTop) {
2083 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2084 py2 += py - pyold;
2085 if (py2 > py1-kMinSize) { py2 = py1-kMinSize; wy = py2; }
2086 if (py2 < py2p) { py2 = py2p; wy = py2; }
2087 if (fixedr) {
2088 Double_t dx = Double_t(TMath::Abs(py2-py1))/parent->VtoPixel(0) *
2090 Int_t npx2 = px1 + parent->UtoPixel(dx);
2091 if (npx2 > px2p)
2092 py2 -= py - pyold;
2093 else
2094 px2 = npx2;
2095 }
2096 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2097 }
2098 if (pBot) {
2099 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2100 py1 += py - pyold;
2101 if (py1 < py2+kMinSize) { py1 = py2+kMinSize; wy = py1; }
2102 if (py1 > py1p) { py1 = py1p; wy = py1; }
2103 if (fixedr) {
2104 Double_t dx = Double_t(TMath::Abs(py2-py1))/parent->VtoPixel(0) *
2106 Int_t npx2 = px1 + parent->UtoPixel(dx);
2107 if (npx2 > px2p)
2108 py1 -= py - pyold;
2109 else
2110 px2 = npx2;
2111 }
2112 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2113 }
2114 if (pL) {
2115 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2116 px1 += px - pxold;
2117 if (px1 > px2-kMinSize) { px1 = px2-kMinSize; wx = px1; }
2118 if (px1 < px1p) { px1 = px1p; wx = px1; }
2119 if (fixedr) {
2120 Double_t dy = Double_t(TMath::Abs(px2-px1))/parent->UtoPixel(1.) /
2122 Int_t npy2 = py1 - TMath::Abs(parent->VtoAbsPixel(dy) -
2123 parent->VtoAbsPixel(0));
2124 if (npy2 < py2p)
2125 px1 -= px - pxold;
2126 else
2127 py2 = npy2;
2128 }
2129 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2130 }
2131 if (pR) {
2132 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2133 px2 += px - pxold;
2134 if (px2 < px1+kMinSize) { px2 = px1+kMinSize; wx = px2; }
2135 if (px2 > px2p) { px2 = px2p; wx = px2; }
2136 if (fixedr) {
2137 Double_t dy = Double_t(TMath::Abs(px2-px1))/parent->UtoPixel(1.) /
2139 Int_t npy2 = py1 - TMath::Abs(parent->VtoAbsPixel(dy) -
2140 parent->VtoAbsPixel(0));
2141 if (npy2 < py2p)
2142 px2 -= px - pxold;
2143 else
2144 py2 = npy2;
2145 }
2146 if (!ropaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow);
2147 }
2148 if (pINSIDE) {
2149 if (!opaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow); // draw the old box
2150 Int_t dx = px - pxold;
2151 Int_t dy = py - pyold;
2152 px1 += dx; py1 += dy; px2 += dx; py2 += dy;
2153 if (px1 < px1p) { dx = px1p - px1; px1 += dx; px2 += dx; wx = px+dx; }
2154 if (px2 > px2p) { dx = px2 - px2p; px1 -= dx; px2 -= dx; wx = px-dx; }
2155 if (py1 > py1p) { dy = py1 - py1p; py1 -= dy; py2 -= dy; wy = py-dy; }
2156 if (py2 < py2p) { dy = py2p - py2; py1 += dy; py2 += dy; wy = py+dy; }
2157 if (!opaque) gVirtualX->DrawBox(px1, py1, px2, py2, TVirtualX::kHollow); // draw the new box
2158 }
2159
2160 if (wx || wy) {
2161 if (wx) px = wx;
2162 if (wy) py = wy;
2163 gVirtualX->Warp(px, py);
2164 }
2165
2166 pxold = px;
2167 pyold = py;
2168
2169 Double_t x1, y1, x2, y2;
2170 x1 = x2 = y1 = y2 = 0;
2171
2172 if ((!fResizing && opaque) || (fResizing && ropaque)) {
2173 if (pA) {
2174 x1 = AbsPixeltoX(pxold);
2175 y1 = AbsPixeltoY(pyt);
2176 x2 = AbsPixeltoX(pxt);
2177 y2 = AbsPixeltoY(pyold);
2178 }
2179 if (pB) {
2180 x1 = AbsPixeltoX(pxl);
2181 y1 = AbsPixeltoY(pyt);
2182 x2 = AbsPixeltoX(pxold);
2183 y2 = AbsPixeltoY(pyold);
2184 }
2185 if (pC) {
2186 x1 = AbsPixeltoX(pxl);
2187 y1 = AbsPixeltoY(pyold);
2188 x2 = AbsPixeltoX(pxold);
2189 y2 = AbsPixeltoY(pyl);
2190 }
2191 if (pD) {
2192 x1 = AbsPixeltoX(pxold);
2193 y1 = AbsPixeltoY(pyold);
2194 x2 = AbsPixeltoX(pxt);
2195 y2 = AbsPixeltoY(pyl);
2196 }
2197 if (pTop || pBot || pL || pR || pINSIDE) {
2198 x1 = AbsPixeltoX(px1);
2199 y1 = AbsPixeltoY(py1);
2200 x2 = AbsPixeltoX(px2);
2201 y2 = AbsPixeltoY(py2);
2202 }
2203
2204 if (px != pxorg || py != pyorg) {
2205
2206 // Get parent corners pixels coordinates
2207 Int_t parentpx1 = fMother->XtoAbsPixel(parent->GetX1());
2208 Int_t parentpx2 = fMother->XtoAbsPixel(parent->GetX2());
2209 Int_t parentpy1 = fMother->YtoAbsPixel(parent->GetY1());
2210 Int_t parentpy2 = fMother->YtoAbsPixel(parent->GetY2());
2211
2212 // Get pad new corners pixels coordinates
2217
2218 // Compute new pad positions in the NDC space of parent
2223 }
2224
2225 // Reset pad parameters and recompute conversion coefficients
2226 ResizePad();
2227
2228 if (pINSIDE) gPad->ShowGuidelines(this, event);
2229 if (pTop) gPad->ShowGuidelines(this, event, 't', true);
2230 if (pBot) gPad->ShowGuidelines(this, event, 'b', true);
2231 if (pL) gPad->ShowGuidelines(this, event, 'l', true);
2232 if (pR) gPad->ShowGuidelines(this, event, 'r', true);
2233 if (pA) gPad->ShowGuidelines(this, event, '1', true);
2234 if (pB) gPad->ShowGuidelines(this, event, '2', true);
2235 if (pC) gPad->ShowGuidelines(this, event, '3', true);
2236 if (pD) gPad->ShowGuidelines(this, event, '4', true);
2237
2238 Modified(kTRUE);
2239 }
2240
2241 break;
2242
2243 case kButton1Up:
2244
2245 if (gROOT->IsEscaped()) {
2246 gROOT->SetEscape(kFALSE);
2247 break;
2248 }
2249
2250 if (opaque||ropaque) {
2251 ShowGuidelines(this, event);
2252 } else {
2253 x1 = x2 = y1 = y2 = 0;
2254
2255 if (pA) {
2256 x1 = AbsPixeltoX(pxold);
2257 y1 = AbsPixeltoY(pyt);
2258 x2 = AbsPixeltoX(pxt);
2259 y2 = AbsPixeltoY(pyold);
2260 }
2261 if (pB) {
2262 x1 = AbsPixeltoX(pxl);
2263 y1 = AbsPixeltoY(pyt);
2264 x2 = AbsPixeltoX(pxold);
2265 y2 = AbsPixeltoY(pyold);
2266 }
2267 if (pC) {
2268 x1 = AbsPixeltoX(pxl);
2269 y1 = AbsPixeltoY(pyold);
2270 x2 = AbsPixeltoX(pxold);
2271 y2 = AbsPixeltoY(pyl);
2272 }
2273 if (pD) {
2274 x1 = AbsPixeltoX(pxold);
2275 y1 = AbsPixeltoY(pyold);
2276 x2 = AbsPixeltoX(pxt);
2277 y2 = AbsPixeltoY(pyl);
2278 }
2279 if (pTop || pBot || pL || pR || pINSIDE) {
2280 x1 = AbsPixeltoX(px1);
2281 y1 = AbsPixeltoY(py1);
2282 x2 = AbsPixeltoX(px2);
2283 y2 = AbsPixeltoY(py2);
2284 }
2285
2286 if (pA || pB || pC || pD || pTop || pL || pR || pBot)
2287 Modified(kTRUE);
2288
2289 gVirtualX->SetLineColor(-1);
2290 gVirtualX->SetLineWidth(-1);
2291
2292 if (px != pxorg || py != pyorg) {
2293
2294 // Get parent corners pixels coordinates
2295 Int_t parentpx1 = fMother->XtoAbsPixel(parent->GetX1());
2296 Int_t parentpx2 = fMother->XtoAbsPixel(parent->GetX2());
2297 Int_t parentpy1 = fMother->YtoAbsPixel(parent->GetY1());
2298 Int_t parentpy2 = fMother->YtoAbsPixel(parent->GetY2());
2299
2300 // Get pad new corners pixels coordinates
2305
2306 // Compute new pad positions in the NDC space of parent
2311 }
2312
2313 // Reset pad parameters and recompute conversion coefficients
2314 ResizePad();
2315
2316
2317 // emit signal
2318 RangeChanged();
2319 }
2320
2321 break;
2322
2323 case kButton1Locate:
2324
2325 ExecuteEvent(kButton1Down, px, py);
2326
2327 while (true) {
2328 px = py = 0;
2329 event = gVirtualX->RequestLocator(1, 1, px, py);
2330
2332
2333 if (event != -1) { // button is released
2334 ExecuteEvent(kButton1Up, px, py);
2335 return;
2336 }
2337 }
2338
2339 case kButton2Down:
2340
2341 Pop();
2342 break;
2343
2344 }
2345}
2346
2347////////////////////////////////////////////////////////////////////////////////
2348/// Execute action corresponding to one event for a TAxis object
2349/// (called by TAxis::ExecuteEvent.)
2350/// This member function is called when an axis is clicked with the locator
2351///
2352/// The axis range is set between the position where the mouse is pressed
2353/// and the position where it is released.
2354///
2355/// If the mouse position is outside the current axis range when it is released
2356/// the axis is unzoomed with the corresponding proportions.
2357///
2358/// Note that the mouse does not need to be in the pad or even canvas
2359/// when it is released.
2360
2362{
2363 if (!IsEditable()) return;
2364 if (!axis) return;
2366
2367 TView *view = GetView();
2368 static Int_t axisNumber;
2369 static Double_t ratio1, ratio2;
2370 static Int_t px1old, py1old, px2old, py2old;
2371 Int_t nbd, inc, bin1, bin2, first, last;
2372 Double_t temp, xmin,xmax;
2373 Bool_t opaque = gPad->OpaqueMoving();
2374 bool resetAxisRange = false;
2375 static std::unique_ptr<TBox> zoombox;
2376 Double_t zbx1=0,zbx2=0,zby1=0,zby2=0;
2377
2378 // The CONT4 option, used to paint TH2, is a special case; it uses a 3D
2379 // drawing technique to paint a 2D plot.
2380 TString opt = axis->GetParent()->GetDrawOption();
2381 opt.ToLower();
2383 if (strstr(opt,"cont4")) {
2384 view = nullptr;
2385 kCont4 = kTRUE;
2386 }
2387
2388 switch (event) {
2389
2390 case kButton1Down:
2391 axisNumber = 1;
2392 if (!strcmp(axis->GetName(),"xaxis")) {
2393 axisNumber = 1;
2394 if (!IsVertical()) axisNumber = 2;
2395 }
2396 if (!strcmp(axis->GetName(),"yaxis")) {
2397 axisNumber = 2;
2398 if (!IsVertical()) axisNumber = 1;
2399 }
2400 if (!strcmp(axis->GetName(),"zaxis")) {
2401 axisNumber = 3;
2402 }
2403 if (view) {
2404 view->GetDistancetoAxis(axisNumber, px, py, ratio1);
2405 } else {
2406 if (axisNumber == 1) {
2407 ratio1 = (AbsPixeltoX(px) - GetUxmin())/(GetUxmax() - GetUxmin());
2410 px2old = px1old;
2412 } else if (axisNumber == 2) {
2413 ratio1 = (AbsPixeltoY(py) - GetUymin())/(GetUymax() - GetUymin());
2417 py2old = py1old;
2418 } else {
2419 ratio1 = (AbsPixeltoY(py) - GetUymin())/(GetUymax() - GetUymin());
2423 py2old = py1old;
2424 }
2425 if (!opaque) {
2427 } else {
2428 if (axisNumber == 1) {
2431 zby1 = GetUymin();
2432 zby2 = GetUymax();
2433 } else if (axisNumber == 2) {
2434 zbx1 = GetUxmin();
2435 zbx2 = GetUxmax();
2438 }
2439 if (GetLogx()) {
2440 zbx1 = TMath::Power(10,zbx1);
2441 zbx2 = TMath::Power(10,zbx2);
2442 }
2443 if (GetLogy()) {
2444 zby1 = TMath::Power(10,zby1);
2445 zby2 = TMath::Power(10,zby2);
2446 }
2447 zoombox = std::make_unique<TBox>(zbx1, zby1, zbx2, zby2);
2448 Int_t ci = TColor::GetColor("#7d7dff");
2449 TColor *zoomcolor = gROOT->GetColor(ci);
2450 if (!TCanvas::SupportAlpha() || !zoomcolor) zoombox->SetFillStyle(3002);
2451 else zoomcolor->SetAlpha(0.5);
2452 zoombox->SetFillColor(ci);
2453 zoombox->Draw();
2454 gPad->Modified();
2455 gPad->Update();
2456 }
2457 }
2458 if (!opaque) gVirtualX->SetLineColor(-1);
2459 // No break !!!
2460
2461 case kButton1Motion:
2462 if (view) {
2463 view->GetDistancetoAxis(axisNumber, px, py, ratio2);
2464 } else {
2466 if (axisNumber == 1) {
2467 ratio2 = (AbsPixeltoX(px) - GetUxmin())/(GetUxmax() - GetUxmin());
2469 } else {
2470 ratio2 = (AbsPixeltoY(py) - GetUymin())/(GetUymax() - GetUymin());
2472 }
2473 if (!opaque) {
2475 } else {
2476 if (axisNumber == 1) {
2479 zby1 = GetUymin();
2480 zby2 = GetUymax();
2481 } else if (axisNumber == 2) {
2482 zbx1 = GetUxmin();
2483 zbx2 = GetUxmax();
2486 }
2487 if (GetLogx()) {
2488 zbx1 = TMath::Power(10,zbx1);
2489 zbx2 = TMath::Power(10,zbx2);
2490 }
2491 if (GetLogy()) {
2492 zby1 = TMath::Power(10,zby1);
2493 zby2 = TMath::Power(10,zby2);
2494 }
2495 if (zoombox) {
2496 zoombox->SetX1(zbx1);
2497 zoombox->SetY1(zby1);
2498 zoombox->SetX2(zbx2);
2499 zoombox->SetY2(zby2);
2500 }
2501 gPad->Modified();
2502 gPad->Update();
2503 }
2504 }
2505 break;
2506
2507 case kWheelUp:
2508 nbd = (axis->GetLast()-axis->GetFirst());
2509 inc = TMath::Max(nbd/100,1);
2510 bin1 = axis->GetFirst()+inc;
2511 bin2 = axis->GetLast()-inc;
2512 bin1 = TMath::Max(bin1, 1);
2513 bin2 = TMath::Min(bin2, axis->GetNbins());
2514 if (bin2>bin1) {
2515 axis->SetRange(bin1,bin2);
2516 gPad->Modified();
2517 gPad->Update();
2518 }
2519 break;
2520
2521 case kWheelDown:
2522 nbd = (axis->GetLast()-axis->GetFirst());
2523 inc = TMath::Max(nbd/100,1);
2524 bin1 = axis->GetFirst()-inc;
2525 bin2 = axis->GetLast()+inc;
2526 bin1 = TMath::Max(bin1, 1);
2527 bin2 = TMath::Min(bin2, axis->GetNbins());
2528 resetAxisRange = (bin1 == 1 && axis->GetFirst() == 1 && bin2 == axis->GetNbins() && axis->GetLast() == axis->GetNbins());
2529 if (bin2>bin1) {
2530 axis->SetRange(bin1,bin2);
2531 }
2533 if (bin2>bin1) {
2534 gPad->Modified();
2535 gPad->Update();
2536 }
2537 break;
2538
2539 case kButton1Up:
2540 if (gROOT->IsEscaped()) {
2541 gROOT->SetEscape(kFALSE);
2542 if (opaque && zoombox)
2543 zoombox.reset();
2544 break;
2545 }
2546
2547 if (view) {
2548 view->GetDistancetoAxis(axisNumber, px, py, ratio2);
2549 if (ratio1 > ratio2) {
2550 temp = ratio1;
2551 ratio1 = ratio2;
2552 ratio2 = temp;
2553 }
2554 if (ratio2 - ratio1 > 0.05) {
2555 TH1 *hobj = (TH1*)axis->GetParent();
2556 if (axisNumber == 3 && hobj && hobj->GetDimension() != 3) {
2557 Float_t zmin = hobj->GetMinimum();
2558 Float_t zmax = hobj->GetMaximum();
2559 if(GetLogz()){
2560 if (zmin <= 0 && zmax > 0) zmin = TMath::Min((Double_t)1,
2561 (Double_t)0.001*zmax);
2562 zmin = TMath::Log10(zmin);
2563 zmax = TMath::Log10(zmax);
2564 }
2565 Float_t newmin = zmin + (zmax-zmin)*ratio1;
2566 Float_t newmax = zmin + (zmax-zmin)*ratio2;
2567 if (newmin < zmin) newmin = hobj->GetBinContent(hobj->GetMinimumBin());
2568 if (newmax > zmax) newmax = hobj->GetBinContent(hobj->GetMaximumBin());
2569 if (GetLogz()){
2570 newmin = TMath::Exp(2.302585092994*newmin);
2571 newmax = TMath::Exp(2.302585092994*newmax);
2572 }
2573 hobj->SetMinimum(newmin);
2574 hobj->SetMaximum(newmax);
2575 hobj->SetBit(TH1::kIsZoomed);
2576 } else {
2577 first = axis->GetFirst();
2578 last = axis->GetLast();
2579 bin1 = first + Int_t((last-first+1)*ratio1);
2580 bin2 = first + Int_t((last-first+1)*ratio2);
2581 bin1 = TMath::Max(bin1, 1);
2582 bin2 = TMath::Min(bin2, axis->GetNbins());
2583 axis->SetRange(bin1, bin2);
2584 }
2585 delete view;
2586 SetView(nullptr);
2587 Modified(kTRUE);
2588 }
2589 } else {
2590 if (axisNumber == 1) {
2591 ratio2 = (AbsPixeltoX(px) - GetUxmin())/(GetUxmax() - GetUxmin());
2592 xmin = GetUxmin() +ratio1*(GetUxmax() - GetUxmin());
2593 xmax = GetUxmin() +ratio2*(GetUxmax() - GetUxmin());
2594 if (GetLogx() && !kCont4) {
2595 xmin = PadtoX(xmin);
2596 xmax = PadtoX(xmax);
2597 }
2598 } else if (axisNumber == 2) {
2599 ratio2 = (AbsPixeltoY(py) - GetUymin())/(GetUymax() - GetUymin());
2600 xmin = GetUymin() +ratio1*(GetUymax() - GetUymin());
2601 xmax = GetUymin() +ratio2*(GetUymax() - GetUymin());
2602 if (GetLogy() && !kCont4) {
2603 xmin = PadtoY(xmin);
2604 xmax = PadtoY(xmax);
2605 }
2606 } else {
2607 ratio2 = (AbsPixeltoY(py) - GetUymin())/(GetUymax() - GetUymin());
2608 xmin = ratio1;
2609 xmax = ratio2;
2610 }
2611 if (xmin > xmax) {
2612 temp = xmin;
2613 xmin = xmax;
2614 xmax = temp;
2615 temp = ratio1;
2616 ratio1 = ratio2;
2617 ratio2 = temp;
2618 }
2619
2620 // xmin and xmax need to be adjusted in case of CONT4.
2621 if (kCont4) {
2622 Double_t low = axis->GetBinLowEdge(axis->GetFirst());
2623 Double_t up = axis->GetBinUpEdge(axis->GetLast());
2624 Double_t xmi = GetUxmin();
2625 Double_t xma = GetUxmax();
2626 xmin = ((xmin-xmi)/(xma-xmi))*(up-low)+low;
2627 xmax = ((xmax-xmi)/(xma-xmi))*(up-low)+low;
2628 }
2629
2630 if (!strcmp(axis->GetName(),"xaxis")) axisNumber = 1;
2631 if (!strcmp(axis->GetName(),"yaxis")) axisNumber = 2;
2632 if (ratio2 - ratio1 > 0.05) {
2633 //update object owning this axis
2634 TH1 *hobj1 = (TH1*)axis->GetParent();
2635 bin1 = axis->FindFixBin(xmin);
2636 bin2 = axis->FindFixBin(xmax);
2637 bin1 = TMath::Max(bin1, 1);
2638 bin2 = TMath::Min(bin2, axis->GetNbins());
2639 if (axisNumber == 1) axis->SetRange(bin1,bin2);
2640 if (axisNumber == 2 && hobj1) {
2641 if (hobj1->GetDimension() == 1) {
2642 if (hobj1->GetNormFactor() != 0) {
2643 Double_t norm = hobj1->GetSumOfWeights()/hobj1->GetNormFactor();
2644 xmin *= norm;
2645 xmax *= norm;
2646 }
2647 hobj1->SetMinimum(xmin);
2648 hobj1->SetMaximum(xmax);
2649 hobj1->SetBit(TH1::kIsZoomed);
2650 } else {
2651 axis->SetRange(bin1,bin2);
2652 }
2653 }
2654 //update all histograms in the pad
2655 TIter next(GetListOfPrimitives());
2656 TObject *obj;
2657 while ((obj= next())) {
2658 if (!obj->InheritsFrom(TH1::Class())) continue;
2659 TH1 *hobj = (TH1*)obj;
2660 if (hobj == hobj1) continue;
2661 bin1 = hobj->GetXaxis()->FindFixBin(xmin);
2662 bin2 = hobj->GetXaxis()->FindFixBin(xmax);
2663 if (axisNumber == 1) {
2664 hobj->GetXaxis()->SetRange(bin1,bin2);
2665 } else if (axisNumber == 2) {
2666 if (hobj->GetDimension() == 1) {
2669 if (hobj->GetNormFactor() != 0) {
2670 Double_t norm = hobj->GetSumOfWeights()/hobj->GetNormFactor();
2671 xxmin *= norm;
2672 xxmax *= norm;
2673 }
2674 hobj->SetMinimum(xxmin);
2675 hobj->SetMaximum(xxmax);
2676 hobj->SetBit(TH1::kIsZoomed);
2677 } else {
2678 bin1 = hobj->GetYaxis()->FindFixBin(xmin);
2679 bin2 = hobj->GetYaxis()->FindFixBin(xmax);
2680 hobj->GetYaxis()->SetRange(bin1,bin2);
2681 }
2682 }
2683 }
2684 Modified(kTRUE);
2685 }
2686 }
2687 if (!opaque) {
2688 gVirtualX->SetLineColor(-1);
2689 } else {
2690 if (zoombox) {
2691 zoombox.reset();
2692 gPad->Modified();
2693 gPad->Update();
2694 }
2695 }
2696 break;
2697 }
2698}
2699
2700////////////////////////////////////////////////////////////////////////////////
2701/// Search if object named name is inside this pad or in pads inside this pad.
2702///
2703/// In case name is in several sub-pads the first one is returned.
2704
2705TObject *TPad::FindObject(const char *name) const
2706{
2707 if (!fPrimitives) return nullptr;
2709 if (found) return found;
2710 TIter next(GetListOfPrimitives());
2711 while (auto cur = next()) {
2712 if (cur->InheritsFrom(TPad::Class())) {
2713 found = ((TPad*)cur)->FindObject(name);
2714 if (found) return found;
2715 }
2716 }
2717 return nullptr;
2718}
2719
2720////////////////////////////////////////////////////////////////////////////////
2721/// Search if obj is in pad or in pads inside this pad.
2722///
2723/// In case obj is in several sub-pads the first one is returned.
2724
2726{
2727 if (!fPrimitives) return nullptr;
2728 TObject *found = fPrimitives->FindObject(obj);
2729 if (found) return found;
2730 TIter next(GetListOfPrimitives());
2731 while (auto cur = next()) {
2732 if (cur->InheritsFrom(TPad::Class())) {
2733 found = ((TPad*)cur)->FindObject(obj);
2734 if (found) return found;
2735 }
2736 }
2737 return nullptr;
2738}
2739
2740////////////////////////////////////////////////////////////////////////////////
2741/// Get canvas identifier.
2742
2744{
2745 return fCanvas ? fCanvas->GetCanvasID() : -1;
2746}
2747
2748////////////////////////////////////////////////////////////////////////////////
2749/// Get canvas implementation pointer if any
2750
2752{
2753 return fCanvas ? fCanvas->GetCanvasImp() : nullptr;
2754}
2755
2756////////////////////////////////////////////////////////////////////////////////
2757/// Get Event.
2758
2760{
2761 return fCanvas ? fCanvas->GetEvent() : 0;
2762}
2763
2764////////////////////////////////////////////////////////////////////////////////
2765/// Get X event.
2766
2768{
2769 return fCanvas ? fCanvas->GetEventX() : 0;
2770}
2771
2772////////////////////////////////////////////////////////////////////////////////
2773/// Get Y event.
2774
2776{
2777 return fCanvas ? fCanvas->GetEventY() : 0;
2778}
2779
2780////////////////////////////////////////////////////////////////////////////////
2781/// Get virtual canvas.
2782
2784{
2785 return fCanvas ? (TVirtualPad*) fCanvas : nullptr;
2786}
2787
2788////////////////////////////////////////////////////////////////////////////////
2789/// Get highlight color.
2790
2792{
2793 return fCanvas ? fCanvas->GetHighLightColor() : 0;
2794}
2795
2796////////////////////////////////////////////////////////////////////////////////
2797/// Static function (see also TPad::SetMaxPickDistance)
2798
2803
2804////////////////////////////////////////////////////////////////////////////////
2805/// Get selected.
2806
2808{
2809 if (fCanvas == this) return nullptr;
2810 return fCanvas ? fCanvas->GetSelected() : nullptr;
2811}
2812
2813////////////////////////////////////////////////////////////////////////////////
2814/// Get selected pad.
2815
2817{
2818 if (fCanvas == this) return nullptr;
2819 return fCanvas ? fCanvas->GetSelectedPad() : nullptr;
2820}
2821
2822////////////////////////////////////////////////////////////////////////////////
2823/// Get save pad.
2824
2826{
2827 if (fCanvas == this) return nullptr;
2828 return fCanvas ? fCanvas->GetPadSave() : nullptr;
2829}
2830
2831////////////////////////////////////////////////////////////////////////////////
2832/// Get Wh.
2833
2835{
2836 return fCanvas ? fCanvas->GetWh() : 0;
2837}
2838
2839////////////////////////////////////////////////////////////////////////////////
2840/// Get Ww.
2841
2843{
2844 return fCanvas ? fCanvas->GetWw() : 0;
2845}
2846
2847////////////////////////////////////////////////////////////////////////////////
2848/// Hide tool tip depending on the event type. Typically tool tips
2849/// are hidden when event is not a kMouseEnter and not a kMouseMotion
2850/// event.
2851
2853{
2854 if (event != kMouseEnter && event != kMouseMotion && fTip)
2855 gPad->CloseToolTip(fTip);
2856}
2857
2858////////////////////////////////////////////////////////////////////////////////
2859/// Is pad in batch mode ?
2860
2862{
2863 return fCanvas ? fCanvas->IsBatch() : kFALSE;
2864}
2865
2866////////////////////////////////////////////////////////////////////////////////
2867/// Is pad retained ?
2868
2870{
2871 return fCanvas ? fCanvas->IsRetained() : kFALSE;
2872}
2873
2874////////////////////////////////////////////////////////////////////////////////
2875/// Is web ?
2877{
2878 return fCanvas ? fCanvas->IsWeb() : kFALSE;
2879}
2880
2881////////////////////////////////////////////////////////////////////////////////
2882/// Is pad moving in opaque mode ?
2883
2885{
2886 return fCanvas ? fCanvas->OpaqueMoving() : kFALSE;
2887}
2888
2889////////////////////////////////////////////////////////////////////////////////
2890/// Is pad resizing in opaque mode ?
2891
2893{
2894 return fCanvas ? fCanvas->OpaqueResizing() : kFALSE;
2895}
2896
2897////////////////////////////////////////////////////////////////////////////////
2898/// Set pad in batch mode.
2899
2901{
2903}
2904
2905////////////////////////////////////////////////////////////////////////////////
2906/// Set canvas size.
2907
2909{
2910 if (fCanvas) fCanvas->SetCanvasSize(ww,wh);
2911}
2912
2913////////////////////////////////////////////////////////////////////////////////
2914/// Set cursor type.
2915
2920
2921////////////////////////////////////////////////////////////////////////////////
2922/// Set double buffer mode ON or OFF.
2923
2928
2929////////////////////////////////////////////////////////////////////////////////
2930/// Set selected.
2931
2933{
2934 if (fCanvas) fCanvas->SetSelected(obj);
2935}
2936
2937////////////////////////////////////////////////////////////////////////////////
2938/// Update pad.
2939
2941{
2942 if (fCanvas) fCanvas->Update();
2943}
2944
2945////////////////////////////////////////////////////////////////////////////////
2946/// Asynchronous pad update.
2947/// In case of web-based canvas triggers update of the canvas on the client side,
2948/// but does not wait that real update is completed. Avoids blocking of caller thread.
2949/// Have to be used if called from other web-based widget to avoid logical dead-locks.
2950/// In case of normal canvas just canvas->Update() is performed.
2951
2953{
2954 if (fCanvas) fCanvas->UpdateAsync();
2955}
2956
2957////////////////////////////////////////////////////////////////////////////////
2958/// Get frame.
2959
2961{
2962 if (!fPrimitives) fPrimitives = new TList;
2964 if (!frame) frame = (TFrame*)GetListOfPrimitives()->FindObject("TFrame");
2965 fFrame = frame;
2966 if (!fFrame) {
2967 if (!frame) fFrame = new TFrame(0,0,1,1);
2977 } else {
2978 // Preexisting and now assigned to fFrame, let's make sure it is not
2979 // deleted twice (the bit might have been set in TPad::Streamer)
2981 }
2982 return fFrame;
2983}
2984
2985////////////////////////////////////////////////////////////////////////////////
2986/// Get primitive.
2987
2989{
2990 if (!fPrimitives) return nullptr;
2991 TIter next(fPrimitives);
2992 TObject *found, *obj;
2993 while ((obj=next())) {
2994 if (!strcmp(name, obj->GetName())) return obj;
2995 if (obj->InheritsFrom(TPad::Class())) continue;
2996 found = obj->FindObject(name);
2997 if (found) return found;
2998 }
2999 return nullptr;
3000}
3001
3002////////////////////////////////////////////////////////////////////////////////
3003/// Get a pointer to subpadnumber of this pad.
3004
3006{
3007 if (!subpadnumber) {
3008 return (TVirtualPad*)this;
3009 }
3010
3011 TObject *obj;
3012 if (!fPrimitives) return nullptr;
3013 TIter next(GetListOfPrimitives());
3014 while ((obj = next())) {
3015 if (obj->InheritsFrom(TVirtualPad::Class())) {
3016 TVirtualPad *pad = (TVirtualPad*)obj;
3017 if (pad->GetNumber() == subpadnumber) return pad;
3018 }
3019 }
3020 return nullptr;
3021}
3022
3023////////////////////////////////////////////////////////////////////////////////
3024/// Return lower and upper bounds of the pad in NDC coordinates.
3025
3026void TPad::GetPadPar(Double_t &xlow, Double_t &ylow, Double_t &xup, Double_t &yup)
3027{
3028 xlow = fXlowNDC;
3029 ylow = fYlowNDC;
3030 xup = fXlowNDC+fWNDC;
3031 yup = fYlowNDC+fHNDC;
3032}
3033
3034////////////////////////////////////////////////////////////////////////////////
3035/// Return pad world coordinates range.
3036
3038{
3039 x1 = fX1;
3040 y1 = fY1;
3041 x2 = fX2;
3042 y2 = fY2;
3043}
3044
3045////////////////////////////////////////////////////////////////////////////////
3046/// Return pad axis coordinates range.
3047
3049{
3050 xmin = fUxmin;
3051 ymin = fUymin;
3052 xmax = fUxmax;
3053 ymax = fUymax;
3054}
3055
3056////////////////////////////////////////////////////////////////////////////////
3057/// Highlight pad.
3058/// do not highlight when printing on Postscript
3059
3061{
3062 if (gVirtualPS && gVirtualPS->TestBit(kPrintingPS)) return;
3063
3064 if (color <= 0) return;
3065
3067
3068 // We do not want to have active(executable) buttons, etc highlighted
3069 // in this manner, unless we want to edit'em
3071 //When doing a DrawClone from the GUI you would do
3072 // - select an empty pad -
3073 // - right click on object -
3074 // - select DrawClone on menu -
3075 //
3076 // Without the SetSelectedPad(); in the HighLight function, the
3077 // above instruction lead to the clone to be drawn in the
3078 // same canvas as the original object. This is because the
3079 // 'right clicking' (via TCanvas::HandleInput) changes gPad
3080 // momentarily such that when DrawClone is called, it is
3081 // not the right value (for DrawClone). Should be FIXED.
3082 gROOT->SetSelectedPad(this);
3083 if (GetBorderMode()>0) {
3084 if (set) PaintBorder(-color, kFALSE);
3086 }
3087 }
3088
3090}
3091
3092////////////////////////////////////////////////////////////////////////////////
3093/// List all primitives in pad.
3094
3096{
3098 std::cout <<IsA()->GetName()<<" fXlowNDC=" <<fXlowNDC<<" fYlowNDC="<<fYlowNDC<<" fWNDC="<<GetWNDC()<<" fHNDC="<<GetHNDC()
3099 <<" Name= "<<GetName()<<" Title= "<<GetTitle()<<" Option="<<option<<std::endl;
3101 if (!fPrimitives) return;
3104}
3105
3106////////////////////////////////////////////////////////////////////////////////
3107/// Increment (i==1) or set (i>1) the number of autocolor in the pad.
3108
3110{
3111 if (opt.Index("pfc")>=0 || opt.Index("plc")>=0 || opt.Index("pmc")>=0) {
3112 if (i==1) fNumPaletteColor++;
3113 else fNumPaletteColor = i;
3114 return fNumPaletteColor;
3115 } else {
3116 return 0;
3117 }
3118}
3119
3120////////////////////////////////////////////////////////////////////////////////
3121/// Get the next autocolor in the pad.
3122
3124{
3125 Int_t i = 0;
3126 Int_t ncolors = gStyle->GetNumberOfColors();
3127 if (fNumPaletteColor>1) {
3128 i = fNextPaletteColor*(ncolors/(fNumPaletteColor-1));
3129 if (i>=ncolors) i = ncolors-1;
3130 }
3133 return gStyle->GetColorPalette(i);
3134}
3135
3136////////////////////////////////////////////////////////////////////////////////
3137/// Initialise the grid used to find empty space when adding a box (Legend) in a pad
3138
3140{
3141 Int_t const cellSize = 10; // Size of an individual grid cell in pixels.
3142
3143 fCGnx = GetWw()/cellSize;
3144 fCGny = GetWh()/cellSize;
3145
3146 // Initialise the collide grid
3147 fCollideGrid.resize(fCGnx*fCGny);
3148 for (int i = 0; i < fCGnx; i++)
3149 for (int j = 0; j < fCGny; j++)
3150 fCollideGrid[i + j * fCGnx] = kTRUE;
3151
3152 // Fill the collide grid
3153 TIter iter(GetListOfPrimitives());
3154
3155 while(auto o = iter()) {
3156 if (o == oi)
3157 continue;
3158 if (o->InheritsFrom(TFrame::Class()))
3160 else if (o->InheritsFrom(TBox::Class()))
3162 else if (o->InheritsFrom(TH1::Class()))
3164 else if (o->InheritsFrom(TGraph::Class()))
3166 else if (o->InheritsFrom(TMultiGraph::Class())) {
3167 TIter nextgraph(((TMultiGraph *)o)->GetListOfGraphs());
3168 while (auto og = nextgraph())
3170 } else if (o->InheritsFrom(THStack::Class())) {
3171 TIter nexthist(((THStack *)o)->GetHists());
3172 while (auto oh = nexthist()) {
3173 if (oh->InheritsFrom(TH1::Class()))
3175 }
3176 }
3177 }
3178}
3179
3180////////////////////////////////////////////////////////////////////////////////
3181/// Check if a box of size w and h collide some primitives in the pad at
3182/// position i,j
3183
3185{
3186 for (int r = i; r < w + i; r++) {
3187 for (int c = j; c < h + j; c++) {
3188 if (!fCollideGrid[r + c * fCGnx])
3189 return kTRUE;
3190 }
3191 }
3192 return kFALSE;
3193}
3194
3195////////////////////////////////////////////////////////////////////////////////
3196/// Place a box in NDC space
3197///
3198/// \return `true` if the box could be placed, `false` if not.
3199///
3200/// \param[in] o pointer to the box to be placed
3201/// \param[in] w box width to be placed
3202/// \param[in] h box height to be placed
3203/// \param[out] xl x position of the bottom left corner of the placed box
3204/// \param[out] yb y position of the bottom left corner of the placed box
3205/// \param[in] option l=left, r=right, t=top, b=bottom, w=within margins. Order determines
3206/// priority for placement. Default is "lb" (prioritises horizontal over vertical)
3207
3209{
3210 FillCollideGrid(o);
3211
3212 Int_t iw = (int)(fCGnx*w);
3213 Int_t ih = (int)(fCGny*h);
3214
3215 Int_t nxbeg = 0;
3216 Int_t nybeg = 0;
3217 Int_t nxend = fCGnx-iw-1;
3218 Int_t nyend = fCGny-ih-1;
3219 Int_t dx = 1;
3220 Int_t dy = 1;
3221
3222 bool isFirstVertical = false;
3223 bool isFirstHorizontal = false;
3224
3225 for (std::size_t i = 0; option[i] != '\0'; ++i) {
3226 char letter = std::tolower(option[i]);
3227 if (letter == 'w') {
3232 } else if (letter == 't' || letter == 'b') {
3234 // go from top to bottom instead of bottom to top
3235 dy = letter == 't' ? -1 : 1;
3236 } else if (letter == 'l' || letter == 'r') {
3238 // go from right to left instead of left to right
3239 dx = letter == 'r' ? -1 : 1;
3240 }
3241 }
3242
3243 if(dx < 0) std::swap(nxbeg, nxend);
3244 if(dy < 0) std::swap(nybeg, nyend);
3245
3246 auto attemptPlacement = [&](Int_t i, Int_t j) {
3247 if (Collide(i, j, iw, ih)) {
3248 return false;
3249 } else {
3250 xl = (Double_t)(i) / (Double_t)(fCGnx);
3251 yb = (Double_t)(j) / (Double_t)(fCGny);
3252 return true;
3253 }
3254 };
3255
3256 if(!isFirstVertical) {
3257 for (Int_t i = nxbeg; i != nxend; i += dx) {
3258 for (Int_t j = nybeg; j != nyend; j += dy) {
3259 if (attemptPlacement(i, j)) return true;
3260 }
3261 }
3262 } else {
3263 // prioritizing vertical over horizontal
3264 for (Int_t j = nybeg; j != nyend; j += dy) {
3265 for (Int_t i = nxbeg; i != nxend; i += dx) {
3266 if (attemptPlacement(i, j)) return true;
3267 }
3268 }
3269 }
3270
3271 return kFALSE;
3272}
3273
3274#define NotFree(i, j) fCollideGrid[TMath::Max(TMath::Min(i+j*fCGnx,fCGnx*fCGny),0)] = kFALSE;
3275
3276////////////////////////////////////////////////////////////////////////////////
3277/// Mark as "not free" the cells along a line.
3278
3280{
3281 NotFree(x1, y1);
3282 NotFree(x2, y2);
3283 Int_t i, j, xt, yt;
3284
3285 // horizontal lines
3286 if (y1==y2) {
3287 for (i=x1+1; i<x2; i++) NotFree(i,y1);
3288 return;
3289 }
3290
3291 // vertical lines
3292 if (x1==x2) {
3293 for (i=y1+1; i<y2; i++) NotFree(x1,i);
3294 return;
3295 }
3296
3297 // other lines
3298 if (TMath::Abs(x2-x1)>TMath::Abs(y2-y1)) {
3299 if (x1>x2) {
3300 xt = x1; x1 = x2; x2 = xt;
3301 yt = y1; y1 = y2; y2 = yt;
3302 }
3303 for (i=x1+1; i<x2; i++) {
3304 j = (Int_t)((Double_t)(y2-y1)*(Double_t)((i-x1)/(Double_t)(x2-x1))+y1);
3305 NotFree(i,j);
3306 NotFree(i,(j+1));
3307 }
3308 } else {
3309 if (y1>y2) {
3310 yt = y1; y1 = y2; y2 = yt;
3311 xt = x1; x1 = x2; x2 = xt;
3312 }
3313 for (j=y1+1; j<y2; j++) {
3314 i = (Int_t)((Double_t)(x2-x1)*(Double_t)((j-y1)/(Double_t)(y2-y1))+x1);
3315 NotFree(i,j);
3316 NotFree((i+1),j);
3317 }
3318 }
3319}
3320
3321////////////////////////////////////////////////////////////////////////////////
3323{
3324 TBox *b = (TBox *)o;
3325 if (fCGnx==0||fCGny==0) return;
3326 Double_t xs = (fX2-fX1)/fCGnx;
3327 Double_t ys = (fY2-fY1)/fCGny;
3328
3329 Int_t x1 = (Int_t)((b->GetX1()-fX1)/xs);
3330 Int_t x2 = (Int_t)((b->GetX2()-fX1)/xs);
3331 Int_t y1 = (Int_t)((b->GetY1()-fY1)/ys);
3332 Int_t y2 = (Int_t)((b->GetY2()-fY1)/ys);
3333 for (int i = x1; i<=x2; i++) {
3334 for (int j = y1; j<=y2; j++) NotFree(i, j);
3335 }
3336}
3337
3338////////////////////////////////////////////////////////////////////////////////
3340{
3341 TFrame *f = (TFrame *)o;
3342 if (fCGnx==0||fCGny==0) return;
3343 Double_t xs = (fX2-fX1)/fCGnx;
3344 Double_t ys = (fY2-fY1)/fCGny;
3345
3346 Int_t x1 = (Int_t)((f->GetX1()-fX1)/xs);
3347 Int_t x2 = (Int_t)((f->GetX2()-fX1)/xs);
3348 Int_t y1 = (Int_t)((f->GetY1()-fY1)/ys);
3349 Int_t y2 = (Int_t)((f->GetY2()-fY1)/ys);
3350 Int_t i;
3351
3352 for (i = x1; i<=x2; i++) {
3353 NotFree(i, y1);
3354 NotFree(i, (y1-1));
3355 NotFree(i, (y1-2));
3356 }
3357 for (i = y1; i<=y2; i++) {
3358 NotFree(x1, i);
3359 NotFree((x1-1), i);
3360 NotFree((x1-2), i);
3361 }
3362}
3363
3364////////////////////////////////////////////////////////////////////////////////
3366{
3367 TGraph *g = (TGraph *)o;
3368 if (fCGnx==0||fCGny==0) return;
3369 Double_t xs = (fX2-fX1)/fCGnx;
3370 Double_t ys = (fY2-fY1)/fCGny;
3371
3372 Int_t n = g->GetN();
3373 Int_t s = TMath::Max(n/10,1);
3374 Double_t x1, x2, y1, y2;
3375 for (Int_t i=s; i<n; i=i+s) {
3376 g->GetPoint(TMath::Max(0,i-s),x1,y1);
3377 g->GetPoint(i ,x2,y2);
3378 if (fLogx) {
3379 if (x1 > 0) x1 = TMath::Log10(x1);
3380 else x1 = fUxmin;
3381 if (x2 > 0) x2 = TMath::Log10(x2);
3382 else x2 = fUxmin;
3383 }
3384 if (fLogy) {
3385 if (y1 > 0) y1 = TMath::Log10(y1);
3386 else y1 = fUymin;
3387 if (y2 > 0) y2 = TMath::Log10(y2);
3388 else y2 = fUymin;
3389 }
3390 LineNotFree((int)((x1-fX1)/xs), (int)((x2-fX1)/xs),
3391 (int)((y1-fY1)/ys), (int)((y2-fY1)/ys));
3392 }
3393}
3394
3395////////////////////////////////////////////////////////////////////////////////
3397{
3398 TH1 *h = (TH1 *)o;
3399 if (fCGnx==0||fCGny==0) return;
3400 if (o->InheritsFrom(TH2::Class())) return;
3401 if (o->InheritsFrom(TH3::Class())) return;
3402
3403 TString name = h->GetName();
3404 if (name.Index("hframe") >= 0) return;
3405
3406 Double_t xs = (fX2-fX1)/fCGnx;
3407 Double_t ys = (fY2-fY1)/fCGny;
3408
3409 bool haserrors = false;
3410 TString drawOption = h->GetDrawOption();
3411 drawOption.ToLower();
3412 drawOption.ReplaceAll("same","");
3413
3414 if (drawOption.Index("hist") < 0) {
3415 if (drawOption.Index("e") >= 0) haserrors = true;
3416 }
3417
3418 Int_t nx = h->GetNbinsX();
3419 Int_t x1, y1, y2;
3420 Int_t i, j;
3421 Double_t x1l, y1l, y2l;
3422
3423 for (i = 1; i<nx; i++) {
3424 if (haserrors) {
3425 x1l = h->GetBinCenter(i);
3426 if (fLogx) {
3427 if (x1l > 0) x1l = TMath::Log10(x1l);
3428 else x1l = fUxmin;
3429 }
3430 x1 = (Int_t)((x1l-fX1)/xs);
3431 y1l = h->GetBinContent(i)-h->GetBinErrorLow(i);
3432 if (fLogy) {
3433 if (y1l > 0) y1l = TMath::Log10(y1l);
3434 else y1l = fUymin;
3435 }
3436 y1 = (Int_t)((y1l-fY1)/ys);
3437 y2l = h->GetBinContent(i)+h->GetBinErrorUp(i);
3438 if (fLogy) {
3439 if (y2l > 0) y2l = TMath::Log10(y2l);
3440 else y2l = fUymin;
3441 }
3442 y2 = (Int_t)((y2l-fY1)/ys);
3443 for (j=y1; j<y2; j++) {
3444 NotFree(x1, j);
3445 }
3446 }
3447 x1l = h->GetBinLowEdge(i);
3448 if (fLogx) {
3449 if (x1l > 0) x1l = TMath::Log10(x1l);
3450 else x1l = fUxmin;
3451 }
3452 x1 = (Int_t)((x1l-fX1)/xs);
3453 y1l = h->GetBinContent(i);
3454 if (fLogy) {
3455 if (y1l > 0) y1l = TMath::Log10(y1l);
3456 else y1l = fUymin;
3457 }
3458 y1 = (Int_t)((y1l-fY1)/ys);
3459 NotFree(x1, y1);
3460 x1l = h->GetBinLowEdge(i)+h->GetBinWidth(i);
3461 if (fLogx) {
3462 if (x1l > 0) x1l = TMath::Log10(x1l);
3463 else x1l = fUxmin;
3464 }
3465 x1 = (int)((x1l-fX1)/xs);
3466 NotFree(x1, y1);
3467 }
3468
3469 // Extra objects in the list of function
3470 TPaveStats *ps = (TPaveStats*)h->GetListOfFunctions()->FindObject("stats");
3471 if (ps) FillCollideGridTBox(ps);
3472}
3473
3474////////////////////////////////////////////////////////////////////////////////
3475/// This method draws the collide grid on top of the canvas. This is used for
3476/// debugging only. At some point it will be removed.
3477
3479{
3480 if (fCGnx==0||fCGny==0) return;
3481
3482 TContext ctxt(this, kTRUE);
3483
3484 TBox box;
3485 box.SetFillColorAlpha(kRed,0.5);
3486
3487 Double_t xs = (fX2-fX1)/fCGnx;
3488 Double_t ys = (fY2-fY1)/fCGny;
3489
3490 Double_t X1L, X2L, Y1L, Y2L;
3491 Double_t t = 0.15;
3492 Double_t Y1, Y2;
3493 Double_t X1 = fX1;
3494 Double_t X2 = X1+xs;
3495
3496 for (int i = 0; i<fCGnx; i++) {
3497 Y1 = fY1;
3498 Y2 = Y1+ys;
3499 for (int j = 0; j<fCGny; j++) {
3500 if (GetLogx()) {
3501 X1L = TMath::Power(10,X1);
3502 X2L = TMath::Power(10,X2);
3503 } else {
3504 X1L = X1;
3505 X2L = X2;
3506 }
3507 if (GetLogy()) {
3508 Y1L = TMath::Power(10,Y1);
3509 Y2L = TMath::Power(10,Y2);
3510 } else {
3511 Y1L = Y1;
3512 Y2L = Y2;
3513 }
3514 if (!fCollideGrid[i + j*fCGnx]) {
3515 box.SetFillColorAlpha(kBlack,t);
3516 box.DrawBox(X1L, Y1L, X2L, Y2L);
3517 } else {
3518 box.SetFillColorAlpha(kRed,t);
3519 box.DrawBox(X1L, Y1L, X2L, Y2L);
3520 }
3521 Y1 = Y2;
3522 Y2 = Y1+ys;
3523 if (t==0.15) t = 0.1;
3524 else t = 0.15;
3525 }
3526 X1 = X2;
3527 X2 = X1+xs;
3528 }
3529}
3530
3531////////////////////////////////////////////////////////////////////////////////
3532/// Short cut to call Modified() and Update() in a single call.
3533/// On Mac with Cocoa, it performs an additional ProcessEvents().
3534
3536{
3537 Modified();
3538 Update();
3539#ifdef R__HAS_COCOA
3541#endif
3542}
3543
3544////////////////////////////////////////////////////////////////////////////////
3545/// Convert x from pad to X.
3546
3548{
3549 if (fLogx && x < 50) return Double_t(TMath::Exp(2.302585092994*x));
3550 return x;
3551}
3552
3553////////////////////////////////////////////////////////////////////////////////
3554/// Convert y from pad to Y.
3555
3557{
3558 if (fLogy && y < 50) return Double_t(TMath::Exp(2.302585092994*y));
3559 return y;
3560}
3561
3562////////////////////////////////////////////////////////////////////////////////
3563/// Convert x from X to pad.
3564
3566{
3567 if (fLogx) {
3568 if (x > 0) x = TMath::Log10(x);
3569 else x = fUxmin;
3570 }
3571 return x;
3572}
3573
3574////////////////////////////////////////////////////////////////////////////////
3575/// Convert y from Y to pad.
3576
3578{
3579 if (fLogy) {
3580 if (y > 0) y = TMath::Log10(y);
3581 else y = fUymin;
3582 }
3583 return y;
3584}
3585
3586////////////////////////////////////////////////////////////////////////////////
3587/// Paint all primitives in pad.
3588
3589void TPad::Paint(Option_t * /*option*/)
3590{
3591 if (!fPrimitives)
3592 fPrimitives = new TList;
3594 fViewer3D->PadPaint(this);
3596 if (GetGLDevice()!=-1 && gVirtualPS) {
3597 TContext ctxt(this, kFALSE);
3598 if (gGLManager) gGLManager->PrintViewer(GetViewer3D());
3599 }
3600 return;
3601 }
3602
3604
3606 fPadPaint = 1;
3607
3608 {
3609 TContext ctxt(this, kTRUE);
3610
3612 PaintDate();
3613
3614 auto lnk = GetListOfPrimitives()->FirstLink();
3615
3616 while (lnk) {
3617 TObject *obj = lnk->GetObject();
3618
3619 // Create a pad 3D viewer if none exists and we encounter a 3D shape
3620 if (!fViewer3D && obj->InheritsFrom(TAtt3D::Class())) {
3621 GetViewer3D("pad");
3622 }
3623
3624 // Open a 3D scene if required
3625 if (fViewer3D && !fViewer3D->BuildingScene()) {
3628 }
3629
3630 obj->Paint(lnk->GetOption());
3631 lnk = lnk->Next();
3632 }
3633 }
3634
3635 fPadPaint = 0;
3637
3638 // Close the 3D scene if we opened it. This must be done after modified
3639 // flag is cleared, as some viewers will invoke another paint by marking pad modified again
3640 if (began3DScene) {
3642 }
3643}
3644
3645////////////////////////////////////////////////////////////////////////////////
3646/// Paint the pad border.
3647/// Draw first a box as a normal filled box
3648
3650{
3651 if (color >= 0) {
3652 TAttLine::Modify(); //Change line attributes only if necessary
3653 TAttFill::Modify(); //Change fill area attributes only if necessary
3654
3655 //With Cocoa we have a transparency. But we also have
3656 //pixmaps, and if you just paint a new content over the old one
3657 //with alpha < 1., you'll be able to see the old content.
3658 if (!gROOT->IsBatch() && gVirtualX->InheritsFrom("TGCocoa") && GetPainter())
3660
3662 }
3663 if (color < 0) color = -color;
3664 // then paint 3d frame (depending on bordermode)
3665 if (IsTransparent()) return;
3666 // Paint a 3D frame around the pad.
3667
3668 if (fBorderMode == 0) return;
3670 if (bordersize <= 0) bordersize = 2;
3671
3672 const Double_t realBsX = bordersize / (GetAbsWNDC() * GetWw()) * (fX2 - fX1);
3673 const Double_t realBsY = bordersize / (GetAbsHNDC() * GetWh()) * (fY2 - fY1);
3674
3675 Short_t px1,py1,px2,py2;
3676 Double_t xl, xt, yl, yt;
3677
3678 // GetColorDark() and GetColorBright() use GetFillColor()
3680 SetFillColor(color);
3682 Color_t light = 0, dark = 0;
3683 if (color != 0) {
3685 dark = TColor::GetColorDark(color);
3686 }
3687
3688 // Compute real left bottom & top right of the box in pixels
3689 px1 = XtoPixel(fX1); py1 = YtoPixel(fY1);
3690 px2 = XtoPixel(fX2); py2 = YtoPixel(fY2);
3691 if (px1 < px2) {xl = fX1; xt = fX2; }
3692 else {xl = fX2; xt = fX1;}
3693 if (py1 > py2) {yl = fY1; yt = fY2;}
3694 else {yl = fY2; yt = fY1;}
3695
3696 Double_t frameXs[7] = {}, frameYs[7] = {};
3697
3698 if (!IsBatch() && GetPainter()) {
3699 // Draw top&left part of the box
3700 frameXs[0] = xl; frameYs[0] = yl;
3701 frameXs[1] = xl + realBsX; frameYs[1] = yl + realBsY;
3702 frameXs[2] = frameXs[1]; frameYs[2] = yt - realBsY;
3703 frameXs[3] = xt - realBsX; frameYs[3] = frameYs[2];
3704 frameXs[4] = xt; frameYs[4] = yt;
3705 frameXs[5] = xl; frameYs[5] = yt;
3706 frameXs[6] = xl; frameYs[6] = yl;
3707
3708 if (fBorderMode == -1) GetPainter()->SetFillColor(dark);
3709 else GetPainter()->SetFillColor(light);
3711
3712 // Draw bottom&right part of the box
3713 frameXs[0] = xl; frameYs[0] = yl;
3714 frameXs[1] = xl + realBsX; frameYs[1] = yl + realBsY;
3715 frameXs[2] = xt - realBsX; frameYs[2] = frameYs[1];
3716 frameXs[3] = frameXs[2]; frameYs[3] = yt - realBsY;
3717 frameXs[4] = xt; frameYs[4] = yt;
3718 frameXs[5] = xt; frameYs[5] = yl;
3719 frameXs[6] = xl; frameYs[6] = yl;
3720
3722 else GetPainter()->SetFillColor(dark);
3724
3725 // If this pad is a button, highlight it
3726 if (InheritsFrom(TButton::Class()) && fBorderMode == -1) {
3727 if (TestBit(kFraming)) { // bit set in TButton::SetFraming
3728 if (GetFillColor() != 2) GetPainter()->SetLineColor(2);
3729 else GetPainter()->SetLineColor(4);
3731 }
3732 }
3733 GetPainter()->SetFillColor(-1);
3735 }
3736
3737 if (!tops) return;
3738
3740}
3741
3742////////////////////////////////////////////////////////////////////////////////
3743/// Paint a frame border with Postscript.
3744
3750
3751////////////////////////////////////////////////////////////////////////////////
3752/// Paint the current date and time if the option `Date` is set on via `gStyle->SetOptDate()`
3753/// Paint the current file name if the option `File` is set on via `gStyle->SetOptFile()`
3754
3756{
3757 if (fCanvas == this) {
3758 if (gStyle->GetOptDate()) {
3759 TDatime dt;
3760 const char *dates;
3761 char iso[16];
3762 if (gStyle->GetOptDate() < 10) {
3763 //by default use format like "Wed Sep 25 17:10:35 2002"
3764 dates = dt.AsString();
3765 } else if (gStyle->GetOptDate() < 20) {
3766 //use ISO format like 2002-09-25
3767 strlcpy(iso,dt.AsSQLString(),16);
3768 dates = iso;
3769 } else {
3770 //use ISO format like 2002-09-25 17:10:35
3771 dates = dt.AsSQLString();
3772 }
3774 tdate.SetTextSize( gStyle->GetAttDate()->GetTextSize());
3775 tdate.SetTextFont( gStyle->GetAttDate()->GetTextFont());
3776 tdate.SetTextColor(gStyle->GetAttDate()->GetTextColor());
3777 tdate.SetTextAlign(gStyle->GetAttDate()->GetTextAlign());
3778 tdate.SetTextAngle(gStyle->GetAttDate()->GetTextAngle());
3779 tdate.SetNDC();
3780 tdate.Paint();
3781 }
3782 if (gStyle->GetOptFile() && gFile) {
3783 TText tfile(1. - gStyle->GetDateX(),gStyle->GetDateY(),gFile->GetName());
3787 tfile.SetTextAlign(31);
3789 tfile.SetNDC();
3790 tfile.Paint();
3791 }
3792 }
3793}
3794
3795////////////////////////////////////////////////////////////////////////////////
3796/// Paint histogram/graph frame.
3797
3799{
3800 if (!fPrimitives) fPrimitives = new TList;
3802 TFrame *frame = GetFrame();
3803 frame->SetX1(xmin);
3804 frame->SetX2(xmax);
3805 frame->SetY1(ymin);
3806 frame->SetY2(ymax);
3807 if (!glist->FindObject(fFrame)) {
3808 glist->AddFirst(frame);
3810 }
3811 frame->Paint();
3812}
3813
3814////////////////////////////////////////////////////////////////////////////////
3815/// Traverse pad hierarchy and (re)paint only modified pads.
3816
3818{
3820 if (IsModified()) {
3821 fViewer3D->PadPaint(this);
3823 }
3825 auto lnk = pList ? pList->FirstLink() : nullptr;
3826 while (lnk) {
3827 auto obj = lnk->GetObject();
3828 if (obj->InheritsFrom(TPad::Class()))
3829 ((TPad*)obj)->PaintModified();
3830 lnk = lnk->Next();
3831 }
3832 return;
3833 }
3834
3836
3838 if (gVirtualPS) {
3840 gVirtualPS = nullptr;
3841 }
3842
3844 fPadPaint = 1;
3845 {
3846 TContext ctxt(this, kTRUE);
3847 if (IsModified() || IsTransparent()) {
3848 if ((fFillStyle < 3026) && (fFillStyle > 3000)) {
3849 if (!gPad->IsBatch() && GetPainter()) GetPainter()->ClearDrawable();
3850 }
3852 }
3853
3854 PaintDate();
3855
3857 auto lnk = pList ? pList->FirstLink() : nullptr;
3858
3859 while (lnk) {
3860 TObject *obj = lnk->GetObject();
3861 if (obj->InheritsFrom(TPad::Class())) {
3862 ((TPad*)obj)->PaintModified();
3863 } else if (IsModified() || IsTransparent()) {
3864
3865 // Create a pad 3D viewer if none exists and we encounter a
3866 // 3D shape
3867 if (!fViewer3D && obj->InheritsFrom(TAtt3D::Class())) {
3868 GetViewer3D("pad");
3869 }
3870
3871 // Open a 3D scene if required
3872 if (fViewer3D && !fViewer3D->BuildingScene()) {
3875 }
3876
3877 obj->Paint(lnk->GetOption());
3878 }
3879 lnk = lnk->Next();
3880 }
3881 }
3882
3883 fPadPaint = 0;
3885
3886 // This must be done after modified flag is cleared, as some
3887 // viewers will invoke another paint by marking pad modified again
3888 if (began3DScene) {
3890 }
3891
3893}
3894
3895////////////////////////////////////////////////////////////////////////////////
3896/// Paint box in CurrentPad World coordinates.
3897///
3898/// - if option[0] = 's' the box is forced to be paint with style=0
3899/// - if option[0] = 'l' the box contour is drawn
3900
3902{
3903 if (!gPad->IsBatch() && GetPainter()) {
3905 Int_t style = style0;
3906 if (option[0] == 's') {
3908 style = 0;
3909 }
3910 if (style) {
3911 if (style > 3000 && style < 4000) {
3912 if (style < 3026) {
3913 // draw stipples with fFillColor foreground
3915 }
3916
3917 if (style >= 3100 && style < 4000) {
3918 Double_t xb[4], yb[4];
3919 xb[0] = x1; xb[1] = x1; xb[2] = x2; xb[3] = x2;
3920 yb[0] = y1; yb[1] = y2; yb[2] = y2; yb[3] = y1;
3922 return;
3923 }
3924 //special case for TAttFillCanvas
3925 if (GetPainter()->GetFillColor() == 10) {
3928 GetPainter()->SetFillColor(10);
3929 }
3930 } else if (style >= 4000 && style <= 4100) {
3931 // For style >=4000 we make the window transparent.
3932 // From 4000 to 4100 the window is 100% transparent to 100% opaque
3933
3934 //ignore this style option when this is the canvas itself
3935 if (this == fMother) {
3936 //It's clear, that virtual X checks a style (4000) and will render a hollow rect!
3938 if (gVirtualX->InheritsFrom("TGCocoa"))
3939 GetPainter()->SetFillStyle(1000);
3941 if (gVirtualX->InheritsFrom("TGCocoa"))
3943 } else {
3944 //draw background by blitting all bottom pads
3945 int px, py;
3946 XYtoAbsPixel(fX1, fY2, px, py);
3947
3948 if (fMother) {
3950 CopyBackgroundPixmaps(fMother, this, px, py);
3951 }
3952
3953 GetPainter()->SetOpacity(style - 4000);
3954 }
3955 } else if (style >= 1000 && style <= 1999) {
3957 } else {
3959 }
3961 } else {
3963 if (option[0] == 's') GetPainter()->SetFillStyle(style0);
3964 }
3965 }
3966
3967 if (gVirtualPS) {
3969 if (option[0] == 's') {
3971 } else {
3972 if (style0 >= 3100 && style0 < 4000) {
3973 Double_t xb[4], yb[4];
3974 xb[0] = x1; xb[1] = x1; xb[2] = x2; xb[3] = x2;
3975 yb[0] = y1; yb[1] = y2; yb[2] = y2; yb[3] = y1;
3977 return;
3978 }
3979 }
3980 gVirtualPS->DrawBox(x1, y1, x2, y2);
3981 if (option[0] == 'l') {
3983 gVirtualPS->DrawBox(x1, y1, x2, y2);
3984 }
3985 if (option[0] == 's' || option[0] == 'l') gVirtualPS->SetFillStyle(style0);
3986 }
3987
3988 Modified();
3989}
3990
3991////////////////////////////////////////////////////////////////////////////////
3992/// Copy pixmaps of pads laying below pad "stop" into pad "stop". This
3993/// gives the effect of pad "stop" being transparent.
3994
3996{
3997 if (!start) return;
3998 TObject *obj;
3999 if (!fPrimitives) fPrimitives = new TList;
4000 TIter next(start->GetListOfPrimitives());
4001 while ((obj = next())) {
4002 if (obj->InheritsFrom(TPad::Class())) {
4003 if (obj == stop) break;
4004 ((TPad*)obj)->CopyBackgroundPixmap(x, y);
4005 ((TPad*)obj)->CopyBackgroundPixmaps((TPad*)obj, stop, x, y);
4006 }
4007 }
4008}
4009
4010////////////////////////////////////////////////////////////////////////////////
4011/// Copy pixmap of this pad as background of the current pad.
4012
4014{
4015 int px, py;
4016 XYtoAbsPixel(fX1, fY2, px, py);
4017 if (GetPainter()) GetPainter()->CopyDrawable(GetPixmapID(), px-x, py-y);
4018}
4019
4020////////////////////////////////////////////////////////////////////////////////
4021
4023{
4024 Warning("TPad::PaintFillArea", "Float_t signature is obsolete. Use Double_t signature.");
4025}
4026
4027////////////////////////////////////////////////////////////////////////////////
4028/// Paint fill area in CurrentPad World coordinates.
4029
4031{
4032 if (nn <3) return;
4033 Int_t n=0;
4037 } else {
4038 xmin = fX1; ymin = fY1; xmax = fX2; ymax = fY2;
4039 }
4040
4041 Int_t nc = 2*nn+1;
4042 std::vector<Double_t> x(nc, 0.);
4043 std::vector<Double_t> y(nc, 0.);
4044
4045 n = ClipPolygon(nn, xx, yy, nc, &x.front(), &y.front(),xmin,ymin,xmax,ymax);
4046 if (!n)
4047 return;
4048
4049 // Paint the fill area with hatches
4051 if (gPad->IsBatch() && GetPainter() && gVirtualPS) fillstyle = gVirtualPS->GetFillStyle();
4052 if (fillstyle >= 3100 && fillstyle < 4000) {
4053 PaintFillAreaHatches(nn, &x.front(), &y.front(), fillstyle);
4054 return;
4055 }
4056
4057 if (!gPad->IsBatch() && GetPainter())
4058 // invoke the graphics subsystem
4059 GetPainter()->DrawFillArea(n, &x.front(), &y.front());
4060
4061 if (gVirtualPS)
4062 gVirtualPS->DrawPS(-n, &x.front(), &y.front());
4063
4064 Modified();
4065}
4066
4067////////////////////////////////////////////////////////////////////////////////
4068/// Paint fill area in CurrentPad NDC coordinates.
4069
4071{
4072 std::vector<Double_t> xw(n), yw(n);
4073 for (int i=0; i<n; i++) {
4074 xw[i] = fX1 + x[i]*(fX2 - fX1);
4075 yw[i] = fY1 + y[i]*(fY2 - fY1);
4076 }
4077 PaintFillArea(n, xw.data(), yw.data(), option);
4078}
4079
4080////////////////////////////////////////////////////////////////////////////////
4081/// This function paints hatched fill area according to the FillStyle value
4082/// The convention for the Hatch is the following:
4083///
4084/// `FillStyle = 3ijk`
4085///
4086/// - i (1-9) : specify the space between each hatch
4087/// 1 = minimum 9 = maximum
4088/// the final spacing is i*GetHatchesSpacing(). The hatches spacing
4089/// is set by SetHatchesSpacing()
4090/// - j (0-9) : specify angle between 0 and 90 degrees
4091/// * 0 = 0
4092/// * 1 = 10
4093/// * 2 = 20
4094/// * 3 = 30
4095/// * 4 = 45
4096/// * 5 = Not drawn
4097/// * 6 = 60
4098/// * 7 = 70
4099/// * 8 = 80
4100/// * 9 = 90
4101/// - k (0-9) : specify angle between 90 and 180 degrees
4102/// * 0 = 180
4103/// * 1 = 170
4104/// * 2 = 160
4105/// * 3 = 150
4106/// * 4 = 135
4107/// * 5 = Not drawn
4108/// * 6 = 120
4109/// * 7 = 110
4110/// * 8 = 100
4111/// * 9 = 90
4112
4114{
4115 static Double_t ang1[10] = { 0., 10., 20., 30., 45.,5., 60., 70., 80., 89.99};
4116 static Double_t ang2[10] = {180.,170.,160.,150.,135.,5.,120.,110.,100., 89.99};
4117
4118 Int_t fasi = FillStyle % 1000;
4119 Int_t idSPA = fasi / 100;
4120 Int_t iAng2 = (fasi - 100 * idSPA) / 10;
4121 Int_t iAng1 = fasi % 10;
4122 Double_t dy = 0.003 * idSPA * gStyle->GetHatchesSpacing();
4124 Int_t lss = 0, lss2 = 0, lcs = 0, lcs2 = 0;
4125
4126 // Save the current line attributes and change to draw hatches
4127 if (!gPad->IsBatch() && GetPainter()) {
4134 }
4135 if (gVirtualPS) {
4142 }
4143
4144 // Draw the hatches
4145 if (ang1[iAng1] != 5.) PaintHatches(dy, ang1[iAng1], nn, xx, yy);
4146 if (ang2[iAng2] != 5.) PaintHatches(dy, ang2[iAng2], nn, xx, yy);
4147
4148 // Restore the line attributes
4149 if (!gPad->IsBatch() && GetPainter()) {
4153 }
4154 if (gVirtualPS) {
4158 }
4159}
4160
4161////////////////////////////////////////////////////////////////////////////////
4162/// This routine draw hatches inclined with the
4163/// angle "angle" and spaced of "dy" in normalized device
4164/// coordinates in the surface defined by n,xx,yy.
4165
4167 Int_t nn, Double_t *xx, Double_t *yy)
4168{
4169 Int_t i, i1, i2, nbi, m, inv;
4171 const Double_t angr = TMath::Pi()*(180.-angle)/180.;
4172 const Double_t epsil = 0.0001;
4173 const Int_t maxnbi = 100;
4174 Double_t xli[maxnbi], xlh[2], ylh[2], xt1, xt2, yt1, yt2;
4175 Double_t ll, x, y, x1, x2, y1, y2, a, b, xi, xip, xin, yi, yip;
4176
4177 Double_t rwxmin = gPad->GetX1();
4178 Double_t rwxmax = gPad->GetX2();
4179 Double_t rwymin = gPad->GetY1();
4180 Double_t rwymax = gPad->GetY2();
4181 ratiox = 1./(rwxmax-rwxmin);
4182 ratioy = 1./(rwymax-rwymin);
4183
4186 if (TMath::Abs(cosa) <= epsil) cosa=0.;
4187 if (TMath::Abs(sina) <= epsil) sina=0.;
4188 sinb = -sina;
4189 cosb = cosa;
4190
4191 // Values needed to compute the hatches in TRUE normalized space (NDC)
4192 Int_t iw = (Int_t)gPad->GetWw();
4193 Int_t ih = (Int_t)gPad->GetWh();
4195 gPad->GetPadPar(x1p,y1p,x2p,y2p);
4196 iw = (Int_t)(iw*x2p)-(Int_t)(iw*x1p);
4197 ih = (Int_t)(ih*y2p)-(Int_t)(ih*y1p);
4200
4201 // Search ymin and ymax
4202 ymin = 1.;
4203 ymax = 0.;
4204 for (i=1; i<=nn; i++) {
4205 x = wndc*ratiox*(xx[i-1]-rwxmin);
4206 y = hndc*ratioy*(yy[i-1]-rwymin);
4207 yrot = sina*x+cosa*y;
4208 if (yrot > ymax) ymax = yrot;
4209 if (yrot < ymin) ymin = yrot;
4210 }
4211 ymax = (Double_t)((Int_t)(ymax/dy))*dy;
4212
4213 for (ycur=ymax; ycur>=ymin; ycur=ycur-dy) {
4214 nbi = 0;
4215 for (i=2; i<=nn+1; i++) {
4216 i2 = i;
4217 i1 = i-1;
4218 if (i == nn+1) i2=1;
4219 x1 = wndc*ratiox*(xx[i1-1]-rwxmin);
4220 y1 = hndc*ratioy*(yy[i1-1]-rwymin);
4221 x2 = wndc*ratiox*(xx[i2-1]-rwxmin);
4222 y2 = hndc*ratioy*(yy[i2-1]-rwymin);
4223 xt1 = cosa*x1-sina*y1;
4224 yt1 = sina*x1+cosa*y1;
4225 xt2 = cosa*x2-sina*y2;
4226 yt2 = sina*x2+cosa*y2;
4227
4228 // Line segment parallel to oy
4229 if (xt1 == xt2) {
4230 if (yt1 < yt2) {
4231 yi = yt1;
4232 yip = yt2;
4233 } else {
4234 yi = yt2;
4235 yip = yt1;
4236 }
4237 if ((yi <= ycur) && (ycur < yip)) {
4238 nbi++;
4239 if (nbi >= maxnbi) return;
4240 xli[nbi-1] = xt1;
4241 }
4242 continue;
4243 }
4244
4245 // Line segment parallel to ox
4246 if (yt1 == yt2) {
4247 if (yt1 == ycur) {
4248 nbi++;
4249 if (nbi >= maxnbi) return;
4250 xli[nbi-1] = xt1;
4251 nbi++;
4252 if (nbi >= maxnbi) return;
4253 xli[nbi-1] = xt2;
4254 }
4255 continue;
4256 }
4257
4258 // Other line segment
4259 a = (yt1-yt2)/(xt1-xt2);
4260 b = (yt2*xt1-xt2*yt1)/(xt1-xt2);
4261 if (xt1 < xt2) {
4262 xi = xt1;
4263 xip = xt2;
4264 } else {
4265 xi = xt2;
4266 xip = xt1;
4267 }
4268 xin = (ycur-b)/a;
4269 if ((xi <= xin) && (xin < xip) &&
4270 (TMath::Min(yt1,yt2) <= ycur) &&
4271 (ycur < TMath::Max(yt1,yt2))) {
4272 nbi++;
4273 if (nbi >= maxnbi) return;
4274 xli[nbi-1] = xin;
4275 }
4276 }
4277
4278 // Sorting of the x coordinates intersections
4279 inv = 0;
4280 m = nbi-1;
4281L30:
4282 for (i=1; i<=m; i++) {
4283 if (xli[i] < xli[i-1]) {
4284 inv++;
4285 ll = xli[i-1];
4286 xli[i-1] = xli[i];
4287 xli[i] = ll;
4288 }
4289 }
4290 m--;
4291 if (inv == 0) goto L50;
4292 inv = 0;
4293 goto L30;
4294
4295 // Draw the hatches
4296L50:
4297 if (nbi%2 != 0) continue;
4298
4299 for (i=1; i<=nbi; i=i+2) {
4300 // Rotate back the hatches
4301 xlh[0] = cosb*xli[i-1]-sinb*ycur;
4302 ylh[0] = sinb*xli[i-1]+cosb*ycur;
4303 xlh[1] = cosb*xli[i] -sinb*ycur;
4304 ylh[1] = sinb*xli[i] +cosb*ycur;
4305 // Convert hatches' positions from true NDC to WC
4306 xlh[0] = (xlh[0]/wndc)*(rwxmax-rwxmin)+rwxmin;
4307 ylh[0] = (ylh[0]/hndc)*(rwymax-rwymin)+rwymin;
4308 xlh[1] = (xlh[1]/wndc)*(rwxmax-rwxmin)+rwxmin;
4309 ylh[1] = (ylh[1]/hndc)*(rwymax-rwymin)+rwymin;
4310 gPad->PaintLine(xlh[0], ylh[0], xlh[1], ylh[1]);
4311 }
4312 }
4313}
4314
4315////////////////////////////////////////////////////////////////////////////////
4316/// Paint line in CurrentPad World coordinates.
4317
4319{
4320 Double_t x[2] = {x1, x2}, y[2] = {y1, y2};
4321
4322 //If line is totally clipped, return
4324 if (Clip(x,y,fUxmin,fUymin,fUxmax,fUymax) == 2) return;
4325 } else {
4326 if (Clip(x,y,fX1,fY1,fX2,fY2) == 2) return;
4327 }
4328
4329 if (!gPad->IsBatch() && GetPainter())
4330 GetPainter()->DrawLine(x[0], y[0], x[1], y[1]);
4331
4332 if (gVirtualPS)
4333 gVirtualPS->DrawPS(2, x, y);
4334
4335 Modified();
4336}
4337
4338////////////////////////////////////////////////////////////////////////////////
4339/// Paint line in normalized coordinates.
4340
4342{
4343 static Double_t xw[2], yw[2];
4344 if (!gPad->IsBatch() && GetPainter())
4346
4347 if (gVirtualPS) {
4348 xw[0] = fX1 + u1*(fX2 - fX1);
4349 xw[1] = fX1 + u2*(fX2 - fX1);
4350 yw[0] = fY1 + v1*(fY2 - fY1);
4351 yw[1] = fY1 + v2*(fY2 - fY1);
4352 gVirtualPS->DrawPS(2, xw, yw);
4353 }
4354
4355 Modified();
4356}
4357
4358////////////////////////////////////////////////////////////////////////////////
4359/// Paint 3-D line in the CurrentPad.
4360
4362{
4363 if (!fView) return;
4364
4365 // convert from 3-D to 2-D pad coordinate system
4366 Double_t xpad[6];
4367 Double_t temp[3];
4368 Int_t i;
4369 for (i=0;i<3;i++) temp[i] = p1[i];
4370 fView->WCtoNDC(temp, &xpad[0]);
4371 for (i=0;i<3;i++) temp[i] = p2[i];
4372 fView->WCtoNDC(temp, &xpad[3]);
4373 PaintLine(xpad[0],xpad[1],xpad[3],xpad[4]);
4374}
4375
4376////////////////////////////////////////////////////////////////////////////////
4377/// Paint 3-D line in the CurrentPad.
4378
4380{
4381 //take into account perspective view
4382 if (!fView) return;
4383 // convert from 3-D to 2-D pad coordinate system
4384 Double_t xpad[6];
4385 Double_t temp[3];
4386 Int_t i;
4387 for (i=0;i<3;i++) temp[i] = p1[i];
4388 fView->WCtoNDC(temp, &xpad[0]);
4389 for (i=0;i<3;i++) temp[i] = p2[i];
4390 fView->WCtoNDC(temp, &xpad[3]);
4391 PaintLine(xpad[0],xpad[1],xpad[3],xpad[4]);
4392}
4393
4394////////////////////////////////////////////////////////////////////////////////
4395/// Paint polyline in CurrentPad World coordinates.
4396
4398{
4399 if (n < 2) return;
4400
4404 } else {
4405 xmin = fX1; ymin = fY1; xmax = fX2; ymax = fY2;
4406 }
4407 Int_t i, i1=-1,np=1;
4408 for (i=0; i<n-1; i++) {
4409 Double_t x1=x[i];
4410 Double_t y1=y[i];
4411 Double_t x2=x[i+1];
4412 Double_t y2=y[i+1];
4413 Int_t iclip = Clip(&x[i],&y[i],xmin,ymin,xmax,ymax);
4414 if (iclip == 2) {
4415 i1 = -1;
4416 continue;
4417 }
4418 np++;
4419 if (i1 < 0) i1 = i;
4420 if (iclip == 0 && i < n-2) continue;
4421 if (!gPad->IsBatch() && GetPainter())
4422 GetPainter()->DrawPolyLine(np, &x[i1], &y[i1]);
4423 if (gVirtualPS) {
4424 gVirtualPS->DrawPS(np, &x[i1], &y[i1]);
4425 }
4426 if (iclip) {
4427 x[i] = x1;
4428 y[i] = y1;
4429 x[i+1] = x2;
4430 y[i+1] = y2;
4431 }
4432 i1 = -1;
4433 np = 1;
4434 }
4435
4436 Modified();
4437}
4438
4439////////////////////////////////////////////////////////////////////////////////
4440/// Paint polyline in CurrentPad World coordinates.
4441///
4442/// If option[0] == 'C' no clipping
4443
4445{
4446 if (n < 2) return;
4447
4452 } else {
4453 xmin = fX1; ymin = fY1; xmax = fX2; ymax = fY2;
4454 if (option && (option[0] == 'C')) mustClip = kFALSE;
4455 }
4456
4457 Int_t i, i1=-1, np=1, iclip=0;
4458
4459 for (i=0; i < n-1; i++) {
4460 Double_t x1=x[i];
4461 Double_t y1=y[i];
4462 Double_t x2=x[i+1];
4463 Double_t y2=y[i+1];
4464 if (mustClip) {
4465 iclip = Clip(&x[i],&y[i],xmin,ymin,xmax,ymax);
4466 if (iclip == 2) {
4467 i1 = -1;
4468 continue;
4469 }
4470 }
4471 np++;
4472 if (i1 < 0) i1 = i;
4473 if (iclip == 0 && i < n-2) continue;
4474 if (!gPad->IsBatch() && GetPainter())
4475 GetPainter()->DrawPolyLine(np, &x[i1], &y[i1]);
4476 if (gVirtualPS) {
4477 gVirtualPS->DrawPS(np, &x[i1], &y[i1]);
4478 }
4479 if (iclip) {
4480 x[i] = x1;
4481 y[i] = y1;
4482 x[i+1] = x2;
4483 y[i+1] = y2;
4484 }
4485 i1 = -1;
4486 np = 1;
4487 }
4488
4489 Modified();
4490}
4491
4492////////////////////////////////////////////////////////////////////////////////
4493/// Paint polyline in CurrentPad NDC coordinates.
4494
4496{
4497 if (n <=0) return;
4498
4499 if (!gPad->IsBatch() && GetPainter())
4501
4502 if (gVirtualPS) {
4503 std::vector<Double_t> xw(n), yw(n);
4504 for (Int_t i=0; i<n; i++) {
4505 xw[i] = fX1 + x[i]*(fX2 - fX1);
4506 yw[i] = fY1 + y[i]*(fY2 - fY1);
4507 }
4508 gVirtualPS->DrawPS(n, xw.data(), yw.data());
4509 }
4510 Modified();
4511}
4512
4513////////////////////////////////////////////////////////////////////////////////
4514/// Paint 3-D polyline in the CurrentPad.
4515
4517{
4518 if (!fView) return;
4519
4520 // Loop on each individual line
4521 for (Int_t i = 1; i < n; i++)
4522 PaintLine3D(&p[3*i-3], &p[3*i]);
4523
4524 Modified();
4525}
4526
4527////////////////////////////////////////////////////////////////////////////////
4528/// Paint polymarker in CurrentPad World coordinates.
4529
4531{
4532 Int_t n = TMath::Abs(nn);
4534 if (nn > 0 || TestBit(TGraph::kClipFrame)) {
4536 } else {
4537 xmin = fX1; ymin = fY1; xmax = fX2; ymax = fY2;
4538 }
4539 Int_t i,i1=-1,np=0;
4540 for (i=0; i<n; i++) {
4541 if (x[i] >= xmin && x[i] <= xmax && y[i] >= ymin && y[i] <= ymax) {
4542 np++;
4543 if (i1 < 0) i1 = i;
4544 if (i < n-1) continue;
4545 }
4546 if (np == 0) continue;
4547 if (!gPad->IsBatch() && GetPainter())
4548 GetPainter()->DrawPolyMarker(np, &x[i1], &y[i1]);
4549 if (gVirtualPS) {
4551 }
4552 i1 = -1;
4553 np = 0;
4554 }
4555 Modified();
4556}
4557
4558////////////////////////////////////////////////////////////////////////////////
4559/// Paint polymarker in CurrentPad World coordinates.
4560
4562{
4563 Int_t n = TMath::Abs(nn);
4565 if (nn > 0 || TestBit(TGraph::kClipFrame)) {
4567 } else {
4568 xmin = fX1; ymin = fY1; xmax = fX2; ymax = fY2;
4569 }
4570 Int_t i,i1=-1,np=0;
4571 for (i=0; i<n; i++) {
4572 if (x[i] >= xmin && x[i] <= xmax && y[i] >= ymin && y[i] <= ymax) {
4573 np++;
4574 if (i1 < 0) i1 = i;
4575 if (i < n-1) continue;
4576 }
4577 if (np == 0) continue;
4578 if (!gPad->IsBatch() && GetPainter())
4579 GetPainter()->DrawPolyMarker(np, &x[i1], &y[i1]);
4580 if (gVirtualPS) {
4582 }
4583 i1 = -1;
4584 np = 0;
4585 }
4586 Modified();
4587}
4588
4589////////////////////////////////////////////////////////////////////////////////
4590/// Paint text in CurrentPad World coordinates.
4591
4593{
4594 Modified();
4595
4596 if (!gPad->IsBatch() && GetPainter())
4598
4599 if (gVirtualPS) gVirtualPS->Text(x, y, text);
4600}
4601
4602////////////////////////////////////////////////////////////////////////////////
4603/// Paint text in CurrentPad World coordinates.
4604
4605void TPad::PaintText(Double_t x, Double_t y, const wchar_t *text)
4606{
4607 Modified();
4608
4609 if (!gPad->IsBatch() && GetPainter())
4611
4612 if (gVirtualPS) gVirtualPS->Text(x, y, text);
4613}
4614
4615////////////////////////////////////////////////////////////////////////////////
4616/// Paint text in CurrentPad NDC coordinates.
4617
4619{
4620 Modified();
4621
4622 if (!gPad->IsBatch() && GetPainter())
4624
4625 if (gVirtualPS) {
4626 Double_t x = fX1 + u*(fX2 - fX1);
4627 Double_t y = fY1 + v*(fY2 - fY1);
4628 gVirtualPS->Text(x, y, text);
4629 }
4630}
4631
4632////////////////////////////////////////////////////////////////////////////////
4633/// Paint text in CurrentPad NDC coordinates.
4634
4636{
4637 Modified();
4638
4639 if (!gPad->IsBatch() && GetPainter())
4641
4642 if (gVirtualPS) {
4643 Double_t x = fX1 + u*(fX2 - fX1);
4644 Double_t y = fY1 + v*(fY2 - fY1);
4645 gVirtualPS->Text(x, y, text);
4646 }
4647}
4648
4649////////////////////////////////////////////////////////////////////////////////
4650/// Search for an object at pixel position px,py.
4651///
4652/// Check if point is in this pad.
4653///
4654/// If yes, check if it is in one of the sub-pads
4655///
4656/// If found in the pad, compute closest distance of approach
4657/// to each primitive.
4658///
4659/// If one distance of approach is found to be within the limit Distancemaximum
4660/// the corresponding primitive is selected and the routine returns.
4661
4663{
4664 //the two following statements are necessary under NT (multithreaded)
4665 //when a TCanvas object is being created and a thread calling TPad::Pick
4666 //before the TPad constructor has completed in the other thread
4667 if (!gPad) return nullptr; //Andy Haas
4668 if (!GetListOfPrimitives()) return nullptr; //Andy Haas
4669
4670 Int_t dist;
4671 // Search if point is in pad itself
4672 Double_t x = AbsPixeltoX(px);
4673 Double_t y = AbsPixeltoY(py);
4674 if (this != gPad->GetCanvas()) {
4675 if (!((x >= fX1 && x <= fX2) && (y >= fY1 && y <= fY2))) return nullptr;
4676 }
4677
4678 // search for a primitive in this pad or its sub-pads
4679 static TObjOptLink dummyLink(nullptr,""); //place holder for when no link available
4680
4681 TContext ctxt(this, kFALSE); // since no drawing will be done, don't use cd() for efficiency reasons
4682
4683 TPad *pick = nullptr;
4684 TPad *picked = this;
4685 pickobj = nullptr;
4687 dummyLink.SetObject(this);
4688 pickobj = &dummyLink;
4689 }
4690
4691 // Loop backwards over the list of primitives. The first non-pad primitive
4692 // found is the selected one. However, we have to keep going down the
4693 // list to see if there is maybe a pad overlaying the primitive. In that
4694 // case look into the pad for a possible primitive. Once a pad has been
4695 // found we can terminate the loop.
4696 Bool_t gotPrim = kFALSE; // true if found a non pad primitive
4698
4699 //We can have 3d stuff in pad. If canvas prefers to draw
4700 //such stuff with OpenGL, the selection of 3d objects is
4701 //a gl viewer business so, in first cycle we do not
4702 //call DistancetoPrimitive for TAtt3D descendants.
4703 //In case of gl we first try to select 2d object first.
4704
4705 while (lnk) {
4706 TObject *obj = lnk->GetObject();
4707
4708 //If canvas prefers GL, all 3d objects must be drawn/selected by
4709 //gl viewer
4710 if (obj->InheritsFrom(TAtt3D::Class()) && fEmbeddedGL) {
4711 lnk = lnk->Prev();
4712 continue;
4713 }
4714
4715 fPadPointer = obj;
4716 if (obj->InheritsFrom(TPad::Class())) {
4717 pick = ((TPad*)obj)->Pick(px, py, pickobj);
4718 if (pick) {
4719 picked = pick;
4720 break;
4721 }
4722 } else if (!gROOT->GetEditorMode()) {
4723 if (!gotPrim) {
4724 if (!obj->TestBit(kCannotPick)) {
4725 dist = obj->DistancetoPrimitive(px, py);
4726 if (dist < fgMaxPickDistance) {
4727 pickobj = lnk;
4728 gotPrim = kTRUE;
4729 if (dist == 0) break;
4730 }
4731 }
4732 }
4733 }
4734
4735 lnk = lnk->Prev();
4736 }
4737
4738 //if no primitive found, check if we have a TView
4739 //if yes, return the view except if you are in the lower or upper X range
4740 //of the pad.
4741 //In case canvas prefers gl, fView existence
4742 //automatically means viewer3d existence. (?)
4743
4744 if (fView && !gotPrim) {
4745 Double_t dx = 0.05*(fUxmax-fUxmin);
4746 if ((x > fUxmin + dx) && (x < fUxmax-dx)) {
4747
4748 if (fEmbeddedGL) {
4749 //No 2d stuff was selected, but we have gl-viewer. Let it select an object in
4750 //scene (or select itself). In any case it'll internally call
4751 //gPad->SetSelected(ptr) as, for example, hist painter does.
4752 py -= Int_t((1 - GetHNDC() - GetYlowNDC()) * GetWh());
4753 px -= Int_t(GetXlowNDC() * GetWw());
4755 }
4756 else
4757 dummyLink.SetObject(fView);
4758 }
4759 }
4760
4761 if (picked->InheritsFrom(TButton::Class())) {
4763 if (!button->IsEditable()) pickobj = nullptr;
4764 }
4765
4766 if (TestBit(kCannotPick)) {
4767
4768 if (picked == this) {
4769 // cannot pick pad itself!
4770 picked = nullptr;
4771 }
4772
4773 }
4774
4775 return picked;
4776}
4777
4778////////////////////////////////////////////////////////////////////////////////
4779/// Pop pad to the top of the stack.
4780
4782{
4784 return;
4785 if (!fPrimitives)
4786 fPrimitives = new TList;
4787 if (this == fMother->GetListOfPrimitives()->Last())
4788 return;
4789
4791 while (auto obj = next())
4792 if (obj == this) {
4793 TString opt = next.GetOption();
4794 fMother->Remove(this, kFALSE); // do not issue modified
4795 fMother->Add(this, opt.Data());
4796 return;
4797 }
4798}
4799
4800////////////////////////////////////////////////////////////////////////////////
4801/// This method is equivalent to `SaveAs("filename")`. See TPad::SaveAs for details.
4802
4803void TPad::Print(const char *filename) const
4804{
4805 ((TPad*)this)->SaveAs(filename);
4806}
4807
4808////////////////////////////////////////////////////////////////////////////////
4809/// Auxiliary function. Returns kTRUE if list contains an object inherited
4810/// from TImage
4811
4813{
4814 TIter next(li);
4815
4816 while (auto obj = next()) {
4817 if (obj->InheritsFrom(TImage::Class())) {
4818 return kTRUE;
4819 } else if (obj->InheritsFrom(TPad::Class())) {
4820 if (ContainsTImage(((TPad*)obj)->GetListOfPrimitives())) {
4821 return kTRUE;
4822 }
4823 }
4824 }
4825 return kFALSE;
4826}
4827
4828////////////////////////////////////////////////////////////////////////////////
4829/// Save Canvas contents in a file in one of various formats.
4830///
4831/// \anchor TPadPrint
4832/// option can be:
4833///
4834/// - `ps`: a Postscript file is produced (default). [See special cases](\ref TPadPrintPS).
4835/// - `Portrait`: Postscript file is produced (Portrait)
4836/// - `Landscape`: Postscript file is produced (Landscape)
4837/// - `eps`: an Encapsulated Postscript file is produced
4838/// - `Preview`: an [Encapsulated Postscript file with preview](\ref TPadPrintPreview) is produced.
4839/// - `pdf`: a PDF file is produced NOTE: TMathText will be converted to TLatex; q.e.d., symbols only available in TMathText will not render properly.
4840/// - `Title:`: The character string after `Title:` becomes a table
4841/// of content entry (for PDF files).
4842/// - `EmbedFonts`: a [PDF file with embedded fonts](\ref TPadPrintEmbedFonts) is generated.
4843/// - `svg`: a SVG file is produced
4844/// - `tex`: a TeX file is produced
4845/// - `Standalone`: a [standalone TeX file](\ref TPadPrintStandalone) is produced.
4846/// - `gif`: a GIF file is produced
4847/// - `gif+NN`: an animated GIF file is produced, where NN is delay in 10ms units NOTE: See other variants for looping animation in TASImage::WriteImage
4848/// - `xpm`: a XPM file is produced
4849/// - `png`: a PNG file is produced
4850/// - `jpg`: a JPEG file is produced. NOTE: JPEG's lossy compression will make all sharp edges fuzzy.
4851/// - `tiff`: a TIFF file is produced
4852/// - `cxx`: a C++ macro file is produced
4853/// - `xml`: a XML file
4854/// - `json`: a JSON file
4855/// - `root`: a ROOT binary file
4856///
4857/// `filename` = 0 - filename is defined by `GetName` and its
4858/// extension is defined with the option
4859///
4860/// When Postscript output is selected (`ps`, `eps`), the canvas is saved
4861/// to `filename.ps` or `filename.eps`. The aspect ratio of the canvas is preserved
4862/// on the Postscript file. When the "ps" option is selected, the Postscript
4863/// page will be landscape format if the canvas is in landscape format, otherwise
4864/// portrait format is selected.
4865///
4866/// The physical size of the Postscript page is the one selected in the
4867/// current style. This size can be modified via TStyle::SetPaperSize.
4868///
4869/// Examples:
4870/// ~~~ {.cpp}
4871/// gStyle->SetPaperSize(TStyle::kA4); //default
4872/// gStyle->SetPaperSize(TStyle::kUSLetter);
4873/// ~~~
4874/// where TStyle::kA4 and TStyle::kUSLetter are defined in the enum
4875/// EPaperSize in TStyle.h
4876///
4877/// An alternative is to call:
4878/// ~~~ {.cpp}
4879/// gStyle->SetPaperSize(20,26); same as kA4
4880/// or gStyle->SetPaperSize(20,24); same as kUSLetter
4881/// ~~~
4882/// The above numbers take into account some margins and are in centimeters.
4883///
4884/// \anchor TPadPrintPreview
4885/// ### The "Preview" option
4886///
4887/// The "Preview" option allows to generate a preview (in the TIFF format) within
4888/// the Encapsulated Postscript file. This preview can be used by programs like
4889/// MSWord to visualize the picture on screen. The "Preview" option relies on the
4890/// ["epstool" command](http://www.cs.wisc.edu/~ghost/gsview/epstool.htm).
4891///
4892/// Example:
4893/// ~~~ {.cpp}
4894/// canvas->Print("example.eps","Preview");
4895/// ~~~
4896///
4897/// \anchor TPadPrintEmbedFonts
4898/// ### The "EmbedFonts" option
4899///
4900/// The "EmbedFonts" option allows to embed the fonts used in a PDF file inside
4901/// that file. This option relies on the ["gs" command](https://ghostscript.com).
4902///
4903/// Example:
4904/// ~~~ {.cpp}
4905/// canvas->Print("example.pdf","EmbedFonts");
4906/// ~~~
4907///
4908/// \anchor TPadPrintStandalone
4909/// ### The "Standalone" option
4910/// The "Standalone" option allows to generate a TeX file ready to be processed by
4911/// tools like `pdflatex`.
4912///
4913/// Example:
4914/// ~~~ {.cpp}
4915/// canvas->Print("example.tex","Standalone");
4916/// ~~~
4917///
4918/// \anchor TPadPrintPS
4919/// ### Writing several canvases to the same Postscript or PDF file:
4920///
4921/// - if the Postscript or PDF file name finishes with "(", the file is not closed
4922/// - if the Postscript or PDF file name finishes with ")" and the file has been opened
4923/// with "(", the file is closed.
4924///
4925/// Example:
4926/// ~~~ {.cpp}
4927/// {
4928/// TCanvas c1("c1");
4929/// h1.Draw();
4930/// c1.Print("c1.ps("); //write canvas and keep the ps file open
4931/// h2.Draw();
4932/// c1.Print("c1.ps"); canvas is added to "c1.ps"
4933/// h3.Draw();
4934/// c1.Print("c1.ps)"); canvas is added to "c1.ps" and ps file is closed
4935/// }
4936/// ~~~
4937/// In the previous example replacing "ps" by "pdf" will create a multi-pages PDF file.
4938///
4939/// Note that the following sequence writes the canvas to "c1.ps" and closes the ps file.:
4940/// ~~~ {.cpp}
4941/// TCanvas c1("c1");
4942/// h1.Draw();
4943/// c1.Print("c1.ps");
4944/// ~~~
4945/// The `TCanvas::Print("file.ps(")` mechanism is very useful, but it can be
4946/// a little inconvenient to have the action of opening/closing a file
4947/// being atomic with printing a page. Particularly if pages are being
4948/// generated in some loop one needs to detect the special cases of first
4949/// and last page and then munge the argument to Print() accordingly.
4950///
4951/// The "[" and "]" can be used instead of "(" and ")".
4952///
4953/// Example:
4954/// ~~~ {.cpp}
4955/// c1.Print("file.ps["); // No actual print, just open file.ps
4956/// for (int i=0; i<10; ++i) {
4957/// // fill canvas for context i
4958/// // ...
4959///
4960/// c1.Print("file.ps"); // actually print canvas to file
4961/// }// end loop
4962/// c1.Print("file.ps]"); // No actual print, just close.
4963/// ~~~
4964/// As before, the same macro is valid for PDF files.
4965///
4966/// It is possible to print a canvas into an animated GIF file by specifying the
4967/// file name as "myfile.gif+" or "myfile.gif+NN", where NN*10ms is delay
4968/// between the subimages' display. If NN is omitted the delay between
4969/// subimages is zero. Each picture is added in the animation thanks to a loop
4970/// similar to the following one:
4971/// ~~~ {.cpp}
4972/// for (int i=0; i<10; ++i) {
4973/// // fill canvas for context i
4974/// // ...
4975///
4976/// c1.Print("file.gif+5"); // print canvas to GIF file with 50ms delays
4977/// }// end loop
4978/// ~~~
4979/// The delay between each frame must be specified in each Print() statement.
4980/// If the file "myfile.gif" already exists, the new frame are appended at
4981/// the end of the file. To avoid this, delete it first with `gSystem->Unlink(myfile.gif);`
4982/// If you want the gif file to repeat or loop forever, check TASImage::WriteImage documentation
4983
4985{
4986 if (!GetCanvas())
4987 return;
4988
4990
4991 // "[" and "]" are special characters for ExpandPathName. When they are at the end
4992 // of the file name (see help) they must be removed before doing ExpandPathName.
4993 if (fs1.EndsWith("[")) {
4994 fs1.Replace((fs1.Length()-1),1," ");
4996 fs1.Replace((fs1.Length()-1),1,"[");
4997 } else if (fs1.EndsWith("]")) {
4998 fs1.Replace((fs1.Length()-1),1," ");
5000 fs1.Replace((fs1.Length()-1),1,"]");
5001 } else {
5003 }
5004
5005 // Set the default option as "Postscript" (Should be a data member of TPad)
5006 const char *opt_default = "ps";
5007
5008 TString opt = !option ? opt_default : option;
5010
5011 if (!fs1.Length()) {
5012 psname = GetName();
5013 psname += opt;
5014 } else {
5015 psname = fs1;
5016 }
5017
5018 // lines below protected against case like c1->SaveAs( "../ps/cs.ps" );
5019 if (psname.BeginsWith('.') && (psname.Contains('/') == 0)) {
5020 psname = GetName();
5021 psname.Append(fs1);
5022 psname.Prepend("/");
5023 psname.Prepend(gEnv->GetValue("Canvas.PrintDirectory","."));
5024 }
5025
5026 // Save pad/canvas in alternative formats
5028 if (strstr(opt, "gif+")) {
5030 image = kTRUE;
5031 } else if (strstr(opt, "gif")) {
5033 image = kTRUE;
5034 } else if (strstr(opt, "png")) {
5036 image = kTRUE;
5037 } else if (strstr(opt, "jpg")) {
5039 image = kTRUE;
5040 } else if (strstr(opt, "tiff")) {
5042 image = kTRUE;
5043 } else if (strstr(opt, "xpm")) {
5045 image = kTRUE;
5046 } else if (strstr(opt, "bmp")) {
5048 image = kTRUE;
5049 }
5050
5051 if (GetCanvas()->IsWeb() && GetPainter() &&
5052 (strstr(opt,"svg") || strstr(opt,"pdf") || (gtype == TImage::kJpeg) || (gtype == TImage::kPng))) {
5053 GetPainter()->SaveImage(this, psname.Data(), gtype);
5054 return;
5055 }
5056
5057 if (!GetCanvas()->IsBatch() && GetPainter())
5059
5060
5061 if (!gROOT->IsBatch() && image) {
5063 Int_t wid = (this == GetCanvas()) ? GetCanvas()->GetCanvasID() : GetPixmapID();
5064 Color_t hc = gPad->GetCanvas()->GetHighLightColor();
5065 gPad->GetCanvas()->SetHighLightColor(-1);
5066 gPad->Modified();
5067 gPad->Update();
5068 if (GetPainter()){
5070 GetPainter()->SaveImage(this, psname.Data(), gtype);
5071 }
5072 if (!gSystem->AccessPathName(psname.Data())) {
5073 Info("Print", "GIF file %s has been created", psname.Data());
5074 }
5075 gPad->GetCanvas()->SetHighLightColor(hc);
5076 return;
5077 }
5078 if (gtype != TImage::kUnknown) {
5079 Color_t hc = gPad->GetCanvas()->GetHighLightColor();
5080 gPad->GetCanvas()->SetHighLightColor(-1);
5081 gPad->Modified();
5082 gPad->Update();
5083 gVirtualX->Update(1);
5084 gSystem->Sleep(30); // synchronize
5085 if (GetPainter()) GetPainter()->SaveImage(this, psname, gtype);
5086 if (!gSystem->AccessPathName(psname)) {
5087 Info("Print", "file %s has been created", psname.Data());
5088 }
5089 gPad->GetCanvas()->SetHighLightColor(hc);
5090 } else {
5091 Warning("Print", "Unsupported image format %s", psname.Data());
5092 }
5093 return;
5094 }
5095
5096 //==============Save pad/canvas as a C++ script==============================
5097 if (strstr(opt,"cxx")) {
5098 GetCanvas()->SaveSource(psname, "");
5099 return;
5100 }
5101
5102 //==============Save pad/canvas as a root file===============================
5103 if (strstr(opt,"root")) {
5104 if (gDirectory) gDirectory->SaveObjectAs(this,psname.Data(),"");
5105 return;
5106 }
5107
5108 //==============Save pad/canvas as a XML file================================
5109 if (strstr(opt,"xml")) {
5110 // Plugin XML driver
5111 if (gDirectory) gDirectory->SaveObjectAs(this,psname.Data(),"");
5112 return;
5113 }
5114
5115 //==============Save pad/canvas as a JSON file================================
5116 if (strstr(opt,"json")) {
5117 if (gDirectory) gDirectory->SaveObjectAs(this,psname.Data(),"");
5118 return;
5119 }
5120
5121 //==============Save pad/canvas as a SVG file================================
5122 if (strstr(opt,"svg")) {
5123 gVirtualPS = (TVirtualPS*)gROOT->GetListOfSpecials()->FindObject(psname);
5124
5126 if (!GetCanvas()->IsBatch() && GetCanvas()->GetCanvasID() == -1) {
5127 noScreen = kTRUE;
5129 }
5130
5131 TContext ctxt(this, kTRUE);
5132
5133 if (!gVirtualPS) {
5134 // Plugin Postscript/SVG driver
5135 if (auto h = gROOT->GetPluginManager()->FindHandler("TVirtualPS", "svg")) {
5136 if (h->LoadPlugin() == -1)
5137 return;
5138 h->ExecPlugin(0);
5139 }
5140 }
5141
5142 // Create a new SVG file
5143 if (gVirtualPS) {
5148 }
5149 Paint();
5150 if (noScreen)
5152
5154 Info("Print", "SVG file %s has been created", psname.Data());
5155
5156 delete gVirtualPS;
5157 gVirtualPS = nullptr;
5158
5159 return;
5160 }
5161
5162 //==============Save pad/canvas as a TeX file================================
5163 if (strstr(opt,"tex") || strstr(opt,"Standalone")) {
5164 gVirtualPS = (TVirtualPS*)gROOT->GetListOfSpecials()->FindObject(psname);
5165
5167 if (!GetCanvas()->IsBatch() && GetCanvas()->GetCanvasID() == -1) {
5168 noScreen = kTRUE;
5170 }
5171
5172 TContext ctxt(this, kTRUE);
5173
5174 if (!gVirtualPS) {
5175 // Plugin Postscript/SVG driver
5176 if (auto h = gROOT->GetPluginManager()->FindHandler("TVirtualPS", "tex")) {
5177 if (h->LoadPlugin() == -1)
5178 return;
5179 h->ExecPlugin(0);
5180 }
5181 }
5182
5184 if (strstr(opt,"Standalone")) standalone = kTRUE;
5185
5186 // Create a new TeX file
5187 if (gVirtualPS) {
5189 if (standalone) gVirtualPS->SetTitle("Standalone");
5193 }
5194 Paint();
5196
5197 if (!gSystem->AccessPathName(psname)) {
5198 if (standalone) {
5199 Info("Print", "Standalone TeX file %s has been created", psname.Data());
5200 } else{
5201 Info("Print", "TeX file %s has been created", psname.Data());
5202 }
5203 }
5204
5205 delete gVirtualPS;
5206 gVirtualPS = nullptr;
5207
5208 return;
5209 }
5210
5211 //==============Save pad/canvas as a Postscript file=========================
5212
5213 // in case we read directly from a Root file and the canvas
5214 // is not on the screen, set batch mode
5215
5218 if (!image) {
5219 // The parenthesis mechanism is only valid for PS and PDF files.
5220 copen = psname.EndsWith("("); if (copen) psname[psname.Length()-1] = 0;
5221 cclose = psname.EndsWith(")"); if (cclose) psname[psname.Length()-1] = 0;
5222 copenb = psname.EndsWith("["); if (copenb) psname[psname.Length()-1] = 0;
5223 ccloseb = psname.EndsWith("]"); if (ccloseb) psname[psname.Length()-1] = 0;
5224 }
5225 gVirtualPS = (TVirtualPS*)gROOT->GetListOfSpecials()->FindObject(psname);
5227 if (copen || copenb) mustClose = kFALSE;
5228 if (cclose || ccloseb) mustClose = kTRUE;
5229
5231 if (!GetCanvas()->IsBatch() && GetCanvas()->GetCanvasID() == -1) {
5232 noScreen = kTRUE;
5234 }
5235 Int_t pstype = 111;
5238 Double_t ratio = ycanvas/xcanvas;
5239 if (ratio < 1) pstype = 112;
5240 if (strstr(opt,"Portrait")) pstype = 111;
5241 if (strstr(opt,"Landscape")) pstype = 112;
5242 if (strstr(opt,"eps")) pstype = 113;
5243 if (strstr(opt,"Preview")) pstype = 113;
5244
5245 TContext ctxt(this, kTRUE);
5247
5248 if (!gVirtualPS || mustOpen) {
5249
5250 const char *pluginName = "ps"; // Plugin Postscript driver
5251 if (strstr(opt,"pdf") || strstr(opt,"Title:") || strstr(opt,"EmbedFonts"))
5252 pluginName = "pdf";
5253 else if (image)
5254 pluginName = "image"; // Plugin TImageDump driver
5255
5256 if (auto h = gROOT->GetPluginManager()->FindHandler("TVirtualPS", pluginName)) {
5257 if (h->LoadPlugin() == -1)
5258 return;
5259 h->ExecPlugin(0);
5260 }
5261
5262 // Create a new Postscript, PDF or image file
5263 if (gVirtualPS)
5265 const Ssiz_t titlePos = opt.Index("Title:");
5266 if (titlePos != kNPOS) {
5267 if (gVirtualPS)
5268 gVirtualPS->SetTitle(opt.Data()+titlePos+6);
5269 opt.Replace(titlePos,opt.Length(),"pdf");
5270 }
5271 if (gVirtualPS)
5273 if (gVirtualPS)
5275 if (!copenb) {
5276 if (!strstr(opt,"pdf") || image) {
5278 }
5279 Paint();
5280 }
5282
5283 if (mustClose) {
5284 gROOT->GetListOfSpecials()->Remove(gVirtualPS);
5285 delete gVirtualPS;
5286 gVirtualPS = psave;
5287 } else {
5288 gROOT->GetListOfSpecials()->Add(gVirtualPS);
5289 gVirtualPS = nullptr;
5290 }
5291
5292 if (!gSystem->AccessPathName(psname)) {
5293 if (!copen) Info("Print", "%s file %s has been created", opt.Data(), psname.Data());
5294 else Info("Print", "%s file %s has been created using the current canvas", opt.Data(), psname.Data());
5295 }
5296 } else {
5297 // Append to existing Postscript, PDF or GIF file
5298 if (!ccloseb) {
5300 Paint();
5301 }
5302 const Ssiz_t titlePos = opt.Index("Title:");
5303 if (titlePos != kNPOS) {
5304 gVirtualPS->SetTitle(opt.Data()+titlePos+6);
5305 opt.Replace(titlePos,opt.Length(),"pdf");
5306 } else {
5307 gVirtualPS->SetTitle("PDF");
5308 }
5309 if (mustClose) {
5310 if (cclose) Info("Print", "Current canvas added to %s file %s and file closed", opt.Data(), psname.Data());
5311 else Info("Print", "%s file %s has been closed", opt.Data(), psname.Data());
5312 gROOT->GetListOfSpecials()->Remove(gVirtualPS);
5313 delete gVirtualPS;
5314 gVirtualPS = nullptr;
5315 } else {
5316 Info("Print", "Current canvas added to %s file %s", opt.Data(), psname.Data());
5317 gVirtualPS = nullptr;
5318 }
5319 }
5320
5321 if (strstr(opt,"Preview"))
5322 gSystem->Exec(TString::Format("epstool --quiet -t6p %s %s", psname.Data(), psname.Data()).Data());
5323 if (strstr(opt,"EmbedFonts")) {
5324 gSystem->Exec(TString::Format("gs -quiet -dSAFER -dNOPLATFONTS -dNOPAUSE -dBATCH -sDEVICE=pdfwrite -dCompatibilityLevel=1.4 -dPDFSETTINGS=/printer -dCompatibilityLevel=1.4 -dMaxSubsetPct=100 -dSubsetFonts=true -dEmbedAllFonts=true -sOutputFile=pdf_temp.pdf -f %s",
5325 psname.Data()).Data());
5326 gSystem->Rename("pdf_temp.pdf", psname.Data());
5327 }
5328
5329}
5330
5331////////////////////////////////////////////////////////////////////////////////
5332/// Set world coordinate system for the pad.
5333/// Emits signal "RangeChanged()", in the slot get the range
5334/// via GetRange().
5335
5337{
5338 if ((x1 >= x2) || (y1 >= y2)) {
5339 Error("Range", "illegal world coordinates range: x1=%f, y1=%f, x2=%f, y2=%f",x1,y1,x2,y2);
5340 return;
5341 }
5342
5343 fUxmin = x1;
5344 fUxmax = x2;
5345 fUymin = y1;
5346 fUymax = y2;
5347
5348 if (fX1 == x1 && fY1 == y1 && fX2 == x2 && fY2 == y2) return;
5349
5350 fX1 = x1;
5351 fY1 = y1;
5352 fX2 = x2;
5353 fY2 = y2;
5354
5355 // compute pad conversion coefficients
5356 ResizePad();
5357
5358 if (gPad == this && GetPainter())
5360
5361 // emit signal
5362 RangeChanged();
5363}
5364
5365////////////////////////////////////////////////////////////////////////////////
5366/// Set axis coordinate system for the pad.
5367/// The axis coordinate system is a subset of the world coordinate system
5368/// xmin,ymin is the origin of the current coordinate system,
5369/// xmax is the end of the X axis, ymax is the end of the Y axis.
5370/// By default a margin of 10 per cent is left on all sides of the pad
5371/// Emits signal "RangeAxisChanged()", in the slot get the axis range
5372/// via GetRangeAxis().
5373
5375{
5376 if ((xmin >= xmax) || (ymin >= ymax)) {
5377 Error("RangeAxis", "illegal axis coordinates range: xmin=%f, ymin=%f, xmax=%f, ymax=%f",
5378 xmin, ymin, xmax, ymax);
5379 return;
5380 }
5381
5382 fUxmin = xmin;
5383 fUymin = ymin;
5384 fUxmax = xmax;
5385 fUymax = ymax;
5386
5387 // emit signal
5389}
5390
5391////////////////////////////////////////////////////////////////////////////////
5392/// Recursively remove object from a pad and its sub-pads.
5393
5395{
5396 if (fCanvas) {
5397 if (obj == fCanvas->GetSelected()) fCanvas->SetSelected(nullptr);
5398 if (obj == fCanvas->GetClickSelected()) fCanvas->SetClickSelected(nullptr);
5399 }
5400 if (obj == fView) fView = nullptr;
5401 if (!fPrimitives) return;
5404 if (nold != fPrimitives->GetSize()) fModified = kTRUE;
5405}
5406
5407////////////////////////////////////////////////////////////////////////////////
5408/// Remove object from list of primitives
5409/// When \par modified set to kTRUE (default) pad will be marked as modified - if object really removed
5410/// Returns result of GetListOfPrimitives()->Remove(obj) or nullptr if list of primitives not exists
5411
5413{
5414 TObject *res = nullptr;
5415 if (fPrimitives)
5416 res = fPrimitives->Remove(obj);
5417 if (res && modified)
5418 Modified();
5419 return res;
5420}
5421
5422////////////////////////////////////////////////////////////////////////////////
5423/// Redraw the frame axis.
5424///
5425/// Redrawing axis may be necessary in case of superimposed histograms
5426/// when one or more histograms have a fill color.
5427///
5428/// Instead of calling this function, it may be more convenient
5429/// to call directly `h1->Draw("sameaxis")` where h1 is the pointer
5430/// to the first histogram drawn in the pad.
5431///
5432/// By default, if the pad has the options gridx or/and gridy activated,
5433/// the grid is not drawn by this function.
5434///
5435/// If option="g" is specified, this will force the drawing of the grid
5436/// on top of the picture
5437///
5438/// To redraw the axis tick marks do:
5439/// ~~~ {.cpp}
5440/// gPad->RedrawAxis();
5441/// ~~~
5442/// To redraw the axis grid do:
5443/// ~~~ {.cpp}
5444/// gPad->RedrawAxis("G");
5445/// ~~~
5446/// To redraw the axis tick marks and the axis grid do:
5447/// ~~~ {.cpp}
5448/// gPad->RedrawAxis();
5449/// gPad->RedrawAxis("G");
5450/// ~~~
5451///
5452/// If option="f" is specified, this will force the drawing of the frame
5453/// around the plot.
5454
5456{
5457 TString opt = option;
5458 opt.ToLower();
5459
5460 TContext ctxt(this, kTRUE);
5461
5462 TH1 *hobj = nullptr;
5463
5464 // Get the first histogram drawing the axis in the list of primitives
5465 if (!fPrimitives) fPrimitives = new TList;
5466 TIter next(fPrimitives);
5467 TObject *obj;
5468 while ((obj = next())) {
5469 if (obj->InheritsFrom(TH1::Class())) {
5470 hobj = (TH1*)obj;
5471 break;
5472 }
5473 if (obj->InheritsFrom(TMultiGraph::Class())) {
5474 TMultiGraph *mg = (TMultiGraph*)obj;
5475 if (mg) hobj = mg->GetHistogram();
5476 break;
5477 }
5478 if (obj->InheritsFrom(TGraph::Class())) {
5479 TGraph *g = (TGraph*)obj;
5480 if (g) hobj = g->GetHistogram();
5481 break;
5482 }
5483 if (obj->InheritsFrom(THStack::Class())) {
5484 THStack *hs = (THStack*)obj;
5485 if (hs) hobj = hs->GetHistogram();
5486 break;
5487 }
5488 }
5489
5490 if (hobj) {
5491 if (opt.Contains("g")) hobj->DrawCopy("sameaxig");
5492 else hobj->DrawCopy("sameaxis");
5493 }
5494
5495 if (opt.Contains("f")) {
5496 auto b = new TBox(gPad->GetUxmin(), gPad->GetUymin(),
5497 gPad->GetUxmax(), gPad->GetUymax());
5498 b->SetFillStyle(0);
5499 b->SetLineStyle(gPad->GetFrameLineStyle());
5500 b->SetLineWidth(gPad->GetFrameLineWidth());
5501 b->SetLineColor(gPad->GetFrameLineColor());
5502 b->Draw();
5503 }
5504}
5505
5506////////////////////////////////////////////////////////////////////////////////
5507/// Compute pad conversion coefficients.
5508///
5509/// ### Conversion from x to px
5510///
5511/// \f[\frac{x-xmin}{xrange} = \frac{px-pxlow}{pxrange}\f]
5512/// with:
5513/// \f[ xrange = xmax-xmin \f]
5514/// \f[ pxrange = pxmax-pxmin \f]
5515///
5516/// \f[
5517/// \Rightarrow px = \frac{pxrange(x-xmin)}{xrange} + pxlow = fXtoPixelk + fXtoPixel \times x
5518/// \f]
5519///
5520/// \f[
5521/// \Rightarrow fXtoPixelk = pxlow - pxrange \frac{xmin}{xrange}
5522/// \f]
5523/// \f[
5524/// fXtoPixel = \frac{pxrange}{xrange}
5525/// \f]
5526/// where:
5527/// \f[
5528/// pxlow = fAbsXlowNDC \times fCw
5529/// \f]
5530/// \f[
5531/// pxrange = fAbsWNDC \times fCw
5532/// \f]
5533///
5534/// ### Conversion from y to py
5535///
5536/// \f[\frac{y-ymin}{yrange} = \frac{py-pylow}{pyrange}\f]
5537/// with:
5538/// \f[ yrange = ymax-ymin \f]
5539/// \f[ pyrange = pymax-pymin \f]
5540///
5541/// \f[
5542/// \Rightarrow py = \frac{pyrange(y-xmin)}{yrange} + pylow = fYtoPixelk + fYtoPixel \times y
5543/// \f]
5544///
5545/// \f[
5546/// \Rightarrow fYtoPixelk = pylow - pyrange \frac{ymin}{yrange}
5547/// \f]
5548/// \f[
5549/// fYtoPixel = \frac{pyrange}{yrange}
5550/// \f]
5551/// where:
5552/// \f[
5553/// pylow = fAbsYlowNDC \times fCh
5554/// \f]
5555/// \f[
5556/// pyrange = fAbsHNDC \times fCh
5557/// \f]
5558///
5559/// ### Conversion from px to x
5560///
5561/// \f[
5562/// \Rightarrow x = \frac{xrange(px-pxlow)}{pxrange}+ xmin = fPixeltoXk + fPixeltoX \times px
5563/// \f]
5564///
5565/// \f[
5566/// \Rightarrow fPixeltoXk = xmin - pxlow \times\frac{xrange}{pxrange}
5567/// \f]
5568/// \f[
5569/// fPixeltoX = \frac{xrange}{pxrange}
5570/// \f]
5571///
5572/// ### Conversion from py to y
5573///
5574/// \f[
5575/// \Rightarrow y = \frac{yrange(py-pylow)}{pyrange}+ ymin = fPixeltoYk + fPixeltoY \times py
5576/// \f]
5577///
5578/// \f[
5579/// \Rightarrow fPixeltoYk = ymin - pylow \times\frac{yrange}{pyrange}
5580/// \f]
5581/// \f[
5582/// fPixeltoY = \frac{yrange}{pyrange}
5583/// \f]
5584///
5585/// ### Computation of the coefficients in case of LOG scales
5586///
5587/// #### Conversion from pixel coordinates to world coordinates
5588///
5589/// \f[
5590/// u = \frac{Log(x) - Log(xmin)}{Log(xmax) - Log(xmin)} = \frac{Log(x/xmin)}{Log(xmax/xmin)} = \frac{px - pxlow}{pxrange}
5591/// \f]
5592///
5593/// \f[ \Rightarrow Log(\frac{x}{xmin}) = u \times Log(\frac{xmax}{xmin}) \f]
5594/// \f[ x = xmin \times e^{(u \times Log(\frac{xmax}{xmin})} \f]
5595/// Let:
5596/// \f[ alfa = \frac{Log(\frac{xmax}{xmin})}{fAbsWNDC} \f]
5597///
5598/// \f[ x = xmin \times e^{(-alfa \times pxlow)} + e^{(alfa \times px)} \f]
5599/// \f[ x = fPixeltoXk \times e^{(fPixeltoX \times px)} \f]
5600/// \f[ ==> fPixeltoXk = xmin \times e^{(-alfa*pxlow)} \f]
5601/// \f[ fPixeltoX = alfa \f]
5602///
5603/// \f[
5604/// v = \frac{Log(y) - Log(ymin)}{Log(ymax) - Log(ymin)} = \frac{Log(y/ymin)}{Log(ymax/ymin)} = \frac{py - pylow}{pyrange}
5605/// \f]
5606/// Let:
5607/// \f[ beta = Log(\frac{ymax}{ymin}) \f]
5608/// \f[ Log(\frac{y}{ymin}) = beta \times pylow - beta \times py \f]
5609/// \f[ \frac{y}{ymin} = e^{(beta \times pylow - beta \times py)} \f]
5610/// \f[ y = ymin \times e^{(beta \times pylow)} \times e^{(-beta \times py)}\f]
5611/// \f[ \Rightarrow y = fPixeltoYk \times e^{(fPixeltoY \times py)} \f]
5612/// \f[ fPixeltoYk = ymin \times e^{(beta \times pylow)} \f]
5613/// \f[ fPixeltoY = -beta \f]
5614///
5615/// #### Conversion from World coordinates to pixel coordinates
5616///
5617/// \f[ px = pxlow + u*pxrange \f]
5618/// \f[ = pxlow + Log(x/xmin)/alfa \f]
5619/// \f[ = pxlow -Log(xmin)/alfa + Log(x)/alfa \f]
5620/// \f[ = fXtoPixelk + fXtoPixel*Log(x) \f]
5621/// \f[ \Rightarrow fXtoPixelk = pxlow -Log(xmin)/alfa \f]
5622/// \f[ \Rightarrow fXtoPixel = 1/alfa \f]
5623///
5624/// \f[ py = pylow - Log(y/ymin)/beta \f]
5625/// \f[ = fYtoPixelk + fYtoPixel*Log(y) \f]
5626/// \f[ \Rightarrow fYtoPixelk = pylow - Log(ymin)/beta \f]
5627/// \f[ fYtoPixel = 1/beta \f]
5628
5630{
5631
5632 if (!gPad) {
5633 Error("ResizePad", "Cannot resize pad. No current pad available.");
5634 return;
5635 }
5636 if (gPad->GetWw()==0.0||gPad->GetWh()==0.0) {
5637 Warning("ResizePad", "gPad has at least one zero dimension.");
5638 return;
5639 }
5640 if (fX1==fX2||fY1==fY2) {
5641 Warning("ResizePad", "The pad has at least one zero dimension.");
5642 return;
5643 }
5644 // Recompute subpad positions in case pad has been moved/resized
5645 TPad *parent = fMother;
5646 if (this == gPad->GetCanvas()) {
5649 fAbsWNDC = fWNDC;
5650 fAbsHNDC = fHNDC;
5651 }
5652 else {
5653 if (parent->GetAbsHNDC()==0.0||parent->GetAbsWNDC()==0.0||fHNDC==0.0||fWNDC==0.0) {
5654 Warning("ResizePad", "The parent pad has at least one zero dimension.");
5655 return;
5656 }
5657 fAbsXlowNDC = fXlowNDC*parent->GetAbsWNDC() + parent->GetAbsXlowNDC();
5658 fAbsYlowNDC = fYlowNDC*parent->GetAbsHNDC() + parent->GetAbsYlowNDC();
5659 fAbsWNDC = fWNDC*parent->GetAbsWNDC();
5660 fAbsHNDC = fHNDC*parent->GetAbsHNDC();
5661 }
5662
5663 Double_t ww = (Double_t)gPad->GetWw();
5664 Double_t wh = (Double_t)gPad->GetWh();
5669
5670 // Linear X axis
5671 Double_t rounding = 0.; // was used before to adjust somehow wrong int trunctation by coordiantes transformation
5672 Double_t xrange = fX2 - fX1;
5673 fXtoAbsPixelk = rounding + pxlow - pxrange*fX1/xrange; //origin at left
5677 fPixeltoXk = fX1;
5679 // Linear Y axis
5680 Double_t yrange = fY2 - fY1;
5681 fYtoAbsPixelk = rounding + pylow - pyrange*fY1/yrange; //origin at top
5685 fPixeltoYk = fY1;
5687
5688 // Coefficients to convert from pad NDC coordinates to pixel coordinates
5689
5696
5697 // Coefficients to convert from canvas pixels to pad world coordinates
5698
5699 // Resize all sub-pads
5700 TObject *obj;
5701 if (!fPrimitives) fPrimitives = new TList;
5702 TIter next(GetListOfPrimitives());
5703 while ((obj = next())) {
5704 if (obj->InheritsFrom(TPad::Class()))
5705 ((TPad*)obj)->ResizePad(option);
5706 }
5707
5708 // Reset all current sizes
5709 if (gPad->IsBatch())
5710 fPixmapID = 0;
5711 else {
5712 if (GetPainter()){
5713 GetPainter()->SetLineWidth(-1);
5714 GetPainter()->SetTextSize(-1);
5715 }
5716 // create or re-create off-screen pixmap
5717 if (fPixmapID) {
5718 int w = TMath::Abs(XtoPixel(fX2) - XtoPixel(fX1));
5719 int h = TMath::Abs(YtoPixel(fY2) - YtoPixel(fY1));
5720 //protection in case of wrong pad parameters.
5721 //without this protection, the OpenPixmap or ResizePixmap crashes with
5722 //the message "Error in <RootX11ErrorHandler>: BadValue (integer parameter out of range for operation)"
5723 //resulting in a frozen xterm
5724 if ( !(TMath::Finite(fX1)) || !(TMath::Finite(fX2))
5725 || !(TMath::Finite(fY1)) || !(TMath::Finite(fY2))
5726 || (TMath::IsNaN(fX1)) || (TMath::IsNaN(fX2))
5727 || (TMath::IsNaN(fY1)) || (TMath::IsNaN(fY2)))
5728 Warning("ResizePad", "Inf/NaN propagated to the pad. Check drawn objects.");
5729 if (w <= 0 || w > 10000) {
5730 Warning("ResizePad", "%s width changed from %d to %d\n",GetName(),w,10);
5731 w = 10;
5732 }
5733 if (h <= 0 || h > 10000) {
5734 Warning("ResizePad", "%s height changed from %d to %d\n",GetName(),h,10);
5735 h = 10;
5736 }
5737 if (fPixmapID == -1) { // this case is handled via the ctor
5739 } else {
5740 if (gVirtualX) {
5741 if (gVirtualX->ResizePixmap(fPixmapID, w, h)) {
5742 Resized();
5743 Modified(kTRUE);
5744 }
5745 }
5746 }
5747 }
5748 }
5749 if (fView) {
5750 if (gPad == this) {
5751 fView->ResizePad();
5752 } else {
5753 TContext ctxt(this, kTRUE);
5754 fView->ResizePad();
5755 }
5756 }
5757}
5758
5759////////////////////////////////////////////////////////////////////////////////
5760/// Save the pad content in a file.
5761///
5762/// The file's format used to save the pad is determined by the `filename` extension:
5763///
5764/// - if `filename` is empty, the file produced is `padname.ps`
5765/// - if `filename` starts with a dot, the padname is added in front
5766/// - if `filename` ends with `.ps`, a Postscript file is produced
5767/// - if `filename` ends with `.eps`, an Encapsulated Postscript file is produced
5768/// - if `filename` ends with `.pdf`, a PDF file is produced NOTE: TMathText will be converted to TLatex; q.e.d., symbols only available in TMathText will not render properly.
5769/// - if `filename` ends with `.svg`, a SVG file is produced
5770/// - if `filename` ends with `.tex`, a TeX file is produced
5771/// - if `filename` ends with `.gif`, a GIF file is produced
5772/// - if `filename` ends with `.gif+NN`, an animated GIF file is produced See comments in TASImage::WriteImage for meaning of NN and other .gif sufix variants
5773/// - if `filename` ends with `.xpm`, a XPM file is produced
5774/// - if `filename` ends with `.png`, a PNG file is produced
5775/// - if `filename` ends with `.bmp`, a BMP file is produced
5776/// - if `filename` ends with `.jpg` or `.jpeg` a JPEG file is produced NOTE: JPEG's lossy compression will make all sharp edges fuzzy.
5777/// - if `filename` ends with `.tiff`, a TIFF file is produced
5778/// - if `filename` ends with `.C`, `.cxx`,`.cpp` or `.cc`, a C++ macro file is produced
5779/// - if `filename` ends with `.root`, a Root file is produced
5780/// - if `filename` ends with `.xml`, a XML file is produced
5781/// - if `filename` ends with `.json`, a JSON file is produced
5782///
5783/// \remarks
5784/// - The parameter `option` is not used.
5785/// - This method calls [TPad::Print(const char *filename, Option_t *option)](\ref TPadPrint)
5786/// the value of `option` is determined by the `filename` extension.
5787/// - Postscript and PDF formats allow to have [several pictures in one file](\ref TPadPrintPS).
5788
5789void TPad::SaveAs(const char *filename, Option_t * /*option*/) const
5790{
5793
5794 if (!lenfil) { psname = GetName(); psname.Append(".ps"); }
5795 else psname = filename;
5796
5797 // lines below protected against case like c1->SaveAs( "../ps/cs.ps" );
5798 if (psname.BeginsWith('.') && (psname.Contains('/') == 0)) {
5799 psname = GetName();
5800 psname.Append(filename);
5801 psname.Prepend("/");
5802 psname.Prepend(gEnv->GetValue("Canvas.PrintDirectory","."));
5803 }
5804
5805 if (psname.EndsWith(".gif"))
5806 ((TPad*)this)->Print(psname,"gif");
5807 else if (psname.Contains(".gif+"))
5808 ((TPad*)this)->Print(psname,"gif+");
5809 else if (psname.EndsWith(".C") || psname.EndsWith(".cxx") || psname.EndsWith(".cpp") || psname.EndsWith(".cc"))
5810 ((TPad*)this)->Print(psname,"cxx");
5811 else if (psname.EndsWith(".root"))
5812 ((TPad*)this)->Print(psname,"root");
5813 else if (psname.EndsWith(".xml"))
5814 ((TPad*)this)->Print(psname,"xml");
5815 else if (psname.EndsWith(".json"))
5816 ((TPad*)this)->Print(psname,"json");
5817 else if (psname.EndsWith(".eps"))
5818 ((TPad*)this)->Print(psname,"eps");
5819 else if (psname.EndsWith(".pdf"))
5820 ((TPad*)this)->Print(psname,"pdf");
5821 else if (psname.EndsWith(".pdf["))
5822 ((TPad*)this)->Print(psname,"pdf");
5823 else if (psname.EndsWith(".pdf]"))
5824 ((TPad*)this)->Print(psname,"pdf");
5825 else if (psname.EndsWith(".pdf("))
5826 ((TPad*)this)->Print(psname,"pdf");
5827 else if (psname.EndsWith(".pdf)"))
5828 ((TPad*)this)->Print(psname,"pdf");
5829 else if (psname.EndsWith(".svg"))
5830 ((TPad*)this)->Print(psname,"svg");
5831 else if (psname.EndsWith(".tex"))
5832 ((TPad*)this)->Print(psname,"tex");
5833 else if (psname.EndsWith(".xpm"))
5834 ((TPad*)this)->Print(psname,"xpm");
5835 else if (psname.EndsWith(".png"))
5836 ((TPad*)this)->Print(psname,"png");
5837 else if (psname.EndsWith(".jpg"))
5838 ((TPad*)this)->Print(psname,"jpg");
5839 else if (psname.EndsWith(".jpeg"))
5840 ((TPad*)this)->Print(psname,"jpg");
5841 else if (psname.EndsWith(".bmp"))
5842 ((TPad*)this)->Print(psname,"bmp");
5843 else if (psname.EndsWith(".tiff"))
5844 ((TPad*)this)->Print(psname,"tiff");
5845 else
5846 ((TPad*)this)->Print(psname,"ps");
5847}
5848
5849////////////////////////////////////////////////////////////////////////////////
5850/// Save primitives in this pad on the C++ source file out.
5851
5852void TPad::SavePrimitive(std::ostream &out, Option_t * option /*= ""*/)
5853{
5854 TContext ctxt(this, kFALSE); // not interactive
5855
5857
5858 // check for space in the pad name
5859 auto p = padName.Index(" ");
5860 if (p != kNPOS)
5861 padName.Resize(p);
5862
5863 TString opt = option;
5864 if (!opt.Contains("toplevel")) {
5865 static Int_t pcounter = 0;
5866 padName += TString::Format("__%d", pcounter++);
5867 padName = gInterpreter->MapCppName(padName);
5868 }
5869
5870 const char *pname = padName.Data();
5871 const char *cname = padName.Data();
5872
5873 if (padName.Length() == 0) {
5874 pname = "unnamed";
5875 if (this == gPad->GetCanvas())
5876 cname = "c1";
5877 else
5878 cname = "pad";
5879 }
5880
5881 // Write pad parameters
5882 if (this != gPad->GetCanvas()) {
5883 out << " \n";
5884 out << "// ------------>Primitives in pad: " << GetName() << "\n";
5885 out << " TPad *" << cname << " = new TPad(\"" << GetName() << "\", \""
5886 << TString(GetTitle()).ReplaceSpecialCppChars() << "\", " << fXlowNDC << ", " << fYlowNDC << ", "
5887 << fXlowNDC + fWNDC << ", " << fYlowNDC + fHNDC << ");\n";
5888 out << " " << cname << "->Draw();\n";
5889 out << " " << cname << "->cd();\n";
5890 }
5891 out << " " << cname << "->Range(" << fX1 << "," << fY1 << "," << fX2 << "," << fY2 << ");\n";
5892 TView *view = GetView();
5893 if (view) {
5894 Double_t rmin[3], rmax[3];
5895 view->GetRange(rmin, rmax);
5896 out << " TView::CreateView(1)->SetRange(" << rmin[0] << ", " << rmin[1] << ", " << rmin[2] << ", " << rmax[0]
5897 << ", " << rmax[1] << ", " << rmax[2] << ");\n";
5898 }
5899
5900 SaveFillAttributes(out, cname, 19, 1001);
5901
5902 if (GetBorderMode() != 1)
5903 out << " " << cname << "->SetBorderMode(" << GetBorderMode() << ");\n";
5904 if (GetBorderSize() != 4)
5905 out << " " << cname << "->SetBorderSize(" << GetBorderSize() << ");\n";
5906 if (GetLogx())
5907 out << " " << cname << "->SetLogx();\n";
5908 if (GetLogy())
5909 out << " " << cname << "->SetLogy();\n";
5910 if (GetLogz())
5911 out << " " << cname << "->SetLogz();\n";
5912 if (GetGridx())
5913 out << " " << cname << "->SetGridx();\n";
5914 if (GetGridy())
5915 out << " " << cname << "->SetGridy();\n";
5916 if (GetTickx())
5917 out << " " << cname << "->SetTickx(" << GetTickx() << ");\n";
5918 if (GetTicky())
5919 out << " " << cname << "->SetTicky(" << GetTicky() << ");\n";
5920 if (GetTheta() != 30)
5921 out << " " << cname << "->SetTheta(" << GetTheta() << ");\n";
5922 if (GetPhi() != 30)
5923 out << " " << cname << "->SetPhi(" << GetPhi() << ");\n";
5924 if (TMath::Abs(fLeftMargin - 0.1) > 0.01)
5925 out << " " << cname << "->SetLeftMargin(" << GetLeftMargin() << ");\n";
5926 if (TMath::Abs(fRightMargin - 0.1) > 0.01)
5927 out << " " << cname << "->SetRightMargin(" << GetRightMargin() << ");\n";
5928 if (TMath::Abs(fTopMargin - 0.1) > 0.01)
5929 out << " " << cname << "->SetTopMargin(" << GetTopMargin() << ");\n";
5930 if (TMath::Abs(fBottomMargin - 0.1) > 0.01)
5931 out << " " << cname << "->SetBottomMargin(" << GetBottomMargin() << ");\n";
5932
5934 out << " " << cname << "->SetFrameFillColor(" << TColor::SavePrimitiveColor(GetFrameFillColor()) << ");\n";
5935 if (GetFrameFillStyle() != 1001)
5936 out << " " << cname << "->SetFrameFillStyle(" << GetFrameFillStyle() << ");\n";
5937 if (GetFrameLineStyle() != 1)
5938 out << " " << cname << "->SetFrameLineStyle(" << GetFrameLineStyle() << ");\n";
5939 if (GetFrameLineColor() != 1)
5940 out << " " << cname << "->SetFrameLineColor(" << TColor::SavePrimitiveColor(GetFrameLineColor()) << ");\n";
5941 if (GetFrameLineWidth() != 1)
5942 out << " " << cname << "->SetFrameLineWidth(" << GetFrameLineWidth() << ");\n";
5943 if (GetFrameBorderMode() != 1)
5944 out << " " << cname << "->SetFrameBorderMode(" << GetFrameBorderMode() << ");\n";
5945 if (GetFrameBorderSize() != 1)
5946 out << " " << cname << "->SetFrameBorderSize(" << GetFrameBorderSize() << ");\n";
5947
5948 TFrame *frame = fFrame;
5949 if (!frame)
5950 frame = (TFrame *)GetPrimitive("TFrame");
5951 if (frame) {
5952 if (frame->GetFillColor() != GetFillColor())
5953 out << " " << cname << "->SetFrameFillColor(" << TColor::SavePrimitiveColor(frame->GetFillColor()) << ");\n";
5954 if (frame->GetFillStyle() != 1001)
5955 out << " " << cname << "->SetFrameFillStyle(" << frame->GetFillStyle() << ");\n";
5956 if (frame->GetLineStyle() != 1)
5957 out << " " << cname << "->SetFrameLineStyle(" << frame->GetLineStyle() << ");\n";
5958 if (frame->GetLineColor() != 1)
5959 out << " " << cname << "->SetFrameLineColor(" << TColor::SavePrimitiveColor(frame->GetLineColor()) << ");\n";
5960 if (frame->GetLineWidth() != 1)
5961 out << " " << cname << "->SetFrameLineWidth(" << frame->GetLineWidth() << ");\n";
5962 if (frame->GetBorderMode() != 1)
5963 out << " " << cname << "->SetFrameBorderMode(" << frame->GetBorderMode() << ");\n";
5964 if (frame->GetBorderSize() != 1)
5965 out << " " << cname << "->SetFrameBorderSize(" << frame->GetBorderSize() << ");\n";
5966 }
5967
5968 TIter next(GetListOfPrimitives());
5969
5970 while (auto obj = next()) {
5971 obj->SavePrimitive(out, (Option_t *)next.GetOption());
5972 if (obj->InheritsFrom(TPad::Class())) {
5973 if (opt.Contains("toplevel"))
5974 out << " " << pname << "->cd();\n";
5975 else
5976 out << " " << cname << "->cd();\n";
5977 }
5978 }
5979 out << " " << cname << "->Modified();\n";
5980}
5981
5982////////////////////////////////////////////////////////////////////////////////
5983/// Fix pad aspect ratio to current value if fixed is true.
5984
5986{
5987 if (fixed) {
5988 if (!fFixedAspectRatio) {
5989 if (fHNDC != 0.)
5991 else {
5992 Error("SetAspectRatio", "cannot fix aspect ratio, height of pad is 0");
5993 return;
5994 }
5996 }
5997 } else {
5999 fAspectRatio = 0;
6000 }
6001}
6002
6003////////////////////////////////////////////////////////////////////////////////
6004/// Set pad editable yes/no
6005/// If a pad is not editable:
6006/// - one cannot modify the pad and its objects via the mouse.
6007/// - one cannot add new objects to the pad
6008
6010{
6011 fEditable = mode;
6012
6013 TObject *obj;
6014 if (!fPrimitives) fPrimitives = new TList;
6015 TIter next(GetListOfPrimitives());
6016 while ((obj = next())) {
6017 if (obj->InheritsFrom(TPad::Class())) {
6018 TPad *pad = (TPad*)obj;
6019 pad->SetEditable(mode);
6020 }
6021 }
6022}
6023
6024////////////////////////////////////////////////////////////////////////////////
6025/// Override TAttFill::FillStyle for TPad because we want to handle style=0
6026/// as style 4000.
6027
6029{
6030 if (fstyle == 0) fstyle = 4000;
6032}
6033
6034////////////////////////////////////////////////////////////////////////////////
6035/// Set Lin/Log scale for X
6036/// - value = 0 X scale will be linear
6037/// - value = 1 X scale will be logarithmic (base 10)
6038/// - value > 1 reserved for possible support of base e or other
6039
6041{
6042 fLogx = value;
6043 delete fView; fView = nullptr;
6044 Modified();
6046}
6047
6048////////////////////////////////////////////////////////////////////////////////
6049/// Set Lin/Log scale for Y
6050/// - value = 0 Y scale will be linear
6051/// - value = 1 Y scale will be logarithmic (base 10)
6052/// - value > 1 reserved for possible support of base e or other
6053
6055{
6056 fLogy = value;
6057 delete fView; fView = nullptr;
6058 Modified();
6060}
6061
6062////////////////////////////////////////////////////////////////////////////////
6063/// Set Lin/Log scale for Z
6064
6066{
6067 fLogz = value;
6068 delete fView; fView = nullptr;
6069 Modified();
6071}
6072
6073////////////////////////////////////////////////////////////////////////////////
6074/// Set canvas range for pad and resize the pad. If the aspect ratio
6075/// was fixed before the call it will be un-fixed.
6076
6078{
6079 // Reorder points to make sure xlow,ylow is bottom left point and
6080 // xup,yup is top right point.
6081 if (xup < xlow) {
6082 Double_t x = xlow;
6083 xlow = xup;
6084 xup = x;
6085 }
6086 if (yup < ylow) {
6087 Double_t y = ylow;
6088 ylow = yup;
6089 yup = y;
6090 }
6091
6092 // Check if the new pad position is valid.
6093 if ((xlow < 0) || (xlow > 1) || (ylow < 0) || (ylow > 1)) {
6094 Error("TPad", "illegal bottom left position: x=%f, y=%f", xlow, ylow);
6095 return;
6096 }
6097 if ((xup < 0) || (xup > 1) || (yup < 0) || (yup > 1)) {
6098 Error("TPad", "illegal top right position: x=%f, y=%f", xup, yup);
6099 return;
6100 }
6101 if (xup-xlow <= 0) {
6102 Error("TPad", "illegal width: %f", xup-xlow);
6103 return;
6104 }
6105 if (yup-ylow <= 0) {
6106 Error("TPad", "illegal height: %f", yup-ylow);
6107 return;
6108 }
6109
6110 fXlowNDC = xlow;
6111 fYlowNDC = ylow;
6112 fXUpNDC = xup;
6113 fYUpNDC = yup;
6114 fWNDC = xup - xlow;
6115 fHNDC = yup - ylow;
6116
6118
6119 ResizePad();
6120}
6121
6122////////////////////////////////////////////////////////////////////////////////
6123/// Set all pad parameters.
6124
6125void TPad::SetPad(const char *name, const char *title,
6126 Double_t xlow, Double_t ylow, Double_t xup, Double_t yup,
6128{
6129 fName = name;
6130 fTitle = title;
6131 SetFillStyle(1001);
6136 if (color >= 0) SetFillColor(color);
6139 else fBorderSize = bordersize;
6141 else fBorderMode = bordermode;
6142
6143 SetPad(xlow, ylow, xup, yup);
6144}
6145
6146////////////////////////////////////////////////////////////////////////////////
6147/// Set the current TView. Delete previous view if view=0
6148
6150{
6151 if (!view) delete fView;
6152 fView = view;
6153}
6154
6155////////////////////////////////////////////////////////////////////////////////
6156/// Set postscript fill area attributes.
6157
6159{
6160 if (gVirtualPS) {
6161 gVirtualPS->SetFillColor(color);
6163 }
6164}
6165
6166////////////////////////////////////////////////////////////////////////////////
6167/// Set postscript line attributes.
6168
6177
6178////////////////////////////////////////////////////////////////////////////////
6179/// Set postscript marker attributes.
6180
6189
6190////////////////////////////////////////////////////////////////////////////////
6191/// Set postscript text attributes.
6192
6194{
6195 if (gVirtualPS) {
6196 gVirtualPS->SetTextAlign(align);
6198 gVirtualPS->SetTextColor(color);
6199 gVirtualPS->SetTextFont(font);
6200 if (font%10 > 2) {
6201 Float_t wh = (Float_t)gPad->XtoPixel(gPad->GetX2());
6202 Float_t hh = (Float_t)gPad->YtoPixel(gPad->GetY1());
6203 Float_t dy;
6204 if (wh < hh) {
6206 tsize = dy/(fX2-fX1);
6207 } else {
6209 tsize = dy/(fY2-fY1);
6210 }
6211 }
6213 }
6214}
6215
6216////////////////////////////////////////////////////////////////////////////////
6217/// Draw Arrows to indicated equal distances of Objects with given BBoxes.
6218/// Used by ShowGuidelines
6219
6221{
6222 Int_t lineColor = TColor::GetColor(239, 202, 0);
6223 Int_t x1,x2,y1,y2;
6224 x1 = x2 = y1 = y2 = 0;
6225 if (mode == 'x') {
6226 if (aBBox.fX<bBBox.fX) {
6227 x1 = aBBox.fX+aBBox.fWidth;
6228 x2 = bBBox.fX;
6229 }
6230 else {
6231 x1 = bBBox.fX+bBBox.fWidth;
6232 x2 = aBBox.fX;
6233 }
6234
6235 if ((aBBox.fY > bBBox.fY) && (aBBox.fY + aBBox.fHeight < bBBox.fY + bBBox.fHeight))
6236 y1 = y2 = aBBox.fY + TMath::Nint(0.5*(Double_t)(aBBox.fHeight))+1;
6237 else if ((bBBox.fY > aBBox.fY) && (bBBox.fY + bBBox.fHeight < aBBox.fY + aBBox.fHeight))
6238 y1 = y2 = bBBox.fY + TMath::Nint(0.5*(Double_t)(bBBox.fHeight))+1;
6239 else if (aBBox.fY>bBBox.fY) y1 = y2 = aBBox.fY-TMath::Nint(0.5*(Double_t)(aBBox.fY-(bBBox.fY+bBBox.fHeight)));
6240 else y1 = y2 = bBBox.fY-TMath::Nint(0.5*(Double_t)(bBBox.fY-(aBBox.fY+aBBox.fHeight)));
6241 }
6242 else if (mode == 'y') {
6243 if (aBBox.fY<bBBox.fY) {
6244 y1 = aBBox.fY+aBBox.fHeight;
6245 y2 = bBBox.fY;
6246 }
6247 else {
6248 y1 = bBBox.fY+bBBox.fHeight;
6249 y2 = aBBox.fY;
6250 }
6251 if ((aBBox.fX > bBBox.fX) && (aBBox.fX + aBBox.fWidth < bBBox.fX + bBBox.fWidth))
6252 x1 = x2 = aBBox.fX + TMath::Nint(0.5*(Double_t)(aBBox.fWidth))+1;
6253 else if ((bBBox.fX > aBBox.fX) && (bBBox.fX + bBBox.fWidth < aBBox.fX + aBBox.fWidth))
6254 x1 = x2 = bBBox.fX + TMath::Nint(0.5*(Double_t)(bBBox.fWidth))+1;
6255 else if (aBBox.fX>bBBox.fX) x1 = x2 = aBBox.fX+TMath::Nint(0.5*(Double_t)(bBBox.fX+bBBox.fWidth-aBBox.fX));
6256 else x1 = x2 = bBBox.fX+TMath::Nint(0.5*(Double_t)(aBBox.fX+aBBox.fWidth-bBBox.fX));
6257 }
6258
6259 TArrow *A = new TArrow(gPad->PixeltoX(x1), gPad->PixeltoY(y1-gPad->VtoPixel(0)), gPad->PixeltoX(x2), gPad->PixeltoY(y2-gPad->VtoPixel(0)), 0.01, "<|>");
6260 A->SetBit(kCanDelete);
6262 A->SetLineWidth(1);
6264 A->Draw();
6265
6266 return;
6267}
6268
6269////////////////////////////////////////////////////////////////////////////////
6270/// struct used by ShowGuidelines to store the distance Field between objects
6271/// in the canvas.
6272
6273struct dField {
6277 char fdir;
6278
6279
6281 : fa(nullptr), fb(nullptr), fdist(0), fdir(' ')
6282 {}
6283
6285 : fa(a), fb(b), fdist(dist), fdir(direction)
6286 {}
6287};
6288
6289////////////////////////////////////////////////////////////////////////////////
6290/// Shows lines to indicate if a TAttBBox2D object is aligned to
6291/// the center or to another object, shows distance arrows if two
6292/// objects on screen have the same distance to another object
6293/// Call from primitive in Execute Event, in ButtonMotion after
6294/// the new coordinates have been set, to 'stick'
6295/// once when button is up to delete lines
6296///
6297/// modes: t (Top), b (bottom), l (left), r (right), i (inside)
6298/// in resize modes (t,b,l,r) only size arrows are sticky
6299///
6300/// in mode, the function gets the point on the element that is clicked to
6301/// move (i) or resize (all others). The expected values are:
6302/// \image html gpad_pad5.png
6303
6304void TPad::ShowGuidelines(TObject *object, const Int_t event, const char mode, const bool cling )
6305{
6306 // When the object is moved with arrow or when the ShowGuideLines flag
6307 // is off we do show guide lines.
6308 if ((event == kArrowKeyRelease) || (event == kArrowKeyPress) ||
6309 !gEnv->GetValue("Canvas.ShowGuideLines", 0)) return;
6310
6311 std::vector<dField> curDist;
6312 std::vector<dField> otherDist;
6313 Int_t pMX, pMY;
6314 Double_t MX, MY;
6316 TList *prims;
6317 UInt_t n;
6319 aBBox = bBBox = Rectangle_t();
6320 TLine *L;
6321 TArrow *A;
6322 Int_t dSizeArrow = 12; // distance of arrows indicating same size from BBox in px
6323 Bool_t movedX, movedY; // make sure the current object is moved just once
6324 movedX = movedY = false;
6325 Bool_t resize = false; // indicates resize mode
6326 Bool_t log = gPad->GetLogx() || gPad->GetLogy();
6327 if (mode != 'i') resize = true;
6328
6329 TPad *is_pad = dynamic_cast<TPad *>( object );
6330
6332
6333 if (is_pad && is_pad->GetMother())
6334 is_pad->GetMother()->cd();
6335
6336 static TPad *tmpGuideLinePad = nullptr;
6337
6338 //delete all existing Guidelines and create new invisible pad
6339 if (tmpGuideLinePad) {
6340 ctxt.PadDeleted(tmpGuideLinePad);
6341 auto guidePadClicked = (object == tmpGuideLinePad); // in case of funny button click combination.
6342 tmpGuideLinePad->Delete();
6343 tmpGuideLinePad = nullptr;
6344 if (guidePadClicked) return;
6345 }
6346
6347 // Get Primitives
6348 prims = gPad->GetListOfPrimitives();
6349 n = TMath::Min(15,prims->GetSize());
6350 Int_t lineColor = TColor::GetColor(239, 202, 0);
6351
6352 TAttBBox2D *cur = dynamic_cast<TAttBBox2D *>( object );
6353 if (cur) {
6354 //create invisible TPad above gPad
6355 if (!tmpGuideLinePad){
6356 tmpGuideLinePad = new TPad("tmpGuideLinePad", "tmpGuideLinePad", 0, 0, 1, 1);
6357 Double_t x1, y1, x2, y2;
6358 gPad->GetRange(x1, y1, x2, y2);
6359 tmpGuideLinePad->Range(x1, y1, x2, y2);
6360 tmpGuideLinePad->SetFillStyle(0);
6361 tmpGuideLinePad->SetFillColor(0);
6362 tmpGuideLinePad->Draw();
6363 tmpGuideLinePad->cd();
6364 gPad->GetRange(x1, y1, x2, y2);
6365 }
6366 if (cling && !log) threshold = 7;
6367 else threshold = 1;
6368
6369 Rectangle_t BBox = cur->GetBBox();
6370 TPoint center = cur->GetBBoxCenter();
6371
6372 otherDist.clear();
6373 curDist.clear();
6374
6375 switch (event) {
6376
6377 case kButton1Down:
6378 case kButton1Motion:
6379 MX = gPad->GetX1() + 0.5 * (gPad->GetX2()-gPad->GetX1());
6380 MY = gPad->GetY1() + 0.5 * (gPad->GetY2()-gPad->GetY1());
6381 pMX = gPad->XtoPixel(MX);
6382 pMY = gPad->YtoPixel(MY);
6383 // Middlelines
6384 if (TMath::Abs(pMX-center.GetX())<threshold) {
6385 if (cling && (!resize)) {
6386 cur->SetBBoxCenterX(pMX);
6387 center = cur->GetBBoxCenter();
6388 BBox = cur->GetBBox();
6389 center = cur->GetBBoxCenter();
6390 }
6391 L = new TLine(MX, gPad->GetY1(), MX, gPad->GetY2());
6392 L->SetBit(kCanDelete);
6393 L->SetLineColor(lineColor);
6394 L->Draw();
6395 }
6396 if (TMath::Abs(pMY-center.GetY())<threshold) {
6397 if (cling && (!resize)) {
6398 cur->SetBBoxCenterY(pMY);
6399 center = cur->GetBBoxCenter();
6400 BBox = cur->GetBBox();
6401 center = cur->GetBBoxCenter();
6402 }
6403 L = new TLine(gPad->GetX1(), MY, gPad->GetX2(), MY);
6404 L->SetBit(kCanDelete);
6405 L->SetLineColor(lineColor);
6406 L->Draw();
6407 }
6408 // Alignment to other objects
6409 for (UInt_t i = 0; i<n; i++) {
6410 TAttBBox2D *other = dynamic_cast<TAttBBox2D *>( prims->At(i) );
6411 if (other) {
6412 if (other != cur) {
6413 TPoint centerOther = other->GetBBoxCenter();
6414 if (TMath::Abs(center.GetX()-centerOther.GetX())<threshold) {
6415 if (cling && (!resize)) {
6416 cur->SetBBoxCenterX(centerOther.GetX());
6417 BBox = cur->GetBBox();
6418 center = cur->GetBBoxCenter();
6419 }
6420 L = new TLine(gPad->PixeltoX(centerOther.GetX()), gPad->PixeltoY(center.GetY()-gPad->VtoPixel(0)),
6421 gPad->PixeltoX(centerOther.GetX()), gPad->PixeltoY(centerOther.GetY()-gPad->VtoPixel(0)));
6422 L->SetLineColor(lineColor);
6423 L->Draw();
6424 L->SetBit(kCanDelete);
6425 }
6426 if (TMath::Abs(center.GetY()-centerOther.GetY())<threshold) {
6427 if (cling && (!resize)) {
6428 cur->SetBBoxCenterY(centerOther.GetY());
6429 BBox = cur->GetBBox();
6430 center = cur->GetBBoxCenter();
6431 }
6432 L = new TLine(gPad->PixeltoX(center.GetX()), gPad->PixeltoY(centerOther.GetY()-gPad->VtoPixel(0)),
6433 gPad->PixeltoX(centerOther.GetX()), gPad->PixeltoY(centerOther.GetY()-gPad->VtoPixel(0)));
6434 L->SetBit(kCanDelete);
6435 L->SetLineColor(lineColor);
6436 L->Draw();
6437 }
6438 }
6439 }
6440 }
6441 // Get Distances between objects
6442 for (UInt_t i = 0; i<n; i++) {
6443 TAttBBox2D *a = dynamic_cast<TAttBBox2D *>( prims->At(i) );
6444 if (a) {
6445 aBBox = a->GetBBox();
6446 for (UInt_t j = i+1; j<n; j++) {
6447 TAttBBox2D *b = dynamic_cast<TAttBBox2D *>( prims->At(j) );
6448 if (b) {
6449 bBBox = b->GetBBox();
6450
6451 //only when bounding boxes overlap in x or y direction
6452 if (((aBBox.fX<bBBox.fX) && (bBBox.fX-aBBox.fX<=aBBox.fWidth))||((aBBox.fX>bBBox.fX) && (aBBox.fX-bBBox.fX<=bBBox.fWidth))){ //BBoxes overlap in x direction
6453 if ((aBBox.fY+aBBox.fHeight<bBBox.fY)||(bBBox.fY+bBBox.fHeight<aBBox.fY)) {//No overlap in Y-direction required
6454 dField abDist = dField();
6455 if (aBBox.fY>bBBox.fY) abDist = dField(a, b, TMath::Abs(aBBox.fY-(bBBox.fY+bBBox.fHeight)), 'y');
6456 else abDist = dField(a, b, TMath::Abs(bBBox.fY-(aBBox.fY+aBBox.fHeight)), 'y');
6457 if ((b != cur)&&(a != cur)) otherDist.push_back(abDist);
6458 else curDist.push_back(abDist);
6459 }
6460 } else if (((aBBox.fY<bBBox.fY) && (bBBox.fY-aBBox.fY<=aBBox.fHeight))||((aBBox.fY>bBBox.fY) && (aBBox.fY-bBBox.fY<=bBBox.fHeight))) { //BBoxes overlap in y direction
6461 if ((aBBox.fX+aBBox.fWidth<bBBox.fX)||(bBBox.fX+bBBox.fWidth<aBBox.fX)) {//No overlap in x-direction required
6462 dField abDist = dField();
6463 if (aBBox.fX>bBBox.fX) abDist = dField(a, b, TMath::Abs(aBBox.fX-(bBBox.fX+bBBox.fWidth)), 'x');
6464 else abDist = dField(a, b, TMath::Abs(bBBox.fX-(aBBox.fX+aBBox.fWidth)), 'x');
6465 if ((b != cur)&&(a != cur)) otherDist.push_back(abDist);
6466 else curDist.push_back(abDist);
6467 }
6468 }
6469 }
6470 }
6471 }
6472 }
6473 // Show equal distances
6474 for (UInt_t i = 0; i<curDist.size(); i++) {
6475 for (UInt_t j = 0; j<otherDist.size(); j++) {
6476 if ((curDist[i].fdir == otherDist[j].fdir) && (otherDist[j].fdir=='x') && (TMath::Abs(curDist[i].fdist-otherDist[j].fdist)<threshold)) {
6477 if (cling && (!movedX) && (!resize)) {
6478 if ((cur->GetBBoxCenter().fX < curDist[i].fb->GetBBoxCenter().fX)||(cur->GetBBoxCenter().fX < curDist[i].fa->GetBBoxCenter().fX))
6479 cur->SetBBoxCenterX(cur->GetBBoxCenter().fX - otherDist[j].fdist + curDist[i].fdist);
6480 else cur->SetBBoxCenterX(cur->GetBBoxCenter().fX + otherDist[j].fdist - curDist[i].fdist);
6481 movedX = true;
6482 }
6483 DrawDist(curDist[i].fa->GetBBox(), curDist[i].fb->GetBBox(), 'x');
6484 DrawDist(otherDist[j].fa->GetBBox(), otherDist[j].fb->GetBBox(), 'x');
6485 }
6486 if ((curDist[i].fdir == otherDist[j].fdir) && (otherDist[j].fdir=='y') && (TMath::Abs(curDist[i].fdist-otherDist[j].fdist)<threshold)) {
6487 if (cling && (!movedY) && (!resize)) {
6488 if ((cur->GetBBoxCenter().fY < curDist[i].fb->GetBBoxCenter().fY)||(cur->GetBBoxCenter().fY < curDist[i].fa->GetBBoxCenter().fY))
6489 cur->SetBBoxCenterY(cur->GetBBoxCenter().fY - otherDist[j].fdist + curDist[i].fdist);
6490 else cur->SetBBoxCenterY(cur->GetBBoxCenter().fY + otherDist[j].fdist - curDist[i].fdist);
6491 movedY = true;
6492 }
6493 DrawDist(curDist[i].fa->GetBBox(), curDist[i].fb->GetBBox(), 'y');
6494 DrawDist(otherDist[j].fa->GetBBox(), otherDist[j].fb->GetBBox(), 'y');
6495 }
6496 }
6497 for (UInt_t j = i; j<curDist.size(); j++) {
6498 if (i!=j) {
6499 if ((curDist[i].fdir == curDist[j].fdir) && (curDist[j].fdir=='x') && (TMath::Abs(curDist[i].fdist-curDist[j].fdist)<threshold)) {
6500 if (cling && (!movedX) && (!resize)) {
6501 if ((cur->GetBBoxCenter().fX < curDist[i].fb->GetBBoxCenter().fX)||(cur->GetBBoxCenter().fX < curDist[i].fa->GetBBoxCenter().fX))
6502 cur->SetBBoxCenterX(cur->GetBBoxCenter().fX - floor(0.5*(curDist[j].fdist - curDist[i].fdist)));
6503 else cur->SetBBoxCenterX(cur->GetBBoxCenter().fX + floor(0.5*(curDist[j].fdist - curDist[i].fdist)));
6504 }
6505 DrawDist(curDist[i].fa->GetBBox(), curDist[i].fb->GetBBox(), 'x');
6506 DrawDist(curDist[j].fa->GetBBox(), curDist[j].fb->GetBBox(), 'x');
6507 }
6508
6509 if ((curDist[i].fdir == curDist[j].fdir) && (curDist[j].fdir=='y') && (TMath::Abs(curDist[i].fdist-curDist[j].fdist)<threshold)) {
6510 if (cling && (!movedY) && (!resize)) {
6511 if ((cur->GetBBoxCenter().fY < curDist[i].fb->GetBBoxCenter().fY)||(cur->GetBBoxCenter().fY < curDist[i].fa->GetBBoxCenter().fY))
6512 cur->SetBBoxCenterY(cur->GetBBoxCenter().fY - floor(0.5*(curDist[j].fdist - curDist[i].fdist)));
6513 else cur->SetBBoxCenterY(cur->GetBBoxCenter().fY + floor(0.5*(curDist[j].fdist - curDist[i].fdist)));
6514 }
6515 DrawDist(curDist[i].fa->GetBBox(), curDist[i].fb->GetBBox(), 'y');
6516 DrawDist(curDist[j].fa->GetBBox(), curDist[j].fb->GetBBox(), 'y');
6517 }
6518 }
6519 }
6520 }
6521 if (resize) {
6522 // Show equal Sizes
6523 for (UInt_t i = 0; i<n; i++) {
6524 TAttBBox2D *a = dynamic_cast<TAttBBox2D *>( prims->At(i) );
6525 if (a && (cur != a)) {
6526 aBBox = a->GetBBox();
6527
6528 if ((TMath::Abs(aBBox.fWidth - BBox.fWidth)<threshold) && (mode != 't') && (mode != 'b')) {
6529 if (cling) {
6530 if (mode == 'l') cur->SetBBoxX1(BBox.fX + BBox.fWidth - aBBox.fWidth);
6531 if (mode == 'r') cur->SetBBoxX2(BBox.fX + aBBox.fWidth);
6532 if ((mode == '1')||(mode == '4')) cur->SetBBoxX1(BBox.fX + BBox.fWidth - aBBox.fWidth);
6533 if ((mode == '2')||(mode == '3')) cur->SetBBoxX2(BBox.fX + aBBox.fWidth);
6534 BBox = cur->GetBBox();
6535 }
6536
6537 A = new TArrow(gPad->PixeltoX(aBBox.fX), gPad->PixeltoY(aBBox.fY-dSizeArrow-gPad->VtoPixel(0)),
6538 gPad->PixeltoX(aBBox.fX+aBBox.fWidth), gPad->PixeltoY(aBBox.fY-dSizeArrow-gPad->VtoPixel(0)), 0.01, "<|>");
6539 A->SetBit(kCanDelete);
6542 A->Draw();
6543
6544 A = new TArrow(gPad->PixeltoX(BBox.fX), gPad->PixeltoY(BBox.fY-dSizeArrow-gPad->VtoPixel(0)),
6545 gPad->PixeltoX(BBox.fX+BBox.fWidth), gPad->PixeltoY(BBox.fY-dSizeArrow-gPad->VtoPixel(0)), 0.01, "<|>");
6546 A->SetBit(kCanDelete);
6549 A->Draw();
6550 }
6551 if ((TMath::Abs(aBBox.fHeight - BBox.fHeight)<threshold) && (mode != 'r') && (mode != 'l')) {
6552 if (cling) {
6553 if (mode == 't') cur->SetBBoxY1(BBox.fY + BBox.fHeight - aBBox.fHeight);
6554 if (mode == 'b') cur->SetBBoxY2(BBox.fY + aBBox.fHeight);
6555 if ((mode == '1')||(mode == '2')) cur->SetBBoxY1(BBox.fY + BBox.fHeight - aBBox.fHeight);
6556 if ((mode == '3')||(mode == '4')) cur->SetBBoxY2(BBox.fY + aBBox.fHeight);
6557 BBox = cur->GetBBox();
6558 }
6559 A = new TArrow(gPad->PixeltoX(aBBox.fX-dSizeArrow), gPad->PixeltoY(aBBox.fY-gPad->VtoPixel(0)),
6560 gPad->PixeltoX(aBBox.fX-dSizeArrow), gPad->PixeltoY(aBBox.fY+aBBox.fHeight-gPad->VtoPixel(0)), 0.01, "<|>");
6561 A->SetBit(kCanDelete);
6564 A->Draw();
6565
6566 A = new TArrow(gPad->PixeltoX(BBox.fX-dSizeArrow), gPad->PixeltoY(BBox.fY-gPad->VtoPixel(0)),
6567 gPad->PixeltoX(BBox.fX-dSizeArrow), gPad->PixeltoY(BBox.fY+BBox.fHeight-gPad->VtoPixel(0)), 0.01, "<|>");
6568 A->SetBit(kCanDelete);
6571 A->Draw();
6572 }
6573 }
6574 }
6575 }
6576
6577 break;
6578
6579 case kButton1Up:
6580 if (tmpGuideLinePad) {
6581 // All the arrows and lines in that pad are also deleted because
6582 // they all have the bit kCanDelete on.
6583 tmpGuideLinePad->Delete();
6584 tmpGuideLinePad = nullptr;
6585 }
6586 break;
6587 }
6588 }
6589
6590 gPad->Modified(kTRUE);
6591}
6592
6593////////////////////////////////////////////////////////////////////////////////
6594/// Return kTRUE if the crosshair has been activated (via SetCrosshair).
6595
6597{
6598 return (Bool_t)GetCrosshair();
6599}
6600
6601////////////////////////////////////////////////////////////////////////////////
6602/// Return the crosshair type (from the mother canvas)
6603/// crosshair type = 0 means no crosshair.
6604
6606{
6607 if (this == (TPad*)fCanvas)
6608 return fCrosshair;
6609 return fCanvas ? fCanvas->GetCrosshair() : 0;
6610}
6611
6612////////////////////////////////////////////////////////////////////////////////
6613/// Set crosshair active/inactive.
6614/// - If crhair != 0, a crosshair will be drawn in the pad and its sub-pads.
6615/// - If the canvas crhair = 1 , the crosshair spans the full canvas.
6616/// - If the canvas crhair > 1 , the crosshair spans only the pad.
6617
6619{
6620 if (!fCanvas) return;
6622 fCrosshairPos = 0;
6623
6624 if (this != (TPad*)fCanvas) fCanvas->SetCrosshair(crhair);
6625}
6626
6627////////////////////////////////////////////////////////////////////////////////
6628/// static function to set the maximum Pick Distance fgMaxPickDistance
6629/// This parameter is used in TPad::Pick to select an object if
6630/// its DistancetoPrimitive returns a value < fgMaxPickDistance
6631/// The default value is 5 pixels. Setting a smaller value will make
6632/// picking more precise but also more difficult
6633
6638
6639////////////////////////////////////////////////////////////////////////////////
6640/// Set tool tip text associated with this pad. The delay is in
6641/// milliseconds (minimum 250). To remove tool tip call method with
6642/// text = 0.
6643
6645{
6646 if (fTip) {
6648 fTip = nullptr;
6649 }
6650
6651 if (text && strlen(text))
6652 fTip = CreateToolTip((TBox*)nullptr, text, delayms);
6653}
6654
6655////////////////////////////////////////////////////////////////////////////////
6656/// Set pad vertical (default) or horizontal
6657
6659{
6660 if (vert) ResetBit(kHori);
6661 else SetBit(kHori);
6662}
6663
6664////////////////////////////////////////////////////////////////////////////////
6665/// Stream a class object.
6666
6668{
6669 UInt_t R__s, R__c;
6672 TObject *obj;
6673 if (b.IsReading()) {
6674 Version_t v = b.ReadVersion(&R__s, &R__c);
6675 if (v > 5) {
6676 if (!gPad)
6677 gPad = new TCanvas(GetName());
6678 fMother = (TPad*)gPad;
6680 TContext ctxt(this, kFALSE);
6681 fPixmapID = -1; // -1 means pixmap will be created by ResizePad()
6682 gReadLevel++;
6683 gROOT->SetReadingObject(kTRUE);
6684
6685 b.ReadClassBuffer(TPad::Class(), this, v, R__s, R__c);
6686
6687 //Set the kCanDelete bit in all objects in the pad such that when the pad
6688 //is deleted all objects in the pad are deleted too.
6689 //Also set must cleanup bit which normally set for all primitives add to pad,
6690 // but may be reset in IO like TH1::Streamer does
6691 TIter next(fPrimitives);
6692 while ((obj = next())) {
6693 obj->SetBit(kCanDelete);
6694 obj->SetBit(kMustCleanup);
6695 }
6696
6697 fModified = kTRUE;
6698 fPadPointer = nullptr;
6699 gReadLevel--;
6700 if (gReadLevel == 0 && IsA() == TPad::Class()) ResizePad();
6701 gROOT->SetReadingObject(kFALSE);
6702 return;
6703 }
6704
6705 //====process old versions before automatic schema evolution
6706 if (v < 5) { //old TPad in single precision
6707 if (v < 3) { //old TPad derived from TWbox
6708 b.ReadVersion(); // TVirtualPad::Streamer(b)
6709 b.ReadVersion(); // TWbox::Streamer(b)
6710 b.ReadVersion(); // TBox::Streamer(b)
6714 b >> single; fX1 = single;
6715 b >> single; fY1 = single;
6716 b >> single; fX2 = single;
6717 b >> single; fY2 = single;
6718 b >> fBorderSize;
6719 b >> fBorderMode;
6721 } else { //new TPad
6724 b >> single; fX1 = single;
6725 b >> single; fY1 = single;
6726 b >> single; fX2 = single;
6727 b >> single; fY2 = single;
6728 b >> fBorderSize;
6729 b >> fBorderMode;
6730 }
6731 b >> fLogx;
6732 b >> fLogy;
6733 b >> fLogz;
6735 b >> single; fXtoPixelk = single;
6736 b >> single; fXtoPixel = single;
6738 b >> single; fYtoPixelk = single;
6739 b >> single; fYtoPixel = single;
6741 b >> single; fUtoPixelk = single;
6742 b >> single; fUtoPixel = single;
6744 b >> single; fVtoPixelk = single;
6745 b >> single; fVtoPixel = single;
6747 b >> single; fPixeltoXk = single;
6748 b >> single; fPixeltoX = single;
6750 b >> single; fPixeltoYk = single;
6751 b >> single; fPixeltoY = single;
6752 b >> single; fXlowNDC = single;
6753 b >> single; fYlowNDC = single;
6754 b >> single; fWNDC = single;
6755 b >> single; fHNDC = single;
6756 b >> single; fAbsXlowNDC = single;
6757 b >> single; fAbsYlowNDC = single;
6758 b >> single; fAbsWNDC = single;
6759 b >> single; fAbsHNDC = single;
6760 b >> single; fUxmin = single;
6761 b >> single; fUymin = single;
6762 b >> single; fUxmax = single;
6763 b >> single; fUymax = single;
6764 } else {
6767 b >> fX1;
6768 b >> fY1;
6769 b >> fX2;
6770 b >> fY2;
6771 b >> fBorderSize;
6772 b >> fBorderMode;
6773 b >> fLogx;
6774 b >> fLogy;
6775 b >> fLogz;
6776 b >> fXtoAbsPixelk;
6777 b >> fXtoPixelk;
6778 b >> fXtoPixel;
6779 b >> fYtoAbsPixelk;
6780 b >> fYtoPixelk;
6781 b >> fYtoPixel;
6782 b >> fUtoAbsPixelk;
6783 b >> fUtoPixelk;
6784 b >> fUtoPixel;
6785 b >> fVtoAbsPixelk;
6786 b >> fVtoPixelk;
6787 b >> fVtoPixel;
6788 b >> fAbsPixeltoXk;
6789 b >> fPixeltoXk;
6790 b >> fPixeltoX;
6791 b >> fAbsPixeltoYk;
6792 b >> fPixeltoYk;
6793 b >> fPixeltoY;
6794 b >> fXlowNDC;
6795 b >> fYlowNDC;
6796 b >> fWNDC;
6797 b >> fHNDC;
6798 b >> fAbsXlowNDC;
6799 b >> fAbsYlowNDC;
6800 b >> fAbsWNDC;
6801 b >> fAbsHNDC;
6802 b >> fUxmin;
6803 b >> fUymin;
6804 b >> fUxmax;
6805 b >> fUymax;
6806 }
6807
6808 if (!gPad)
6809 gPad = new TCanvas(GetName());
6810 if (gReadLevel == 0)
6812 else
6813 fMother = (TPad *)gPad;
6814 if (!fMother)
6815 fMother = (TPad *)gPad;
6816 if (fMother)
6818 gPad = fMother;
6819 fPixmapID = -1; // -1 means pixmap will be created by ResizePad()
6820 //-------------------------
6821 // read objects and their drawing options
6822 // b >> fPrimitives;
6823 gReadLevel++;
6824 gROOT->SetReadingObject(kTRUE);
6825 fPrimitives = new TList;
6826 b >> nobjects;
6827 if (nobjects > 0) {
6828 TContext ctxt(this, kFALSE);
6829 char drawoption[64];
6830 for (Int_t i = 0; i < nobjects; i++) {
6831 b >> obj;
6832 b >> nch;
6833 b.ReadFastArray(drawoption,nch);
6835 gPad = this; // gPad may be modified in b >> obj if obj is a pad
6836 }
6837 }
6838 gReadLevel--;
6839 gROOT->SetReadingObject(kFALSE);
6840 //////////////////////////////////////////////////////////////////////////
6841
6842 if (v > 3) {
6843 b >> fExecs;
6844 }
6845 fName.Streamer(b);
6846 fTitle.Streamer(b);
6847 b >> fPadPaint;
6848 fModified = kTRUE;
6849 b >> fGridx;
6850 b >> fGridy;
6851 b >> fFrame;
6852 b >> fView;
6853 if (v < 5) {
6854 b >> single; fTheta = single;
6855 b >> single; fPhi = single;
6856 } else {
6857 b >> fTheta;
6858 b >> fPhi;
6859 }
6860 fPadPointer = nullptr;
6861 b >> fNumber;
6862 b >> fAbsCoord;
6863 if (v > 1) {
6864 b >> fTickx;
6865 b >> fTicky;
6866 } else {
6867 fTickx = fTicky = 0;
6868 }
6869 if (gReadLevel == 0 && IsA() == TPad::Class()) ResizePad();
6870 b.CheckByteCount(R__s, R__c, TPad::IsA());
6871 //====end of old versions
6872
6873 } else {
6874 b.WriteClassBuffer(TPad::Class(),this);
6875 }
6876}
6877
6878////////////////////////////////////////////////////////////////////////////////
6879/// Force a copy of current style for all objects in pad.
6880
6882{
6883 if (gStyle->IsReading()) {
6895 fLogx = gStyle->GetOptLogx();
6896 fLogy = gStyle->GetOptLogy();
6897 fLogz = gStyle->GetOptLogz();
6898 } else {
6913 }
6914
6915 if (!fPrimitives) fPrimitives = new TList;
6916 TIter next(GetListOfPrimitives());
6917 TObject *obj;
6918
6919 while ((obj = next())) {
6920 obj->UseCurrentStyle();
6921 }
6922
6923 TPaveText *title = (TPaveText*)FindObject("title");
6924 if (title) {
6925 if (gStyle->IsReading()) {
6927 title->SetTextFont(gStyle->GetTitleFont(""));
6930 if (!gStyle->GetOptTitle()) delete title;
6931 } else {
6933 gStyle->SetTitleFont(title->GetTextFont());
6936 }
6937 }
6939
6940 if (gStyle->IsReading()) Modified();
6941}
6942
6943////////////////////////////////////////////////////////////////////////////////
6944/// Loop and sleep until a primitive with name=pname is found in the pad.
6945///
6946/// If emode is given, the editor is automatically set to emode, ie
6947/// it is not required to have the editor control bar.
6948///
6949/// The possible values for emode are:
6950/// - emode = "" (default). User will select the mode via the editor bar
6951/// - emode = "Arc", "Line", "Arrow", "Button", "Diamond", "Ellipse",
6952/// - emode = "Pad","pave", "PaveLabel","PaveText", "PavesText",
6953/// - emode = "PolyLine", "CurlyLine", "CurlyArc", "Text", "Marker", "CutG"
6954///
6955/// If emode is specified and it is not valid, "PolyLine" is assumed. If emode
6956/// is not specified or ="", an attempt is to use pname[1...]
6957///
6958/// for example if pname="TArc", emode="Arc" will be assumed.
6959/// When this function is called within a macro, the macro execution
6960/// is suspended until a primitive corresponding to the arguments
6961/// is found in the pad.
6962///
6963/// If CRTL/C is typed in the pad, the function returns 0.
6964///
6965/// While this function is executing, one can use the mouse, interact
6966/// with the graphics pads, use the Inspector, Browser, TreeViewer, etc.
6967///
6968/// Examples:
6969/// ~~~ {.cpp}
6970/// c1.WaitPrimitive(); // Return the first created primitive
6971/// // whatever it is.
6972/// // If a double-click with the mouse is executed
6973/// // in the pad or any key pressed, the function
6974/// // returns 0.
6975/// c1.WaitPrimitive("ggg"); // Set the editor in mode "PolyLine/Graph"
6976/// // Create a polyline, then using the context
6977/// // menu item "SetName", change the name
6978/// // of the created TGraph to "ggg"
6979/// c1.WaitPrimitive("TArc");// Set the editor in mode "Arc". Returns
6980/// // as soon as a TArc object is created.
6981/// c1.WaitPrimitive("lat","Text"); // Set the editor in Text/Latex mode.
6982/// // Create a text object, then Set its name to "lat"
6983/// ~~~
6984/// The following macro waits for 10 primitives of any type to be created.
6985///
6986/// ~~~ {.cpp}
6987///{
6988/// TCanvas c1("c1");
6989/// TObject *obj;
6990/// for (Int_t i=0;i<10;i++) {
6991/// obj = gPad->WaitPrimitive();
6992/// if (!obj) break;
6993/// printf("Loop i=%d, found objIsA=%s, name=%s\n",
6994/// i,obj->ClassName(),obj->GetName());
6995/// }
6996///}
6997/// ~~~
6998///
6999/// If ROOT runs in batch mode a call to this method does nothing.
7000
7001TObject *TPad::WaitPrimitive(const char *pname, const char *emode)
7002{
7003 if (!gPad || IsWeb())
7004 return nullptr;
7005
7006 if (emode && strlen(emode)) gROOT->SetEditorMode(emode);
7007 if (gROOT->GetEditorMode() == 0 && pname && strlen(pname) > 2) gROOT->SetEditorMode(&pname[1]);
7008
7009 if (!fPrimitives) fPrimitives = new TList;
7011 TObject *oldlast = gPad->GetListOfPrimitives() ? gPad->GetListOfPrimitives()->Last() : nullptr;
7012 TObject *obj = nullptr;
7014 Bool_t hasname = pname && (strlen(pname) > 0);
7015 if (!pname[0] && !emode[0]) testlast = kTRUE;
7016 if (testlast) gROOT->SetEditorMode();
7017 while (!gSystem->ProcessEvents() && gROOT->GetSelectedPad() && gPad) {
7018 if (gROOT->GetEditorMode() == 0) {
7019 if (hasname) {
7020 obj = FindObject(pname);
7021 if (obj) return obj;
7022 }
7023 if (testlast) {
7024 if (!gPad->GetListOfPrimitives()) return nullptr;
7025 obj = gPad->GetListOfPrimitives()->Last();
7026 if (obj != oldlast) return obj;
7027 Int_t event = GetEvent();
7028 if (event == kButton1Double || event == kKeyPress) {
7029 //the following statement is required against other loop executions
7030 //before returning
7031 fCanvas->HandleInput((EEventType)-1,0,0);
7032 return nullptr;
7033 }
7034 }
7035 }
7036 gSystem->Sleep(10);
7037 }
7038
7039 return nullptr;
7040}
7041
7042////////////////////////////////////////////////////////////////////////////////
7043/// Create a tool tip and return its pointer.
7044
7046{
7047 if (gPad->IsBatch()) return nullptr;
7048 return (TObject*)gROOT->ProcessLineFast(TString::Format("new TGToolTip((TBox*)0x%zx,\"%s\",%d)",
7049 (size_t)box,text,(Int_t)delayms).Data());
7050}
7051
7052////////////////////////////////////////////////////////////////////////////////
7053/// Delete tool tip object.
7054
7056{
7057 // delete tip;
7058 if (!tip) return;
7059 gROOT->ProcessLineFast(TString::Format("delete (TGToolTip*)0x%zx", (size_t)tip).Data());
7060}
7061
7062////////////////////////////////////////////////////////////////////////////////
7063/// Reset tool tip, i.e. within time specified in CreateToolTip the
7064/// tool tip will pop up.
7065
7067{
7068 if (!tip) return;
7069 // tip->Reset(this);
7070 gROOT->ProcessLineFast(TString::Format("((TGToolTip*)0x%zx)->Reset((TPad*)0x%zx)",
7071 (size_t)tip,(size_t)this).Data());
7072}
7073
7074////////////////////////////////////////////////////////////////////////////////
7075/// Hide tool tip.
7076
7078{
7079 if (!tip) return;
7080 // tip->Hide();
7081 gROOT->ProcessLineFast(TString::Format("((TGToolTip*)0x%zx)->Hide()", (size_t)tip).Data());
7082}
7083
7084////////////////////////////////////////////////////////////////////////////////
7085/// Deprecated: use TPad::GetViewer3D() instead
7086
7088{
7089 ::Info("TPad::x3d()", "This function is deprecated. Use %s->GetViewer3D(\"x3d\") instead",this->GetName());
7090
7091 // Default on GetViewer3D is pad - for x3d it was x3d...
7092 if (!type || !type[0]) {
7093 type = "x3d";
7094 }
7096}
7097
7098////////////////////////////////////////////////////////////////////////////////
7099/// Create/obtain handle to 3D viewer. Valid types are:
7100/// - 'pad' - pad drawing via TViewer3DPad
7101/// any others registered with plugin manager supporting TVirtualViewer3D
7102/// If an invalid/null type is requested then the current viewer is returned
7103/// (if any), otherwise a default 'pad' type is returned
7104
7106{
7108
7109 if ((!type || !*type || (strstr(type, "gl") && !strstr(type, "ogl"))) && (!fCanvas || !fCanvas->UseGL()))
7110 type = "pad";
7111
7112 if (type && *type) {
7113 if (gPluginMgr->FindHandler("TVirtualViewer3D", type))
7114 validType = kTRUE;
7115 }
7116
7117 // Invalid/null type requested?
7118 if (!validType) {
7119 // Return current viewer if there is one
7120 if (fViewer3D)
7121 return fViewer3D;
7122 // otherwise default to the pad
7123 else
7124 type = "pad";
7125 }
7126
7127 // Ensure we can create the new viewer before removing any existing one
7128 TVirtualViewer3D *newViewer = nullptr;
7129
7131
7132 // External viewers need to be created via plugin manager via interface...
7133 if (!strstr(type,"pad")) {
7135
7136 if (!newViewer) {
7137 Warning("GetViewer3D", "Cannot create 3D viewer of type: %s", type);
7138 // Return the existing viewer
7139 return fViewer3D;
7140 }
7141
7142 if (strstr(type, "gl") && !strstr(type, "ogl")) {
7145 Modified();
7146 } else {
7148 }
7149
7150 } else {
7151 newViewer = new TViewer3DPad(*this);
7152 }
7153
7154 // If we had a previous viewer destroy it now
7155 // In this case we do take responsibility for destroying viewer
7156 // c.f. ReleaseViewer3D
7157 delete fViewer3D;
7158
7159 // Set and return new viewer
7161
7162 // Ensure any new external viewer is painted
7163 // For internal TViewer3DPad type we assume this is being
7164 // create on demand due to a paint - so this is not required
7165 if (createdExternal) {
7166 Modified();
7167 Update();
7168 }
7169
7170 return fViewer3D;
7171}
7172
7173////////////////////////////////////////////////////////////////////////////////
7174/// Release current (external) viewer
7175
7177{
7178 fViewer3D = nullptr;
7179
7180 // We would like to ensure the pad is repainted
7181 // when external viewer is closed down. However
7182 // a modify/paint call here will repaint the pad
7183 // before the external viewer window actually closes.
7184 // So the pad would have to be redraw twice over.
7185 // Currently we just have to live with the pad staying blank
7186 // any click in pad will refresh.
7187}
7188
7189////////////////////////////////////////////////////////////////////////////////
7190/// Get GL device.
7191
7193{
7194 return fGLDevice;
7195}
7196
7197////////////////////////////////////////////////////////////////////////////////
7198/// Emit RecordPave() signal.
7199
7201{
7202 Emit("RecordPave(const TObject*)", (Longptr_t)obj);
7203}
7204
7205////////////////////////////////////////////////////////////////////////////////
7206/// Emit RecordLatex() signal.
7207
7209{
7210 Emit("RecordLatex(const TObject*)", (Longptr_t)obj);
7211}
7212
7213////////////////////////////////////////////////////////////////////////////////
7214/// Get pad painter from TCanvas.
7215
7217{
7218 if (!fCanvas) return nullptr;
7219 return fCanvas->GetCanvasPainter();
7220}
7221
7222////////////////////////////////////////////////////////////////////////////////
7223/// Return the bounding Box of the Pad
7224
7226{
7227 Rectangle_t BBox{0, 0, 0, 0};
7228 if (gPad) {
7229 BBox.fX = gPad->XtoPixel(fXlowNDC * (gPad->GetX2() - gPad->GetX1()) + gPad->GetX1());
7230 BBox.fY = gPad->YtoPixel((fYlowNDC + fHNDC) * (gPad->GetY2() - gPad->GetY1()) + gPad->GetY1());
7231 BBox.fWidth = gPad->XtoPixel((fXlowNDC + fWNDC) * (gPad->GetX2() - gPad->GetX1()) + gPad->GetX1()) -
7232 gPad->XtoPixel(fXlowNDC * (gPad->GetX2() - gPad->GetX1()) + gPad->GetX1());
7233 BBox.fHeight = gPad->YtoPixel((fYlowNDC) * (gPad->GetY2() - gPad->GetY1()) + gPad->GetY1()) -
7234 gPad->YtoPixel((fYlowNDC + fHNDC) * (gPad->GetY2() - gPad->GetY1()) + gPad->GetY1());
7235 }
7236 return BBox;
7237}
7238
7239////////////////////////////////////////////////////////////////////////////////
7240/// Return the center of the Pad as TPoint in pixels
7241
7243{
7244 TPoint p(0, 0);
7245 if (gPad) {
7246 Double_t x = ((fXlowNDC + 0.5 * fWNDC) * (gPad->GetX2() - gPad->GetX1())) + gPad->GetX1();
7247 Double_t y = ((fYlowNDC + 0.5 * fHNDC) * (gPad->GetY2() - gPad->GetY1())) + gPad->GetY1();
7248 p.SetX(gPad->XtoPixel(x));
7249 p.SetY(gPad->YtoPixel(y));
7250 }
7251 return p;
7252}
7253
7254////////////////////////////////////////////////////////////////////////////////
7255/// Set center of the Pad
7256
7258{
7259 if (!gPad)
7260 return;
7261 fXlowNDC = (gPad->PixeltoX(p.GetX()) - gPad->GetX1()) / (gPad->GetX2() - gPad->GetX1()) - 0.5 * fWNDC;
7262 fYlowNDC =
7263 (gPad->PixeltoY(p.GetY() - gPad->VtoPixel(0)) - gPad->GetY1()) / (gPad->GetY2() - gPad->GetY1()) - 0.5 * fHNDC;
7264 ResizePad();
7265}
7266
7267////////////////////////////////////////////////////////////////////////////////
7268/// Set X coordinate of the center of the Pad
7269
7271{
7272 if (!gPad)
7273 return;
7274 fXlowNDC = (gPad->PixeltoX(x) - gPad->GetX1())/(gPad->GetX2()-gPad->GetX1())-0.5*fWNDC;
7275 ResizePad();
7276}
7277
7278////////////////////////////////////////////////////////////////////////////////
7279/// Set Y coordinate of the center of the Pad
7280
7282{
7283 if (!gPad)
7284 return;
7285 fYlowNDC = (gPad->PixeltoY(y - gPad->VtoPixel(0)) - gPad->GetY1()) / (gPad->GetY2() - gPad->GetY1()) - 0.5 * fHNDC;
7286 ResizePad();
7287}
7288
7289////////////////////////////////////////////////////////////////////////////////
7290/// Set lefthandside of BoundingBox to a value
7291/// (resize in x direction on left)
7292
7294{
7295 if (!gPad)
7296 return;
7297 fXlowNDC = (gPad->PixeltoX(x) - gPad->GetX1()) / (gPad->GetX2() - gPad->GetX1());
7299 ResizePad();
7300}
7301
7302////////////////////////////////////////////////////////////////////////////////
7303/// Set right hand side of BoundingBox to a value
7304/// (resize in x direction on right)
7305
7307{
7308 if (!gPad)
7309 return;
7310 fWNDC = (gPad->PixeltoX(x) - gPad->GetX1()) / (gPad->GetX2() - gPad->GetX1()) - fXlowNDC;
7311 ResizePad();
7312}
7313
7314////////////////////////////////////////////////////////////////////////////////
7315/// Set top of BoundingBox to a value (resize in y direction on top)
7316
7318{
7319 if (!gPad)
7320 return;
7321 fHNDC = (gPad->PixeltoY(y - gPad->VtoPixel(0)) - gPad->GetY1()) / (gPad->GetY2() - gPad->GetY1()) - fYlowNDC;
7322 ResizePad();
7323}
7324
7325////////////////////////////////////////////////////////////////////////////////
7326/// Set bottom of BoundingBox to a value
7327/// (resize in y direction on bottom)
7328
7330{
7331 if (!gPad)
7332 return;
7333 fYlowNDC = (gPad->PixeltoY(y - gPad->VtoPixel(0)) - gPad->GetY1()) / (gPad->GetY2() - gPad->GetY1());
7335 ResizePad();
7336}
7337
7338////////////////////////////////////////////////////////////////////////////////
7339/// Mark pad modified
7340/// Will be repainted when TCanvas::Update() will be called next time
7341
7343{
7344 if (!fModified && flag) Emit("Modified()");
7345 fModified = flag;
7346}
7347
7348////////////////////////////////////////////////////////////////////////////////
7349/// Convert absolute pixel into X/Y coordinates
7350
7356
7357
7358////////////////////////////////////////////////////////////////////////////////
7359/// Convert pixel to X coordinate
7360
7362{
7363 if (fAbsCoord) return fAbsPixeltoXk + px*fPixeltoX;
7364 else return fPixeltoXk + px*fPixeltoX;
7365}
7366
7367////////////////////////////////////////////////////////////////////////////////
7368/// Convert pixel to Y coordinate
7369
7371{
7372 if (fAbsCoord) return fAbsPixeltoYk + py*fPixeltoY;
7373 else return fPixeltoYk + py*fPixeltoY;
7374}
7375
7376////////////////////////////////////////////////////////////////////////////////
7377/// Convert pixel to X/Y coordinates
7378
7384
7385////////////////////////////////////////////////////////////////////////////////
7386/// Convert X/Y into absolute pixel coordinates
7387
7393
7394////////////////////////////////////////////////////////////////////////////////
7395/// Convert X/Y into pixel coordinates
7396
7398{
7399 xpixel = XtoPixel(x);
7400 ypixel = YtoPixel(y);
7401}
7402
7403////////////////////////////////////////////////////////////////////////////////
7404/// Convert X NDC to pixel
7405
7407{
7408 Double_t val;
7409 if (fAbsCoord) val = fUtoAbsPixelk + u*fUtoPixel;
7410 else val = u*fUtoPixel;
7411 if (val < -kMaxPixel) return -kMaxPixel;
7412 if (val > kMaxPixel) return kMaxPixel;
7413 return TMath::Nint(val);
7414}
7415
7416////////////////////////////////////////////////////////////////////////////////
7417/// Convert Y NDC to pixel
7418
7420{
7421 Double_t val;
7422 if (fAbsCoord) val = fVtoAbsPixelk + v*fVtoPixel;
7423 else val = fVtoPixelk + v*fVtoPixel;
7424 if (val < -kMaxPixel) return -kMaxPixel;
7425 if (val > kMaxPixel) return kMaxPixel;
7426 return TMath::Nint(val);
7427}
7428
7429////////////////////////////////////////////////////////////////////////////////
7430/// Convert X NDC to absolute pixel
7431
7436
7437////////////////////////////////////////////////////////////////////////////////
7438/// Convert Y NDC to absolute pixel
7439
7444
7445////////////////////////////////////////////////////////////////////////////////
7446/// Convert X coordinate to absolute pixel
7447
7449{
7451 if (val < -kMaxPixel) return -kMaxPixel;
7452 if (val > kMaxPixel) return kMaxPixel;
7453 return TMath::Nint(val);
7454}
7455
7456////////////////////////////////////////////////////////////////////////////////
7457/// Convert X coordinate to pixel
7458
7460{
7461 Double_t val;
7462 if (fAbsCoord) val = fXtoAbsPixelk + x*fXtoPixel;
7463 else val = fXtoPixelk + x*fXtoPixel;
7464 if (val < -kMaxPixel) return -kMaxPixel;
7465 if (val > kMaxPixel) return kMaxPixel;
7466 return TMath::Nint(val);
7467}
7468
7469////////////////////////////////////////////////////////////////////////////////
7470/// Convert Y coordinate to absolute pixel
7471
7473{
7475 if (val < -kMaxPixel) return -kMaxPixel;
7476 if (val > kMaxPixel) return kMaxPixel;
7477 return TMath::Nint(val);
7478}
7479
7480////////////////////////////////////////////////////////////////////////////////
7481/// Convert Y coordinate to pixel
7482
7484{
7485 Double_t val;
7486 if (fAbsCoord) val = fYtoAbsPixelk + y*fYtoPixel;
7487 else val = fYtoPixelk + y*fYtoPixel;
7488 if (val < -kMaxPixel) return -kMaxPixel;
7489 if (val > kMaxPixel) return kMaxPixel;
7490 return TMath::Nint(val);
7491}
EEventType
Definition Buttons.h:15
@ kMouseMotion
Definition Buttons.h:23
@ kWheelUp
Definition Buttons.h:18
@ kButton2Down
Definition Buttons.h:17
@ kKeyPress
Definition Buttons.h:20
@ kArrowKeyRelease
Definition Buttons.h:21
@ kButton1Double
Definition Buttons.h:24
@ kButton1Motion
Definition Buttons.h:20
@ kButton1Up
Definition Buttons.h:19
@ kWheelDown
Definition Buttons.h:18
@ kArrowKeyPress
Definition Buttons.h:21
@ kMouseLeave
Definition Buttons.h:23
@ kButton1Down
Definition Buttons.h:17
@ kButton1Locate
Definition Buttons.h:22
@ kMouseEnter
Definition Buttons.h:23
@ kMarker
Definition Buttons.h:34
@ kCurlyArc
Definition Buttons.h:38
@ kPad
Definition Buttons.h:30
@ kPolyLine
Definition Buttons.h:28
@ kDiamond
Definition Buttons.h:37
@ kPave
Definition Buttons.h:31
@ kArrow
Definition Buttons.h:33
@ kPaveText
Definition Buttons.h:32
@ kCutG
Definition Buttons.h:38
@ kLine
Definition Buttons.h:33
@ kPavesText
Definition Buttons.h:32
@ kCurlyLine
Definition Buttons.h:38
@ kPaveLabel
Definition Buttons.h:31
@ kButton
Definition Buttons.h:37
@ kEllipse
Definition Buttons.h:32
@ kText
Definition Buttons.h:30
@ kArc
Definition Buttons.h:33
ECursor
Definition GuiTypes.h:372
@ kRightSide
Definition GuiTypes.h:373
@ kBottomSide
Definition GuiTypes.h:373
@ kTopLeft
Definition GuiTypes.h:372
@ kBottomRight
Definition GuiTypes.h:372
@ kTopSide
Definition GuiTypes.h:373
@ kLeftSide
Definition GuiTypes.h:373
@ kMove
Definition GuiTypes.h:374
@ kTopRight
Definition GuiTypes.h:372
@ kBottomLeft
Definition GuiTypes.h:372
@ kHand
Definition GuiTypes.h:374
@ kCross
Definition GuiTypes.h:374
const Int_t kMaxPixel
Max value for an int.
Definition GuiTypes.h:369
#define SafeDelete(p)
Definition RConfig.hxx:541
#define d(i)
Definition RSha256.hxx:102
#define b(i)
Definition RSha256.hxx:100
#define f(i)
Definition RSha256.hxx:104
#define c(i)
Definition RSha256.hxx:101
#define g(i)
Definition RSha256.hxx:105
#define a(i)
Definition RSha256.hxx:99
#define h(i)
Definition RSha256.hxx:106
short Style_t
Definition RtypesCore.h:82
bool Bool_t
Definition RtypesCore.h:63
int Int_t
Definition RtypesCore.h:45
short Color_t
Definition RtypesCore.h:85
float Size_t
Definition RtypesCore.h:89
long Longptr_t
Definition RtypesCore.h:75
short Version_t
Definition RtypesCore.h:65
long Long_t
Definition RtypesCore.h:54
short Width_t
Definition RtypesCore.h:84
float Float_t
Definition RtypesCore.h:57
short Short_t
Definition RtypesCore.h:39
constexpr Bool_t kFALSE
Definition RtypesCore.h:94
double Double_t
Definition RtypesCore.h:59
constexpr Ssiz_t kNPOS
Definition RtypesCore.h:117
constexpr Bool_t kTRUE
Definition RtypesCore.h:93
const char Option_t
Definition RtypesCore.h:66
@ kRed
Definition Rtypes.h:66
@ kBlack
Definition Rtypes.h:65
ROOT::Detail::TRangeCast< T, true > TRangeDynCast
TRangeDynCast is an adapter class that allows the typed iteration through a TCollection.
#define gDirectory
Definition TDirectory.h:384
R__EXTERN TEnv * gEnv
Definition TEnv.h:170
#define gFile
Definition TFile.h:434
winID h TVirtualViewer3D TVirtualGLPainter p
Option_t Option_t option
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t cursor
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char ColorStruct_t color const char filename
Option_t Option_t SetFillStyle
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize wid
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t np
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t r
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char cname
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void value
Option_t Option_t TPoint TPoint const char x2
Option_t Option_t TPoint TPoint const char x1
Option_t Option_t TPoint TPoint angle
Option_t Option_t TPoint TPoint const char mode
Option_t Option_t TPoint TPoint const char y2
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void SetCursor
Option_t Option_t SetFillColor
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t WindowAttributes_t Float_t Float_t Float_t Int_t Int_t UInt_t UInt_t Rectangle_t Int_t Int_t Window_t TString Int_t GCValues_t GetPrimarySelectionOwner GetDisplay GetScreen GetColormap GetNativeEvent const char const char dpyName wid window const char font_name cursor keysym reg const char only_if_exist regb h Point_t winding char text const char depth char const char Int_t count const char ColorStruct_t color const char Pixmap_t Pixmap_t PictureAttributes_t attr const char char ret_data h unsigned char height h Atom_t Int_t ULong_t ULong_t unsigned char prop_list Atom_t Atom_t Atom_t Time_t type
Option_t Option_t style
Option_t Option_t TPoint TPoint const char GetTextMagnitude GetFillStyle GetLineColor GetLineWidth GetMarkerStyle GetTextAlign GetTextColor GetTextSize void char Point_t Rectangle_t button
Option_t Option_t TPoint TPoint const char text
Option_t Option_t TPoint TPoint const char y1
char name[80]
Definition TGX11.cxx:110
const Int_t kMAXLEVELS
Definition TGeometry.h:27
float xmin
float ymin
float xmax
float ymax
#define gInterpreter
@ kMustCleanup
Definition TObject.h:374
static Int_t gReadLevel
Definition TPad.cxx:66
static Bool_t ContainsTImage(TList *li)
Auxiliary function.
Definition TPad.cxx:4812
#define NotFree(i, j)
Definition TPad.cxx:3274
R__EXTERN TPluginManager * gPluginMgr
#define ClassImpQ(name)
Definition TQObject.h:283
R__EXTERN TVirtualMutex * gROOTMutex
Definition TROOT.h:63
#define gROOT
Definition TROOT.h:406
R__EXTERN TStyle * gStyle
Definition TStyle.h:442
R__EXTERN TSystem * gSystem
Definition TSystem.h:572
#define gGLManager
Definition TVirtualGL.h:159
#define R__LOCKGUARD(mutex)
R__EXTERN TVirtualPS * gVirtualPS
Definition TVirtualPS.h:81
#define gPad
R__EXTERN Int_t(* gThreadXAR)(const char *xact, Int_t nb, void **ar, Int_t *iret)
#define gVirtualX
Definition TVirtualX.h:337
#define snprintf
Definition civetweb.c:1540
Draw all kinds of Arrows.
Definition TArrow.h:29
void Draw(Option_t *option="") override
Draw this arrow with its current attributes.
Definition TArrow.cxx:119
static TClass * Class()
Abstract base class for elements drawn in the editor.
Definition TAttBBox2D.h:19
virtual void Streamer(TBuffer &)
virtual Color_t GetFillColor() const
Return the fill area color.
Definition TAttFill.h:31
virtual Style_t GetFillStyle() const
Return the fill area style.
Definition TAttFill.h:32
virtual void Modify()
Change current fill area attributes if necessary.
Definition TAttFill.cxx:216
Style_t fFillStyle
Fill area style.
Definition TAttFill.h:24
virtual void SetFillColor(Color_t fcolor)
Set the fill area color.
Definition TAttFill.h:38
static TClass * Class()
virtual Bool_t IsTransparent() const
Definition TAttFill.h:47
virtual void SetFillStyle(Style_t fstyle)
Set the fill area style.
Definition TAttFill.h:40
virtual void SaveFillAttributes(std::ostream &out, const char *name, Int_t coldef=1, Int_t stydef=1001)
Save fill attributes as C++ statement(s) on output stream out.
Definition TAttFill.cxx:239
virtual void Streamer(TBuffer &)
virtual Color_t GetLineColor() const
Return the line color.
Definition TAttLine.h:35
virtual void SetLineStyle(Style_t lstyle)
Set the line style.
Definition TAttLine.h:44
virtual Width_t GetLineWidth() const
Return the line width.
Definition TAttLine.h:37
virtual void SetLineWidth(Width_t lwidth)
Set the line width.
Definition TAttLine.h:45
static TClass * Class()
Width_t fLineWidth
Line width.
Definition TAttLine.h:25
virtual void SetLineColor(Color_t lcolor)
Set the line color.
Definition TAttLine.h:42
virtual Style_t GetLineStyle() const
Return the line style.
Definition TAttLine.h:36
virtual void Modify()
Change current line attributes if necessary.
Definition TAttLine.cxx:247
virtual void SetMarkerColor(Color_t mcolor=1)
Set the marker color.
Definition TAttMarker.h:39
virtual void SetMarkerStyle(Style_t mstyle=1)
Set the marker style.
Definition TAttMarker.h:41
virtual void SetMarkerSize(Size_t msize=1)
Set the marker size.
Definition TAttMarker.h:46
static TClass * Class()
virtual void SetBottomMargin(Float_t bottommargin)
Set Pad bottom margin in fraction of the pad height.
Definition TAttPad.cxx:99
Color_t GetFrameFillColor() const
Definition TAttPad.h:53
virtual void SetLeftMargin(Float_t leftmargin)
Set Pad left margin in fraction of the pad width.
Definition TAttPad.cxx:109
Color_t GetFrameLineColor() const
Definition TAttPad.h:54
Style_t GetFrameLineStyle() const
Definition TAttPad.h:56
Float_t fRightMargin
RightMargin.
Definition TAttPad.h:22
Style_t GetFrameFillStyle() const
Definition TAttPad.h:55
Float_t fLeftMargin
LeftMargin.
Definition TAttPad.h:21
Float_t fTopMargin
TopMargin.
Definition TAttPad.h:24
Float_t GetLeftMargin() const
Definition TAttPad.h:44
Width_t GetFrameLineWidth() const
Definition TAttPad.h:57
Float_t GetBottomMargin() const
Definition TAttPad.h:43
virtual void SetRightMargin(Float_t rightmargin)
Set Pad right margin in fraction of the pad width.
Definition TAttPad.cxx:119
Float_t GetRightMargin() const
Definition TAttPad.h:45
Int_t GetFrameBorderMode() const
Definition TAttPad.h:59
virtual void SetTopMargin(Float_t topmargin)
Set Pad top margin in fraction of the pad height.
Definition TAttPad.cxx:129
Width_t GetFrameBorderSize() const
Definition TAttPad.h:58
Float_t fBottomMargin
BottomMargin.
Definition TAttPad.h:23
Float_t GetTopMargin() const
Definition TAttPad.h:46
virtual void Streamer(TBuffer &)
Stream an object of class TAttPad.
Definition TAttPad.cxx:150
virtual Float_t GetTextSize() const
Return the text size.
Definition TAttText.h:38
virtual void SetTextAlign(Short_t align=11)
Set the text alignment.
Definition TAttText.h:44
virtual Short_t GetTextAlign() const
Return the text alignment.
Definition TAttText.h:34
virtual Font_t GetTextFont() const
Return the text font.
Definition TAttText.h:37
virtual Color_t GetTextColor() const
Return the text color.
Definition TAttText.h:36
virtual void SetTextAngle(Float_t tangle=0)
Set the text angle.
Definition TAttText.h:45
virtual Float_t GetTextAngle() const
Return the text angle.
Definition TAttText.h:35
virtual void SetTextColor(Color_t tcolor=1)
Set the text color.
Definition TAttText.h:46
virtual void SetTextFont(Font_t tfont=62)
Set the text font.
Definition TAttText.h:48
virtual void SetTextSize(Float_t tsize=1)
Set the text size.
Definition TAttText.h:49
Class to manage histogram axis.
Definition TAxis.h:32
@ kAxisRange
Definition TAxis.h:66
virtual Double_t GetBinLowEdge(Int_t bin) const
Return low edge of bin.
Definition TAxis.cxx:522
virtual Int_t FindFixBin(Double_t x) const
Find bin number corresponding to abscissa x
Definition TAxis.cxx:423
Int_t GetLast() const
Return last bin on the axis i.e.
Definition TAxis.cxx:473
Int_t GetNbins() const
Definition TAxis.h:127
virtual TObject * GetParent() const
Definition TAxis.h:130
virtual void SetRange(Int_t first=0, Int_t last=0)
Set the viewing range for the axis using bin numbers.
Definition TAxis.cxx:1046
virtual Double_t GetBinUpEdge(Int_t bin) const
Return up edge of bin.
Definition TAxis.cxx:532
Int_t GetFirst() const
Return first bin on the axis i.e.
Definition TAxis.cxx:462
Each class (see TClass) has a linked list of its base class(es).
Definition TBaseClass.h:33
TClass * GetClassPointer(Bool_t load=kTRUE)
Get pointer to the base class TClass.
Create a Box.
Definition TBox.h:22
virtual void SetY2(Double_t y2)
Definition TBox.h:65
static TClass * Class()
virtual void SetX1(Double_t x1)
Definition TBox.h:62
virtual void SetX2(Double_t x2)
Definition TBox.h:63
virtual void SetY1(Double_t y1)
Definition TBox.h:64
Using a TBrowser one can browse all ROOT objects.
Definition TBrowser.h:37
Buffer base class used for serializing objects.
Definition TBuffer.h:43
A TButton object is a user interface object.
Definition TButton.h:18
static TClass * Class()
ABC describing GUI independent main window (with menubar, scrollbars and a drawing area).
Definition TCanvasImp.h:30
The Canvas class.
Definition TCanvas.h:23
UInt_t GetWindowHeight() const
Definition TCanvas.h:162
TObject * GetClickSelected() const
Definition TCanvas.h:142
void ClearPadSave()
Definition TCanvas.h:140
TVirtualPad * GetClickSelectedPad() const
Definition TCanvas.h:147
void SetClickSelectedPad(TPad *pad)
Definition TCanvas.h:211
void SetSelectedPad(TPad *pad)
Definition TCanvas.h:210
void SetDoubleBuffer(Int_t mode=1) override
Set Double Buffer On/Off.
Definition TCanvas.cxx:1973
TCanvasImp * GetCanvasImp() const override
Definition TCanvas.h:158
Bool_t IsRetained() const override
Definition TCanvas.h:176
static Bool_t SupportAlpha()
Static function returning "true" if transparency is supported.
Definition TCanvas.cxx:2475
Int_t GetEventY() const override
Definition TCanvas.h:137
Bool_t IsBatch() const override
Definition TCanvas.h:171
TVirtualPadPainter * GetCanvasPainter()
Access and (probably) creation of pad painter.
Definition TCanvas.cxx:2615
Color_t GetHighLightColor() const override
Definition TCanvas.h:138
Bool_t IsGrayscale()
Check whether this canvas is to be drawn in grayscale mode.
Definition TCanvas.cxx:2566
void SaveSource(const char *filename="", Option_t *option="")
Save primitives in this canvas as a C++ macro file.
Definition TCanvas.cxx:1812
virtual void HandleInput(EEventType button, Int_t x, Int_t y)
Handle Input Events.
Definition TCanvas.cxx:1234
void UpdateAsync() override
Asynchronous pad update.
Definition TCanvas.cxx:2544
TObject * GetSelected() const override
Definition TCanvas.h:141
Int_t GetEventX() const override
Definition TCanvas.h:136
void SetCanvasSize(UInt_t ww, UInt_t wh) override
Set Width and Height of canvas to ww and wh respectively.
Definition TCanvas.cxx:1950
TVirtualPad * GetSelectedPad() const override
Definition TCanvas.h:146
void SetCursor(ECursor cursor) override
Set cursor.
Definition TCanvas.cxx:1964
Int_t GetCanvasID() const override
Definition TCanvas.h:157
UInt_t GetWindowWidth() const
Definition TCanvas.h:161
void FeedbackMode(Bool_t set)
Turn rubberband feedback mode on or off.
Definition TCanvas.cxx:1128
void SetClickSelected(TObject *obj)
Definition TCanvas.h:209
TVirtualPad * GetPadSave() const override
Definition TCanvas.h:139
void Update() override
Update canvas pad buffers.
Definition TCanvas.cxx:2488
virtual void Cleared(TVirtualPad *pad)
Emit pad Cleared signal.
Definition TCanvas.cxx:770
UInt_t GetWw() const override
Definition TCanvas.h:163
Bool_t OpaqueMoving() const override
Definition TCanvas.h:180
UInt_t GetWh() const override
Definition TCanvas.h:164
static TClass * Class()
void SetSelected(TObject *obj) override
Set selected canvas.
Definition TCanvas.cxx:2121
Int_t GetEvent() const override
Definition TCanvas.h:135
Bool_t IsWeb() const override
Is web canvas.
Definition TCanvas.cxx:1496
void SetBatch(Bool_t batch=kTRUE) override
Toggle batch mode.
Definition TCanvas.cxx:1932
Bool_t UseGL() const
Definition TCanvas.h:228
Bool_t OpaqueResizing() const override
Definition TCanvas.h:181
TClass instances represent classes, structs and namespaces in the ROOT type system.
Definition TClass.h:84
TList * GetListOfMethods(Bool_t load=kTRUE)
Return list containing the TMethods of a class.
Definition TClass.cxx:3922
Int_t GetNmethods()
Return the number of methods of this class Note that in case the list of methods is not yet created,...
Definition TClass.cxx:4683
Int_t GetNdata()
Return the number of data members of this class Note that in case the list of data members is not yet...
Definition TClass.cxx:4664
TList * GetListOfDataMembers(Bool_t load=kTRUE)
Return list containing the TDataMembers of a class.
Definition TClass.cxx:3880
TList * GetListOfBases()
Return list containing the TBaseClass(es) of a class.
Definition TClass.cxx:3746
void ls(Option_t *option="") const override
List (ls) all objects in this collection.
void Browse(TBrowser *b) override
Browse this collection (called by TBrowser).
virtual Int_t GetSize() const
Return the capacity of the collection, i.e.
The color creation and management class.
Definition TColor.h:22
static Int_t GetColor(const char *hexcolor)
Static method returning color number for color specified by hex color string of form: "#rrggbb",...
Definition TColor.cxx:1927
static Int_t GetColorBright(Int_t color)
Static function: Returns the bright color number corresponding to n If the TColor object does not exi...
Definition TColor.cxx:2096
static Int_t GetColorDark(Int_t color)
Static function: Returns the dark color number corresponding to n If the TColor object does not exist...
Definition TColor.cxx:2139
static TString SavePrimitiveColor(Int_t ci)
Convert color in C++ statement which can be used in SetColor directives Produced statement either inc...
Definition TColor.cxx:2557
static void SetGrayscale(Bool_t set=kTRUE)
Set whether all colors should return grayscale values.
Definition TColor.cxx:2605
static void Pave(Int_t event, Int_t px, Int_t py, Int_t mode)
Create a new pavetext in gPad.
static void Line(Int_t event, Int_t px, Int_t py, Int_t mode)
Create a new line/arrow in this gPad.
static void PolyLine(Int_t event, Int_t px, Int_t py, Int_t mode)
Create a new PolyLine in gPad.
static void Text(Int_t event, Int_t px, Int_t py, Int_t mode)
Create a new TLatex at the cursor position in gPad.
static void Ellipse(Int_t event, Int_t px, Int_t py, Int_t mode)
Create a new arc/ellipse in this gPad.
static void Pad(Int_t event, Int_t px, Int_t py, Int_t)
Create a new pad in gPad.
All ROOT classes may have RTTI (run time type identification) support added.
Definition TDataMember.h:31
This class stores the date and time with a precision of one second in an unsigned 32 bit word (950130...
Definition TDatime.h:37
virtual Int_t GetValue(const char *name, Int_t dflt) const
Returns the integer value for a resource.
Definition TEnv.cxx:491
TExec is a utility class that can be used to execute a C++ command when some event happens in a pad.
Definition TExec.h:26
Define a Frame.
Definition TFrame.h:19
static TClass * Class()
void Paint(Option_t *option="") override
Paint this wbox with its current attributes.
Definition TFrame.cxx:128
void UseCurrentStyle() override
Replace current frame attributes by current style.
Definition TFrame.cxx:158
A TGraph is an object made of two arrays X and Y with npoints each.
Definition TGraph.h:41
static TClass * Class()
@ kClipFrame
Clip to the frame boundary.
Definition TGraph.h:75
1-D histogram with a float per channel (see TH1 documentation)
Definition TH1.h:647
TH1 is the base class of all histogram classes in ROOT.
Definition TH1.h:59
static TClass * Class()
@ kNoStats
Don't draw stats box.
Definition TH1.h:176
@ kIsZoomed
Bit set when zooming on Y axis.
Definition TH1.h:180
static TClass * Class()
static TClass * Class()
The Histogram stack class.
Definition THStack.h:40
static TClass * Class()
EImageFileTypes
Definition TImage.h:36
@ kBmp
Definition TImage.h:45
@ kPng
Definition TImage.h:40
@ kJpeg
Definition TImage.h:41
@ kXpm
Definition TImage.h:37
@ kAnimGif
Definition TImage.h:55
@ kUnknown
Definition TImage.h:54
@ kTiff
Definition TImage.h:49
@ kGif
Definition TImage.h:48
static TClass * Class()
Option_t * GetOption() const
This class displays a legend box (TPaveText) containing several legend entries.
Definition TLegend.h:23
Use the TLine constructor to create a simple line.
Definition TLine.h:22
Iterator of linked list.
Definition TList.h:191
Option_t * GetOption() const override
Returns the object option stored in the list.
Definition TList.cxx:1143
A doubly linked list.
Definition TList.h:38
void Clear(Option_t *option="") override
Remove all objects from the list.
Definition TList.cxx:400
virtual TObjLink * LastLink() const
Definition TList.h:105
TObject * FindObject(const char *name) const override
Find an object in this list using its name.
Definition TList.cxx:576
void RecursiveRemove(TObject *obj) override
Remove object from this collection and recursively remove the object from all other objects (and coll...
Definition TList.cxx:762
void Add(TObject *obj) override
Definition TList.h:81
TObject * Remove(TObject *obj) override
Remove object from the list.
Definition TList.cxx:820
void AddLast(TObject *obj) override
Add object at the end of the list.
Definition TList.cxx:150
TObject * Last() const override
Return the last object in the list. Returns 0 when list is empty.
Definition TList.cxx:691
virtual TObjLink * FirstLink() const
Definition TList.h:102
void AddFirst(TObject *obj) override
Add object at the beginning of the list.
Definition TList.cxx:98
Each ROOT class (see TClass) has a linked list of methods.
Definition TMethod.h:38
A TMultiGraph is a collection of TGraph (or derived) objects.
Definition TMultiGraph.h:34
TH1F * GetHistogram()
Returns a pointer to the histogram used to draw the axis.
static TClass * Class()
virtual void SetTitle(const char *title="")
Set the title of the TNamed.
Definition TNamed.cxx:174
const char * GetName() const override
Returns name of object.
Definition TNamed.h:49
const char * GetTitle() const override
Returns title of object.
Definition TNamed.h:50
static TClass * Class()
virtual void SetName(const char *name)
Set the name of the TNamed.
Definition TNamed.cxx:150
Mother of all ROOT objects.
Definition TObject.h:41
virtual Int_t DistancetoPrimitive(Int_t px, Int_t py)
Computes distance from point (px,py) to the object.
Definition TObject.cxx:284
virtual const char * GetName() const
Returns name of object.
Definition TObject.cxx:457
R__ALWAYS_INLINE Bool_t TestBit(UInt_t f) const
Definition TObject.h:205
virtual void Streamer(TBuffer &)
Stream an object of class TObject.
Definition TObject.cxx:972
virtual const char * ClassName() const
Returns name of class to which the object belongs.
Definition TObject.cxx:226
virtual void UseCurrentStyle()
Set current style settings in this object This function is called when either TCanvas::UseCurrentStyl...
Definition TObject.cxx:885
virtual void Warning(const char *method, const char *msgfmt,...) const
Issue warning message.
Definition TObject.cxx:1057
virtual TObject * FindObject(const char *name) const
Must be redefined in derived classes.
Definition TObject.cxx:421
void SetBit(UInt_t f, Bool_t set)
Set or unset the user status bits as specified in f.
Definition TObject.cxx:864
virtual Bool_t InheritsFrom(const char *classname) const
Returns kTRUE if object inherits from class "classname".
Definition TObject.cxx:543
virtual void Error(const char *method, const char *msgfmt,...) const
Issue error message.
Definition TObject.cxx:1071
virtual const char * GetTitle() const
Returns title of object.
Definition TObject.cxx:501
void MakeZombie()
Definition TObject.h:53
virtual void Draw(Option_t *option="")
Default Draw method for all objects.
Definition TObject.cxx:293
virtual void Paint(Option_t *option="")
This method must be overridden if a class wants to paint itself.
Definition TObject.cxx:625
void ResetBit(UInt_t f)
Definition TObject.h:204
@ kCannotPick
if object in a pad cannot be picked
Definition TObject.h:73
@ kCanDelete
if object in a list can be deleted
Definition TObject.h:68
@ kMustCleanup
if object destructor must call RecursiveRemove()
Definition TObject.h:70
The most important graphics class in the ROOT system.
Definition TPad.h:28
Short_t GetBorderMode() const override
Definition TPad.h:199
Bool_t OpaqueMoving() const override
Is pad moving in opaque mode ?
Definition TPad.cxx:2884
void PaintTextNDC(Double_t u, Double_t v, const char *text) override
Paint text in CurrentPad NDC coordinates.
Definition TPad.cxx:4618
Int_t GetTicky() const override
Definition TPad.h:239
virtual Int_t Clip(Float_t *x, Float_t *y, Float_t xclipl, Float_t yclipb, Float_t xclipr, Float_t yclipt)
Clipping routine: Cohen Sutherland algorithm.
Definition TPad.cxx:768
virtual void HideToolTip(Int_t event)
Hide tool tip depending on the event type.
Definition TPad.cxx:2852
Double_t fAbsYlowNDC
Absolute Y top left corner of pad in NDC [0,1].
Definition TPad.h:70
Double_t fXtoAbsPixelk
Conversion coefficient for X World to absolute pixel.
Definition TPad.h:41
void SetAttMarkerPS(Color_t color, Style_t style, Size_t msize) override
Set postscript marker attributes.
Definition TPad.cxx:6181
virtual void DivideSquare(Int_t n, Float_t xmargin=0.01, Float_t ymargin=0.01, Int_t color=0)
"n" is the total number of sub-pads.
Definition TPad.cxx:1341
void AbsCoordinates(Bool_t set) override
Definition TPad.h:167
Double_t AbsPixeltoY(Int_t py) override
Definition TPad.h:169
static TClass * Class()
Bool_t IsBatch() const override
Is pad in batch mode ?
Definition TPad.cxx:2861
Double_t GetUymax() const override
Returns the maximum y-coordinate value visible on the pad. If log axis the returned value is in decad...
Definition TPad.h:234
void AddExec(const char *name, const char *command) override
Add a new TExec object to the list of Execs.
Definition TPad.cxx:498
Double_t fWNDC
Width of pad along X in Normalized Coordinates (NDC)
Definition TPad.h:66
Int_t NextPaletteColor() override
Get the next autocolor in the pad.
Definition TPad.cxx:3123
Int_t VtoPixel(Double_t v) const override
Convert Y NDC to pixel.
Definition TPad.cxx:7419
void PaintBorder(Color_t color, Bool_t tops)
Paint the pad border.
Definition TPad.cxx:3649
Double_t GetPhi() const override
Definition TPad.h:225
Bool_t IsEditable() const override
Definition TPad.h:272
void FillCollideGrid(TObject *o)
Initialise the grid used to find empty space when adding a box (Legend) in a pad.
Definition TPad.cxx:3139
void SetView(TView *view=nullptr) override
Set the current TView. Delete previous view if view=0.
Definition TPad.cxx:6149
TVirtualViewer3D * GetViewer3D(Option_t *type="") override
Create/obtain handle to 3D viewer.
Definition TPad.cxx:7105
Double_t fPixeltoYk
Conversion coefficient for pixel to Y World.
Definition TPad.h:59
void PaintHatches(Double_t dy, Double_t angle, Int_t nn, Double_t *xx, Double_t *yy)
This routine draw hatches inclined with the angle "angle" and spaced of "dy" in normalized device coo...
Definition TPad.cxx:4166
void PaintLine3D(Float_t *p1, Float_t *p2) override
Paint 3-D line in the CurrentPad.
Definition TPad.cxx:4361
static Int_t fgMaxPickDistance
Maximum Pick Distance.
Definition TPad.h:115
void ResizePad(Option_t *option="") override
Compute pad conversion coefficients.
Definition TPad.cxx:5629
void PaintPolyMarker(Int_t n, Float_t *x, Float_t *y, Option_t *option="") override
Paint polymarker in CurrentPad World coordinates.
Definition TPad.cxx:4530
Double_t fPhi
phi angle to view as lego/surface
Definition TPad.h:80
Double_t fPixeltoY
yworld = fPixeltoYk + fPixeltoY*ypixel
Definition TPad.h:60
virtual void RecordLatex(const TObject *obj)
Emit RecordLatex() signal.
Definition TPad.cxx:7208
Double_t fAbsXlowNDC
Absolute X top left corner of pad in NDC [0,1].
Definition TPad.h:69
Double_t fVtoPixelk
Conversion coefficient for V NDC to pixel.
Definition TPad.h:52
Bool_t fGridx
Set to true if grid along X.
Definition TPad.h:100
TObject * fPadView3D
! 3D View of this TPad
Definition TPad.h:114
void CopyBackgroundPixmaps(TPad *start, TPad *stop, Int_t x, Int_t y)
Copy pixmaps of pads laying below pad "stop" into pad "stop".
Definition TPad.cxx:3995
Double_t GetUxmax() const override
Returns the maximum x-coordinate value visible on the pad. If log axis the returned value is in decad...
Definition TPad.h:232
void Divide(Int_t nx=1, Int_t ny=1, Float_t xmargin=0.01, Float_t ymargin=0.01, Int_t color=0) override
Automatic pad generation by division.
Definition TPad.cxx:1249
void SavePrimitive(std::ostream &out, Option_t *option="") override
Save primitives in this pad on the C++ source file out.
Definition TPad.cxx:5852
Double_t fXtoPixel
xpixel = fXtoPixelk + fXtoPixel*xworld
Definition TPad.h:43
TList * fExecs
List of commands to be executed when a pad event occurs.
Definition TPad.h:108
Double_t PadtoY(Double_t y) const override
Convert y from pad to Y.
Definition TPad.cxx:3556
Int_t fTickx
Set to 1 if tick marks along X.
Definition TPad.h:89
void SetBBoxY1(const Int_t y) override
Set top of BoundingBox to a value (resize in y direction on top)
Definition TPad.cxx:7317
Int_t fTicky
Set to 1 if tick marks along Y.
Definition TPad.h:90
Int_t GetPixmapID() const override
Definition TPad.h:266
virtual void XYtoAbsPixel(Double_t x, Double_t y, Int_t &xpixel, Int_t &ypixel) const
Convert X/Y into absolute pixel coordinates.
Definition TPad.cxx:7388
Bool_t GetGridx() const override
Definition TPad.h:235
Double_t fX2
X of upper X coordinate.
Definition TPad.h:38
Bool_t PlaceBox(TObject *o, Double_t w, Double_t h, Double_t &xl, Double_t &yb, Option_t *option="lb") override
Place a box in NDC space.
Definition TPad.cxx:3208
void SetLogz(Int_t value=1) override
Set Lin/Log scale for Z.
Definition TPad.cxx:6065
Bool_t fEmbeddedGL
!
Definition TPad.h:87
Int_t UtoAbsPixel(Double_t u) const override
Convert X NDC to absolute pixel.
Definition TPad.cxx:7432
void Browse(TBrowser *b) override
Browse pad.
Definition TPad.cxx:523
Int_t XtoPixel(Double_t x) const override
Convert X coordinate to pixel.
Definition TPad.cxx:7459
void SetAttFillPS(Color_t color, Style_t style) override
Set postscript fill area attributes.
Definition TPad.cxx:6158
Int_t fCGnx
! Size of the collide grid along x
Definition TPad.h:119
void SetBBoxCenterY(const Int_t y) override
Set Y coordinate of the center of the Pad.
Definition TPad.cxx:7281
Double_t fPixeltoX
xworld = fPixeltoXk + fPixeltoX*xpixel
Definition TPad.h:57
void CopyBackgroundPixmap(Int_t x, Int_t y)
Copy pixmap of this pad as background of the current pad.
Definition TPad.cxx:4013
~TPad() override
Pad destructor.
Definition TPad.cxx:379
Bool_t fCopyGLDevice
!
Definition TPad.h:86
Double_t fYtoPixel
ypixel = fYtoPixelk + fYtoPixel*yworld
Definition TPad.h:46
void DeleteToolTip(TObject *tip) override
Delete tool tip object.
Definition TPad.cxx:7055
void Close(Option_t *option="") override
Delete all primitives in pad and pad itself.
Definition TPad.cxx:1070
Double_t fAbsWNDC
Absolute Width of pad along X in NDC.
Definition TPad.h:71
TObject * Remove(TObject *obj, Bool_t modified=kTRUE) override
Remove object from list of primitives When.
Definition TPad.cxx:5412
UInt_t GetWw() const override
Get Ww.
Definition TPad.cxx:2842
void PaintModified() override
Traverse pad hierarchy and (re)paint only modified pads.
Definition TPad.cxx:3817
void SetEditable(Bool_t mode=kTRUE) override
Set pad editable yes/no If a pad is not editable:
Definition TPad.cxx:6009
const char * GetTitle() const override
Returns title of object.
Definition TPad.h:261
void PaintDate()
Paint the current date and time if the option Date is set on via gStyle->SetOptDate() Paint the curre...
Definition TPad.cxx:3755
static void SetMaxPickDistance(Int_t maxPick=5)
static function to set the maximum Pick Distance fgMaxPickDistance This parameter is used in TPad::Pi...
Definition TPad.cxx:6634
void SetBBoxX2(const Int_t x) override
Set right hand side of BoundingBox to a value (resize in x direction on right)
Definition TPad.cxx:7306
void SetBBoxX1(const Int_t x) override
Set lefthandside of BoundingBox to a value (resize in x direction on left)
Definition TPad.cxx:7293
virtual Int_t ClippingCode(Double_t x, Double_t y, Double_t xcl1, Double_t ycl1, Double_t xcl2, Double_t ycl2)
Compute the endpoint codes for TPad::Clip.
Definition TPad.cxx:901
Double_t GetUymin() const override
Returns the minimum y-coordinate value visible on the pad. If log axis the returned value is in decad...
Definition TPad.h:230
Double_t fX1
X of lower X coordinate.
Definition TPad.h:36
TList * GetListOfPrimitives() const override
Definition TPad.h:245
void SetFillStyle(Style_t fstyle) override
Override TAttFill::FillStyle for TPad because we want to handle style=0 as style 4000.
Definition TPad.cxx:6028
TH1F * DrawFrame(Double_t xmin, Double_t ymin, Double_t xmax, Double_t ymax, const char *title="") override
Draw an empty pad frame with X and Y axis.
Definition TPad.cxx:1678
Double_t fVtoPixel
ypixel = fVtoPixelk + fVtoPixel*vndc
Definition TPad.h:53
TCanvasImp * GetCanvasImp() const override
Get canvas implementation pointer if any.
Definition TPad.cxx:2751
Int_t GetEvent() const override
Get Event.
Definition TPad.cxx:2759
Double_t PadtoX(Double_t x) const override
Convert x from pad to X.
Definition TPad.cxx:3547
virtual void PixeltoXY(Int_t xpixel, Int_t ypixel, Double_t &x, Double_t &y)
Convert pixel to X/Y coordinates.
Definition TPad.cxx:7379
virtual void DrawCrosshair()
Function called to draw a crosshair in the canvas.
Definition TPad.cxx:1605
Double_t YtoPad(Double_t y) const override
Convert y from Y to pad.
Definition TPad.cxx:3577
virtual void RangeChanged()
Definition TPad.h:316
Double_t fUymin
Minimum value on the Y axis.
Definition TPad.h:75
void SetPad(const char *name, const char *title, Double_t xlow, Double_t ylow, Double_t xup, Double_t yup, Color_t color=35, Short_t bordersize=5, Short_t bordermode=-1) override
Set all pad parameters.
Definition TPad.cxx:6125
void SetCursor(ECursor cursor) override
Set cursor type.
Definition TPad.cxx:2916
Int_t GetCanvasID() const override
Get canvas identifier.
Definition TPad.cxx:2743
Int_t fLogz
(=0 if Z linear scale, =1 if log scale)
Definition TPad.h:93
Double_t fYtoPixelk
Conversion coefficient for Y World to pixel.
Definition TPad.h:45
void UpdateAsync() override
Asynchronous pad update.
Definition TPad.cxx:2952
TPad()
Pad default constructor.
Definition TPad.cxx:132
Double_t AbsPixeltoX(Int_t px) override
Definition TPad.h:168
void UseCurrentStyle() override
Force a copy of current style for all objects in pad.
Definition TPad.cxx:6881
static Int_t GetMaxPickDistance()
Static function (see also TPad::SetMaxPickDistance)
Definition TPad.cxx:2799
Int_t VtoAbsPixel(Double_t v) const override
Convert Y NDC to absolute pixel.
Definition TPad.cxx:7440
void Range(Double_t x1, Double_t y1, Double_t x2, Double_t y2) override
Set world coordinate system for the pad.
Definition TPad.cxx:5336
Double_t fUtoPixelk
Conversion coefficient for U NDC to pixel.
Definition TPad.h:49
Double_t fPixeltoXk
Conversion coefficient for pixel to X World.
Definition TPad.h:56
Bool_t IsModified() const override
Definition TPad.h:274
Double_t fY1
Y of lower Y coordinate.
Definition TPad.h:37
Int_t GetEventY() const override
Get Y event.
Definition TPad.cxx:2775
Int_t fGLDevice
! OpenGL off-screen pixmap identifier
Definition TPad.h:85
Double_t fYlowNDC
Y bottom left corner of pad in NDC [0,1].
Definition TPad.h:63
TObject * WaitPrimitive(const char *pname="", const char *emode="") override
Loop and sleep until a primitive with name=pname is found in the pad.
Definition TPad.cxx:7001
void SetAttTextPS(Int_t align, Float_t angle, Color_t color, Style_t font, Float_t tsize) override
Set postscript text attributes.
Definition TPad.cxx:6193
Bool_t fModified
Set to true when pad is modified.
Definition TPad.h:99
TLegend * BuildLegend(Double_t x1=0.3, Double_t y1=0.21, Double_t x2=0.3, Double_t y2=0.21, const char *title="", Option_t *option="") override
Build a legend from the graphical objects in the pad.
Definition TPad.cxx:555
virtual TPad * Pick(Int_t px, Int_t py, TObjLink *&pickobj)
Search for an object at pixel position px,py.
Definition TPad.cxx:4662
void Update() override
Update pad.
Definition TPad.cxx:2940
Int_t fNumber
pad number identifier
Definition TPad.h:88
void PaintFillAreaNDC(Int_t n, Double_t *x, Double_t *y, Option_t *option="") override
Paint fill area in CurrentPad NDC coordinates.
Definition TPad.cxx:4070
Double_t PixeltoX(Int_t px) override
Convert pixel to X coordinate.
Definition TPad.cxx:7361
Double_t fAbsPixeltoXk
Conversion coefficient for absolute pixel to X World.
Definition TPad.h:55
void Clear(Option_t *option="") override
Delete all pad primitives.
Definition TPad.cxx:722
Int_t YtoPixel(Double_t y) const override
Convert Y coordinate to pixel.
Definition TPad.cxx:7483
Int_t GetTickx() const override
Definition TPad.h:238
void PaintLine(Double_t x1, Double_t y1, Double_t x2, Double_t y2) override
Paint line in CurrentPad World coordinates.
Definition TPad.cxx:4318
Int_t GetGLDevice() override
Get GL device.
Definition TPad.cxx:7192
Double_t fAspectRatio
ratio of w/h in case of fixed ratio
Definition TPad.h:82
virtual void RecordPave(const TObject *obj)
Emit RecordPave() signal.
Definition TPad.cxx:7200
void PaintBorderPS(Double_t xl, Double_t yl, Double_t xt, Double_t yt, Int_t bmode, Int_t bsize, Int_t dark, Int_t light) override
Paint a frame border with Postscript.
Definition TPad.cxx:3745
Double_t fUymax
Maximum value on the Y axis.
Definition TPad.h:77
void SetLogy(Int_t value=1) override
Set Lin/Log scale for Y.
Definition TPad.cxx:6054
virtual Int_t ClipPolygon(Int_t n, Double_t *x, Double_t *y, Int_t nn, Double_t *xc, Double_t *yc, Double_t xclipl, Double_t yclipb, Double_t xclipr, Double_t yclipt)
Clip polygon using the Sutherland-Hodgman algorithm.
Definition TPad.cxx:950
void ExecuteEventAxis(Int_t event, Int_t px, Int_t py, TAxis *axis) override
Execute action corresponding to one event for a TAxis object (called by TAxis::ExecuteEvent....
Definition TPad.cxx:2361
void HighLight(Color_t col=kRed, Bool_t set=kTRUE) override
Highlight pad.
Definition TPad.cxx:3060
void SetBatch(Bool_t batch=kTRUE) override
Set pad in batch mode.
Definition TPad.cxx:2900
TCanvas * fCanvas
! Pointer to mother canvas
Definition TPad.h:106
TVirtualPad * GetMother() const override
Definition TPad.h:259
TVirtualViewer3D * fViewer3D
! Current 3D viewer
Definition TPad.h:123
virtual void x3d(Option_t *type="")
Deprecated: use TPad::GetViewer3D() instead.
Definition TPad.cxx:7087
Bool_t HasCrosshair() const override
Return kTRUE if the crosshair has been activated (via SetCrosshair).
Definition TPad.cxx:6596
Bool_t IsRetained() const override
Is pad retained ?
Definition TPad.cxx:2869
Bool_t Collide(Int_t i, Int_t j, Int_t w, Int_t h)
Check if a box of size w and h collide some primitives in the pad at position i,j.
Definition TPad.cxx:3184
Int_t DistancetoPrimitive(Int_t px, Int_t py) override
Compute distance from point px,py to a box.
Definition TPad.cxx:1174
Bool_t fFixedAspectRatio
True if fixed aspect ratio.
Definition TPad.h:104
void PaintFillArea(Int_t n, Float_t *x, Float_t *y, Option_t *option="") override
Definition TPad.cxx:4022
void Modified(Bool_t flag=true) override
Mark pad modified Will be repainted when TCanvas::Update() will be called next time.
Definition TPad.cxx:7342
void RecursiveRemove(TObject *obj) override
Recursively remove object from a pad and its sub-pads.
Definition TPad.cxx:5394
Bool_t HasFixedAspectRatio() const override
Definition TPad.h:270
void CloseToolTip(TObject *tip) override
Hide tool tip.
Definition TPad.cxx:7077
Double_t GetUxmin() const override
Returns the minimum x-coordinate value visible on the pad. If log axis the returned value is in decad...
Definition TPad.h:228
void SetToolTipText(const char *text, Long_t delayms=1000) override
Set tool tip text associated with this pad.
Definition TPad.cxx:6644
void PaintPolyLine(Int_t n, Float_t *x, Float_t *y, Option_t *option="") override
Paint polyline in CurrentPad World coordinates.
Definition TPad.cxx:4397
void ls(Option_t *option="") const override
List all primitives in pad.
Definition TPad.cxx:3095
TView * GetView() const override
Definition TPad.h:254
void ModifiedUpdate() override
Short cut to call Modified() and Update() in a single call.
Definition TPad.cxx:3535
Double_t fVtoAbsPixelk
Conversion coefficient for V NDC to absolute pixel.
Definition TPad.h:51
TVirtualPad * GetPadSave() const override
Get save pad.
Definition TPad.cxx:2825
void SetAttLinePS(Color_t color, Style_t style, Width_t lwidth) override
Set postscript line attributes.
Definition TPad.cxx:6169
TClass * IsA() const override
Definition TPad.h:419
virtual void Resized()
Definition TPad.h:323
TVirtualPad * GetVirtCanvas() const override
Get virtual canvas.
Definition TPad.cxx:2783
void DeleteExec(const char *name) override
Remove TExec name from the list of Execs.
Definition TPad.cxx:1158
void Streamer(TBuffer &) override
Stream a class object.
Definition TPad.cxx:6667
TString fTitle
Pad title.
Definition TPad.h:110
void CopyPixmaps() override
Copy the sub-pixmaps of the pad to the canvas.
Definition TPad.cxx:1143
void CopyPixmap() override
Copy the pixmap of the pad to the canvas.
Definition TPad.cxx:1129
Double_t GetY1() const override
Definition TPad.h:242
Int_t UtoPixel(Double_t u) const override
Convert X NDC to pixel.
Definition TPad.cxx:7406
TPoint GetBBoxCenter() override
Return the center of the Pad as TPoint in pixels.
Definition TPad.cxx:7242
void FillCollideGridTFrame(TObject *o)
Definition TPad.cxx:3339
Bool_t GetGridy() const override
Definition TPad.h:236
void LineNotFree(Int_t x1, Int_t x2, Int_t y1, Int_t y2)
Mark as "not free" the cells along a line.
Definition TPad.cxx:3279
virtual void AbsPixeltoXY(Int_t xpixel, Int_t ypixel, Double_t &x, Double_t &y)
Convert absolute pixel into X/Y coordinates.
Definition TPad.cxx:7351
Double_t fAbsHNDC
Absolute Height of pad along Y in NDC.
Definition TPad.h:72
void ExecuteEvent(Int_t event, Int_t px, Int_t py) override
Execute action corresponding to one event.
Definition TPad.cxx:1786
TObject * fTip
! tool tip associated with box
Definition TPad.h:33
void SetCanvasSize(UInt_t ww, UInt_t wh) override
Set canvas size.
Definition TPad.cxx:2908
Int_t GetLogz() const override
Definition TPad.h:258
virtual void XYtoPixel(Double_t x, Double_t y, Int_t &xpixel, Int_t &ypixel) const
Convert X/Y into pixel coordinates.
Definition TPad.cxx:7397
virtual void AutoExec()
Execute the list of Execs when a pad event occurs.
Definition TPad.cxx:508
Bool_t fAbsCoord
Use absolute coordinates.
Definition TPad.h:102
Int_t fNumPaletteColor
Number of objects with an automatic color.
Definition TPad.h:116
Int_t fCrosshairPos
Position of crosshair.
Definition TPad.h:96
void FillCollideGridTGraph(TObject *o)
Definition TPad.cxx:3365
void SetFixedAspectRatio(Bool_t fixed=kTRUE) override
Fix pad aspect ratio to current value if fixed is true.
Definition TPad.cxx:5985
Short_t GetBorderSize() const override
Definition TPad.h:200
void RedrawAxis(Option_t *option="") override
Redraw the frame axis.
Definition TPad.cxx:5455
void DrawDist(Rectangle_t aBBox, Rectangle_t bBBox, char mode)
Draw Arrows to indicated equal distances of Objects with given BBoxes.
Definition TPad.cxx:6220
Int_t fLogx
(=0 if X linear scale, =1 if log scale)
Definition TPad.h:91
Double_t GetAbsWNDC() const override
Definition TPad.h:222
Int_t YtoAbsPixel(Double_t y) const override
Convert Y coordinate to absolute pixel.
Definition TPad.cxx:7472
Double_t fUtoPixel
xpixel = fUtoPixelk + fUtoPixel*undc
Definition TPad.h:50
Int_t fCrosshair
Crosshair type (0 if no crosshair requested)
Definition TPad.h:95
void PaintFillAreaHatches(Int_t n, Double_t *x, Double_t *y, Int_t FillStyle)
This function paints hatched fill area according to the FillStyle value The convention for the Hatch ...
Definition TPad.cxx:4113
void RangeAxis(Double_t xmin, Double_t ymin, Double_t xmax, Double_t ymax) override
Set axis coordinate system for the pad.
Definition TPad.cxx:5374
Double_t fUtoAbsPixelk
Conversion coefficient for U NDC to absolute pixel.
Definition TPad.h:48
void ResetToolTip(TObject *tip) override
Reset tool tip, i.e.
Definition TPad.cxx:7066
Double_t GetTheta() const override
Definition TPad.h:226
TList * fPrimitives
->List of primitives (subpads)
Definition TPad.h:107
UInt_t GetWh() const override
Get Wh.
Definition TPad.cxx:2834
TCanvas * GetCanvas() const override
Definition TPad.h:262
Short_t fBorderSize
pad bordersize in pixels
Definition TPad.h:97
void Add(TObject *obj, Option_t *opt="", Bool_t modified=kTRUE) override
Add an object to list of primitives with speicified draw option When.
Definition TPad.cxx:418
TView * fView
! Pointer to 3-D view (if one exists)
Definition TPad.h:112
Rectangle_t GetBBox() override
Return the bounding Box of the Pad.
Definition TPad.cxx:7225
void Paint(Option_t *option="") override
Paint all primitives in pad.
Definition TPad.cxx:3589
void FillCollideGridTBox(TObject *o)
Definition TPad.cxx:3322
Double_t fTheta
theta angle to view as lego/surface
Definition TPad.h:79
void DrawCollideGrid()
This method draws the collide grid on top of the canvas.
Definition TPad.cxx:3478
TString fName
Pad name.
Definition TPad.h:109
void SetVertical(Bool_t vert=kTRUE) override
Set pad vertical (default) or horizontal.
Definition TPad.cxx:6658
void FillCollideGridTH1(TObject *o)
Definition TPad.cxx:3396
void GetPadPar(Double_t &xlow, Double_t &ylow, Double_t &xup, Double_t &yup) override
Return lower and upper bounds of the pad in NDC coordinates.
Definition TPad.cxx:3026
void PaintText(Double_t x, Double_t y, const char *text) override
Paint text in CurrentPad World coordinates.
Definition TPad.cxx:4592
Int_t fPadPaint
Set to 1 while painting the pad.
Definition TPad.h:94
static void DrawColorTable()
Static function to Display Color Table in a pad.
Definition TPad.cxx:1723
Double_t GetXlowNDC() const override
Definition TPad.h:212
void SaveAs(const char *filename="", Option_t *option="") const override
Save the pad content in a file.
Definition TPad.cxx:5789
Int_t fPixmapID
! Off-screen pixmap identifier
Definition TPad.h:84
Bool_t fEditable
True if canvas is editable.
Definition TPad.h:103
Double_t GetYlowNDC() const override
Definition TPad.h:213
TObject * FindObject(const char *name) const override
Search if object named name is inside this pad or in pads inside this pad.
Definition TPad.cxx:2705
Color_t GetHighLightColor() const override
Get highlight color.
Definition TPad.cxx:2791
Bool_t OpaqueResizing() const override
Is pad resizing in opaque mode ?
Definition TPad.cxx:2892
Double_t fXUpNDC
Definition TPad.h:64
std::vector< Bool_t > fCollideGrid
! Grid used to find empty space when adding a box (Legend) in a pad
Definition TPad.h:118
TVirtualPad * cd(Int_t subpadnumber=0) override
Set Current pad.
Definition TPad.cxx:693
Int_t GetLogy() const override
Definition TPad.h:257
void PaintLineNDC(Double_t u1, Double_t v1, Double_t u2, Double_t v2) override
Paint line in normalized coordinates.
Definition TPad.cxx:4341
Double_t PixeltoY(Int_t py) override
Convert pixel to Y coordinate.
Definition TPad.cxx:7370
void Print(const char *filename="") const override
This method is equivalent to SaveAs("filename"). See TPad::SaveAs for details.
Definition TPad.cxx:4803
Int_t GetEventX() const override
Get X event.
Definition TPad.cxx:2767
TFrame * GetFrame() override
Get frame.
Definition TPad.cxx:2960
Double_t fYUpNDC
Definition TPad.h:65
Double_t fYtoAbsPixelk
Conversion coefficient for Y World to absolute pixel.
Definition TPad.h:44
Double_t fXtoPixelk
Conversion coefficient for X World to pixel.
Definition TPad.h:42
Int_t fLogy
(=0 if Y linear scale, =1 if log scale)
Definition TPad.h:92
TFrame * fFrame
! Pointer to 2-D frame (if one exists)
Definition TPad.h:111
TVirtualPadPainter * GetPainter() override
Get pad painter from TCanvas.
Definition TPad.cxx:7216
void Draw(Option_t *option="") override
Draw Pad in Current pad (re-parent pad if necessary).
Definition TPad.cxx:1364
virtual void Closed()
Definition TPad.h:184
Double_t fHNDC
Height of pad along Y in Normalized Coordinates (NDC)
Definition TPad.h:67
void ShowGuidelines(TObject *object, const Int_t event, const char mode='i', const bool cling=true) override
Shows lines to indicate if a TAttBBox2D object is aligned to the center or to another object,...
Definition TPad.cxx:6304
Int_t GetCrosshair() const
Return the crosshair type (from the mother canvas) crosshair type = 0 means no crosshair.
Definition TPad.cxx:6605
void GetRangeAxis(Double_t &xmin, Double_t &ymin, Double_t &xmax, Double_t &ymax) override
Return pad axis coordinates range.
Definition TPad.cxx:3048
void PaintBox(Double_t x1, Double_t y1, Double_t x2, Double_t y2, Option_t *option="") override
Paint box in CurrentPad World coordinates.
Definition TPad.cxx:3901
void DrawClassObject(const TObject *obj, Option_t *option="") override
Draw class inheritance tree of the class to which obj belongs.
Definition TPad.cxx:1403
Int_t fCGny
! Size of the collide grid along y
Definition TPad.h:120
Double_t fXlowNDC
X bottom left corner of pad in NDC [0,1].
Definition TPad.h:62
TObject * GetPrimitive(const char *name) const override
Get primitive.
Definition TPad.cxx:2988
Double_t fUxmin
Minimum value on the X axis.
Definition TPad.h:74
Double_t GetAbsHNDC() const override
Definition TPad.h:223
void SetBBoxCenter(const TPoint &p) override
Set center of the Pad.
Definition TPad.cxx:7257
Bool_t IsWeb() const override
Is web ?
Definition TPad.cxx:2876
void SetSelected(TObject *obj) override
Set selected.
Definition TPad.cxx:2932
TObject * GetSelected() const override
Get selected.
Definition TPad.cxx:2807
void GetRange(Double_t &x1, Double_t &y1, Double_t &x2, Double_t &y2) override
Return pad world coordinates range.
Definition TPad.cxx:3037
void PaintPolyLineNDC(Int_t n, Double_t *x, Double_t *y, Option_t *option="") override
Paint polyline in CurrentPad NDC coordinates.
Definition TPad.cxx:4495
Bool_t IsVertical() const override
Definition TPad.h:276
Int_t IncrementPaletteColor(Int_t i, TString opt) override
Increment (i==1) or set (i>1) the number of autocolor in the pad.
Definition TPad.cxx:3109
void PaintPadFrame(Double_t xmin, Double_t ymin, Double_t xmax, Double_t ymax) override
Paint histogram/graph frame.
Definition TPad.cxx:3798
Double_t GetAbsYlowNDC() const override
Definition TPad.h:221
Double_t fUxmax
Maximum value on the X axis.
Definition TPad.h:76
void AddFirst(TObject *obj, Option_t *opt="", Bool_t modified=kTRUE) override
Add an object as first in list of primitives with speicified draw option When.
Definition TPad.cxx:439
Double_t fY2
Y of upper Y coordinate.
Definition TPad.h:39
Double_t fAbsPixeltoYk
Conversion coefficient for absolute pixel to Y World.
Definition TPad.h:58
TVirtualPad * GetSelectedPad() const override
Get selected pad.
Definition TPad.cxx:2816
void PaintPolyLine3D(Int_t n, Double_t *p) override
Paint 3-D polyline in the CurrentPad.
Definition TPad.cxx:4516
TVirtualPad * GetPad(Int_t subpadnumber) const override
Get a pointer to subpadnumber of this pad.
Definition TPad.cxx:3005
Short_t fBorderMode
Bordermode (-1=down, 0 = no border, 1=up)
Definition TPad.h:98
void SetLogx(Int_t value=1) override
Set Lin/Log scale for X.
Definition TPad.cxx:6040
void ReleaseViewer3D(Option_t *type="") override
Release current (external) viewer.
Definition TPad.cxx:7176
void SetCrosshair(Int_t crhair=1) override
Set crosshair active/inactive.
Definition TPad.cxx:6618
void SetDoubleBuffer(Int_t mode=1) override
Set double buffer mode ON or OFF.
Definition TPad.cxx:2924
Int_t fNextPaletteColor
Next automatic color.
Definition TPad.h:117
void SetBBoxCenterX(const Int_t x) override
Set X coordinate of the center of the Pad.
Definition TPad.cxx:7270
Int_t GetLogx() const override
Definition TPad.h:256
TObject * fPadPointer
! free pointer
Definition TPad.h:113
Double_t GetX2() const override
Definition TPad.h:241
TObject * CreateToolTip(const TBox *b, const char *text, Long_t delayms) override
Create a tool tip and return its pointer.
Definition TPad.cxx:7045
@ kCannotMove
Fixed position.
Definition TPad.h:158
@ kClearAfterCR
Clear after CR.
Definition TPad.h:159
@ kHori
Pad is horizontal.
Definition TPad.h:155
@ kPrintingPS
PS Printing.
Definition TPad.h:157
@ kFraming
Frame is requested.
Definition TPad.h:154
Double_t GetWNDC() const override
Get width of pad along X in Normalized Coordinates (NDC)
Definition TPad.h:215
void Pop() override
Pop pad to the top of the stack.
Definition TPad.cxx:4781
Double_t GetAbsXlowNDC() const override
Definition TPad.h:220
Double_t GetHNDC() const override
Get height of pad along Y in Normalized Coordinates (NDC)
Definition TPad.h:217
TPad * fMother
! pointer to mother of the list
Definition TPad.h:105
const char * GetName() const override
Returns name of object.
Definition TPad.h:260
Int_t XtoAbsPixel(Double_t x) const override
Convert X coordinate to absolute pixel.
Definition TPad.cxx:7448
Bool_t fGridy
Set to true if grid along Y.
Definition TPad.h:101
void SetBBoxY2(const Int_t y) override
Set bottom of BoundingBox to a value (resize in y direction on bottom)
Definition TPad.cxx:7329
Double_t XtoPad(Double_t x) const override
Convert x from X to pad.
Definition TPad.cxx:3565
The histogram statistics painter class.
Definition TPaveStats.h:18
A Pave (see TPave) with text, lines or/and boxes inside.
Definition TPaveText.h:21
virtual TText * AddText(Double_t x1, Double_t y1, const char *label)
Add a new Text line to this pavetext at given coordinates.
virtual TLine * AddLine(Double_t x1=0, Double_t y1=0, Double_t x2=0, Double_t y2=0)
Add a new graphics line to this pavetext.
virtual TBox * AddBox(Double_t x1, Double_t y1, Double_t x2, Double_t y2)
Add a new graphics box to this pavetext.
void Draw(Option_t *option="") override
Draw this pavetext with its current attributes.
Int_t GetBorderSize() const
Definition TPave.h:56
virtual void SetBorderSize(Int_t bordersize=4)
Sets the border size of the TPave box and shadow.
Definition TPave.h:79
static TClass * Class()
TPluginHandler * FindHandler(const char *base, const char *uri=nullptr)
Returns the handler if there exists a handler for the specified URI.
SCoord_t GetY() const
Definition TPoint.h:47
SCoord_t GetX() const
Definition TPoint.h:46
void Emit(const char *signal, const T &arg)
Activate signal with single parameter.
Definition TQObject.h:164
static Int_t IncreaseDirLevel()
Increase the indentation level for ls().
Definition TROOT.cxx:2887
static void IndentLevel()
Functions used by ls() to indent an object hierarchy.
Definition TROOT.cxx:2895
static Int_t DecreaseDirLevel()
Decrease the indentation level for ls().
Definition TROOT.cxx:2747
Basic string class.
Definition TString.h:139
Ssiz_t Length() const
Definition TString.h:417
void ToLower()
Change string to lower-case.
Definition TString.cxx:1182
TString & ReplaceSpecialCppChars()
Find special characters which are typically used in printf() calls and replace them by appropriate es...
Definition TString.cxx:1114
TString & Replace(Ssiz_t pos, Ssiz_t n, const char *s)
Definition TString.h:694
const char * Data() const
Definition TString.h:376
virtual void Streamer(TBuffer &)
Stream a string object.
Definition TString.cxx:1412
static TString Format(const char *fmt,...)
Static method which formats a string using a printf style format descriptor and return a TString.
Definition TString.cxx:2378
void Form(const char *fmt,...)
Formats a string using a printf style format descriptor.
Definition TString.cxx:2356
Bool_t Contains(const char *pat, ECaseCompare cmp=kExact) const
Definition TString.h:632
Ssiz_t Index(const char *pat, Ssiz_t i=0, ECaseCompare cmp=kExact) const
Definition TString.h:651
Int_t GetOptLogy() const
Definition TStyle.h:250
void SetPadBorderMode(Int_t mode=1)
Definition TStyle.h:361
void SetPadTopMargin(Float_t margin=0.1)
Definition TStyle.h:363
void SetOptLogx(Int_t logx=1)
Definition TStyle.h:333
void SetPadBottomMargin(Float_t margin=0.1)
Definition TStyle.h:362
Int_t GetOptTitle() const
Definition TStyle.h:248
Int_t GetPadTickX() const
Definition TStyle.h:219
Bool_t IsReading() const
Definition TStyle.h:300
Color_t GetPadColor() const
Definition TStyle.h:210
void SetPadRightMargin(Float_t margin=0.1)
Definition TStyle.h:365
void SetTitleFont(Style_t font=62, Option_t *axis="X")
Definition TStyle.cxx:1777
Float_t GetPadRightMargin() const
Definition TStyle.h:216
void SetTitleBorderSize(Width_t size=2)
Definition TStyle.h:412
Float_t GetDateX() const
Definition TStyle.h:199
Style_t GetTitleFont(Option_t *axis="X") const
Return title font.
Definition TStyle.cxx:1218
void SetPadTickY(Int_t ticky)
Definition TStyle.h:369
Color_t GetTitleFillColor() const
Definition TStyle.h:273
void SetPadTickX(Int_t tickx)
Definition TStyle.h:368
Int_t GetOptDate() const
Definition TStyle.h:244
Bool_t GetPadGridY() const
Definition TStyle.h:218
void SetPadGridX(Bool_t gridx)
Definition TStyle.h:366
void SetTitleTextColor(Color_t color=1)
Definition TStyle.h:409
Float_t GetPadLeftMargin() const
Definition TStyle.h:215
Double_t GetHatchesSpacing() const
Definition TStyle.h:203
Bool_t GetPadGridX() const
Definition TStyle.h:217
void SetPadLeftMargin(Float_t margin=0.1)
Definition TStyle.h:364
void SetPadGridY(Bool_t gridy)
Definition TStyle.h:367
void SetOptLogy(Int_t logy=1)
Definition TStyle.h:334
Int_t GetOptFile() const
Definition TStyle.h:245
TAttText * GetAttDate()
Definition TStyle.h:170
Int_t GetPadTickY() const
Definition TStyle.h:220
Width_t GetPadBorderSize() const
Definition TStyle.h:211
Width_t GetTitleBorderSize() const
Definition TStyle.h:277
Int_t GetColorPalette(Int_t i) const
Return color number i in current palette.
Definition TStyle.cxx:1103
void SetTitleFillColor(Color_t color=1)
Definition TStyle.h:408
Float_t GetPadBottomMargin() const
Definition TStyle.h:213
void SetOptLogz(Int_t logz=1)
Definition TStyle.h:335
void SetPadColor(Color_t color=19)
Definition TStyle.h:359
Color_t GetTitleTextColor() const
Definition TStyle.h:274
Int_t GetOptLogx() const
Definition TStyle.h:249
Float_t GetDateY() const
Definition TStyle.h:200
Int_t GetPadBorderMode() const
Definition TStyle.h:212
Int_t GetNumberOfColors() const
Return number of colors in the color palette.
Definition TStyle.cxx:1177
Int_t GetOptLogz() const
Definition TStyle.h:251
void SetPadBorderSize(Width_t size=1)
Definition TStyle.h:360
Int_t GetHatchesLineWidth() const
Definition TStyle.h:202
Float_t GetPadTopMargin() const
Definition TStyle.h:214
virtual Bool_t ExpandPathName(TString &path)
Expand a pathname getting rid of special shell characters like ~.
Definition TSystem.cxx:1285
virtual Int_t Exec(const char *shellcmd)
Execute a command.
Definition TSystem.cxx:653
virtual Bool_t AccessPathName(const char *path, EAccessMode mode=kFileExists)
Returns FALSE if one can access a file using the specified access mode.
Definition TSystem.cxx:1307
virtual int Rename(const char *from, const char *to)
Rename a file.
Definition TSystem.cxx:1361
virtual void Sleep(UInt_t milliSec)
Sleep milliSec milli seconds.
Definition TSystem.cxx:437
virtual Bool_t ProcessEvents()
Process pending events (GUI, timers, sockets).
Definition TSystem.cxx:416
Base class for several text objects.
Definition TText.h:22
void Paint(Option_t *option="") override
Paint this text with its current attributes.
Definition TText.cxx:687
virtual void SetNDC(Bool_t isNDC=kTRUE)
Set NDC mode on if isNDC = kTRUE, off otherwise.
Definition TText.cxx:817
See TView3D.
Definition TView.h:25
virtual void WCtoNDC(const Float_t *pw, Float_t *pn)=0
virtual Int_t GetDistancetoAxis(Int_t axis, Int_t px, Int_t py, Double_t &ratio)=0
virtual void ResizePad()=0
virtual void GetRange(Float_t *min, Float_t *max)=0
Provides 3D viewer interface (TVirtualViewer3D) support on a pad.
TVirtualPS is an abstract interface to Postscript, PDF, SVG.
Definition TVirtualPS.h:30
virtual void Text(Double_t x, Double_t y, const char *string)=0
virtual void NewPage()=0
virtual void DrawPS(Int_t n, Float_t *xw, Float_t *yw)=0
virtual void DrawBox(Double_t x1, Double_t y1, Double_t x2, Double_t y2)=0
virtual void DrawPolyMarker(Int_t n, Float_t *x, Float_t *y)=0
virtual void Open(const char *filename, Int_t type=-111)=0
virtual void DrawFrame(Double_t xl, Double_t yl, Double_t xt, Double_t yt, Int_t mode, Int_t border, Int_t dark, Int_t light)=0
To make it possible to use GL for 2D graphic in a TPad/TCanvas.
virtual void DrawBox(Double_t x1, Double_t y1, Double_t x2, Double_t y2, EBoxMode mode)=0
virtual void ClearDrawable()=0
virtual void SetOpacity(Int_t percent)=0
virtual void DrawPolyLine(Int_t n, const Double_t *x, const Double_t *y)=0
virtual Color_t GetLineColor() const =0
virtual void SetLineStyle(Style_t lstyle)=0
virtual void SetFillColor(Color_t fcolor)=0
virtual void DestroyDrawable(Int_t device)=0
virtual void CopyDrawable(Int_t device, Int_t px, Int_t py)=0
virtual void DrawLine(Double_t x1, Double_t y1, Double_t x2, Double_t y2)=0
virtual void DrawFillArea(Int_t n, const Double_t *x, const Double_t *y)=0
virtual Int_t CreateDrawable(UInt_t w, UInt_t h)=0
virtual void SetLineColor(Color_t lcolor)=0
virtual Style_t GetLineStyle() const =0
virtual void SaveImage(TVirtualPad *pad, const char *fileName, Int_t type) const =0
virtual void DrawLineNDC(Double_t u1, Double_t v1, Double_t u2, Double_t v2)=0
virtual void SetFillStyle(Style_t fstyle)=0
virtual void SetTextSize(Float_t tsize=1)=0
virtual void SetLineWidth(Width_t lwidth)=0
virtual void DrawPolyLineNDC(Int_t n, const Double_t *u, const Double_t *v)=0
virtual void DrawTextNDC(Double_t u, Double_t v, const char *text, ETextMode mode)=0
virtual Width_t GetLineWidth() const =0
virtual void SelectDrawable(Int_t device)=0
virtual void DrawPolyMarker(Int_t n, const Double_t *x, const Double_t *y)=0
virtual void DrawText(Double_t x, Double_t y, const char *text, ETextMode mode)=0
virtual Style_t GetFillStyle() const =0
virtual void InvalidateCS()
Empty definition.
small helper class to store/restore gPad context in TPad methods
Definition TVirtualPad.h:61
TVirtualPad is an abstract base class for the Pad and Canvas classes.
Definition TVirtualPad.h:51
virtual Int_t YtoAbsPixel(Double_t y) const =0
virtual Double_t GetX2() const =0
virtual Int_t XtoAbsPixel(Double_t x) const =0
virtual Double_t GetY1() const =0
void Streamer(TBuffer &) override
Stream an object of class TVirtualPad.
virtual Int_t VtoPixel(Double_t v) const =0
static TClass * Class()
virtual Int_t VtoAbsPixel(Double_t v) const =0
virtual void RangeAxisChanged()
virtual Double_t GetY2() const =0
virtual Int_t UtoPixel(Double_t u) const =0
Bool_t fResizing
Definition TVirtualPad.h:54
virtual Short_t GetBorderSize() const =0
virtual Bool_t IsEditable() const =0
virtual Double_t GetX1() const =0
Abstract 3D shapes viewer.
virtual Bool_t BuildingScene() const =0
virtual void EndScene()=0
static TVirtualViewer3D * Viewer3D(TVirtualPad *pad=nullptr, Option_t *type="")
Create a Viewer 3D of specified type.
virtual void PadPaint(TVirtualPad *)
virtual void BeginScene()=0
virtual Bool_t CanLoopOnPrimitives() const
Short_t GetBorderSize() const
Definition TWbox.h:41
Short_t GetBorderMode() const
Definition TWbox.h:40
virtual void SetBorderMode(Short_t bordermode)
Definition TWbox.h:51
virtual void SetBorderSize(Short_t bordersize)
Definition TWbox.h:52
TPaveText * pt
void box(Int_t pat, Double_t x1, Double_t y1, Double_t x2, Double_t y2)
Definition fillpatterns.C:1
std::ostream & Info()
Definition hadd.cxx:171
Double_t y[n]
Definition legend1.C:17
Double_t x[n]
Definition legend1.C:17
const Int_t n
Definition legend1.C:16
TGraphErrors * gr
Definition legend1.C:25
Double_t ex[n]
Definition legend1.C:17
leg
Definition legend1.C:34
R__ALWAYS_INLINE bool HasBeenDeleted(const TObject *obj)
Check if the TObject's memory has been deleted.
Definition TObject.h:408
void CallRecursiveRemoveIfNeeded(TObject &obj)
call RecursiveRemove for obj if gROOT is valid and obj.TestBit(kMustCleanup) is true.
Definition TROOT.h:395
Bool_t IsNaN(Double_t x)
Definition TMath.h:896
Int_t Nint(T x)
Round to nearest integer. Rounds half integers to the nearest even integer.
Definition TMath.h:697
Short_t Max(Short_t a, Short_t b)
Returns the largest of a and b.
Definition TMathBase.h:250
Double_t Exp(Double_t x)
Returns the base-e exponential function of x, which is e raised to the power x.
Definition TMath.h:713
Double_t Floor(Double_t x)
Rounds x downward, returning the largest integral value that is not greater than x.
Definition TMath.h:684
Double_t Ceil(Double_t x)
Rounds x upward, returning the smallest integral value that is not less than x.
Definition TMath.h:672
Int_t Finite(Double_t x)
Check if it is finite with a mask in order to be consistent in presence of fast math.
Definition TMath.h:774
Double_t Log(Double_t x)
Returns the natural logarithm of x.
Definition TMath.h:760
Double_t Sqrt(Double_t x)
Returns the square root of x.
Definition TMath.h:666
LongDouble_t Power(LongDouble_t x, LongDouble_t y)
Returns x raised to the power y.
Definition TMath.h:725
Short_t Min(Short_t a, Short_t b)
Returns the smallest of a and b.
Definition TMathBase.h:198
Double_t Cos(Double_t)
Returns the cosine of an angle of x radians.
Definition TMath.h:598
constexpr Double_t Pi()
Definition TMath.h:37
Double_t Sin(Double_t)
Returns the sine of an angle of x radians.
Definition TMath.h:592
Double_t Log10(Double_t x)
Returns the common (base-10) logarithm of x.
Definition TMath.h:766
Short_t Abs(Short_t d)
Returns the absolute value of parameter Short_t d.
Definition TMathBase.h:123
void inv(rsa_NUMBER *, rsa_NUMBER *, rsa_NUMBER *)
Definition rsaaux.cxx:949
Rectangle structure (maps to the X11 XRectangle structure)
Definition GuiTypes.h:361
Short_t fX
Definition GuiTypes.h:362
UShort_t fHeight
Definition GuiTypes.h:363
Short_t fY
Definition GuiTypes.h:362
UShort_t fWidth
Definition GuiTypes.h:363
struct used by ShowGuidelines to store the distance Field between objects in the canvas.
Definition TPad.cxx:6273
TAttBBox2D * fb
Definition TPad.cxx:6275
dField()
Definition TPad.cxx:6280
char fdir
Definition TPad.cxx:6277
dField(TAttBBox2D *a, TAttBBox2D *b, Int_t dist, char direction)
Definition TPad.cxx:6284
TAttBBox2D * fa
Definition TPad.cxx:6274
Int_t fdist
Definition TPad.cxx:6276
TMarker m
Definition textangle.C:8