LArSoft  v10_04_05
Liquid Argon Software toolkit - https://larsoft.org/
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
BoxBoundedGeo.cxx
Go to the documentation of this file.
1 
7 
8 // LArSoft libraries
10 
11 // ROOT libraries
12 #include "Math/GenVector/Cartesian3D.h"
13 #include "Math/GenVector/DisplacementVector3D.h"
14 #include "Math/GenVector/PositionVector3D.h"
15 
16 #include <array>
17 #include <type_traits>
18 #include <vector>
19 
20 namespace geo {
21  //----------------------------------------------------------------------------
22  bool BoxBoundedGeo::ContainsPosition(TVector3 const& point, double wiggle /* = 1.0 */) const
23  {
24  return ContainsPosition(vect::toPoint(point), wiggle);
25  }
26 
27  //----------------------------------------------------------------------------
28  bool BoxBoundedGeo::ContainsPosition(double const* point, double wiggle /* = 1.0 */) const
29  {
30  return ContainsPosition(vect::makePointFromCoords(point), wiggle);
31  }
32 
33  //----------------------------------------------------------------------------
34  std::vector<Point_t> BoxBoundedGeo::GetIntersections(Point_t const& TrajectoryStart,
35  Vector_t const& TrajectoryDirect) const
36  {
37  std::vector<Point_t> IntersectionPoints;
38  std::vector<double> LineParameters;
39 
40  // Generate normal vectors and offsets for every plane of the box
41  // All normal vectors are headed outwards
42  // BUG the double brace syntax is required to work around clang bug 21629
43  // (https://bugs.llvm.org/show_bug.cgi?id=21629)
44  static std::array<Vector_t, 6U> const NormalVectors = {{
45  -Xaxis<Vector_t>(),
46  Xaxis<Vector_t>(), // anode, cathode,
47  -Yaxis<Vector_t>(),
48  Yaxis<Vector_t>(), // bottom, top,
49  -Zaxis<Vector_t>(),
50  Zaxis<Vector_t>() // upstream, downstream
51  }};
52  // BUG the double brace syntax is required to work around clang bug 21629
53  // (https://bugs.llvm.org/show_bug.cgi?id=21629)
54  std::array<Point_t, 6U> const NormalVectorOffsets = {{
55  Point_t{Min().X(), Min().Y(), Min().Z()}, // Anode offset
56  Point_t{Max().X(), Min().Y(), Min().Z()}, // Cathode
57  Point_t{Min().X(), Min().Y(), Min().Z()}, // Bottom
58  Point_t{Min().X(), Max().Y(), Min().Z()}, // Top
59  Point_t{Min().X(), Min().Y(), Min().Z()}, // upstream
60  Point_t{Min().X(), Min().Y(), Max().Z()} // downstream
61  }};
62 
63  // Loop over all surfaces of the box
64  for (unsigned int face_no = 0; face_no < NormalVectors.size(); face_no++) {
65  // Check if trajectory and surface are not parallel
66  if (NormalVectors[face_no].Dot(TrajectoryDirect)) {
67  // Calculate the line parameter for the intersection points
68  LineParameters.push_back(
69  NormalVectors[face_no].Dot(NormalVectorOffsets.at(face_no) - TrajectoryStart) /
70  NormalVectors[face_no].Dot(TrajectoryDirect));
71  }
72  else
73  continue;
74 
75  // Calculate intersection point using the line parameter
76  IntersectionPoints.push_back(TrajectoryStart + LineParameters.back() * TrajectoryDirect);
77 
78  // Coordinate which should be ignored when checking for limits added by Christoph
79  // Rudolf von Rohr 05/21/2016
80  unsigned int NoCheckCoord;
81 
82  // Calculate NoCheckCoord out of the face_no
83  if (face_no % 2) {
84  // Convert odd face number to coordinate
85  NoCheckCoord = (face_no - 1) / 2;
86  }
87  else {
88  // Convert even face number to coordinate
89  NoCheckCoord = face_no / 2;
90  }
91 
92  // Loop over all three space coordinates
93  unsigned int coord = 0;
94  for (auto extractCoord : vect::coordReaders<Point_t>()) {
95  auto const lastPointCoord = vect::bindCoord(IntersectionPoints.back(), extractCoord);
96  auto const minCoord = vect::bindCoord(c_min, extractCoord);
97  auto const maxCoord = vect::bindCoord(c_max, extractCoord);
98 
99  // Changed by Christoph Rudolf von Rohr 05/21/2016 Then check if point is not
100  // within the surface limits at this coordinate, without looking at the plane
101  // normal vector coordinate. We can assume, that our algorithm already found this
102  // coordinate correctily. In rare cases, looking for boundaries in this
103  // coordinate causes unexpected behavior due to floating point inaccuracies.
104  if (coord++ != NoCheckCoord &&
105  ((lastPointCoord() > maxCoord()) || (lastPointCoord() < minCoord))) {
106  // if off limits, get rid of the useless data and break the coordinate loop
107  LineParameters.pop_back();
108  IntersectionPoints.pop_back();
109  break;
110  }
111  } // coordinate loop
112  } // Surcaces loop
113 
114  // sort points according to their parameter value (first is entry, second is exit)
115  if (LineParameters.size() == 2 && LineParameters.front() > LineParameters.back()) {
116  std::swap(IntersectionPoints.front(), IntersectionPoints.back());
117  }
118 
119  return IntersectionPoints;
120  } // GetIntersections()
121 
122  //----------------------------------------------------------------------------
123  std::vector<TVector3> BoxBoundedGeo::GetIntersections(TVector3 const& TrajectoryStart,
124  TVector3 const& TrajectoryDirect) const
125  {
126  std::vector<TVector3> intersections;
127  for (auto const& point : GetIntersections(Point_t(TrajectoryStart), Vector_t(TrajectoryDirect)))
128  intersections.emplace_back(point.X(), point.Y(), point.Z());
129  return intersections;
130  }
131 
132  //----------------------------------------------------------------------------
134  {
135  for (auto coordMan : vect::coordManagers<Point_t>()) {
136  auto min = vect::bindCoord(c_min, coordMan);
137  auto max = vect::bindCoord(c_max, coordMan);
138  if (min() > max()) {
139  auto temp = min();
140  min = max();
141  max = temp;
142  }
143  } // for
144  } // BoxBoundedGeo::SortCoordinates()
145 
146  //----------------------------------------------------------------------------
147 
148 } // namespace geo
ROOT::Math::DisplacementVector3D< ROOT::Math::Cartesian3D< double >, ROOT::Math::GlobalCoordinateSystemTag > Vector_t
Type for representation of momenta in 3D space.
Definition: geo_vectors.h:160
Point_t Max() const
Returns the corner point with the largest coordinates.
GENVECTOR_CONSTEXPR Point_t makePointFromCoords(Coords &&coords)
Creates a geo::Point_t from its coordinates (see makeFromCoords()).
auto coord(Vector &v, unsigned int n) noexcept
Returns an object to manage the coordinate n of a vector.
constexpr auto bindCoord(Vector const &v, CoordReader_t< Vector > helper)
Binds the specified constant vector to the coordinate reader.
Utilities to extend the interface of geometry vectors.This library provides facilities that can be us...
Point_t toPoint(Point const &p)
Convert the specified point into a geo::Point_t.
Provides a base class aware of world box coordinates.
Coords_t c_max
maximum coordinates (x, y, z)
ROOT::Math::PositionVector3D< ROOT::Math::Cartesian3D< double >, ROOT::Math::GlobalCoordinateSystemTag > Point_t
Type for representation of position in physical 3D space.
Definition: geo_vectors.h:180
Point_t Min() const
Returns the corner point with the smallest coordinates.
Coords_t c_min
minimum coordinates (x, y, z)
void SortCoordinates()
Makes sure each coordinate of the minimum point is smaller than maximum.
std::vector< TVector3 > GetIntersections(TVector3 const &TrajectoryStart, TVector3 const &TrajectoryDirect) const
Calculates the entry and exit points of a trajectory on the box surface.
ROOT libraries.
bool ContainsPosition(Point_t const &point, double wiggle=1.0) const
Returns whether this volume contains the specified point.