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

#include <G4MultiUnion.hh>

+ Inheritance diagram for G4MultiUnion:

Public Member Functions

 G4MultiUnion ()
 
 G4MultiUnion (const G4String &name)
 
 ~G4MultiUnion () override
 
void AddNode (G4VSolid &solid, const G4Transform3D &trans)
 
void AddNode (G4VSolid *solid, const G4Transform3D &trans)
 
 G4MultiUnion (const G4MultiUnion &rhs)
 
G4MultiUnionoperator= (const G4MultiUnion &rhs)
 
const G4Transform3DGetTransformation (G4int index) const
 
G4VSolidGetSolid (G4int index) const
 
G4int GetNumberOfSolids () const
 
EInside Inside (const G4ThreeVector &aPoint) const override
 
EInside InsideIterator (const G4ThreeVector &aPoint) const
 
G4double DistanceToIn (const G4ThreeVector &aPoint) const override
 
G4double DistanceToOut (const G4ThreeVector &aPoint) const override
 
void SetAccurateSafety (G4bool flag)
 
G4double DistanceToIn (const G4ThreeVector &aPoint, const G4ThreeVector &aDirection) const override
 
G4double DistanceToOut (const G4ThreeVector &aPoint, const G4ThreeVector &aDirection, const G4bool calcNorm=false, G4bool *validNorm=nullptr, G4ThreeVector *aNormalVector=nullptr) const override
 
G4double DistanceToInNoVoxels (const G4ThreeVector &aPoint, const G4ThreeVector &aDirection) const
 
G4double DistanceToOutVoxels (const G4ThreeVector &aPoint, const G4ThreeVector &aDirection, G4ThreeVector *aNormalVector) const
 
G4double DistanceToOutVoxelsCore (const G4ThreeVector &aPoint, const G4ThreeVector &aDirection, G4ThreeVector *aNormalVector, G4bool &aConvex, std::vector< G4int > &candidates) const
 
G4double DistanceToOutNoVoxels (const G4ThreeVector &aPoint, const G4ThreeVector &aDirection, G4ThreeVector *aNormalVector) const
 
G4ThreeVector SurfaceNormal (const G4ThreeVector &aPoint) const override
 
void Extent (EAxis aAxis, G4double &aMin, G4double &aMax) const
 
void BoundingLimits (G4ThreeVector &aMin, G4ThreeVector &aMax) const override
 
G4bool CalculateExtent (const EAxis pAxis, const G4VoxelLimits &pVoxelLimit, const G4AffineTransform &pTransform, G4double &pMin, G4double &pMax) const override
 
G4double GetCubicVolume () override
 
G4double GetSurfaceArea () override
 
G4int GetNumOfConstituents () const override
 
G4bool IsFaceted () const override
 
G4VSolidClone () const override
 
G4GeometryType GetEntityType () const override
 
void Voxelize ()
 
EInside InsideNoVoxels (const G4ThreeVector &aPoint) const
 
G4VoxelizerGetVoxels () const
 
std::ostream & StreamInfo (std::ostream &os) const override
 
G4ThreeVector GetPointOnSurface () const override
 
void DescribeYourselfTo (G4VGraphicsScene &scene) const override
 
G4PolyhedronCreatePolyhedron () const override
 
G4PolyhedronGetPolyhedron () const override
 
 G4MultiUnion (__void__ &)
 
- 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)
 
void DumpInfo () const
 
virtual G4VisExtent GetExtent () 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
 

Friends

class G4Voxelizer
 

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 G4VSolid
G4double kCarTolerance
 

Detailed Description

Definition at line 53 of file G4MultiUnion.hh.

Constructor & Destructor Documentation

◆ G4MultiUnion() [1/4]

G4MultiUnion::G4MultiUnion ( )
inline

Definition at line 59 of file G4MultiUnion.hh.

59: G4VSolid("") {}
G4VSolid(const G4String &name)
Definition G4VSolid.cc:57

Referenced by Clone(), G4MultiUnion(), operator=(), and ~G4MultiUnion().

◆ G4MultiUnion() [2/4]

G4MultiUnion::G4MultiUnion ( const G4String & name)

Definition at line 57 of file G4MultiUnion.cc.

58 : G4VSolid(name)
59{
60 SetName(name);
61 fSolids.clear();
62 fTransformObjs.clear();
64}
G4double GetRadialTolerance() const
static G4GeometryTolerance * GetInstance()
void SetName(const G4String &name)
Definition G4VSolid.cc:127

◆ ~G4MultiUnion()

G4MultiUnion::~G4MultiUnion ( )
overridedefault

◆ G4MultiUnion() [3/4]

G4MultiUnion::G4MultiUnion ( const G4MultiUnion & rhs)

Definition at line 92 of file G4MultiUnion.cc.

93 : G4VSolid(rhs), fCubicVolume(rhs.fCubicVolume),
94 fSurfaceArea(rhs.fSurfaceArea),
95 kRadTolerance(rhs.kRadTolerance), fAccurate(rhs.fAccurate)
96{
97}

◆ G4MultiUnion() [4/4]

G4MultiUnion::G4MultiUnion ( __void__ & a)

Definition at line 101 of file G4MultiUnion.cc.

102 : G4VSolid(a)
103{
104}

Member Function Documentation

◆ AddNode() [1/2]

void G4MultiUnion::AddNode ( G4VSolid & solid,
const G4Transform3D & trans )

Definition at line 71 of file G4MultiUnion.cc.

72{
73 fSolids.push_back(&solid);
74 fTransformObjs.push_back(trans); // Store a local copy of transformations
75}

Referenced by G4tgbVolume::FindOrConstructG4Solid(), and G4GDMLReadSolids::MultiUnionNodeRead().

◆ AddNode() [2/2]

void G4MultiUnion::AddNode ( G4VSolid * solid,
const G4Transform3D & trans )

Definition at line 78 of file G4MultiUnion.cc.

79{
80 fSolids.push_back(solid);
81 fTransformObjs.push_back(trans); // Store a local copy of transformations
82}

◆ BoundingLimits()

void G4MultiUnion::BoundingLimits ( G4ThreeVector & aMin,
G4ThreeVector & aMax ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 603 of file G4MultiUnion.cc.

605{
606 Extent(kXAxis, aMin[0], aMax[0]);
607 Extent(kYAxis, aMin[1], aMax[1]);
608 Extent(kZAxis, aMin[2], aMax[2]);
609}
void Extent(EAxis aAxis, G4double &aMin, G4double &aMax) const
@ kYAxis
Definition geomdefs.hh:56
@ kXAxis
Definition geomdefs.hh:55
@ kZAxis
Definition geomdefs.hh:57

Referenced by CalculateExtent().

◆ CalculateExtent()

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

Implements G4VSolid.

Definition at line 613 of file G4MultiUnion.cc.

617{
618 G4ThreeVector bmin, bmax;
619
620 // Get bounding box
621 BoundingLimits(bmin,bmax);
622
623 // Find extent
624 G4BoundingEnvelope bbox(bmin,bmax);
625 return bbox.CalculateExtent(pAxis,pVoxelLimit,pTransform,pMin,pMax);
626}
CLHEP::Hep3Vector G4ThreeVector
void BoundingLimits(G4ThreeVector &aMin, G4ThreeVector &aMax) const override

◆ Clone()

G4VSolid * G4MultiUnion::Clone ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 85 of file G4MultiUnion.cc.

86{
87 return new G4MultiUnion(*this);
88}

◆ CreatePolyhedron()

G4Polyhedron * G4MultiUnion::CreatePolyhedron ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 970 of file G4MultiUnion.cc.

971{
973 {
974 HepPolyhedronProcessor processor;
976
977 G4VSolid* solidA = GetSolid(0);
978 const G4Transform3D transform0 = GetTransformation(0);
979 G4DisplacedSolid dispSolidA("placedA", solidA, transform0);
980
981 auto top = new G4Polyhedron(*dispSolidA.GetPolyhedron());
982
983 for (G4int i = 1; i < GetNumberOfSolids(); ++i)
984 {
985 G4VSolid* solidB = GetSolid(i);
986 const G4Transform3D transform = GetTransformation(i);
987 G4DisplacedSolid dispSolidB("placedB", solidB, transform);
988 G4Polyhedron* operand = dispSolidB.GetPolyhedron();
989 processor.push_back(operation, *operand);
990 }
991
992 if (processor.execute(*top))
993 {
994 return top;
995 }
996 else
997 {
998 return nullptr;
999 }
1000 }
1001 else
1002 {
1004 }
1005}
HepGeom::Transform3D G4Transform3D
int G4int
Definition G4Types.hh:85
static G4VBooleanProcessor * GetExternalBooleanProcessor()
const G4Transform3D & GetTransformation(G4int index) const
G4int GetNumberOfSolids() const
G4VSolid * GetSolid(G4int index) const
virtual G4PolyhedronArbitrary * Process(const G4VSolid *)
bool execute(HepPolyhedron &)
void push_back(Operation, const HepPolyhedron &)

Referenced by GetPolyhedron().

◆ DescribeYourselfTo()

void G4MultiUnion::DescribeYourselfTo ( G4VGraphicsScene & scene) const
overridevirtual

Implements G4VSolid.

Definition at line 964 of file G4MultiUnion.cc.

965{
966 scene.AddSolid (*this);
967}
virtual void AddSolid(const G4Box &)=0

◆ DistanceToIn() [1/2]

G4double G4MultiUnion::DistanceToIn ( const G4ThreeVector & aPoint) const
overridevirtual

Implements G4VSolid.

Definition at line 747 of file G4MultiUnion.cc.

748{
749 // Estimates the isotropic safety from a point outside the current solid to
750 // any of its surfaces. The algorithm may be accurate or should provide a fast
751 // underestimate.
752
753 if (!fAccurate) { return fVoxels.DistanceToBoundingBox(point); }
754
755 const std::vector<G4VoxelBox>& boxes = fVoxels.GetBoxes();
756 G4double safetyMin = kInfinity;
757 G4ThreeVector localPoint;
758
759 std::size_t numNodes = fSolids.size();
760 for (std::size_t j = 0; j < numNodes; ++j)
761 {
762 G4ThreeVector dxyz;
763 if (j > 0)
764 {
765 const G4ThreeVector& pos = boxes[j].pos;
766 const G4ThreeVector& hlen = boxes[j].hlen;
767 for (auto i = 0; i <= 2; ++i)
768 // distance to middle point - hlength => distance from point to border
769 // of x,y,z
770 if ((dxyz[i] = std::abs(point[i] - pos[i]) - hlen[i]) > safetyMin)
771 continue;
772
773 G4double d2xyz = 0.;
774 for (auto i = 0; i <= 2; ++i)
775 if (dxyz[i] > 0) d2xyz += dxyz[i] * dxyz[i];
776
777 // minimal distance is at least this, but could be even higher. therefore,
778 // we can stop if previous was already lower, let us check if it does any
779 // chance to be better tha previous values...
780 if (d2xyz >= safetyMin * safetyMin)
781 {
782 continue;
783 }
784 }
785 const G4Transform3D& transform = fTransformObjs[j];
786 localPoint = GetLocalPoint(transform, point);
787 G4VSolid& solid = *fSolids[j];
788
789 G4double safety = solid.DistanceToIn(localPoint);
790 if (safety <= 0) return safety;
791 // it was detected, that the point is not located outside
792 if (safetyMin > safety) safetyMin = safety;
793 }
794 return safetyMin;
795}
double G4double
Definition G4Types.hh:83
virtual G4double DistanceToIn(const G4ThreeVector &p, const G4ThreeVector &v) const =0

◆ DistanceToIn() [2/2]

G4double G4MultiUnion::DistanceToIn ( const G4ThreeVector & aPoint,
const G4ThreeVector & aDirection ) const
overridevirtual

Implements G4VSolid.

Definition at line 195 of file G4MultiUnion.cc.

197{
198 G4double minDistance = kInfinity;
199 G4ThreeVector direction = aDirection.unit();
200 G4double shift = fVoxels.DistanceToFirst(aPoint, direction);
201 if (shift == kInfinity) return shift;
202
203 G4ThreeVector currentPoint = aPoint;
204 if (shift != 0.0) currentPoint += direction * shift;
205
206 G4SurfBits exclusion(fVoxels.GetBitsPerSlice());
207 std::vector<G4int> candidates, curVoxel(3);
208 fVoxels.GetVoxel(curVoxel, currentPoint);
209
210 do
211 {
212 {
213 if (fVoxels.GetCandidatesVoxelArray(curVoxel, candidates, &exclusion) != 0)
214 {
215 G4double distance = DistanceToInCandidates(aPoint, direction,
216 candidates, exclusion);
217 if (minDistance > distance) minDistance = distance;
218 if (distance < shift) break;
219 }
220 }
221 shift = fVoxels.DistanceToNext(aPoint, direction, curVoxel);
222 }
223 while (minDistance > shift);
224
225 return minDistance;
226}
Hep3Vector unit() const

◆ DistanceToInNoVoxels()

G4double G4MultiUnion::DistanceToInNoVoxels ( const G4ThreeVector & aPoint,
const G4ThreeVector & aDirection ) const

Definition at line 136 of file G4MultiUnion.cc.

138{
139 G4ThreeVector direction = aDirection.unit();
140 G4ThreeVector localPoint, localDirection;
141 G4double minDistance = kInfinity;
142
143 std::size_t numNodes = fSolids.size();
144 for (std::size_t i = 0 ; i < numNodes ; ++i)
145 {
146 G4VSolid& solid = *fSolids[i];
147 const G4Transform3D& transform = fTransformObjs[i];
148
149 localPoint = GetLocalPoint(transform, aPoint);
150 localDirection = GetLocalVector(transform, direction);
151
152 G4double distance = solid.DistanceToIn(localPoint, localDirection);
153 if (minDistance > distance) minDistance = distance;
154 }
155 return minDistance;
156}

◆ DistanceToOut() [1/2]

G4double G4MultiUnion::DistanceToOut ( const G4ThreeVector & aPoint) const
overridevirtual

Implements G4VSolid.

Definition at line 710 of file G4MultiUnion.cc.

711{
712 // Estimates isotropic distance to the surface of the solid. This must
713 // be either accurate or an underestimate.
714 // Two modes: - default/fast mode, sacrificing accuracy for speed
715 // - "precise" mode, requests accurate value if available.
716
717 std::vector<G4int> candidates;
718 G4ThreeVector localPoint;
719 G4double safetyMin = kInfinity;
720
721 // In general, the value return by DistanceToIn(p) will not be the exact
722 // but only an undervalue (cf. overlaps)
723 fVoxels.GetCandidatesVoxelArray(point, candidates);
724
725 std::size_t limit = candidates.size();
726 for (std::size_t i = 0; i < limit; ++i)
727 {
728 G4int candidate = candidates[i];
729
730 // The coordinates of the point are modified so as to fit the intrinsic
731 // solid local frame:
732 const G4Transform3D& transform = fTransformObjs[candidate];
733 localPoint = GetLocalPoint(transform, point);
734 G4VSolid& solid = *fSolids[candidate];
735 if (solid.Inside(localPoint) == EInside::kInside)
736 {
737 G4double safety = solid.DistanceToOut(localPoint);
738 if (safetyMin > safety) safetyMin = safety;
739 }
740 }
741 if (safetyMin == kInfinity) safetyMin = 0; // we are not inside
742
743 return safetyMin;
744}
virtual EInside Inside(const G4ThreeVector &p) const =0
virtual G4double DistanceToOut(const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm=false, G4bool *validNorm=nullptr, G4ThreeVector *n=nullptr) const =0
@ kInside
Definition geomdefs.hh:70

◆ DistanceToOut() [2/2]

G4double G4MultiUnion::DistanceToOut ( const G4ThreeVector & aPoint,
const G4ThreeVector & aDirection,
const G4bool calcNorm = false,
G4bool * validNorm = nullptr,
G4ThreeVector * aNormalVector = nullptr ) const
overridevirtual

Implements G4VSolid.

Definition at line 280 of file G4MultiUnion.cc.

285{
286 return DistanceToOutVoxels(aPoint, aDirection, aNormal);
287}
G4double DistanceToOutVoxels(const G4ThreeVector &aPoint, const G4ThreeVector &aDirection, G4ThreeVector *aNormalVector) const

◆ DistanceToOutNoVoxels()

G4double G4MultiUnion::DistanceToOutNoVoxels ( const G4ThreeVector & aPoint,
const G4ThreeVector & aDirection,
G4ThreeVector * aNormalVector ) const

Definition at line 229 of file G4MultiUnion.cc.

232{
233 // Computes distance from a point presumably outside the solid to the solid
234 // surface. Ignores first surface if the point is actually inside.
235 // Early return infinity in case the safety to any surface is found greater
236 // than the proposed step aPstep.
237 // The normal vector to the crossed surface is filled only in case the box
238 // is crossed, otherwise aNormal->IsNull() is true.
239
240 // algorithm:
241 G4ThreeVector direction = aDirection.unit();
242 G4ThreeVector localPoint, localDirection;
243 G4int ignoredSolid = -1;
244 G4double resultDistToOut = 0;
245 G4ThreeVector currentPoint = aPoint;
246
247 auto numNodes = (G4int)fSolids.size();
248 for (auto i = 0; i < numNodes; ++i)
249 {
250 if (i != ignoredSolid)
251 {
252 G4VSolid& solid = *fSolids[i];
253 const G4Transform3D& transform = fTransformObjs[i];
254 localPoint = GetLocalPoint(transform, currentPoint);
255 localDirection = GetLocalVector(transform, direction);
256 EInside location = solid.Inside(localPoint);
257 if (location != EInside::kOutside)
258 {
259 G4double distance = solid.DistanceToOut(localPoint, localDirection,
260 false, nullptr, aNormal);
261 if (distance < kInfinity)
262 {
263 if (resultDistToOut == kInfinity) resultDistToOut = 0;
264 if (distance > 0)
265 {
266 currentPoint = GetGlobalPoint(transform, localPoint
267 + distance*localDirection);
268 resultDistToOut += distance;
269 ignoredSolid = i; // skip the solid which we have just left
270 i = -1; // force the loop to continue from 0
271 }
272 }
273 }
274 }
275 }
276 return resultDistToOut;
277}
EInside
Definition geomdefs.hh:67
@ kOutside
Definition geomdefs.hh:68

◆ DistanceToOutVoxels()

G4double G4MultiUnion::DistanceToOutVoxels ( const G4ThreeVector & aPoint,
const G4ThreeVector & aDirection,
G4ThreeVector * aNormalVector ) const

Definition at line 290 of file G4MultiUnion.cc.

293{
294 // Computes distance from a point presumably inside the solid to the solid
295 // surface. Ignores first surface along each axis systematically (for points
296 // inside or outside. Early returns zero in case the second surface is behind
297 // the starting point.
298 // o The proposed step is ignored.
299 // o The normal vector to the crossed surface is always filled.
300
301 // In the case the considered point is located inside the G4MultiUnion
302 // structure, the treatments are as follows:
303 // - investigation of the candidates for the passed point
304 // - progressive moving of the point towards the surface, along the
305 // passed direction
306 // - processing of the normal
307
308 G4ThreeVector direction = aDirection.unit();
309 std::vector<G4int> candidates;
310 G4double distance = 0;
311 std::size_t numNodes = 2*fSolids.size();
312 std::size_t count=0;
313
314 if (fVoxels.GetCandidatesVoxelArray(aPoint, candidates) != 0)
315 {
316 // For normal case for which we presume the point is inside
317 G4ThreeVector localPoint, localDirection, localNormal;
318 G4ThreeVector currentPoint = aPoint;
319 G4SurfBits exclusion(fVoxels.GetBitsPerSlice());
320 G4bool notOutside;
321 G4ThreeVector maxNormal;
322
323 do
324 {
325 notOutside = false;
326
327 G4double maxDistance = -kInfinity;
328 G4int maxCandidate = 0;
329 G4ThreeVector maxLocalPoint;
330
331 std::size_t limit = candidates.size();
332 for (std::size_t i = 0 ; i < limit ; ++i)
333 {
334 G4int candidate = candidates[i];
335 // ignore the current component (that you just got out of) since
336 // numerically the propagated point will be on its surface
337
338 G4VSolid& solid = *fSolids[candidate];
339 const G4Transform3D& transform = fTransformObjs[candidate];
340
341 // The coordinates of the point are modified so as to fit the
342 // intrinsic solid local frame:
343 localPoint = GetLocalPoint(transform, currentPoint);
344
345 // DistanceToOut at least for Trd sometimes return non-zero value
346 // even from points that are outside. Therefore, this condition
347 // must currently be here, otherwise it would not work.
348 // But it means it would be slower.
349
350 if (solid.Inside(localPoint) != EInside::kOutside)
351 {
352 notOutside = true;
353
354 localDirection = GetLocalVector(transform, direction);
355
356 // propagate with solid.DistanceToOut
357 G4double shift = solid.DistanceToOut(localPoint, localDirection,
358 false, nullptr, &localNormal);
359 if (maxDistance < shift)
360 {
361 maxDistance = shift;
362 maxCandidate = candidate;
363 maxNormal = localNormal;
364 }
365 }
366 }
367
368 if (notOutside)
369 {
370 const G4Transform3D& transform = fTransformObjs[maxCandidate];
371
372 // convert from local normal
373 if (aNormal != nullptr) *aNormal = GetGlobalVector(transform, maxNormal);
374
375 distance += maxDistance;
376 currentPoint += maxDistance * direction;
377 if(maxDistance == 0.) ++count;
378
379 // the current component will be ignored
380 exclusion.SetBitNumber(maxCandidate);
381 EInside location = InsideWithExclusion(currentPoint, &exclusion);
382
383 // perform a Inside
384 // it should be excluded current solid from checking
385 // we have to collect the maximum distance from all given candidates.
386 // such "maximum" candidate should be then used for finding next
387 // candidates
388 if (location == EInside::kOutside)
389 {
390 // else return cumulated distances to outside of the traversed
391 // components
392 break;
393 }
394 // if inside another component, redo 1 to 3 but add the next
395 // DistanceToOut on top of the previous.
396
397 // and fill the candidates for the corresponding voxel (just
398 // exiting current component along direction)
399 candidates.clear();
400
401 fVoxels.GetCandidatesVoxelArray(currentPoint, candidates, &exclusion);
402 exclusion.ResetBitNumber(maxCandidate);
403 }
404 }
405 while ((notOutside) && (count < numNodes));
406 }
407
408 return distance;
409}
bool G4bool
Definition G4Types.hh:86

Referenced by DistanceToOut().

◆ DistanceToOutVoxelsCore()

G4double G4MultiUnion::DistanceToOutVoxelsCore ( const G4ThreeVector & aPoint,
const G4ThreeVector & aDirection,
G4ThreeVector * aNormalVector,
G4bool & aConvex,
std::vector< G4int > & candidates ) const

◆ Extent()

void G4MultiUnion::Extent ( EAxis aAxis,
G4double & aMin,
G4double & aMax ) const

Definition at line 538 of file G4MultiUnion.cc.

539{
540 // Determines the bounding box for the considered instance of "UMultipleUnion"
542
543 auto numNodes = (G4int)fSolids.size();
544 for (auto i = 0 ; i < numNodes ; ++i)
545 {
546 G4VSolid& solid = *fSolids[i];
547 G4Transform3D transform = GetTransformation(i);
548 solid.BoundingLimits(min, max);
549
550 TransformLimits(min, max, transform);
551
552 if (i == 0)
553 {
554 switch (aAxis)
555 {
556 case kXAxis:
557 aMin = min.x();
558 aMax = max.x();
559 break;
560 case kYAxis:
561 aMin = min.y();
562 aMax = max.y();
563 break;
564 case kZAxis:
565 aMin = min.z();
566 aMax = max.z();
567 break;
568 default:
569 break;
570 }
571 }
572 else
573 {
574 // Determine the min/max on the considered axis:
575 switch (aAxis)
576 {
577 case kXAxis:
578 if (min.x() < aMin)
579 aMin = min.x();
580 if (max.x() > aMax)
581 aMax = max.x();
582 break;
583 case kYAxis:
584 if (min.y() < aMin)
585 aMin = min.y();
586 if (max.y() > aMax)
587 aMax = max.y();
588 break;
589 case kZAxis:
590 if (min.z() < aMin)
591 aMin = min.z();
592 if (max.z() > aMax)
593 aMax = max.z();
594 break;
595 default:
596 break;
597 }
598 }
599 }
600}
virtual void BoundingLimits(G4ThreeVector &pMin, G4ThreeVector &pMax) const
Definition G4VSolid.cc:680
T max(const T t1, const T t2)
brief Return the largest of the two arguments
T min(const T t1, const T t2)
brief Return the smallest of the two arguments

Referenced by BoundingLimits().

◆ GetCubicVolume()

G4double G4MultiUnion::GetCubicVolume ( )
overridevirtual

Reimplemented from G4VSolid.

Definition at line 125 of file G4MultiUnion.cc.

126{
127 if (fCubicVolume == 0.0)
128 {
129 fCubicVolume = EstimateCubicVolume(1000000, 0.001);
130 }
131 return fCubicVolume;
132}
G4double EstimateCubicVolume(G4int nStat, G4double epsilon) const
Definition G4VSolid.cc:218

◆ GetEntityType()

G4GeometryType G4MultiUnion::GetEntityType ( ) const
inlineoverridevirtual

Implements G4VSolid.

Definition at line 124 of file G4MultiUnion.hh.

124{ return "G4MultiUnion"; }

◆ GetNumberOfSolids()

G4int G4MultiUnion::GetNumberOfSolids ( ) const
inline

Definition at line 207 of file G4MultiUnion.hh.

208{
209 return G4int(fSolids.size());
210}

Referenced by CreatePolyhedron(), G4tgbGeometryDumper::DumpMultiUnionVolume(), and G4GDMLWriteSolids::MultiUnionWrite().

◆ GetNumOfConstituents()

G4int G4MultiUnion::GetNumOfConstituents ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 808 of file G4MultiUnion.cc.

809{
810 G4int num = 0;
811 for (const auto solid : fSolids) { num += solid->GetNumOfConstituents(); }
812 return num;
813}

◆ GetPointOnSurface()

G4ThreeVector G4MultiUnion::GetPointOnSurface ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 943 of file G4MultiUnion.cc.

944{
945 G4ThreeVector point;
946
947 G4long size = fSolids.size();
948
949 do
950 {
951 G4long rnd = G4RandFlat::shootInt(G4long(0), size);
952 G4VSolid& solid = *fSolids[rnd];
953 point = solid.GetPointOnSurface();
954 const G4Transform3D& transform = fTransformObjs[rnd];
955 point = GetGlobalPoint(transform, point);
956 }
957 while (Inside(point) != EInside::kSurface);
958
959 return point;
960}
long G4long
Definition G4Types.hh:87
EInside Inside(const G4ThreeVector &aPoint) const override
virtual G4ThreeVector GetPointOnSurface() const
Definition G4VSolid.cc:152
@ kSurface
Definition geomdefs.hh:69

◆ GetPolyhedron()

G4Polyhedron * G4MultiUnion::GetPolyhedron ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 1008 of file G4MultiUnion.cc.

1009{
1010 if (fpPolyhedron == nullptr ||
1011 fRebuildPolyhedron ||
1012 fpPolyhedron->GetNumberOfRotationStepsAtTimeOfCreation() !=
1013 fpPolyhedron->GetNumberOfRotationSteps())
1014 {
1015 G4AutoLock l(&polyhedronMutex);
1016 delete fpPolyhedron;
1017 fpPolyhedron = CreatePolyhedron();
1018 fRebuildPolyhedron = false;
1019 l.unlock();
1020 }
1021 return fpPolyhedron;
1022}
G4TemplateAutoLock< G4Mutex > G4AutoLock
G4Polyhedron * CreatePolyhedron() const override

◆ GetSolid()

G4VSolid * G4MultiUnion::GetSolid ( G4int index) const
inline

Definition at line 201 of file G4MultiUnion.hh.

202{
203 return fSolids[index];
204}

Referenced by CreatePolyhedron(), G4tgbGeometryDumper::DumpMultiUnionVolume(), and G4GDMLWriteSolids::MultiUnionWrite().

◆ GetSurfaceArea()

G4double G4MultiUnion::GetSurfaceArea ( )
overridevirtual

Reimplemented from G4VSolid.

Definition at line 798 of file G4MultiUnion.cc.

799{
800 if (fSurfaceArea == 0.0)
801 {
802 fSurfaceArea = EstimateSurfaceArea(1000000, 0.001);
803 }
804 return fSurfaceArea;
805}
G4double EstimateSurfaceArea(G4int nStat, G4double ell) const
Definition G4VSolid.cc:280

◆ GetTransformation()

const G4Transform3D & G4MultiUnion::GetTransformation ( G4int index) const
inline

Definition at line 195 of file G4MultiUnion.hh.

196{
197 return fTransformObjs[index];
198}

Referenced by CreatePolyhedron(), G4tgbGeometryDumper::DumpMultiUnionVolume(), Extent(), InsideNoVoxels(), and G4GDMLWriteSolids::MultiUnionWrite().

◆ GetVoxels()

G4Voxelizer & G4MultiUnion::GetVoxels ( ) const
inline

Definition at line 189 of file G4MultiUnion.hh.

190{
191 return (G4Voxelizer&)fVoxels;
192}
friend class G4Voxelizer

◆ Inside()

EInside G4MultiUnion::Inside ( const G4ThreeVector & aPoint) const
overridevirtual

Implements G4VSolid.

Definition at line 490 of file G4MultiUnion.cc.

491{
492 // Classify point location with respect to solid:
493 // o eInside - inside the solid
494 // o eSurface - close to surface within tolerance
495 // o eOutside - outside the solid
496
497 // Hitherto, it is considered that only parallelepipedic nodes can be
498 // added to the container
499
500 // Implementation using voxelisation techniques:
501 // ---------------------------------------------
502
503 // return InsideIterator(aPoint);
504
505 EInside location = InsideWithExclusion(aPoint);
506 return location;
507}

Referenced by GetPointOnSurface().

◆ InsideIterator()

EInside G4MultiUnion::InsideIterator ( const G4ThreeVector & aPoint) const

◆ InsideNoVoxels()

EInside G4MultiUnion::InsideNoVoxels ( const G4ThreeVector & aPoint) const

Definition at line 510 of file G4MultiUnion.cc.

511{
512 G4ThreeVector localPoint;
513 EInside location = EInside::kOutside;
514 G4int countSurface = 0;
515
516 auto numNodes = (G4int)fSolids.size();
517 for (auto i = 0 ; i < numNodes ; ++i)
518 {
519 G4VSolid& solid = *fSolids[i];
520 G4Transform3D transform = GetTransformation(i);
521
522 // The coordinates of the point are modified so as to fit the
523 // intrinsic solid local frame:
524 localPoint = GetLocalPoint(transform, aPoint);
525
526 location = solid.Inside(localPoint);
527
528 if (location == EInside::kSurface)
529 ++countSurface;
530
531 if (location == EInside::kInside) return EInside::kInside;
532 }
533 if (countSurface != 0) return EInside::kSurface;
534 return EInside::kOutside;
535}

◆ IsFaceted()

G4bool G4MultiUnion::IsFaceted ( ) const
overridevirtual

Reimplemented from G4VSolid.

Definition at line 816 of file G4MultiUnion.cc.

817{
818 for (const auto solid : fSolids) { if (!solid->IsFaceted()) return false; }
819 return true;
820}

◆ operator=()

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

Definition at line 108 of file G4MultiUnion.cc.

109{
110 // Check assignment to self
111 //
112 if (this == &rhs)
113 {
114 return *this;
115 }
116
117 // Copy base class data
118 //
120
121 return *this;
122}
G4VSolid & operator=(const G4VSolid &rhs)
Definition G4VSolid.cc:107

◆ SetAccurateSafety()

void G4MultiUnion::SetAccurateSafety ( G4bool flag)
inline

Definition at line 213 of file G4MultiUnion.hh.

214{
215 fAccurate = flag;
216}

◆ StreamInfo()

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

Implements G4VSolid.

Definition at line 917 of file G4MultiUnion.cc.

918{
919 G4long oldprc = os.precision(16);
920 os << "-----------------------------------------------------------\n"
921 << " *** Dump for solid - " << GetName() << " ***\n"
922 << " ===================================================\n"
923 << " Solid type: G4MultiUnion\n"
924 << " Parameters: \n";
925 std::size_t numNodes = fSolids.size();
926 for (std::size_t i = 0 ; i < numNodes ; ++i)
927 {
928 G4VSolid& solid = *fSolids[i];
929 solid.StreamInfo(os);
930 const G4Transform3D& transform = fTransformObjs[i];
931 os << " Translation is " << transform.getTranslation() << " \n";
932 os << " Rotation is :" << " \n";
933 os << " " << transform.getRotation() << "\n";
934 }
935 os << " \n"
936 << "-----------------------------------------------------------\n";
937 os.precision(oldprc);
938
939 return os;
940}
G4String GetName() const
virtual std::ostream & StreamInfo(std::ostream &os) const =0
CLHEP::HepRotation getRotation() const
CLHEP::Hep3Vector getTranslation() const

◆ SurfaceNormal()

G4ThreeVector G4MultiUnion::SurfaceNormal ( const G4ThreeVector & aPoint) const
overridevirtual

Implements G4VSolid.

Definition at line 629 of file G4MultiUnion.cc.

630{
631 // Computes the localNormal on a surface and returns it as a unit vector.
632 // Must return a valid vector. (even if the point is not on the surface).
633 //
634 // On an edge or corner, provide an average localNormal of all facets within
635 // tolerance
636 // NOTE: the tolerance value used in here is not yet the global surface
637 // tolerance - we will have to revise this value - TODO
638
639 std::vector<G4int> candidates;
640 G4ThreeVector localPoint, normal, localNormal;
641 G4double safety = kInfinity;
642 G4int node = 0;
643
644 ///////////////////////////////////////////////////////////////////////////
645 // Important comment: Cases for which the point is located on an edge or
646 // on a vertice remain to be treated
647
648 // determine weather we are in voxel area
649 if (fVoxels.GetCandidatesVoxelArray(aPoint, candidates) != 0)
650 {
651 std::size_t limit = candidates.size();
652 for (std::size_t i = 0 ; i < limit ; ++i)
653 {
654 G4int candidate = candidates[i];
655 const G4Transform3D& transform = fTransformObjs[candidate];
656
657 // The coordinates of the point are modified so as to fit the intrinsic
658 // solid local frame:
659 localPoint = GetLocalPoint(transform, aPoint);
660 G4VSolid& solid = *fSolids[candidate];
661 EInside location = solid.Inside(localPoint);
662
663 if (location == EInside::kSurface)
664 {
665 // normal case when point is on surface, we pick first solid
666 normal = GetGlobalVector(transform, solid.SurfaceNormal(localPoint));
667 return normal.unit();
668 }
669 else
670 {
671 // collect the smallest safety and remember solid node
672 G4double s = (location == EInside::kInside)
673 ? solid.DistanceToOut(localPoint)
674 : solid.DistanceToIn(localPoint);
675 if (s < safety)
676 {
677 safety = s;
678 node = candidate;
679 }
680 }
681 }
682 // on none of the solids, the point was not on the surface
683 G4VSolid& solid = *fSolids[node];
684 const G4Transform3D& transform = fTransformObjs[node];
685 localPoint = GetLocalPoint(transform, aPoint);
686
687 normal = GetGlobalVector(transform, solid.SurfaceNormal(localPoint));
688 return normal.unit();
689 }
690 else
691 {
692 // for the case when point is certainly outside:
693
694 // find a solid in union with the smallest safety
695 node = SafetyFromOutsideNumberNode(aPoint, safety);
696 G4VSolid& solid = *fSolids[node];
697
698 const G4Transform3D& transform = fTransformObjs[node];
699 localPoint = GetLocalPoint(transform, aPoint);
700
701 // evaluate normal for point at this found solid
702 // and transform multi-union coordinates
703 normal = GetGlobalVector(transform, solid.SurfaceNormal(localPoint));
704
705 return normal.unit();
706 }
707}
virtual G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const =0

◆ Voxelize()

void G4MultiUnion::Voxelize ( )

Definition at line 823 of file G4MultiUnion.cc.

824{
825 fVoxels.Voxelize(fSolids, fTransformObjs);
826}

Referenced by G4tgbVolume::FindOrConstructG4Solid(), and G4GDMLReadSolids::MultiUnionRead().

Friends And Related Symbol Documentation

◆ G4Voxelizer

friend class G4Voxelizer
friend

Definition at line 55 of file G4MultiUnion.hh.

Referenced by G4Voxelizer, and GetVoxels().


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