Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4ExtrudedSolid Class Reference

#include <G4ExtrudedSolid.hh>

+ Inheritance diagram for G4ExtrudedSolid:

Classes

struct  ZSection
 

Public Member Functions

 G4ExtrudedSolid (const G4String &pName, const std::vector< G4TwoVector > &polygon, const std::vector< ZSection > &zsections)
 
 G4ExtrudedSolid (const G4String &pName, const std::vector< G4TwoVector > &polygon, G4double halfZ, const G4TwoVector &off1=G4TwoVector(0., 0.), G4double scale1=1., const G4TwoVector &off2=G4TwoVector(0., 0.), G4double scale2=1.)
 
 ~G4ExtrudedSolid () override
 
G4int GetNofVertices () const
 
G4TwoVector GetVertex (G4int index) const
 
std::vector< G4TwoVectorGetPolygon () const
 
G4int GetNofZSections () const
 
ZSection GetZSection (G4int index) const
 
std::vector< ZSectionGetZSections () const
 
EInside Inside (const G4ThreeVector &p) const override
 
G4ThreeVector SurfaceNormal (const G4ThreeVector &p) const override
 
G4double DistanceToIn (const G4ThreeVector &p, const G4ThreeVector &v) const override
 
G4double DistanceToIn (const G4ThreeVector &p) const override
 
G4double DistanceToOut (const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm=false, G4bool *validNorm=nullptr, G4ThreeVector *n=nullptr) const override
 
G4double DistanceToOut (const G4ThreeVector &p) const override
 
void BoundingLimits (G4ThreeVector &pMin, G4ThreeVector &pMax) const override
 
G4bool CalculateExtent (const EAxis pAxis, const G4VoxelLimits &pVoxelLimit, const G4AffineTransform &pTransform, G4double &pMin, G4double &pMax) const override
 
G4GeometryType GetEntityType () const override
 
G4bool IsFaceted () const override
 
G4VSolidClone () const override
 
std::ostream & StreamInfo (std::ostream &os) const override
 
 G4ExtrudedSolid (__void__ &)
 
 G4ExtrudedSolid (const G4ExtrudedSolid &rhs)
 
G4ExtrudedSolidoperator= (const G4ExtrudedSolid &rhs)
 
- Public Member Functions inherited from G4TessellatedSolid
 G4TessellatedSolid ()
 
 ~G4TessellatedSolid () override
 
 G4TessellatedSolid (const G4String &name)
 
 G4TessellatedSolid (__void__ &)
 
 G4TessellatedSolid (const G4TessellatedSolid &ts)
 
G4TessellatedSolidoperator= (const G4TessellatedSolid &right)
 
G4TessellatedSolidoperator+= (const G4TessellatedSolid &right)
 
G4bool AddFacet (G4VFacet *aFacet)
 
G4VFacetGetFacet (G4int i) const
 
G4int GetNumberOfFacets () const
 
G4int GetFacetIndex (const G4ThreeVector &p) const
 
EInside Inside (const G4ThreeVector &p) const override
 
G4ThreeVector SurfaceNormal (const G4ThreeVector &p) const override
 
G4double DistanceToIn (const G4ThreeVector &p, const G4ThreeVector &v) const override
 
G4double DistanceToIn (const G4ThreeVector &p) const override
 
G4double DistanceToOut (const G4ThreeVector &p) const override
 
G4double DistanceToOut (const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm, G4bool *validNorm, G4ThreeVector *norm) const override
 
virtual G4bool Normal (const G4ThreeVector &p, G4ThreeVector &n) const
 
virtual G4double SafetyFromOutside (const G4ThreeVector &p, G4bool aAccurate=false) const
 
virtual G4double SafetyFromInside (const G4ThreeVector &p, G4bool aAccurate=false) const
 
G4GeometryType GetEntityType () const override
 
G4bool IsFaceted () const override
 
std::ostream & StreamInfo (std::ostream &os) const override
 
G4VSolidClone () const override
 
G4ThreeVector GetPointOnSurface () const override
 
G4double GetSurfaceArea () override
 
G4double GetCubicVolume () override
 
void SetSolidClosed (const G4bool t)
 
G4bool GetSolidClosed () const
 
G4int CheckStructure () const
 
void SetMaxVoxels (G4int max)
 
G4VoxelizerGetVoxels ()
 
G4bool CalculateExtent (const EAxis pAxis, const G4VoxelLimits &pVoxelLimit, const G4AffineTransform &pTransform, G4double &pMin, G4double &pMax) const override
 
void BoundingLimits (G4ThreeVector &pMin, G4ThreeVector &pMax) const override
 
G4double GetMinXExtent () const
 
G4double GetMaxXExtent () const
 
G4double GetMinYExtent () const
 
G4double GetMaxYExtent () const
 
G4double GetMinZExtent () const
 
G4double GetMaxZExtent () const
 
G4PolyhedronCreatePolyhedron () const override
 
G4PolyhedronGetPolyhedron () const override
 
void DescribeYourselfTo (G4VGraphicsScene &scene) const override
 
G4VisExtent GetExtent () const override
 
G4int AllocatedMemoryWithoutVoxels ()
 
G4int AllocatedMemory ()
 
void DisplayAllocatedMemory ()
 
- Public Member Functions inherited from G4VSolid
 G4VSolid (const G4String &name)
 
virtual ~G4VSolid ()
 
G4bool operator== (const G4VSolid &s) const
 
G4String GetName () const
 
void SetName (const G4String &name)
 
G4double GetTolerance () const
 
virtual void ComputeDimensions (G4VPVParameterisation *p, const G4int n, const G4VPhysicalVolume *pRep)
 
virtual G4int GetNumOfConstituents () const
 
void DumpInfo () const
 
virtual const G4VSolidGetConstituentSolid (G4int no) const
 
virtual G4VSolidGetConstituentSolid (G4int no)
 
virtual const G4DisplacedSolidGetDisplacedSolidPtr () const
 
virtual G4DisplacedSolidGetDisplacedSolidPtr ()
 
 G4VSolid (__void__ &)
 
 G4VSolid (const G4VSolid &rhs)
 
G4VSolidoperator= (const G4VSolid &rhs)
 
G4double EstimateCubicVolume (G4int nStat, G4double epsilon) const
 
G4double EstimateSurfaceArea (G4int nStat, G4double ell) const
 

Additional Inherited Members

- Protected Member Functions inherited from G4VSolid
void CalculateClippedPolygonExtent (G4ThreeVectorList &pPolygon, const G4VoxelLimits &pVoxelLimit, const EAxis pAxis, G4double &pMin, G4double &pMax) const
 
void ClipCrossSection (G4ThreeVectorList *pVertices, const G4int pSectionIndex, const G4VoxelLimits &pVoxelLimit, const EAxis pAxis, G4double &pMin, G4double &pMax) const
 
void ClipBetweenSections (G4ThreeVectorList *pVertices, const G4int pSectionIndex, const G4VoxelLimits &pVoxelLimit, const EAxis pAxis, G4double &pMin, G4double &pMax) const
 
void ClipPolygon (G4ThreeVectorList &pPolygon, const G4VoxelLimits &pVoxelLimit, const EAxis pAxis) const
 
- Protected Attributes inherited from G4TessellatedSolid
G4double kCarToleranceHalf
 
- Protected Attributes inherited from G4VSolid
G4double kCarTolerance
 

Detailed Description

Definition at line 71 of file G4ExtrudedSolid.hh.

Constructor & Destructor Documentation

◆ G4ExtrudedSolid() [1/4]

G4ExtrudedSolid::G4ExtrudedSolid ( const G4String & pName,
const std::vector< G4TwoVector > & polygon,
const std::vector< ZSection > & zsections )

Definition at line 66 of file G4ExtrudedSolid.cc.

69 : G4TessellatedSolid(pName),
70 fNv(polygon.size()),
71 fNz(zsections.size()),
72 fIsConvex(false),
73 fGeometryType("G4ExtrudedSolid"),
74 fSolidType(0)
75{
76 // General constructor
77
78 // First check input parameters
79
80 if (fNv < 3)
81 {
82 std::ostringstream message;
83 message << "Number of vertices in polygon < 3 - " << pName;
84 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0002",
85 FatalErrorInArgument, message);
86 }
87
88 if (fNz < 2)
89 {
90 std::ostringstream message;
91 message << "Number of z-sides < 2 - " << pName;
92 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0002",
93 FatalErrorInArgument, message);
94 }
95
96 for ( std::size_t i=0; i<fNz-1; ++i )
97 {
98 if ( zsections[i].fZ > zsections[i+1].fZ )
99 {
100 std::ostringstream message;
101 message << "Z-sections have to be ordered by z value (z0 < z1 < z2...) - "
102 << pName;
103 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0002",
104 FatalErrorInArgument, message);
105 }
106 if ( std::fabs( zsections[i+1].fZ - zsections[i].fZ ) < kCarToleranceHalf )
107 {
108 std::ostringstream message;
109 message << "Z-sections with the same z position are not supported - "
110 << pName;
111 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0001",
112 FatalException, message);
113 }
114 }
115
116 // Copy polygon
117 //
118 fPolygon = polygon;
119
120 // Remove collinear and coincident vertices, if any
121 //
122 std::vector<G4int> removedVertices;
123 G4GeomTools::RemoveRedundantVertices(fPolygon,removedVertices,
124 2*kCarTolerance);
125 if (!removedVertices.empty())
126 {
127 std::size_t nremoved = removedVertices.size();
128 std::ostringstream message;
129 message << "The following "<< nremoved
130 << " vertices have been removed from polygon in " << pName
131 << "\nas collinear or coincident with other vertices: "
132 << removedVertices[0];
133 for (std::size_t i=1; i<nremoved; ++i) message << ", " << removedVertices[i];
134 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids1001",
135 JustWarning, message);
136 }
137
138 fNv = fPolygon.size();
139 if (fNv < 3)
140 {
141 std::ostringstream message;
142 message << "Number of vertices in polygon after removal < 3 - " << pName;
143 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0002",
144 FatalErrorInArgument, message);
145 }
146
147 // Check if polygon vertices are defined clockwise
148 // (the area is positive if polygon vertices are defined anti-clockwise)
149 //
150 if (G4GeomTools::PolygonArea(fPolygon) > 0.)
151 {
152 // Polygon vertices are defined anti-clockwise, we revert them
153 // G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids1001",
154 // JustWarning,
155 // "Polygon vertices defined anti-clockwise, reverting polygon");
156 std::reverse(fPolygon.begin(),fPolygon.end());
157 }
158
159 // Copy z-sections
160 //
161 fZSections = zsections;
162
163 G4bool result = MakeFacets();
164 if (!result)
165 {
166 std::ostringstream message;
167 message << "Making facets failed - " << pName;
168 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0003",
169 FatalException, message);
170 }
171 fIsConvex = G4GeomTools::IsConvex(fPolygon);
172
173 ComputeProjectionParameters();
174
175 // Check if the solid is a right prism, if so then set lateral planes
176 //
177 if ((fNz == 2)
178 && (fZSections[0].fScale == 1) && (fZSections[1].fScale == 1)
179 && (fZSections[0].fOffset == G4TwoVector(0,0))
180 && (fZSections[1].fOffset == G4TwoVector(0,0)))
181 {
182 fSolidType = (fIsConvex) ? 1 : 2; // 1 - convex, 2 - non-convex right prism
183 ComputeLateralPlanes();
184 }
185}
@ JustWarning
@ FatalException
@ FatalErrorInArgument
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
CLHEP::Hep2Vector G4TwoVector
bool G4bool
Definition G4Types.hh:86
static void RemoveRedundantVertices(G4TwoVectorList &polygon, std::vector< G4int > &iout, G4double tolerance=0.0)
static G4double PolygonArea(const G4TwoVectorList &polygon)
static G4bool IsConvex(const G4TwoVectorList &polygon)
G4double kCarTolerance
Definition G4VSolid.hh:306

Referenced by Clone(), G4ExtrudedSolid(), and operator=().

◆ G4ExtrudedSolid() [2/4]

G4ExtrudedSolid::G4ExtrudedSolid ( const G4String & pName,
const std::vector< G4TwoVector > & polygon,
G4double halfZ,
const G4TwoVector & off1 = G4TwoVector(0.,0.),
G4double scale1 = 1.,
const G4TwoVector & off2 = G4TwoVector(0.,0.),
G4double scale2 = 1. )

Definition at line 189 of file G4ExtrudedSolid.cc.

194 : G4TessellatedSolid(pName),
195 fNv(polygon.size()),
196 fNz(2),
197 fGeometryType("G4ExtrudedSolid")
198{
199 // Special constructor for solid with 2 z-sections
200
201 // First check input parameters
202 //
203 if (fNv < 3)
204 {
205 std::ostringstream message;
206 message << "Number of vertices in polygon < 3 - " << pName;
207 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0002",
208 FatalErrorInArgument, message);
209 }
210
211 // Copy polygon
212 //
213 fPolygon = polygon;
214
215 // Remove collinear and coincident vertices, if any
216 //
217 std::vector<G4int> removedVertices;
218 G4GeomTools::RemoveRedundantVertices(fPolygon,removedVertices,
219 2*kCarTolerance);
220 if (!removedVertices.empty())
221 {
222 std::size_t nremoved = removedVertices.size();
223 std::ostringstream message;
224 message << "The following "<< nremoved
225 << " vertices have been removed from polygon in " << pName
226 << "\nas collinear or coincident with other vertices: "
227 << removedVertices[0];
228 for (std::size_t i=1; i<nremoved; ++i) message << ", " << removedVertices[i];
229 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids1001",
230 JustWarning, message);
231 }
232
233 fNv = fPolygon.size();
234 if (fNv < 3)
235 {
236 std::ostringstream message;
237 message << "Number of vertices in polygon after removal < 3 - " << pName;
238 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0002",
239 FatalErrorInArgument, message);
240 }
241
242 // Check if polygon vertices are defined clockwise
243 // (the area is positive if polygon vertices are defined anti-clockwise)
244 //
245 if (G4GeomTools::PolygonArea(fPolygon) > 0.)
246 {
247 // Polygon vertices are defined anti-clockwise, we revert them
248 // G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids1001",
249 // JustWarning,
250 // "Polygon vertices defined anti-clockwise, reverting polygon");
251 std::reverse(fPolygon.begin(),fPolygon.end());
252 }
253
254 // Copy z-sections
255 //
256 fZSections.emplace_back(-dz, off1, scale1);
257 fZSections.emplace_back( dz, off2, scale2);
258
259 G4bool result = MakeFacets();
260 if (!result)
261 {
262 std::ostringstream message;
263 message << "Making facets failed - " << pName;
264 G4Exception("G4ExtrudedSolid::G4ExtrudedSolid()", "GeomSolids0003",
265 FatalException, message);
266 }
267 fIsConvex = G4GeomTools::IsConvex(fPolygon);
268
269 ComputeProjectionParameters();
270
271 // Check if the solid is a right prism, if so then set lateral planes
272 //
273 if ((scale1 == 1) && (scale2 == 1)
274 && (off1 == G4TwoVector(0,0)) && (off2 == G4TwoVector(0,0)))
275 {
276 fSolidType = (fIsConvex) ? 1 : 2; // 1 - convex, 2 - non-convex right prism
277 ComputeLateralPlanes();
278 }
279}

◆ ~G4ExtrudedSolid()

G4ExtrudedSolid::~G4ExtrudedSolid ( )
override

Definition at line 322 of file G4ExtrudedSolid.cc.

323{
324 // Destructor
325}

◆ G4ExtrudedSolid() [3/4]

G4ExtrudedSolid::G4ExtrudedSolid ( __void__ & a)

Definition at line 283 of file G4ExtrudedSolid.cc.

284 : G4TessellatedSolid(a), fGeometryType("G4ExtrudedSolid")
285{
286 // Fake default constructor - sets only member data and allocates memory
287 // for usage restricted to object persistency.
288}

◆ G4ExtrudedSolid() [4/4]

G4ExtrudedSolid::G4ExtrudedSolid ( const G4ExtrudedSolid & rhs)
default

Member Function Documentation

◆ BoundingLimits()

void G4ExtrudedSolid::BoundingLimits ( G4ThreeVector & pMin,
G4ThreeVector & pMax ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 1368 of file G4ExtrudedSolid.cc.

1370{
1371 G4double xmin0 = kInfinity, xmax0 = -kInfinity;
1372 G4double ymin0 = kInfinity, ymax0 = -kInfinity;
1373
1374 for (G4int i=0; i<GetNofVertices(); ++i)
1375 {
1376 G4double x = fPolygon[i].x();
1377 if (x < xmin0) xmin0 = x;
1378 if (x > xmax0) xmax0 = x;
1379 G4double y = fPolygon[i].y();
1380 if (y < ymin0) ymin0 = y;
1381 if (y > ymax0) ymax0 = y;
1382 }
1383
1384 G4double xmin = kInfinity, xmax = -kInfinity;
1385 G4double ymin = kInfinity, ymax = -kInfinity;
1386
1387 G4int nsect = GetNofZSections();
1388 for (G4int i=0; i<nsect; ++i)
1389 {
1390 ZSection zsect = GetZSection(i);
1391 G4double dx = zsect.fOffset.x();
1392 G4double dy = zsect.fOffset.y();
1393 G4double scale = zsect.fScale;
1394 xmin = std::min(xmin,xmin0*scale+dx);
1395 xmax = std::max(xmax,xmax0*scale+dx);
1396 ymin = std::min(ymin,ymin0*scale+dy);
1397 ymax = std::max(ymax,ymax0*scale+dy);
1398 }
1399
1400 G4double zmin = GetZSection(0).fZ;
1401 G4double zmax = GetZSection(nsect-1).fZ;
1402
1403 pMin.set(xmin,ymin,zmin);
1404 pMax.set(xmax,ymax,zmax);
1405
1406 // Check correctness of the bounding box
1407 //
1408 if (pMin.x() >= pMax.x() || pMin.y() >= pMax.y() || pMin.z() >= pMax.z())
1409 {
1410 std::ostringstream message;
1411 message << "Bad bounding box (min >= max) for solid: "
1412 << GetName() << " !"
1413 << "\npMin = " << pMin
1414 << "\npMax = " << pMax;
1415 G4Exception("G4ExtrudedSolid::BoundingLimits()",
1416 "GeomMgt0001", JustWarning, message);
1417 DumpInfo();
1418 }
1419}
double G4double
Definition G4Types.hh:83
int G4int
Definition G4Types.hh:85
double z() const
double x() const
double y() const
void set(double x, double y, double z)
ZSection GetZSection(G4int index) const
G4int GetNofZSections() const
G4int GetNofVertices() const
G4String GetName() const
void DumpInfo() const

Referenced by CalculateExtent().

◆ CalculateExtent()

G4bool G4ExtrudedSolid::CalculateExtent ( const EAxis pAxis,
const G4VoxelLimits & pVoxelLimit,
const G4AffineTransform & pTransform,
G4double & pMin,
G4double & pMax ) const
overridevirtual

Implements G4VSolid.

Definition at line 1425 of file G4ExtrudedSolid.cc.

1429{
1430 G4ThreeVector bmin, bmax;
1431 G4bool exist;
1432
1433 // Check bounding box (bbox)
1434 //
1435 BoundingLimits(bmin,bmax);
1436 G4BoundingEnvelope bbox(bmin,bmax);
1437#ifdef G4BBOX_EXTENT
1438 return bbox.CalculateExtent(pAxis,pVoxelLimit,pTransform,pMin,pMax);
1439#endif
1440 if (bbox.BoundingBoxVsVoxelLimits(pAxis,pVoxelLimit,pTransform,pMin,pMax))
1441 {
1442 return exist = pMin < pMax;
1443 }
1444
1445 // To find the extent, the base polygon is subdivided in triangles.
1446 // The extent is calculated as cumulative extent of the parts
1447 // formed by extrusion of the triangles
1448 //
1449 G4TwoVectorList triangles;
1450 G4double eminlim = pVoxelLimit.GetMinExtent(pAxis);
1451 G4double emaxlim = pVoxelLimit.GetMaxExtent(pAxis);
1452
1453 // triangulate the base polygon
1454 if (!G4GeomTools::TriangulatePolygon(fPolygon,triangles))
1455 {
1456 std::ostringstream message;
1457 message << "Triangulation of the base polygon has failed for solid: "
1458 << GetName() << " !"
1459 << "\nExtent has been calculated using boundary box";
1460 G4Exception("G4ExtrudedSolid::CalculateExtent()",
1461 "GeomMgt1002",JustWarning,message);
1462 return bbox.CalculateExtent(pAxis,pVoxelLimit,pTransform,pMin,pMax);
1463 }
1464
1465 // allocate vector lists
1466 G4int nsect = GetNofZSections();
1467 std::vector<const G4ThreeVectorList *> polygons;
1468 polygons.resize(nsect);
1469 for (G4int k=0; k<nsect; ++k) { polygons[k] = new G4ThreeVectorList(3); }
1470
1471 // main loop along triangles
1472 pMin = kInfinity;
1473 pMax = -kInfinity;
1474 G4int ntria = (G4int)triangles.size()/3;
1475 for (G4int i=0; i<ntria; ++i)
1476 {
1477 G4int i3 = i*3;
1478 for (G4int k=0; k<nsect; ++k) // extrude triangle
1479 {
1480 ZSection zsect = GetZSection(k);
1481 G4double z = zsect.fZ;
1482 G4double dx = zsect.fOffset.x();
1483 G4double dy = zsect.fOffset.y();
1484 G4double scale = zsect.fScale;
1485
1486 auto ptr = const_cast<G4ThreeVectorList*>(polygons[k]);
1487 auto iter = ptr->begin();
1488 G4double x0 = triangles[i3+0].x()*scale+dx;
1489 G4double y0 = triangles[i3+0].y()*scale+dy;
1490 iter->set(x0,y0,z);
1491 iter++;
1492 G4double x1 = triangles[i3+1].x()*scale+dx;
1493 G4double y1 = triangles[i3+1].y()*scale+dy;
1494 iter->set(x1,y1,z);
1495 iter++;
1496 G4double x2 = triangles[i3+2].x()*scale+dx;
1497 G4double y2 = triangles[i3+2].y()*scale+dy;
1498 iter->set(x2,y2,z);
1499 }
1500
1501 // set sub-envelope and adjust extent
1502 G4double emin,emax;
1503 G4BoundingEnvelope benv(polygons);
1504 if (!benv.CalculateExtent(pAxis,pVoxelLimit,pTransform,emin,emax)) continue;
1505 if (emin < pMin) pMin = emin;
1506 if (emax > pMax) pMax = emax;
1507 if (eminlim > pMin && emaxlim < pMax) break; // max possible extent
1508 }
1509 // free memory
1510 for (G4int k=0; k<nsect; ++k) { delete polygons[k]; polygons[k]=nullptr;}
1511 return (pMin < pMax);
1512}
std::vector< G4ThreeVector > G4ThreeVectorList
std::vector< G4TwoVector > G4TwoVectorList
CLHEP::Hep3Vector G4ThreeVector
void BoundingLimits(G4ThreeVector &pMin, G4ThreeVector &pMax) const override
static G4bool TriangulatePolygon(const G4TwoVectorList &polygon, G4TwoVectorList &result)
G4double GetMinExtent(const EAxis pAxis) const
G4double GetMaxExtent(const EAxis pAxis) const

◆ Clone()

G4VSolid * G4ExtrudedSolid::Clone ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 847 of file G4ExtrudedSolid.cc.

848{
849 return new G4ExtrudedSolid(*this);
850}
G4ExtrudedSolid(const G4String &pName, const std::vector< G4TwoVector > &polygon, const std::vector< ZSection > &zsections)

◆ DistanceToIn() [1/2]

G4double G4ExtrudedSolid::DistanceToIn ( const G4ThreeVector & p) const
overridevirtual

Implements G4VSolid.

Definition at line 1216 of file G4ExtrudedSolid.cc.

1217{
1218 switch (fSolidType)
1219 {
1220 case 1: // convex right prism
1221 {
1222 G4double dist = std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
1223 std::size_t np = fPlanes.size();
1224 for (std::size_t i=0; i<np; ++i)
1225 {
1226 G4double dd = fPlanes[i].a*p.x() + fPlanes[i].b*p.y() + fPlanes[i].d;
1227 if (dd > dist) dist = dd;
1228 }
1229 return (dist > 0) ? dist : 0.;
1230 }
1231 case 2: // non-convex right prism
1232 {
1233 G4bool in = PointInPolygon(p);
1234 if (in)
1235 {
1236 G4double distz= std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
1237 return (distz > 0) ? distz : 0;
1238 }
1239 else
1240 {
1241 G4double distz= std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
1242 G4double dd = DistanceToPolygonSqr(p);
1243 if (distz > 0) dd += distz*distz;
1244 return std::sqrt(dd);
1245 }
1246 }
1247 }
1248
1249 // General case: use tessellated solid
1251}
G4double DistanceToIn(const G4ThreeVector &p, const G4ThreeVector &v) const override

◆ DistanceToIn() [2/2]

G4double G4ExtrudedSolid::DistanceToIn ( const G4ThreeVector & p,
const G4ThreeVector & v ) const
overridevirtual

Implements G4VSolid.

Definition at line 1155 of file G4ExtrudedSolid.cc.

1157{
1158 G4double z0 = fZSections[0].fZ;
1159 G4double z1 = fZSections[fNz-1].fZ;
1160 if ((p.z() <= z0 + kCarToleranceHalf) && v.z() <= 0) return kInfinity;
1161 if ((p.z() >= z1 - kCarToleranceHalf) && v.z() >= 0) return kInfinity;
1162
1163 switch (fSolidType)
1164 {
1165 case 1: // convex right prism
1166 {
1167 // Intersection with Z planes
1168 //
1169 G4double dz = (z1 - z0)*0.5;
1170 G4double pz = p.z() - dz - z0;
1171
1172 G4double invz = (v.z() == 0) ? DBL_MAX : -1./v.z();
1173 G4double ddz = (invz < 0) ? dz : -dz;
1174 G4double tzmin = (pz + ddz)*invz;
1175 G4double tzmax = (pz - ddz)*invz;
1176
1177 // Intersection with lateral planes
1178 //
1179 std::size_t np = fPlanes.size();
1180 G4double txmin = tzmin, txmax = tzmax;
1181 for (std::size_t i=0; i<np; ++i)
1182 {
1183 G4double cosa = fPlanes[i].a*v.x()+fPlanes[i].b*v.y();
1184 G4double dist = fPlanes[i].a*p.x()+fPlanes[i].b*p.y()+fPlanes[i].d;
1185 if (dist >= -kCarToleranceHalf)
1186 {
1187 if (cosa >= 0) { return kInfinity; }
1188 G4double tmp = -dist/cosa;
1189 if (txmin < tmp) { txmin = tmp; }
1190 }
1191 else if (cosa > 0)
1192 {
1193 G4double tmp = -dist/cosa;
1194 if (txmax > tmp) { txmax = tmp; }
1195 }
1196 }
1197
1198 // Find distance
1199 //
1200 G4double tmin = txmin, tmax = txmax;
1201 if (tmax <= tmin + kCarToleranceHalf) // touch or no hit
1202 {
1203 return kInfinity;
1204 }
1205 return (tmin < kCarToleranceHalf) ? 0. : tmin;
1206 }
1207 case 2: // non-convex right prism
1208 {
1209 }
1210 }
1212}
#define DBL_MAX
Definition templates.hh:62

◆ DistanceToOut() [1/2]

G4double G4ExtrudedSolid::DistanceToOut ( const G4ThreeVector & p) const
overridevirtual

Implements G4VSolid.

Definition at line 1337 of file G4ExtrudedSolid.cc.

1338{
1339 switch (fSolidType)
1340 {
1341 case 1: // convex right prism
1342 {
1343 G4double dist = std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
1344 std::size_t np = fPlanes.size();
1345 for (std::size_t i=0; i<np; ++i)
1346 {
1347 G4double dd = fPlanes[i].a*p.x() + fPlanes[i].b*p.y() + fPlanes[i].d;
1348 if (dd > dist) dist = dd;
1349 }
1350 return (dist < 0) ? -dist : 0.;
1351 }
1352 case 2: // non-convex right prism
1353 {
1354 G4double distz = std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
1355 G4bool in = PointInPolygon(p);
1356 if (distz >= 0 || (!in)) return 0; // point is outside
1357 return std::min(-distz,std::sqrt(DistanceToPolygonSqr(p)));
1358 }
1359 }
1360
1361 // General case: use tessellated solid
1363}
G4double DistanceToOut(const G4ThreeVector &p) const override

◆ DistanceToOut() [2/2]

G4double G4ExtrudedSolid::DistanceToOut ( const G4ThreeVector & p,
const G4ThreeVector & v,
const G4bool calcNorm = false,
G4bool * validNorm = nullptr,
G4ThreeVector * n = nullptr ) const
overridevirtual

Implements G4VSolid.

Definition at line 1255 of file G4ExtrudedSolid.cc.

1260{
1261 G4bool getnorm = calcNorm;
1262 if (getnorm) *validNorm = true;
1263
1264 G4double z0 = fZSections[0].fZ;
1265 G4double z1 = fZSections[fNz-1].fZ;
1266 if ((p.z() <= z0 + kCarToleranceHalf) && v.z() < 0)
1267 {
1268 if (getnorm) n->set(0,0,-1);
1269 return 0;
1270 }
1271 if ((p.z() >= z1 - kCarToleranceHalf) && v.z() > 0)
1272 {
1273 if (getnorm) n->set(0,0,1);
1274 return 0;
1275 }
1276
1277 switch (fSolidType)
1278 {
1279 case 1: // convex right prism
1280 {
1281 // Intersection with Z planes
1282 //
1283 G4double dz = (z1 - z0)*0.5;
1284 G4double pz = p.z() - 0.5 * (z0 + z1);
1285
1286 G4double vz = v.z();
1287 G4double tmax = (vz == 0) ? DBL_MAX : (std::copysign(dz,vz) - pz)/vz;
1288 G4int iside = (vz < 0) ? -4 : -2; // little trick: (-4+3)=-1, (-2+3)=+1
1289
1290 // Intersection with lateral planes
1291 //
1292 std::size_t np = fPlanes.size();
1293 for (std::size_t i=0; i<np; ++i)
1294 {
1295 G4double cosa = fPlanes[i].a*v.x()+fPlanes[i].b*v.y();
1296 if (cosa > 0)
1297 {
1298 G4double dist = fPlanes[i].a*p.x()+fPlanes[i].b*p.y()+fPlanes[i].d;
1299 if (dist >= -kCarToleranceHalf)
1300 {
1301 if (getnorm) n->set(fPlanes[i].a, fPlanes[i].b, fPlanes[i].c);
1302 return 0;
1303 }
1304 G4double tmp = -dist/cosa;
1305 if (tmax > tmp) { tmax = tmp; iside = (G4int)i; }
1306 }
1307 }
1308
1309 // Set normal, if required, and return distance
1310 //
1311 if (getnorm)
1312 {
1313 if (iside < 0)
1314 { n->set(0, 0, iside + 3); } // (-4+3)=-1, (-2+3)=+1
1315 else
1316 { n->set(fPlanes[iside].a, fPlanes[iside].b, fPlanes[iside].c); }
1317 }
1318 return tmax;
1319 }
1320 case 2: // non-convex right prism
1321 {
1322 }
1323 }
1324
1325 // Override the base class function to redefine validNorm
1326 // (the solid can be concave)
1327
1328 G4double distOut =
1329 G4TessellatedSolid::DistanceToOut(p, v, calcNorm, validNorm, n);
1330 if (validNorm != nullptr) { *validNorm = fIsConvex; }
1331
1332 return distOut;
1333}

◆ GetEntityType()

G4GeometryType G4ExtrudedSolid::GetEntityType ( ) const
overridevirtual

Implements G4VSolid.

Definition at line 831 of file G4ExtrudedSolid.cc.

832{
833 // Return entity type
834
835 return fGeometryType;
836}

◆ GetNofVertices()

G4int G4ExtrudedSolid::GetNofVertices ( ) const
inline

◆ GetNofZSections()

G4int G4ExtrudedSolid::GetNofZSections ( ) const
inline

◆ GetPolygon()

std::vector< G4TwoVector > G4ExtrudedSolid::GetPolygon ( ) const
inline

◆ GetVertex()

G4TwoVector G4ExtrudedSolid::GetVertex ( G4int index) const
inline

◆ GetZSection()

ZSection G4ExtrudedSolid::GetZSection ( G4int index) const
inline

◆ GetZSections()

std::vector< ZSection > G4ExtrudedSolid::GetZSections ( ) const
inline

◆ Inside()

EInside G4ExtrudedSolid::Inside ( const G4ThreeVector & p) const
overridevirtual

Implements G4VSolid.

Definition at line 854 of file G4ExtrudedSolid.cc.

855{
856 switch (fSolidType)
857 {
858 case 1: // convex right prism
859 {
860 G4double dist = std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
861 if (dist > kCarToleranceHalf) { return kOutside; }
862
863 std::size_t np = fPlanes.size();
864 for (std::size_t i=0; i<np; ++i)
865 {
866 G4double dd = fPlanes[i].a*p.x() + fPlanes[i].b*p.y() + fPlanes[i].d;
867 if (dd > dist) { dist = dd; }
868 }
869 if (dist > kCarToleranceHalf) { return kOutside; }
870 return (dist > -kCarToleranceHalf) ? kSurface : kInside;
871 }
872 case 2: // non-convex right prism
873 {
874 G4double distz = std::max(fZSections[0].fZ-p.z(),p.z()-fZSections[1].fZ);
875 if (distz > kCarToleranceHalf) { return kOutside; }
876
877 G4bool in = PointInPolygon(p);
878 if (distz > -kCarToleranceHalf && in) { return kSurface; }
879
880 G4double dd = DistanceToPolygonSqr(p) - kCarToleranceHalf*kCarToleranceHalf;
881 if (in)
882 {
883 return (dd >= 0) ? kInside : kSurface;
884 }
885 else
886 {
887 return (dd > 0) ? kOutside : kSurface;
888 }
889 }
890 }
891
892 // Override the base class function as it fails in case of concave polygon.
893 // Project the point in the original polygon scale and check if it is inside
894 // for each triangle.
895
896 // Check first if outside extent
897 //
898 if ( p.x() < GetMinXExtent() - kCarToleranceHalf ||
904 {
905 // G4cout << "G4ExtrudedSolid::Outside extent: " << p << G4endl;
906 return kOutside;
907 }
908
909 // Project point p(z) to the polygon scale p0
910 //
911 G4TwoVector pscaled = ProjectPoint(p);
912
913 // Check if on surface of polygon
914 //
915 for ( G4int i=0; i<(G4int)fNv; ++i )
916 {
917 G4int j = (i+1) % fNv;
918 if ( IsSameLineSegment(pscaled, fPolygon[i], fPolygon[j]) )
919 {
920 // G4cout << "G4ExtrudedSolid::Inside return Surface (on polygon) "
921 // << G4endl;
922
923 return kSurface;
924 }
925 }
926
927 // Now check if inside triangles
928 //
929 auto it = fTriangles.cbegin();
930 G4bool inside = false;
931 do // Loop checking, 13.08.2015, G.Cosmo
932 {
933 if ( IsPointInside(fPolygon[(*it)[0]], fPolygon[(*it)[1]],
934 fPolygon[(*it)[2]], pscaled) ) { inside = true; }
935 ++it;
936 } while ( (!inside) && (it != fTriangles.cend()) );
937
938 if ( inside )
939 {
940 // Check if on surface of z sides
941 //
942 if ( std::fabs( p.z() - fZSections[0].fZ ) < kCarToleranceHalf ||
943 std::fabs( p.z() - fZSections[fNz-1].fZ ) < kCarToleranceHalf )
944 {
945 // G4cout << "G4ExtrudedSolid::Inside return Surface (on z side)"
946 // << G4endl;
947
948 return kSurface;
949 }
950
951 // G4cout << "G4ExtrudedSolid::Inside return Inside" << G4endl;
952
953 return kInside;
954 }
955
956 // G4cout << "G4ExtrudedSolid::Inside return Outside " << G4endl;
957
958 return kOutside;
959}
G4double GetMinYExtent() const
G4double GetMinZExtent() const
G4double GetMaxYExtent() const
G4double GetMaxZExtent() const
G4double GetMaxXExtent() const
G4double GetMinXExtent() const
@ kInside
Definition geomdefs.hh:70
@ kOutside
Definition geomdefs.hh:68
@ kSurface
Definition geomdefs.hh:69

◆ IsFaceted()

G4bool G4ExtrudedSolid::IsFaceted ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 840 of file G4ExtrudedSolid.cc.

841{
842 return true;
843}

◆ operator=()

G4ExtrudedSolid & G4ExtrudedSolid::operator= ( const G4ExtrudedSolid & rhs)

Definition at line 296 of file G4ExtrudedSolid.cc.

297{
298 // Check assignment to self
299 //
300 if (this == &rhs) { return *this; }
301
302 // Copy base class data
303 //
305
306 // Copy data
307 //
308 fNv = rhs.fNv; fNz = rhs.fNz;
309 fPolygon = rhs.fPolygon; fZSections = rhs.fZSections;
310 fTriangles = rhs.fTriangles; fIsConvex = rhs.fIsConvex;
311 fGeometryType = rhs.fGeometryType;
312 fSolidType = rhs.fSolidType; fPlanes = rhs.fPlanes;
313 fLines = rhs.fLines; fLengths = rhs.fLengths;
314 fKScales = rhs.fKScales; fScale0s = rhs.fScale0s;
315 fKOffsets = rhs.fKOffsets; fOffset0s = rhs.fOffset0s;
316
317 return *this;
318}
G4TessellatedSolid & operator=(const G4TessellatedSolid &right)

◆ StreamInfo()

std::ostream & G4ExtrudedSolid::StreamInfo ( std::ostream & os) const
overridevirtual

Implements G4VSolid.

Definition at line 1516 of file G4ExtrudedSolid.cc.

1517{
1518 G4long oldprc = os.precision(16);
1519 os << "-----------------------------------------------------------\n"
1520 << " *** Dump for solid - " << GetName() << " ***\n"
1521 << " ===================================================\n"
1522 << " Solid geometry type: " << fGeometryType << G4endl;
1523
1524 if ( fIsConvex)
1525 { os << " Convex polygon; list of vertices:" << G4endl; }
1526 else
1527 { os << " Concave polygon; list of vertices:" << G4endl; }
1528
1529 for ( std::size_t i=0; i<fNv; ++i )
1530 {
1531 os << std::setw(5) << "#" << i
1532 << " vx = " << fPolygon[i].x()/mm << " mm"
1533 << " vy = " << fPolygon[i].y()/mm << " mm" << G4endl;
1534 }
1535
1536 os << " Sections:" << G4endl;
1537 for ( std::size_t iz=0; iz<fNz; ++iz )
1538 {
1539 os << " z = " << fZSections[iz].fZ/mm << " mm "
1540 << " x0= " << fZSections[iz].fOffset.x()/mm << " mm "
1541 << " y0= " << fZSections[iz].fOffset.y()/mm << " mm "
1542 << " scale= " << fZSections[iz].fScale << G4endl;
1543 }
1544
1545/*
1546 // Triangles (for debugging)
1547 os << G4endl;
1548 os << " Triangles:" << G4endl;
1549 os << " Triangle # vertex1 vertex2 vertex3" << G4endl;
1550
1551 G4int counter = 0;
1552 std::vector< std::vector<G4int> >::const_iterator it;
1553 for ( it = fTriangles.begin(); it != fTriangles.end(); it++ ) {
1554 std::vector<G4int> triangle = *it;
1555 os << std::setw(10) << counter++
1556 << std::setw(10) << triangle[0] << std::setw(10) << triangle[1]
1557 << std::setw(10) << triangle[2]
1558 << G4endl;
1559 }
1560*/
1561 os.precision(oldprc);
1562
1563 return os;
1564}
long G4long
Definition G4Types.hh:87
#define G4endl
Definition G4ios.hh:67

◆ SurfaceNormal()

G4ThreeVector G4ExtrudedSolid::SurfaceNormal ( const G4ThreeVector & p) const
overridevirtual

Implements G4VSolid.

Definition at line 963 of file G4ExtrudedSolid.cc.

964{
965 G4int nsurf = 0;
966 G4double nx = 0., ny = 0., nz = 0.;
967 switch (fSolidType)
968 {
969 case 1: // convex right prism
970 {
971 if (std::abs(p.z() - fZSections[0].fZ) <= kCarToleranceHalf)
972 {
973 nz = -1; ++nsurf;
974 }
975 if (std::abs(p.z() - fZSections[1].fZ) <= kCarToleranceHalf)
976 {
977 nz = 1; ++nsurf;
978 }
979 for (std::size_t i=0; i<fNv; ++i)
980 {
981 G4double dd = fPlanes[i].a*p.x() + fPlanes[i].b*p.y() + fPlanes[i].d;
982 if (std::abs(dd) > kCarToleranceHalf) continue;
983 nx += fPlanes[i].a;
984 ny += fPlanes[i].b;
985 ++nsurf;
986 }
987 break;
988 }
989 case 2: // non-convex right prism
990 {
991 if (std::abs(p.z() - fZSections[0].fZ) <= kCarToleranceHalf)
992 {
993 nz = -1; ++nsurf;
994 }
995 if (std::abs(p.z() - fZSections[1].fZ) <= kCarToleranceHalf)
996 {
997 nz = 1; ++nsurf;
998 }
999
1000 G4double sqrCarToleranceHalf = kCarToleranceHalf*kCarToleranceHalf;
1001 for (std::size_t i=0, k=fNv-1; i<fNv; k=i++)
1002 {
1003 G4double ix = p.x() - fPolygon[i].x();
1004 G4double iy = p.y() - fPolygon[i].y();
1005 G4double u = fPlanes[i].a*iy - fPlanes[i].b*ix;
1006 if (u < 0)
1007 {
1008 if (ix*ix + iy*iy > sqrCarToleranceHalf) continue;
1009 }
1010 else if (u > fLengths[i])
1011 {
1012 G4double kx = p.x() - fPolygon[k].x();
1013 G4double ky = p.y() - fPolygon[k].y();
1014 if (kx*kx + ky*ky > sqrCarToleranceHalf) continue;
1015 }
1016 else
1017 {
1018 G4double dd = fPlanes[i].a*p.x() + fPlanes[i].b*p.y() + fPlanes[i].d;
1019 if (dd*dd > sqrCarToleranceHalf) continue;
1020 }
1021 nx += fPlanes[i].a;
1022 ny += fPlanes[i].b;
1023 ++nsurf;
1024 }
1025 break;
1026 }
1027 default:
1028 {
1030 }
1031 }
1032
1033 // Return normal (right prism)
1034 //
1035 if (nsurf == 1)
1036 {
1037 return { nx,ny,nz };
1038 }
1039 else if (nsurf != 0) // edge or corner
1040 {
1041 return G4ThreeVector(nx,ny,nz).unit();
1042 }
1043 else
1044 {
1045 // Point is not on the surface, compute approximate normal
1046 //
1047#ifdef G4CSGDEBUG
1048 std::ostringstream message;
1049 G4long oldprc = message.precision(16);
1050 message << "Point p is not on surface (!?) of solid: "
1051 << GetName() << G4endl;
1052 message << "Position:\n";
1053 message << " p.x() = " << p.x()/mm << " mm\n";
1054 message << " p.y() = " << p.y()/mm << " mm\n";
1055 message << " p.z() = " << p.z()/mm << " mm";
1056 G4cout.precision(oldprc) ;
1057 G4Exception("G4TesselatedSolid::SurfaceNormal(p)", "GeomSolids1002",
1058 JustWarning, message );
1059 DumpInfo();
1060#endif
1061 return ApproxSurfaceNormal(p);
1062 }
1063}
G4GLOB_DLL std::ostream G4cout
Hep3Vector unit() const
G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const override

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