63 using namespace CLHEP;
91 if ( pDz <= 0 || pDy1 <= 0 || pDx1 <= 0 ||
92 pDx2 <= 0 || pDy2 <= 0 || pDx3 <= 0 || pDx4 <= 0 )
94 std::ostringstream message;
95 message <<
"Invalid length parameters for Solid: " <<
GetName() <<
G4endl
97 << pDx1 <<
", " << pDx2 <<
", " << pDx3 <<
", " << pDx4 <<
G4endl
98 <<
" Y - " << pDy1 <<
", " << pDy2 <<
G4endl
137 && pt[0].
z() == pt[1].
z() && pt[0].
z() == pt[2].
z()
138 && pt[0].
z() == pt[3].
z()
140 && pt[4].
z() == pt[5].
z() && pt[4].
z() == pt[6].
z()
141 && pt[4].
z() == pt[7].
z()
143 && pt[0].y() == pt[1].y() && pt[2].y() == pt[3].y()
144 && pt[4].y() == pt[5].y() && pt[6].y() == pt[7].y()
145 && std::fabs( pt[0].y() + pt[2].y() + pt[4].y() + pt[6].y() ) <
kCarTolerance
146 && std::fabs( pt[0].x() + pt[1].x() + pt[4].x() + pt[5].x() +
147 pt[2].x() + pt[3].x() + pt[6].x() + pt[7].x() ) <
kCarTolerance ) )
149 std::ostringstream message;
150 message <<
"Invalid vertice coordinates for Solid: " <<
GetName();
163 "Face at ~-Y not planar.");
172 std::ostringstream message;
173 message <<
"Face at ~+Y not planar for Solid: " <<
GetName();
184 std::ostringstream message;
185 message <<
"Face at ~-X not planar for Solid: " <<
GetName();
195 std::ostringstream message;
196 message <<
"Face at ~+X not planar for Solid: " <<
GetName();
202 fDy1 = ((pt[2]).y()-(pt[1]).y())*0.5;
203 fDx1 = ((pt[1]).x()-(pt[0]).x())*0.5;
204 fDx2 = ((pt[3]).x()-(pt[2]).x())*0.5;
205 fTalpha1 = ((pt[2]).x()+(pt[3]).x()-(pt[1]).x()-(pt[0]).x())*0.25/
fDy1;
207 fDy2 = ((pt[6]).y()-(pt[5]).y())*0.5;
208 fDx3 = ((pt[5]).x()-(pt[4]).x())*0.5;
209 fDx4 = ((pt[7]).x()-(pt[6]).x())*0.5;
210 fTalpha2 = ((pt[6]).x()+(pt[7]).x()-(pt[5]).x()-(pt[4]).x())*0.25/
fDy2;
228 if ( pZ<=0 || pY<=0 || pX<=0 || pLTX<=0 || pLTX>pX )
230 std::ostringstream message;
231 message <<
"Invalid length parameters for Solid: " <<
GetName();
274 std::ostringstream message;
275 message <<
"Face at ~-Y not planar for Solid: " <<
GetName();
285 std::ostringstream message;
286 message <<
"Face at ~+Y not planar for Solid: " <<
GetName();
296 std::ostringstream message;
297 message <<
"Face at ~-X not planar for Solid: " <<
GetName();
307 std::ostringstream message;
308 message <<
"Face at ~+X not planar for Solid: " <<
GetName();
326 if ( pDz<=0 || pDy1<=0 || pDx1<=0 || pDx2<=0 || pDy2<=0 )
328 std::ostringstream message;
329 message <<
"Invalid length parameters for Solid: " <<
GetName();
372 std::ostringstream message;
373 message <<
"Face at ~-Y not planar for Solid: " <<
GetName();
383 std::ostringstream message;
384 message <<
"Face at ~+Y not planar for Solid: " <<
GetName();
394 std::ostringstream message;
395 message <<
"Face at ~-X not planar for Solid: " <<
GetName();
405 std::ostringstream message;
406 message <<
"Face at ~+X not planar for Solid: " <<
GetName();
425 if ( pDz<=0 || pDy<=0 || pDx<=0 )
427 std::ostringstream message;
428 message <<
"Invalid length parameters for Solid: " <<
GetName();
471 std::ostringstream message;
472 message <<
"Face at ~-Y not planar for Solid: " <<
GetName();
482 std::ostringstream message;
483 message <<
"Face at ~+Y not planar for Solid: " <<
GetName();
493 std::ostringstream message;
494 message <<
"Face at ~-X not planar for Solid: " <<
GetName();
504 std::ostringstream message;
505 message <<
"Face at ~+X not planar for Solid: " <<
GetName();
518 :
G4CSGSolid (pName), fDz(1.), fTthetaCphi(0.), fTthetaSphi(0.),
519 fDy1(1.), fDx1(1.), fDx2(1.), fTalpha1(0.),
520 fDy2(1.), fDx3(1.), fDx4(1.), fTalpha2(0.)
531 :
G4CSGSolid(a), fDz(1.), fTthetaCphi(0.), fTthetaSphi(0.),
532 fDy1(1.), fDx1(1.), fDx2(1.), fTalpha1(0.),
533 fDy2(1.), fDx3(1.), fDx4(1.), fTalpha2(0.)
552 fTthetaCphi(rhs.fTthetaCphi), fTthetaSphi(rhs.fTthetaSphi),
553 fDy1(rhs.fDy1), fDx1(rhs.fDx1), fDx2(rhs.fDx2), fTalpha1(rhs.fTalpha1),
554 fDy2(rhs.fDy2), fDx3(rhs.fDx3), fDx4(rhs.fDx4), fTalpha2(rhs.fTalpha2)
556 for (
size_t i=0; i<4; ++i)
573 if (
this == &rhs) {
return *
this; }
585 for (
size_t i=0; i<4; ++i)
613 if ( pDz<=0 || pDy1<=0 || pDx1<=0 || pDx2<=0 || pDy2<=0 || pDx3<=0 || pDx4<=0 )
615 std::ostringstream message;
616 message <<
"Invalid Length Parameters for Solid: " <<
GetName() <<
G4endl
618 << pDx1 <<
", " << pDx2 <<
", " << pDx3 <<
", " << pDx4 <<
G4endl
619 <<
" Y - " << pDy1 <<
", " << pDy2 <<
G4endl
621 G4Exception(
"G4Trap::SetAllParameters()",
"GeomSolids0002",
676 std::ostringstream message;
677 message <<
"Face at ~-Y not planar for Solid: " <<
GetName();
678 G4Exception(
"G4Trap::MakePlanes()",
"GeomSolids0002",
687 std::ostringstream message;
688 message <<
"Face at ~+Y not planar for Solid: " <<
GetName();
689 G4Exception(
"G4Trap::MakePlanes()",
"GeomSolids0002",
698 std::ostringstream message;
699 message <<
"Face at ~-X not planar for Solid: " <<
GetName();
700 G4Exception(
"G4Trap::MakePlanes()",
"GeomSolids0002",
709 std::ostringstream message;
710 message <<
"Face at ~+X not planar for Solid: " <<
GetName();
711 G4Exception(
"G4Trap::MakePlanes()",
"GeomSolids0002",
741 Vcross = v12.cross(v13);
763 a = +(p4.y() - p2.y())*(p3.z() - p1.z())
764 - (p3.y() - p1.y())*(p4.z() - p2.z());
766 b = -(p4.x() - p2.x())*(p3.z() - p1.z())
767 + (p3.x() - p1.x())*(p4.z() - p2.z());
769 c = +(p4.x() - p2.x())*(p3.y() - p1.y())
770 - (p3.x() - p1.x())*(p4.y() - p2.y());
772 sd = std::sqrt( a*a + b*b + c*c );
782 std::ostringstream message;
783 message <<
"Invalid parameters: norm.mod() <= 0, for Solid: "
785 G4Exception(
"G4Trap::MakePlanes()",
"GeomSolids0002",
790 plane.
d = -( plane.
a*p1.x() + plane.
b*p1.y() + plane.
c*p1.z() );
819 G4double xMin, xMax, yMin, yMax, zMin, zMax;
877 temp[0] = pt[0].y()+(pt[4].y()-pt[0].y())*(zMin-pt[0].
z())
878 /(pt[4].
z()-pt[0].z()) ;
879 temp[1] = pt[0].y()+(pt[4].y()-pt[0].y())*(zMax-pt[0].
z())
880 /(pt[4].
z()-pt[0].z()) ;
881 temp[2] = pt[2].y()+(pt[6].y()-pt[2].y())*(zMin-pt[2].
z())
882 /(pt[6].
z()-pt[2].z()) ;
883 temp[3] = pt[2].y()+(pt[6].y()-pt[2].y())*(zMax-pt[2].
z())
884 /(pt[6].
z()-pt[2].z()) ;
889 for( i = 0 ; i < 4 ; i++ )
891 if( temp[i] > yMax ) yMax = temp[i] ;
892 if( temp[i] < yMin ) yMin = temp[i] ;
913 temp[0] = pt[0].x()+(pt[4].x()-pt[0].x())
914 *(zMin-pt[0].
z())/(pt[4].
z()-pt[0].z()) ;
915 temp[1] = pt[0].x()+(pt[4].x()-pt[0].x())
916 *(zMax-pt[0].
z())/(pt[4].
z()-pt[0].z()) ;
917 temp[2] = pt[2].x()+(pt[6].x()-pt[2].x())
918 *(zMin-pt[2].
z())/(pt[6].
z()-pt[2].z()) ;
919 temp[3] = pt[2].x()+(pt[6].x()-pt[2].x())
920 *(zMax-pt[2].
z())/(pt[6].
z()-pt[2].z()) ;
921 temp[4] = pt[3].x()+(pt[7].x()-pt[3].x())
922 *(zMin-pt[3].
z())/(pt[7].
z()-pt[3].z()) ;
923 temp[5] = pt[3].x()+(pt[7].x()-pt[3].x())
924 *(zMax-pt[3].
z())/(pt[7].
z()-pt[3].z()) ;
925 temp[6] = pt[1].x()+(pt[5].x()-pt[1].x())
926 *(zMin-pt[1].
z())/(pt[5].
z()-pt[1].z()) ;
927 temp[7] = pt[1].x()+(pt[5].x()-pt[1].x())
928 *(zMax-pt[1].
z())/(pt[5].
z()-pt[1].z()) ;
933 for( i = 0 ; i < 8 ; i++ )
935 if( temp[i] > xMax) xMax = temp[i] ;
936 if( temp[i] < xMin) xMin = temp[i] ;
984 G4bool existsAfterClip = false ;
996 for(
G4int nv = 0 ; nv < 8 ; nv++ )
998 if( (*vertices)[nv].x() > xMax ) xMax = (*vertices)[nv].x();
999 if( (*vertices)[nv].y() > yMax ) yMax = (*vertices)[nv].y();
1000 if( (*vertices)[nv].
z() > zMax ) zMax = (*vertices)[nv].z();
1002 if( (*vertices)[nv].x() < xMin ) xMin = (*vertices)[nv].x();
1003 if( (*vertices)[nv].y() < yMin ) yMin = (*vertices)[nv].y();
1004 if( (*vertices)[nv].
z() < zMin ) zMin = (*vertices)[nv].z();
1088 existsAfterClip=
true;
1096 flag = existsAfterClip ;
1115 for ( i = 0;i < 4;i++ )
1129 for ( i = 0; i < 4; i++ )
1149 G4int i, noSurfaces = 0;
1154 for (i = 0; i < 4; i++)
1163 distz = std::fabs( std::fabs( p.z() ) -
fDz );
1188 if (distmx <= delta)
1198 if (distmy <= delta)
1206 if ( p.z() >= 0.) sumnorm += nZ;
1209 if ( noSurfaces == 0 )
1212 G4Exception(
"G4Trap::SurfaceNormal(p)",
"GeomSolids1002",
1217 else if ( noSurfaces == 1 ) norm = sumnorm;
1218 else norm = sumnorm.unit();
1241 safez=std::fabs(std::fabs(p.z())-
fDz);
1285 smin = (-
fDz-p.z())/v.z();
1292 else if (v.z() < 0 )
1294 max = -
fDz - p.z() ;
1298 smin=(
fDz-p.z())/v.z();
1394 safe=std::fabs(p.z())-
fDz;
1399 if (Dist > safe) safe=Dist;
1676 std::ostringstream message;
1677 G4int oldprc = message.precision(16);
1678 message <<
"Undefined side for valid surface normal to solid."
1680 <<
"Position:" << G4endl << G4endl
1681 <<
"p.x() = " << p.x()/
mm <<
" mm" << G4endl
1682 <<
"p.y() = " << p.y()/
mm <<
" mm" << G4endl
1683 <<
"p.z() = " << p.z()/
mm <<
" mm" << G4endl << G4endl
1684 <<
"Direction:" << G4endl << G4endl
1685 <<
"v.x() = " << v.x() << G4endl
1686 <<
"v.y() = " << v.y() << G4endl
1687 <<
"v.z() = " << v.z() << G4endl << G4endl
1688 <<
"Proposed distance :" << G4endl << G4endl
1689 <<
"snxt = " << snxt/
mm <<
" mm" <<
G4endl;
1690 message.precision(oldprc);
1691 G4Exception(
"G4Trap::DistanceToOut(p,v,..)",
"GeomSolids1002",
1719 G4cout.precision(oldprc) ;
1721 "GeomSolids1002",
JustWarning,
"Point p is outside !?" );
1725 safe=
fDz-std::fabs(p.z());
1733 if (Dist<safe) safe=Dist;
1756 vertices->reserve(8);
1788 "Error in allocation of vertices. Out of memory !");
1808 return new G4Trap(*
this);
1817 G4int oldprc = os.precision(16);
1818 os <<
"-----------------------------------------------------------\n"
1819 <<
" *** Dump for solid - " <<
GetName() <<
" ***\n"
1820 <<
" ===================================================\n"
1821 <<
" Solid type: G4Trap\n"
1822 <<
" Parameters: \n"
1823 <<
" half length Z: " <<
fDz/
mm <<
" mm \n"
1824 <<
" half length Y of face -fDz: " <<
fDy1/
mm <<
" mm \n"
1825 <<
" half length X of side -fDy1, face -fDz: " <<
fDx1/
mm <<
" mm \n"
1826 <<
" half length X of side +fDy1, face -fDz: " <<
fDx2/
mm <<
" mm \n"
1827 <<
" half length Y of face +fDz: " <<
fDy2/
mm <<
" mm \n"
1828 <<
" half length X of side -fDy2, face +fDz: " <<
fDx3/
mm <<
" mm \n"
1829 <<
" half length X of side +fDy2, face +fDz: " <<
fDx4/
mm <<
" mm \n"
1832 <<
" std::tan(alpha), -fDz: " <<
fTalpha1/
degree <<
" degrees \n"
1833 <<
" std::tan(alpha), +fDz: " <<
fTalpha2/
degree <<
" degrees \n"
1834 <<
" trap side plane equations:\n"
1843 <<
"-----------------------------------------------------------\n";
1844 os.precision(oldprc);
1859 G4double lambda1, lambda2, chose, aOne, aTwo;
1868 w.z()*v.x() - w.x()*v.z(),
1869 w.x()*v.y() - w.y()*v.x());
1871 aOne = 0.5*Area.mag();
1874 t.z()*u.x() - t.x()*u.z(),
1875 t.x()*u.y() - t.y()*u.x());
1877 aTwo = 0.5*Area.mag();
1883 if( (chose>=0.) && (chose < aOne) )
1887 return (p2+lambda1*v+lambda2*w);
1895 return (p0+lambda1*t+lambda2*u);
1904 G4double aOne, aTwo, aThree, aFour, aFive, aSix, chose;
1935 if( (chose>=0.) && (chose<aOne) )
1937 else if( (chose>=aOne) && (chose<aOne+aTwo) )
1939 else if( (chose>=aOne+aTwo) && (chose<aOne+aTwo+aThree) )
1941 else if( (chose>=aOne+aTwo+aThree) && (chose<aOne+aTwo+aThree+aFour) )
1943 else if( (chose>=aOne+aTwo+aThree+aFour)
1944 && (chose<aOne+aTwo+aThree+aFour+aFive) )
ThreeVector shoot(const G4int Ap, const G4int Af)
G4ThreeVector GetPointOnPlane(G4ThreeVector p0, G4ThreeVector p1, G4ThreeVector p2, G4ThreeVector p3, G4double &area) const
G4GeometryType GetEntityType() const
static const G4double kInfinity
G4double GetMinYExtent() const
CLHEP::Hep3Vector G4ThreeVector
G4Trap(const G4String &pName, G4double pDz, G4double pTheta, G4double pPhi, G4double pDy1, G4double pDx1, G4double pDx2, G4double pAlp1, G4double pDy2, G4double pDx3, G4double pDx4, G4double pAlp2)
G4bool IsYLimited() const
G4Trap & operator=(const G4Trap &rhs)
G4bool IsXLimited() const
virtual void AddSolid(const G4Box &)=0
static double normal(HepRandomEngine *eptr)
G4double GetMaxXExtent() const
G4double DistanceToOut(const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm=false, G4bool *validNorm=0, G4ThreeVector *n=0) const
G4double GetMinZExtent() const
G4GLOB_DLL std::ostream G4cout
G4Polyhedron * fpPolyhedron
void SetAllParameters(G4double pDz, G4double pTheta, G4double pPhi, G4double pDy1, G4double pDx1, G4double pDx2, G4double pAlp1, G4double pDy2, G4double pDx3, G4double pDx4, G4double pAlp2)
G4bool CalculateExtent(const EAxis pAxis, const G4VoxelLimits &pVoxelLimit, const G4AffineTransform &pTransform, G4double &pMin, G4double &pMax) const
std::vector< G4ThreeVector > G4ThreeVectorList
G4ThreeVectorList * CreateRotatedVertices(const G4AffineTransform &pTransform) const
void ComputeDimensions(G4VPVParameterisation *p, const G4int n, const G4VPhysicalVolume *pRep)
G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *comments)
virtual void ComputeDimensions(G4Box &, const G4int, const G4VPhysicalVolume *) const
G4double GetMinXExtent() const
std::ostream & StreamInfo(std::ostream &os) const
G4double GetMaxZExtent() const
T max(const T t1, const T t2)
brief Return the largest of the two arguments
G4ThreeVector ApproxSurfaceNormal(const G4ThreeVector &p) const
static const double degree
G4ThreeVector GetPointOnSurface() const
G4double GetMaxYExtent() const
const G4double kCoplanar_Tolerance
G4Polyhedron * CreatePolyhedron() const
G4double DistanceToIn(const G4ThreeVector &p, const G4ThreeVector &v) const
G4bool MakePlane(const G4ThreeVector &p1, const G4ThreeVector &p2, const G4ThreeVector &p3, const G4ThreeVector &p4, TrapSidePlane &plane)
EInside Inside(const G4ThreeVector &p) const
G4CSGSolid & operator=(const G4CSGSolid &rhs)
G4bool IsZLimited() const
void DescribeYourselfTo(G4VGraphicsScene &scene) const