Garfield++ 5.0
A toolkit for the detailed simulation of particle detectors based on ionisation measurement in gases and semiconductors
Loading...
Searching...
No Matches
Garfield::ViewSignal Class Reference

Plot the signal computed by a sensor as a ROOT histogram. More...

#include <ViewSignal.hh>

+ Inheritance diagram for Garfield::ViewSignal:

Public Member Functions

 ViewSignal ()
 Constructor.
 
 ~ViewSignal ()=default
 Destructor.
 
void SetSensor (Sensor *s)
 Set the sensor from which to retrieve the signal.
 
void PlotSignal (const std::string &label, const std::string &optTotal="t", const std::string &optPrompt="", const std::string &optDelayed="", const bool same=false)
 
TH1D * GetHistogram (const char h='t')
 
void SetRangeX (const double xmin, const double xmax)
 Set the x-axis limits explicitly.
 
void UnsetRangeX ()
 Remove the user-defined x-axis limits.
 
void SetRangeY (const double ymin, const double ymax)
 Set the y-axis limits explicitly.
 
void UnsetRangeY ()
 Remove the user-defined y-axis limits.
 
void SetLabelY (const std::string &label)
 Override the default y-axis label.
 
void EnableLegend (const bool on=true)
 Draw a legend on the plot or not.
 
void SetColourTotal (const short col)
 Set the (ROOT) colour with which to draw the total signal.
 
void SetColourElectrons (const short col)
 Set the (ROOT) colour with which to draw the electron component.
 
void SetColourIons (const short col)
 Set the (ROOT) colour with which to draw the hole/ion component.
 
void SetColourHoles (const short col)
 Set the (ROOT) colour with which to draw the hole/ion component.
 
void SetColourDelayed (const short colTotal, const short colElectrons=kYellow - 7, const short colIons=kRed - 9)
 Set the (ROOT) colours with which to draw the delayed signal(s).
 
- Public Member Functions inherited from Garfield::ViewBase
 ViewBase ()=delete
 Default constructor.
 
 ViewBase (const std::string &name)
 Constructor.
 
virtual ~ViewBase ()=default
 Destructor.
 
void SetCanvas (TPad *pad)
 Set the canvas to be painted on.
 
void SetCanvas ()
 Unset an external canvas.
 
TPad * GetCanvas ()
 Retrieve the canvas.
 
void SetArea (const double xmin, const double ymin, const double xmax, const double ymax)
 
virtual void SetArea (const double xmin, const double ymin, const double zmin, const double xmax, const double ymax, const double zmax)
 Set a bounding box (if applicable).
 
void SetArea ()
 
virtual void SetPlane (const double fx, const double fy, const double fz, const double x0, const double y0, const double z0)
 
virtual void SetPlane (const double fx, const double fy, const double fz, const double x0, const double y0, const double z0, const double hx, const double hy, const double hz)
 Set the projection plane specifying a hint for the in-plane x axis.
 
void Rotate (const double angle)
 Rotate the viewing plane (angle in radian).
 
void SetPlaneXY ()
 Set the viewing plane to x-y.
 
void SetPlaneXZ ()
 Set the viewing plane to x-z.
 
void SetPlaneYZ ()
 Set the viewing plane to y-z.
 
void SetPlaneZX ()
 Set the viewing plane to z-x.
 
void SetPlaneZY ()
 Set the viewing plane to z-y.
 
void EnableDebugging (const bool on=true)
 Switch on/off debugging output.
 

Additional Inherited Members

- Static Public Member Functions inherited from Garfield::ViewBase
static std::string FindUnusedFunctionName (const std::string &s)
 Find an unused function name.
 
static std::string FindUnusedHistogramName (const std::string &s)
 Find an unused histogram name.
 
static std::string FindUnusedCanvasName (const std::string &s)
 Find an unused canvas name.
 
- Protected Member Functions inherited from Garfield::ViewBase
void UpdateProjectionMatrix ()
 
template<typename T>
void ToPlane (const T x, const T y, const T z, T &xp, T &yp) const
 
template<typename T>
bool InBox (const std::array< T, 3 > &x) const
 
void Clip (const std::array< float, 3 > &x0, const std::array< float, 3 > &x1, std::array< float, 3 > &xc) const
 
void DrawLine (const std::vector< std::array< float, 3 > > &xl, const short col, const short lw)
 
std::string LabelX ()
 
std::string LabelY ()
 
std::string PlaneDescription ()
 
bool PlotLimits (Sensor *sensor, double &xmin, double &ymin, double &xmax, double &ymax) const
 
bool PlotLimits (Component *cmp, double &xmin, double &ymin, double &xmax, double &ymax) const
 
bool PlotLimitsFromUserBox (double &xmin, double &ymin, double &xmax, double &ymax) const
 
bool PlotLimits (std::array< double, 3 > &bbmin, std::array< double, 3 > &bbmax, double &xmin, double &ymin, double &xmax, double &ymax) const
 
- Static Protected Member Functions inherited from Garfield::ViewBase
static bool RangeSet (TVirtualPad *)
 
static void SetRange (TVirtualPad *pad, const double x0, const double y0, const double x1, const double y1)
 
- Protected Attributes inherited from Garfield::ViewBase
std::string m_className = "ViewBase"
 
bool m_debug = false
 
bool m_userPlotLimits = false
 
double m_xMinPlot = -1.
 
double m_xMaxPlot = 1.
 
double m_yMinPlot = -1.
 
double m_yMaxPlot = 1.
 
bool m_userBox = false
 
double m_xMinBox = -1.
 
double m_xMaxBox = 1.
 
double m_yMinBox = -1.
 
double m_yMaxBox = 1.
 
double m_zMinBox = -1.
 
double m_zMaxBox = 1.
 
std::array< std::array< double, 3 >, 3 > m_proj
 
std::array< double, 4 > m_plane {{0, 0, 1, 0}}
 
std::array< std::array< double, 3 >, 3 > m_prmat
 

Detailed Description

Plot the signal computed by a sensor as a ROOT histogram.

Definition at line 19 of file ViewSignal.hh.

Constructor & Destructor Documentation

◆ ViewSignal()

Garfield::ViewSignal::ViewSignal ( )

Constructor.

Definition at line 16 of file ViewSignal.cc.

16: ViewBase("ViewSignal") {}
ViewBase()=delete
Default constructor.

◆ ~ViewSignal()

Garfield::ViewSignal::~ViewSignal ( )
default

Destructor.

Member Function Documentation

◆ EnableLegend()

void Garfield::ViewSignal::EnableLegend ( const bool on = true)
inline

Draw a legend on the plot or not.

Definition at line 77 of file ViewSignal.hh.

77{ m_legend = on; }

◆ GetHistogram()

TH1D * Garfield::ViewSignal::GetHistogram ( const char h = 't')
inline

Retrieve the histogram for the total, prompt and delayed induced charge or signal.

Parameters
hhistogram to be returned ('t': total, 'e': electron-induced, 'i': ion/hole-induced).

Definition at line 58 of file ViewSignal.hh.

58 {
59 return h == 'e' ? m_hSignalElectrons.get()
60 : h == 'i' ? m_hSignalIons.get() : m_hSignal.get();
61 }

◆ PlotSignal()

void Garfield::ViewSignal::PlotSignal ( const std::string & label,
const std::string & optTotal = "t",
const std::string & optPrompt = "",
const std::string & optDelayed = "",
const bool same = false )

Plot the signal.

Parameters
labelIdentifier (weighting field) of the signal to be plotted.
optTotalString containing information about the total signals you want to plot. The syntax is the first letter of the charge carrier signal component you want to plot: "t" for total, "e" for electron and "i" for ion/hole. "tei" enables all three components. The total signal is always plotted.
optPromptString containing information about the prompt signal components you want to plot. The syntax is identical to the one described above.
optDelayedString containing information about the delayed signal components you want to plot. The syntax is identical to the one described above.
sameFlag to keep existing plots on the canvas or not.

Definition at line 61 of file ViewSignal.cc.

65 {
66 const bool totalT = true;
67 const bool totalP =
68 settingPrompt.find("t") != std::string::npos ? true : false;
69 const bool totalD =
70 settingDelayed.find("t") != std::string::npos ? true : false;
71 const bool electronT =
72 settingTotal.find("e") != std::string::npos ? true : false;
73 const bool electronP =
74 settingPrompt.find("e") != std::string::npos ? true : false;
75 const bool electronD =
76 settingDelayed.find("e") != std::string::npos ? true : false;
77 const bool ionT = settingTotal.find("i") != std::string::npos ? true : false;
78 const bool ionP = settingPrompt.find("i") != std::string::npos ? true : false;
79 const bool ionD =
80 settingDelayed.find("i") != std::string::npos ? true : false;
81
82 constexpr int lineThickness = 5;
83 constexpr double tol = 1e-50;
84
85 if (!m_sensor) {
86 std::cerr << m_className << "::PlotSignal: Sensor is not defined.\n";
87 return;
88 }
89
90 auto canvas = GetCanvas();
91 canvas->cd();
92 canvas->SetTitle("Signal");
93
94 unsigned int nBins = 100;
95 double t0 = 0., dt = 1.;
96 m_sensor->GetTimeWindow(t0, dt, nBins);
97 const double t1 = t0 + nBins * dt;
98
99 std::string xlabel = "time [ns]";
100 std::string ylabel = m_labelY;
101 if (ylabel.empty()) {
102 ylabel = m_sensor->IsIntegrated(label) ? "signal [fC]" : "signal [fC / ns]";
103 }
104 unsigned int nPlots = same ? 1 : 0;
105 if (!RangeSet(gPad)) nPlots = 0;
106
107 if (totalT) {
108 const auto hname = FindUnusedHistogramName("hSignal_");
109 m_hSignal.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
110 m_hSignal->SetDirectory(nullptr);
111 m_hSignal->SetLineColor(m_colTotal);
112 for (unsigned int i = 0; i < nBins; ++i) {
113 const double sig = m_sensor->GetSignal(label, i, 0);
114 if (!std::isnan(sig) && std::abs(sig) < tol) {
115 m_hSignal->SetBinContent(i + 1, 0.);
116 } else {
117 m_hSignal->SetBinContent(i + 1, sig);
118 }
119 }
120
121 m_hSignal->SetLineWidth(lineThickness);
122
123 const std::string opt = nPlots > 0 ? "same" : "";
124 ++nPlots;
125
126 // Get and plot threshold crossings.
127 const auto nCrossings = m_sensor->GetNumberOfThresholdCrossings();
128 if (nCrossings > 0) {
129 TGraph gCrossings;
130 gCrossings.SetMarkerStyle(20);
131 gCrossings.SetMarkerColor(m_colTotal);
132 std::vector<double> xp;
133 std::vector<double> yp;
134 double time = 0., level = 0.;
135 bool rise = true;
136 for (unsigned int i = 0; i < nCrossings; ++i) {
137 if (m_sensor->GetThresholdCrossing(i, time, level, rise)) {
138 xp.push_back(time);
139 yp.push_back(level);
140 }
141 }
142 gCrossings.DrawGraph(xp.size(), xp.data(), yp.data(), "psame");
143 }
144 DrawHistogram(m_hSignal.get(), opt, xlabel, ylabel);
145 }
146
147 if (totalD) {
148 const auto hname = FindUnusedHistogramName("hDelayedSignal_");
149 m_hDelayedSignal.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
150 m_hDelayedSignal->SetDirectory(nullptr);
151 m_hDelayedSignal->SetLineColor(m_colDelayed[3]);
152 m_hDelayedSignal->SetLineStyle(7);
153 m_hDelayedSignal->SetStats(0);
154 for (unsigned int i = 0; i < nBins; ++i) {
155 const double sig = m_sensor->GetSignal(label, i, 2);
156 if (!std::isnan(sig) && std::abs(sig) < tol) {
157 m_hDelayedSignal->SetBinContent(i + 1, 0.);
158 } else {
159 m_hDelayedSignal->SetBinContent(i + 1, sig);
160 }
161 }
162 m_hDelayedSignal->SetLineWidth(lineThickness);
163 m_hDelayedSignal->DrawCopy("same");
164 }
165
166 if (totalP) {
167 const auto hname = FindUnusedHistogramName("hPromptSignal_");
168 m_hPromptSignal.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
169 m_hPromptSignal->SetDirectory(nullptr);
170 m_hPromptSignal->SetLineColor(m_colPrompt[0]);
171 m_hPromptSignal->SetLineStyle(2);
172 m_hPromptSignal->SetStats(0);
173 for (unsigned int i = 0; i < nBins; ++i) {
174 const double sig = m_sensor->GetSignal(label, i, 1);
175 if (!std::isnan(sig) && std::abs(sig) < tol) {
176 m_hPromptSignal->SetBinContent(i + 1, 0.);
177 } else {
178 m_hPromptSignal->SetBinContent(i + 1, sig);
179 }
180 }
181
182 m_hPromptSignal->SetLineWidth(lineThickness);
183 m_hPromptSignal->DrawCopy("same");
184 }
185
186 if (electronT) {
187 const auto hname = FindUnusedHistogramName("hSignalElectrons_");
188 m_hSignalElectrons.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
189 m_hSignalElectrons->SetDirectory(nullptr);
190 m_hSignalElectrons->SetLineColor(m_colElectrons);
191 for (unsigned int i = 0; i < nBins; ++i) {
192 const double sig = m_sensor->GetElectronSignal(label, i);
193 m_hSignalElectrons->SetBinContent(i + 1, sig);
194 }
195 m_hSignalElectrons->SetLineWidth(lineThickness);
196 m_hSignalElectrons->DrawCopy("same");
197 }
198
199 if (electronD) {
200 const auto hname = FindUnusedHistogramName("m_hDelayedElectrons_");
201 m_hDelayedElectrons.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
202 m_hDelayedElectrons->SetDirectory(nullptr);
203 m_hDelayedElectrons->SetLineColor(m_colDelayed[4]);
204 m_hDelayedElectrons->SetLineStyle(7);
205 m_hDelayedElectrons->SetStats(0);
206 for (unsigned int i = 0; i < nBins; ++i) {
207 const double sig = m_sensor->GetDelayedElectronSignal(label, i);
208 if (!std::isnan(sig) && std::abs(sig) < tol) {
209 m_hDelayedElectrons->SetBinContent(i + 1, 0.);
210 } else {
211 m_hDelayedElectrons->SetBinContent(i + 1, sig);
212 }
213 }
214 m_hDelayedElectrons->SetLineWidth(lineThickness);
215 m_hDelayedElectrons->DrawCopy("same");
216 }
217
218 if (electronP) {
219 const auto hname = FindUnusedHistogramName("m_hPromptElectrons_");
220 m_hPromptElectrons.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
221 m_hPromptElectrons->SetDirectory(nullptr);
222 m_hPromptElectrons->SetLineColor(m_colPrompt[1]);
223 m_hPromptElectrons->SetLineStyle(2);
224 m_hPromptElectrons->SetStats(0);
225 for (unsigned int i = 0; i < nBins; ++i) {
226 const double sig = m_sensor->GetElectronSignal(label, i) -
227 m_sensor->GetDelayedElectronSignal(label, i);
228 if (!std::isnan(sig) && std::abs(sig) < tol) {
229 m_hPromptElectrons->SetBinContent(i + 1, 0.);
230 } else {
231 m_hPromptElectrons->SetBinContent(i + 1, sig);
232 }
233 }
234
235 m_hPromptElectrons->SetLineWidth(lineThickness);
236 m_hPromptElectrons->DrawCopy("same");
237 }
238
239 if (ionT) {
240 const auto hname = FindUnusedHistogramName("m_hSignalIons_");
241 m_hSignalIons.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
242 m_hSignalIons->SetDirectory(nullptr);
243 m_hSignalIons->SetLineColor(m_colIons);
244 for (unsigned int i = 0; i < nBins; ++i) {
245 const double sig = m_sensor->GetIonSignal(label, i);
246 m_hSignalIons->SetBinContent(i + 1, sig);
247 }
248 m_hSignalIons->SetLineWidth(lineThickness);
249 m_hSignalIons->DrawCopy("same");
250 }
251
252 if (ionD) {
253 const auto hname = FindUnusedHistogramName("m_hDelayedIons_");
254 m_hDelayedIons.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
255 m_hDelayedIons->SetDirectory(nullptr);
256 m_hDelayedIons->SetLineColor(m_colDelayed[5]);
257 m_hDelayedIons->SetLineStyle(7);
258 m_hDelayedIons->SetStats(0);
259 for (unsigned int i = 0; i < nBins; ++i) {
260 const double sig = m_sensor->GetDelayedIonSignal(label, i);
261 if (!std::isnan(sig) && std::abs(sig) < tol) {
262 m_hDelayedIons->SetBinContent(i + 1, 0.);
263 } else {
264 m_hDelayedIons->SetBinContent(i + 1, sig);
265 }
266 }
267 m_hDelayedIons->SetLineWidth(lineThickness);
268 m_hDelayedIons->DrawCopy("same");
269 }
270
271 if (ionP) {
272 std::cerr << m_className << "::ionP.\n";
273 const auto hname = FindUnusedHistogramName("m_hPromptIons_");
274 m_hPromptIons.reset(new TH1D(hname.c_str(), "", nBins, t0, t1));
275 m_hPromptIons->SetDirectory(nullptr);
276 m_hPromptIons->SetLineColor(m_colPrompt[2]);
277 m_hPromptIons->SetLineStyle(2);
278 m_hPromptIons->SetStats(0);
279 for (unsigned int i = 0; i < nBins; ++i) {
280 const double sig = m_sensor->GetIonSignal(label, i) -
281 m_sensor->GetDelayedIonSignal(label, i);
282 if (!std::isnan(sig) && std::abs(sig) < tol) {
283 m_hPromptIons->SetBinContent(i + 1, 0.);
284 } else {
285 m_hPromptIons->SetBinContent(i + 1, sig);
286 }
287 }
288
289 m_hPromptIons->SetLineWidth(lineThickness);
290 m_hPromptIons->DrawCopy("same");
291 }
292
293 if (m_legend) {
294 TLegend* leg = new TLegend(0.7, 0.7, 0.9, 0.9);
295 leg->SetHeader("Induced current components");
296 if (totalT) leg->AddEntry(m_hSignal.get(), "Total induced signal", "l");
297 if (totalP) {
298 leg->AddEntry(m_hPromptSignal.get(), "Prompt induced signal", "l");
299 }
300 if (totalD) {
301 leg->AddEntry(m_hDelayedSignal.get(), "Delayed induced signal", "l");
302 }
303 if (electronT) {
304 leg->AddEntry(m_hSignalElectrons.get(), "Electron induced signal", "l");
305 }
306 if (electronP) {
307 leg->AddEntry(m_hPromptElectrons.get(), "Electron prompt induced signal",
308 "l");
309 }
310 if (electronD) {
311 leg->AddEntry(m_hDelayedElectrons.get(),
312 "Electron delayed induced signal", "l");
313 }
314 if (ionT) leg->AddEntry(m_hSignalIons.get(), "Ion induced signal", "l");
315 if (ionP) {
316 leg->AddEntry(m_hPromptIons.get(), "Ion/hole prompt induced signal", "l");
317 }
318 if (ionD) {
319 leg->AddEntry(m_hDelayedIons.get(), "Ion/hole delayed induced signal",
320 "l");
321 }
322 leg->Draw();
323 }
324 gPad->Update();
325}
static std::string FindUnusedHistogramName(const std::string &s)
Find an unused histogram name.
Definition ViewBase.cc:327
std::string m_className
Definition ViewBase.hh:82
TPad * GetCanvas()
Retrieve the canvas.
Definition ViewBase.cc:74
static bool RangeSet(TVirtualPad *)
Definition ViewBase.cc:83

Referenced by main(), and Garfield::Sensor::PlotSignal().

◆ SetColourDelayed()

void Garfield::ViewSignal::SetColourDelayed ( const short colTotal,
const short colElectrons = kYellow - 7,
const short colIons = kRed - 9 )
inline

Set the (ROOT) colours with which to draw the delayed signal(s).

Definition at line 89 of file ViewSignal.hh.

91 {
92 m_colDelayed = {colTotal, colElectrons, colIons};
93 }

◆ SetColourElectrons()

void Garfield::ViewSignal::SetColourElectrons ( const short col)
inline

Set the (ROOT) colour with which to draw the electron component.

Definition at line 82 of file ViewSignal.hh.

82{ m_colElectrons = col; }

◆ SetColourHoles()

void Garfield::ViewSignal::SetColourHoles ( const short col)
inline

Set the (ROOT) colour with which to draw the hole/ion component.

Definition at line 86 of file ViewSignal.hh.

86{ m_colIons = col; }

◆ SetColourIons()

void Garfield::ViewSignal::SetColourIons ( const short col)
inline

Set the (ROOT) colour with which to draw the hole/ion component.

Definition at line 84 of file ViewSignal.hh.

84{ m_colIons = col; }

◆ SetColourTotal()

void Garfield::ViewSignal::SetColourTotal ( const short col)
inline

Set the (ROOT) colour with which to draw the total signal.

Definition at line 80 of file ViewSignal.hh.

80{ m_colTotal = col; }

◆ SetLabelY()

void Garfield::ViewSignal::SetLabelY ( const std::string & label)
inline

Override the default y-axis label.

Definition at line 74 of file ViewSignal.hh.

74{ m_labelY = label; }

◆ SetRangeX()

void Garfield::ViewSignal::SetRangeX ( const double xmin,
const double xmax )

Set the x-axis limits explicitly.

Definition at line 26 of file ViewSignal.cc.

26 {
27 if (fabs(xmax - xmin) < Small) {
28 std::cerr << m_className << "::SetRangeX: Invalid range.\n";
29 return;
30 }
31 m_xmin = std::min(xmin, xmax);
32 m_xmax = std::max(xmin, xmax);
33 m_userRangeX = true;
34}
DoubleAc fabs(const DoubleAc &f)
Definition DoubleAc.h:615

◆ SetRangeY()

void Garfield::ViewSignal::SetRangeY ( const double ymin,
const double ymax )

Set the y-axis limits explicitly.

Definition at line 36 of file ViewSignal.cc.

36 {
37 if (fabs(ymax - ymin) < Small) {
38 std::cerr << m_className << "::SetRangeY: Invalid range.\n";
39 return;
40 }
41 m_ymin = std::min(ymin, ymax);
42 m_ymax = std::max(ymin, ymax);
43 m_userRangeY = true;
44}

◆ SetSensor()

void Garfield::ViewSignal::SetSensor ( Sensor * s)

Set the sensor from which to retrieve the signal.

Definition at line 18 of file ViewSignal.cc.

18 {
19 if (!s) {
20 std::cerr << m_className << "::SetSensor: Null pointer.\n";
21 return;
22 }
23 m_sensor = s;
24}

Referenced by main(), and Garfield::Sensor::PlotSignal().

◆ UnsetRangeX()

void Garfield::ViewSignal::UnsetRangeX ( )
inline

Remove the user-defined x-axis limits.

Definition at line 66 of file ViewSignal.hh.

66{ m_userRangeX = false; }

◆ UnsetRangeY()

void Garfield::ViewSignal::UnsetRangeY ( )
inline

Remove the user-defined y-axis limits.

Definition at line 71 of file ViewSignal.hh.

71{ m_userRangeY = false; }

The documentation for this class was generated from the following files: