37#if !defined(G4GEOM_USE_UGENERICTRAP)
67 const std::vector<G4TwoVector>& vertices)
71 CheckParameters(halfZ, vertices);
72 ComputeLateralSurfaces();
74 ComputeScratchLength();
100 halfTolerance(rhs.halfTolerance), fScratch(rhs.fScratch),
101 fDz(rhs.fDz), fVertices(rhs.fVertices), fIsTwisted(rhs.fIsTwisted),
102 fMinBBox(rhs.fMinBBox), fMaxBBox(rhs.fMaxBBox),
103 fVisSubdivisions(rhs.fVisSubdivisions),
104 fSurfaceArea(rhs.fSurfaceArea), fCubicVolume(rhs.fCubicVolume)
106 for (
auto i = 0; i < 5; ++i) { fTwist[i] = rhs.fTwist[i]; }
107 for (
auto i = 0; i < 4; ++i) { fDelta[i] = rhs.fDelta[i]; }
108 for (
auto i = 0; i < 8; ++i) { fPlane[i] = rhs.fPlane[i]; }
109 for (
auto i = 0; i < 4; ++i) { fSurf[i] = rhs.fSurf[i]; }
110 for (
auto i = 0; i < 4; ++i) { fArea[i] = rhs.fArea[i]; }
120 if (
this == &rhs) {
return *
this; }
126 halfTolerance = rhs.halfTolerance; fScratch = rhs.fScratch;
127 fDz = rhs.fDz; fVertices = rhs.fVertices; fIsTwisted = rhs.fIsTwisted;
128 fMinBBox = rhs.fMinBBox; fMaxBBox = rhs.fMaxBBox;
129 fVisSubdivisions = rhs.fVisSubdivisions;
130 fSurfaceArea = rhs.fSurfaceArea; fCubicVolume = rhs.fCubicVolume;
132 for (
auto i = 0; i < 8; ++i) { fVertices[i] = rhs.fVertices[i]; }
133 for (
auto i = 0; i < 5; ++i) { fTwist[i] = rhs.fTwist[i]; }
134 for (
auto i = 0; i < 4; ++i) { fDelta[i] = rhs.fDelta[i]; }
135 for (
auto i = 0; i < 8; ++i) { fPlane[i] = rhs.fPlane[i]; }
136 for (
auto i = 0; i < 4; ++i) { fSurf[i] = rhs.fSurf[i]; }
137 for (
auto i = 0; i < 4; ++i) { fArea[i] = rhs.fArea[i]; }
139 fRebuildPolyhedron =
false;
140 delete fpPolyhedron; fpPolyhedron =
nullptr;
156 for (
auto i = 0; i < 4; ++i) { pp[i] = fVertices[i] + fDelta[i]*z; }
160 for (
auto i = 0; i < 4; ++i)
164 G4double dd = fSurf[i].D*px + fSurf[i].E*py + fSurf[i].F*pz + fSurf[i].G;
165 dist = std::max(dd, dist);
174 G4double leng = std::sqrt(dx*dx + dy*dy);
175 G4double dd = (dx*(py - a.
y()) - dy*(px - a.
x()))/leng;
176 dist = std::max(dd, dist);
179 return (dist > halfTolerance) ?
kOutside :
189 G4double halfToleranceSquared = halfTolerance*halfTolerance;
196 for (
auto i = 0; i < 4; ++i) { pp[i] = fVertices[i] + fDelta[i]*tz; }
200 nz = std::copysign(
G4double(std::abs(dz) <= halfTolerance), pz);
203 for (
auto i = 0; i < 4; ++i)
207 G4double dd = fSurf[i].D*px + fSurf[i].E*py + fSurf[i].F*pz + fSurf[i].G;
208 if (std::abs(dd) <= halfTolerance)
223 G4double dd = dx*(py - a.
y()) - dy*(px - a.
x());
224 if (dd*dd <= halfToleranceSquared*ll)
226 G4double x = fSurf[i].A*pz + fSurf[i].D;
227 G4double y = fSurf[i].B*pz + fSurf[i].E;
228 G4double z = fSurf[i].A*px + fSurf[i].B*py + 2.*fSurf[i].C*pz + fSurf[i].F;
229 G4double inv = 1./std::sqrt(x*x + y*y + z*z);
238 G4double mag2 = nx*nx + ny*ny + nz*nz;
239 if (mag2 == 0.)
return ApproxSurfaceNormal(p);
241 G4double inv = (mag == 1.) ? 1. : 1./mag;
242 return { nx*inv, ny*inv, nz*inv };
253 std::ostringstream message;
254 message.precision(16);
255 message <<
"Point p is not on surface of solid: " <<
GetName() <<
" !?\n"
256 <<
"Position: " << p <<
" is "
259 G4Exception(
"G4GenericTrap::ApproxSurfaceNormal(p)",
"GeomSolids1002",
269 for (
auto i = 0; i < 4; ++i) { pp[i] = fVertices[i] + fDelta[i]*tz; }
272 for (
auto i = 0; i < 4; ++i)
276 G4double d = fSurf[i].D*px + fSurf[i].E*py + fSurf[i].F*pz + fSurf[i].G;
277 if (d > dist) { dist = d; iside = i; }
286 G4double leng = std::sqrt(dx*dx + dy*dy);
287 G4double d = (dx*(py - a.
y()) - dy*(px - a.
x()))/leng;
288 if (d > dist) { dist = d; iside = i; }
292 G4double distz = std::abs(pz) - fDz;
293 if (distz >= dist)
return { 0., 0., std::copysign(1., pz) };
295 G4double x = fSurf[iside].A*pz + fSurf[iside].D;
296 G4double y = fSurf[iside].B*pz + fSurf[iside].E;
297 G4double z = fSurf[iside].A*px + fSurf[iside].B*py + 2.*fSurf[iside].C*pz + fSurf[iside].F;
298 G4double inv = 1./std::sqrt(x*x + y*y + z*z);
299 return { x*inv, y*inv, z*inv };
315 if (std::abs(pz) - fDz >= -halfTolerance && pz*vz >= 0) {
return kInfinity; }
316 G4double invz = (vz == 0) ? kInfinity : -1./vz;
317 G4double dz = std::copysign(fDz,invz);
322 for (
auto k = 0; k < 4; ++k)
324 if (fTwist[k] != 0)
continue;
325 const G4GenericTrapPlane& surf = fPlane[2*k];
326 G4double cosa = surf.A*vx + surf.B*vy + surf.C*vz;
327 G4double dist = surf.A*px + surf.B*py + surf.C*pz + surf.D;
328 if (dist >= -halfTolerance)
330 if (cosa >= 0.) {
return kInfinity; }
332 xin = std::max(tmp, xin);
336 if (cosa > 0) { xout = std::min(-dist/cosa, xout); }
337 if (cosa < 0) { xin = std::max(-dist/cosa, xin); }
344 for (
auto i = 0; i < 4; ++i)
346 if (fTwist[i] == 0)
continue;
349 const G4GenericTrapPlane& surf1 = fPlane[2*i];
350 G4double cosa = surf1.A*vx + surf1.B*vy + surf1.C*vz;
351 G4double dist = surf1.A*px + surf1.B*py + surf1.C*pz + surf1.D;
352 if (dist >= -halfTolerance)
354 if (cosa >= 0.) {
return kInfinity; }
356 tin = std::max(tmp, tin);
360 if (cosa > 0) { tout = std::min(-dist/cosa, tout); }
361 if (cosa < 0) { tin = std::max(-dist/cosa, tin); }
365 const G4GenericTrapPlane& surf2 = fPlane[2*i + 1];
366 cosa = surf2.A*vx + surf2.B*vy + surf2.C*vz;
367 dist = surf2.A*px + surf2.B*py + surf2.C*pz + surf2.D;
368 if (dist >= -halfTolerance)
370 if (cosa >= 0.) {
return kInfinity; }
372 tin = std::max(tmp, tin);
376 if (cosa > 0) { tout = std::min(-dist/cosa, tout); }
377 if (cosa < 0) { tin = std::max(-dist/cosa, tin); }
380 if (tout - tin <= halfTolerance) {
return kInfinity; }
384 G4double t0 = 0., x0 = px, y0 = py, z0 = pz;
385 if (x0 < fMinBBox.x() - halfTolerance ||
386 y0 < fMinBBox.y() - halfTolerance ||
387 z0 < fMinBBox.z() - halfTolerance ||
388 x0 > fMaxBBox.x() + halfTolerance ||
389 y0 > fMaxBBox.y() + halfTolerance ||
390 z0 > fMaxBBox.z() + halfTolerance)
403 if (tin - xin < halfTolerance) ttin[0] = xin;
404 if (xout - tout < halfTolerance) ttout[0] = xout;
405 G4double tminimal = tin - halfTolerance;
406 G4double tmaximal = tout + halfTolerance;
409 for (
auto i = 0; i < 4; ++i)
411 if (fTwist[i] == 0)
continue;
414 G4double ABC = fSurf[i].A*vx + fSurf[i].B*vy + fSurf[i].C*vz;
415 G4double ABCF = fSurf[i].A*x0 + fSurf[i].B*y0 + fSurf[i].C*z0 + fSurf[i].F;
417 G4double B = 0.5*(fSurf[i].D*vx + fSurf[i].E*vy + ABCF*vz + ABC*z0);
418 G4double C = fSurf[i].D*x0 + fSurf[i].E*y0 + ABCF*z0 + fSurf[i].G;
419 if (std::abs(
A) <= EPSILON)
422 if (std::abs(B) <= EPSILON)
432 G4double leng = std::sqrt(dx*dx + dy*dy);
433 G4double dist = dx*(y0 - a.
y()) - dy*(x0 - a.
x());
434 if (dist >= -halfTolerance*leng) {
return kInfinity; }
444 const G4GenericTrapSurface& surf = fSurf[i];
447 G4double nz = surf.A*x + surf.B*y + 2.*surf.C*z + surf.F;
449 if (nx*vx + ny*vy + nz*vz >= 0.)
451 auto k = (i == 3) ? 0 : i + 1;
457 G4double leng = std::sqrt(dx*dx + dy*dy);
458 G4double dist = dx*(py - a.
y()) - dy*(px - a.
x());
459 if (dist >= -halfTolerance*leng) {
return kInfinity; }
461 if (tmp < tminimal || tmp > tmaximal)
continue;
462 if (std::abs(tmp - ttout[0]) < halfTolerance)
continue;
468 else { ttout[1] = std::min(tmp, ttout[1]); }
472 if (tmp < tminimal || tmp > tmaximal)
continue;
473 if (std::abs(tmp - ttin[0]) < halfTolerance)
continue;
479 else { ttin[1] = std::min(tmp, ttin[1]); }
486 if (
D < 0.25*fScratch*fScratch*
A*
A)
488 if (
A > 0)
return kInfinity;
493 G4double tmp = -B - std::copysign(std::sqrt(
D), B);
496 G4double tsurfin = std::min(t1, t2);
497 G4double tsurfout = std::max(t1, t2);
498 if (
A < 0) std::swap(tsurfin, tsurfout);
500 if (tsurfin >= tminimal && tsurfin <= tmaximal)
502 if (std::abs(tsurfin - ttin[0]) >= halfTolerance)
504 if (tsurfin < ttin[0])
509 else { ttin[1] = std::min(tsurfin, ttin[1]); }
512 if (tsurfout >= tminimal && tsurfout <= tmaximal)
514 if (std::abs(tsurfout - ttout[0]) >= halfTolerance)
516 if (tsurfout < ttout[0])
521 else { ttout[1] = std::min(tsurfout, ttout[1]); }
528 if (ttin[0] ==
DBL_MAX) {
return kInfinity; }
534 G4double distout = (distin >= ttout[0] - halfTolerance) ? ttout[1] : ttout[0];
535 G4double dist = (distout <= halfTolerance || distout - distin <= halfTolerance) ? kInfinity : distin;
536 return (dist < halfTolerance) ? 0. : dist;
540 if (ttin[1] < ttout[0])
542 G4double distin = ttin[1], distout = ttout[0];
543 G4double dist = (distout <= halfTolerance || distout - distin <= halfTolerance) ? kInfinity : distin;
544 return (dist < halfTolerance) ? 0. : dist;
548 G4double distin1 = ttin[0], distout1 = ttout[0];
549 G4double dist1 = (distout1 <= halfTolerance || distout1 - distin1 <= halfTolerance) ? kInfinity : distin1;
550 if (dist1 != kInfinity) {
return (dist1 < halfTolerance) ? 0. : dist1; }
553 G4double distin2 = ttin[1], distout2 = ttout[1];
554 G4double dist2 = (distout2 <= halfTolerance || distout2 - distin2 <= halfTolerance) ? kInfinity : distin2;
555 return (dist2 < halfTolerance) ? 0. : dist2;
569 for (
auto i = 0; i < 4; ++i) { pp[i] = fVertices[i] + fDelta[i]*z; }
573 for (
auto i = 0; i < 4; ++i)
577 G4double dd = fSurf[i].D*px + fSurf[i].E*py + fSurf[i].F*pz + fSurf[i].G;
578 dist = std::max(dd, dist);
584 G4double cx = pp[j].x() - pp[i].x();
585 G4double cy = pp[j].y() - pp[i].y();
586 G4double d = (pp[i].x() - px)*cy + (py - pp[i].y())*cx;
587 G4ThreeVector na(-cy, cx, fDelta[i].x()*cy - fDelta[i].y()*cx);
588 G4ThreeVector nb(-cy, cx, fDelta[j].x()*cy - fDelta[j].y()*cx);
591 G4double distab = (amag2 > bmag2) ? d/std::sqrt(amag2) : d/std::sqrt(bmag2);
592 dist = std::max(distab, dist);
595 return (dist > 0.) ? dist : 0.;
613 if ((std::abs(pz) - fDz) >= -halfTolerance && pz*vz > 0.)
618 n->set(0., 0., std::copysign(1., pz));
622 G4double tout = (vz == 0) ?
DBL_MAX : (std::copysign(fDz, vz) - pz)/vz;
623 G4int iface = (vz < 0) ? -4 : -2;
625 for (
auto i = 0; i < 4; ++i)
627 if (fTwist[i] != 0)
continue;
628 const G4GenericTrapPlane& surf = fPlane[2*i];
629 G4double cosa = surf.A*vx + surf.B*vy + surf.C*vz;
632 G4double dist = surf.A*px + surf.B*py + surf.C*pz + surf.D;
633 if (dist >= -halfTolerance)
638 n->set(surf.A, surf.B, surf.C);
643 if (tout > tmp) { tout = tmp; iface = i; }
650 for (
auto i = 0; i < 4; ++i)
652 if (fTwist[i] == 0)
continue;
655 const G4GenericTrapSurface& surf = fSurf[i];
656 G4double ABC = surf.A*vx + surf.B*vy + surf.C*vz;
657 G4double ABCF = surf.A*px + surf.B*py + surf.C*pz + surf.F;
659 G4double B = 0.5*(surf.D*vx + surf.E*vy + ABCF*vz + ABC*pz);
660 G4double C = surf.D*px + surf.E*py + ABCF*pz + surf.G;
662 if (std::abs(
A) <= EPSILON)
665 if (std::abs(B) <= EPSILON) {
continue; }
675 G4double nz = surf.A*x + surf.B*y + 2.*surf.C*z + surf.F;
677 if (nx*vx + ny*vy + nz*vz > 0.)
685 G4double leng = std::sqrt(dx*dx + dy*dy);
686 G4double dist = dx*(py - a.
y()) - dy*(px - a.
x());
687 if (dist >= -halfTolerance*leng)
692 G4double normx = surf.A*pz + surf.D;
693 G4double normy = surf.B*pz + surf.E;
694 G4double normz = surf.A*px + surf.B*py + 2.*surf.C*pz + surf.F;
695 G4double inv = 1./std::sqrt(normx*normx + normy*normy + normz*normz);
696 n->set(normx*inv, normy*inv, normz*inv);
700 if (tout > tmp) { tout = tmp; iface = i; }
707 if (
D < 0.25*fScratch*fScratch*
A*
A)
716 G4double leng = std::sqrt(dx*dx + dy*dy);
717 G4double dist = dx*(py - a.
y()) - dy*(px - a.
x());
718 if (dist <= -halfTolerance*leng) {
continue; }
719 if (
A > 0 || dist > halfTolerance*leng)
726 G4double nz = surf.A*px + surf.B*py + 2.*surf.C*pz + surf.F;
727 G4double inv = 1./std::sqrt(nx*nx + ny*ny + nz*nz);
728 n->set(nx*inv, ny*inv, nz*inv);
736 G4double tmp = -B - std::copysign(std::sqrt(
D), B);
744 if (std::abs(tmax) < std::abs(tmin))
continue;
745 if (tmin <= halfTolerance)
758 G4double leng = std::sqrt(dx*dx + dy*dy);
759 G4double dist = dx*(y - a.
y()) - dy*(x - a.
x());
760 if (dist <= -halfTolerance*leng)
continue;
766 G4double nz = surf.A*px + surf.B*py + 2.*surf.C*pz + surf.F;
767 G4double inv = 1./std::sqrt(nx*nx + ny*ny + nz*nz);
768 n->set(nx*inv, ny*inv, nz*inv);
772 if (tout > tmin) { tout = tmin; iface = i; }
776 if (tmax < halfTolerance)
783 G4double nz = surf.A*px + surf.B*py + 2.*surf.C*pz + surf.F;
784 G4double inv = 1./std::sqrt(nx*nx + ny*ny + nz*nz);
785 n->set(nx*inv, ny*inv, nz*inv);
789 if (tout > tmax) { tout = tmax; iface = i; }
795 if (tout < halfTolerance) tout = 0.;
801 n->set(0, 0, iface + 3);
805 const G4GenericTrapSurface& surf = fSurf[iface];
806 if (fTwist[iface] == 0)
809 n->set(surf.D, surf.E, surf.F);
819 G4double nz = surf.A*x + surf.B*y + 2.*surf.C*z + surf.F;
820 G4double inv = 1./std::sqrt(nx*nx + ny*ny + nz*nz);
821 n->set(nx*inv, ny*inv, nz*inv);
839 for (
auto i = 0; i < 4; ++i) { pp[i] = fVertices[i] + fDelta[i]*z; }
843 for (
auto i = 0; i < 4; ++i)
847 G4double dd = fSurf[i].D*px + fSurf[i].E*py + fSurf[i].F*pz + fSurf[i].G;
848 dist = std::max(dd, dist);
854 G4double cx = pp[j].x() - pp[i].x();
855 G4double cy = pp[j].y() - pp[i].y();
856 G4double d = (pp[i].x() - px)*cy + (py - pp[i].y())*cx;
857 G4ThreeVector na(-cy, cx, fDelta[i].x()*cy - fDelta[i].y()*cx);
858 G4ThreeVector nb(-cy, cx, fDelta[j].x()*cy - fDelta[j].y()*cx);
861 G4double distab = (amag2 > bmag2) ? d/std::sqrt(amag2) : d/std::sqrt(bmag2);
862 dist = std::max(distab, dist);
865 return (dist < 0.) ? -dist : 0.;
901 return exist = pMin < pMax;
913 for (
G4int i = 0; i < 4; ++i)
917 baseA[2*i].set(va.
x(), va.
y(),-dz);
918 baseB[2*i].set(vb.
x(), vb.
y(), dz);
920 for (
G4int i = 0; i < 4; ++i)
922 G4int k1 = 2*i, k2 = (2*i + 2)%8;
923 G4double ax = (baseA[k2].x() - baseA[k1].x());
924 G4double ay = (baseA[k2].y() - baseA[k1].y());
925 G4double bx = (baseB[k2].x() - baseB[k1].x());
926 G4double by = (baseB[k2].y() - baseB[k1].y());
928 baseA[k1+1] = (znorm < 0.0) ? baseA[k2] : baseA[k1];
929 baseB[k1+1] = (znorm < 0.0) ? baseB[k1] : baseB[k2];
932 std::vector<const G4ThreeVectorList *> polygons(2);
933 polygons[0] = &baseA;
934 polygons[1] = &baseB;
947 return {
"G4GenericTrap" };
956 return (!fIsTwisted);
974 G4long oldprc = os.precision(16);
975 os <<
"-----------------------------------------------------------\n"
976 <<
" *** Dump for solid - " <<
GetName() <<
" ***\n"
977 <<
" ===================================================\n"
979 <<
" half length Z: " << fDz/mm <<
"\n"
980 <<
" list of vertices:\n";
981 for (
G4int i = 0; i < 8; ++i)
983 os <<
" #" << i <<
" " << fVertices[i] <<
"\n";
985 os <<
"-----------------------------------------------------------\n";
986 os.precision(oldprc);
996 if (fArea[0] + fArea[1] + fArea[2] + fArea[3] == 0.)
999 fArea[0] = GetLateralFaceArea(0);
1000 fArea[1] = GetLateralFaceArea(1);
1001 fArea[2] = GetLateralFaceArea(2);
1002 fArea[3] = GetLateralFaceArea(3);
1006 constexpr G4int iface[6][4] =
1007 { {0,1,2,3}, {0,4,5,1}, {1,5,6,2}, {2,6,7,3}, {3,7,4,0}, {4,5,6,7} };
1009 G4bool isTwisted[6] = {
false};
1010 for (
auto i = 0; i < 4; ++i) { isTwisted[i + 1] = (fTwist[i] != 0.0); }
1017 ssurf[0] = (
A.x()*B.y() -
A.y()*B.x())*0.5;
1018 ssurf[1] = ssurf[0] + fArea[0];
1019 ssurf[2] = ssurf[1] + fArea[1];
1020 ssurf[3] = ssurf[2] + fArea[2];
1021 ssurf[4] = ssurf[3] + fArea[3];
1022 ssurf[5] = ssurf[4] + (C.x()*
D.y() - C.y()*
D.x())*.5;
1026 for (k = 0; k < 5; ++k) {
if (select <= ssurf[k])
break; }
1028 G4int i0 = iface[k][0];
1029 G4int i1 = iface[k][1];
1030 G4int i2 = iface[k][2];
1031 G4int i3 = iface[k][3];
1033 pp[0].set(fVertices[i0].x(), fVertices[i0].y(), ((k == 5) ? fDz : -fDz));
1034 pp[1].set(fVertices[i1].x(), fVertices[i1].y(), ((k == 0) ? -fDz : fDz));
1035 pp[2].set(fVertices[i2].x(), fVertices[i2].y(), ((k == 0) ? -fDz : fDz));
1036 pp[3].set(fVertices[i3].x(), fVertices[i3].y(), ((k == 5) ? fDz : -fDz));
1041 G4double maxlength = std::max((pp[2] - pp[1]).mag(), (pp[3] - pp[0]).mag());
1042 point = (pp[0] + pp[1] + pp[2] + pp[3])*0.25;
1043 for (
auto i = 0; i < 10000; ++i)
1049 if (v >= 1.)
continue;
1050 point = p0 + (p1 - p0)*v;
1058 if (u + v > 1.) { u = 1. - u; v = 1. - v; }
1059 G4double ss = (((pp[2] - pp[0]).cross(pp[3] - pp[0])).mag())*0.5;
1060 G4int j = (select > ssurf[k] - ss) ? 3 : 1;
1061 point = pp[0] + (pp[2] - pp[0])*u + (pp[j] - pp[0])*v;
1072 if (fCubicVolume == 0.0)
1084 G4double CB = C.x()*B.y() - C.y()*B.x();
1086 fCubicVolume = ((AB + CD)/3. + (AD + CB)/6.)*fDz;
1088 return fCubicVolume;
1095G4double G4GenericTrap::GetLateralFaceArea(
G4int iface)
const
1097 G4int i1 = iface, i2 = (i1 + 1)%4, i3 = i1 + 4, i4 = i2 + 4;
1100 if (fTwist[iface] == 0.0)
1102 G4ThreeVector p1(fVertices[i1].x(), fVertices[i1].y(),-fDz);
1103 G4ThreeVector p2(fVertices[i2].x(), fVertices[i2].y(),-fDz);
1104 G4ThreeVector p3(fVertices[i3].x(), fVertices[i3].y(), fDz);
1105 G4ThreeVector p4(fVertices[i4].x(), fVertices[i4].y(), fDz);
1106 return ((p4 - p1).cross(p3 - p2)).mag()*0.5;
1110 constexpr G4int NSTEP = 250;
1113 G4double x21 = fVertices[i2].x() - fVertices[i1].x();
1114 G4double y21 = fVertices[i2].y() - fVertices[i1].y();
1115 G4double x31 = fVertices[i3].x() - fVertices[i1].x();
1116 G4double y31 = fVertices[i3].y() - fVertices[i1].y();
1117 G4double x42 = fVertices[i4].x() - fVertices[i2].x();
1118 G4double y42 = fVertices[i4].y() - fVertices[i2].y();
1119 G4double x43 = fVertices[i4].x() - fVertices[i3].x();
1120 G4double y43 = fVertices[i4].y() - fVertices[i3].y();
1131 for (
G4int i = 0; i < NSTEP; ++i)
1143 G4double R1 = std::sqrt(aa + bb + cc);
1145 G4double log1 = std::log(std::abs(sqrtAA*R1 + 2.*aa + bb));
1146 G4double log0 = std::log(std::abs(sqrtAA*R0 + bb));
1148 area += 0.5*R1 + 0.25*bb*invAA*(R1 - R0) + IIJJ*invSqrtAA*(log1 - log0);
1159 if (fSurfaceArea == 0.0)
1165 G4double S_bot = (
A.x()*B.y() -
A.y()*B.x())*0.5;
1166 G4double S_top = (C.x()*
D.y() - C.y()*
D.x())*0.5;
1167 fArea[0] = GetLateralFaceArea(0);
1168 fArea[1] = GetLateralFaceArea(1);
1169 fArea[2] = GetLateralFaceArea(2);
1170 fArea[3] = GetLateralFaceArea(3);
1171 fSurfaceArea = S_bot + S_top + fArea[0] + fArea[1] + fArea[2] + fArea[3];
1173 return fSurfaceArea;
1181G4GenericTrap::CheckParameters(
G4double halfZ,
1182 const std::vector<G4TwoVector>& vertices)
1186 if (vertices.size() != 8)
1188 std::ostringstream message;
1189 message <<
"Number of vertices is " << vertices.size()
1190 <<
" instead of 8 for Solid: " <<
GetName() <<
" !";
1191 G4Exception(
"G4GenericTrap::CheckParameters()",
"GeomSolids0002",
1199 std::ostringstream message;
1200 message <<
"Z-dimension is too small or negative (dZ = " << fDz <<
" mm)"
1201 <<
" for Solid: " <<
GetName() <<
" !";
1202 G4Exception(
"G4GenericTrap::CheckParameters()",
"GeomSolids0002",
1208 for (
auto i = 0; i < 8; ++i) { fVertices.at(i) = vertices[i]; }
1214 G4double zbot = diag1.
x()*diag2.
y() - diag1.
y()*diag2.
x();
1221 G4double ztop = diag3.
x()*diag4.
y() - diag3.
y()*diag4.
x();
1226 std::ostringstream message;
1227 message <<
"Vertices of the bottom and top polygons are defined in opposite directions !\n";
1229 G4Exception(
"G4GenericTrap::CheckParameters()",
"GeomSolids0002",
1232 if ((zbot > 0.) || (ztop > 0.))
1234 std::swap(fVertices[1], fVertices[3]);
1235 std::swap(fVertices[5], fVertices[7]);
1236 std::ostringstream message;
1237 message <<
"Vertices re-ordered in Solid: " <<
GetName() <<
" !\n"
1238 <<
"Vertices of the bottom and top polygons must be defined in a clockwise direction.";
1239 G4Exception(
"G4GenericTrap::CheckParameters()",
"GeomSolids1001",
1245 G4int ndegenerated = 0;
1246 for (
auto i = 0; i < 4; ++i)
1249 G4double lbot = (fVertices[j] - fVertices[i]).mag();
1250 G4double ltop = (fVertices[j + 4] - fVertices[i + 4]).mag();
1253 if (ndegenerated > 1 ||
1256 std::ostringstream message;
1257 message <<
"Degenerated solid !\n";
1259 G4Exception(
"G4GenericTrap::CheckParameters()",
"GeomSolids0002",
1266 for (
auto i = 0; i < 4; ++i)
1273 if (!isConvex)
break;
1274 G4TwoVector edge3 = fVertices[j + 4] - fVertices[i + 4];
1275 G4TwoVector edge4 = fVertices[k + 4] - fVertices[j + 4];
1277 if (!isConvex)
break;
1281 std::ostringstream message;
1282 message <<
"The bottom and top faces must be convex polygons !\n";
1284 G4Exception(
"G4GenericTrap::CheckParameters()",
"GeomSolids0002",
1293void G4GenericTrap::ComputeLateralSurfaces()
1295 for (
auto i = 0; i < 4; ++i)
1300 G4ThreeVector p3(fVertices[j + 4].x(), fVertices[j + 4].y(), fDz);
1301 G4ThreeVector p4(fVertices[i + 4].x(), fVertices[i + 4].y(), fDz);
1306 G4double zcross = ebot.
x()*etop.
y() - ebot.
y()*etop.
x();
1308 if (std::min(lbot, ltop) <
kCarTolerance || std::abs(zcross) < eps)
1319 G4TwoVector vij = (vi - vl).unit() + (vj - vk).unit();
1326 normal = ((p4 - p1).cross(p3 - p2)).
unit();
1328 fSurf[i].D = fPlane[2*i].A = fPlane[2*i + 1].A = normal.
x();
1329 fSurf[i].E = fPlane[2*i].B = fPlane[2*i + 1].B = normal.
y();
1330 fSurf[i].F = fPlane[2*i].C = fPlane[2*i + 1].C = normal.
z();
1331 fSurf[i].G = fPlane[2*i].D = fPlane[2*i + 1].D = -normal.
dot((p1 + p2 + p3 + p4)/4.);
1336 G4double angle = std::acos(ebot.
dot(etop)/(lbot*ltop));
1337 if (angle > CLHEP::halfpi)
1339 std::ostringstream message;
1340 message <<
"Twist on " << angle/CLHEP::deg
1341 <<
" degrees, should not be more than 90 degrees !";
1343 G4Exception(
"G4GenericTrap::ComputeLateralSurfaces()",
"GeomSolids0002",
1346 fTwist[i] = std::copysign(angle, zcross);
1348 fSurf[i].A = 2.*fDz*(p4.y() - p3.y() - p2.y() + p1.y());
1349 fSurf[i].B =-2.*fDz*(p4.x() - p3.x() - p2.x() + p1.x());
1350 fSurf[i].C = ((p4.x() - p2.x())*(p3.y() - p1.y()) - (p4.y() - p2.y())*(p3.x() - p1.x()));
1351 fSurf[i].D = 2.*fDz*fDz*(p4.y() - p3.y() + p2.y() - p1.y());
1352 fSurf[i].E =-2.*fDz*fDz*(p4.x() - p3.x() + p2.x() - p1.x());
1353 fSurf[i].F = 2.*fDz*(p4.x()*p3.y() - p3.x()*p4.y() - p2.x()*p1.y() + p1.x()*p2.y());
1354 fSurf[i].G = fDz*fDz*((p4.x() + p2.x())*(p3.y() + p1.y()) - (p3.x() + p1.x())*(p4.y() + p2.y()));
1357 fSurf[i].A /= magnitude;
1358 fSurf[i].B /= magnitude;
1359 fSurf[i].C /= magnitude;
1360 fSurf[i].D /= magnitude;
1361 fSurf[i].E /= magnitude;
1362 fSurf[i].F /= magnitude;
1363 fSurf[i].G /= magnitude;
1369 normal1 = ((p2 - p1).cross(p4 - p1)).
unit();
1370 normal2 = ((p3 - p4).cross(p1 - p4)).
unit();
1376 normal1 = ((p3 - p2).cross(p1 - p2)).
unit();
1377 normal2 = ((p2 - p3).cross(p4 - p3)).
unit();
1381 fPlane[2*i].A = normal1.
x();
1382 fPlane[2*i].B = normal1.
y();
1383 fPlane[2*i].C = normal1.
z();
1384 fPlane[2*i].D = -normal1.
dot(c1);
1385 fPlane[2*i + 1].A = normal2.
x();
1386 fPlane[2*i + 1].B = normal2.
y();
1387 fPlane[2*i + 1].C = normal2.
z();
1388 fPlane[2*i + 1].D = -normal2.
dot(c2);
1390 fDelta[i] = (fVertices[i + 4] - fVertices[i])/(2*fDz);
1398void G4GenericTrap::ComputeBoundingBox()
1401 minX = maxX = fVertices[0].x();
1402 minY = maxY = fVertices[0].y();
1403 for (
auto i = 1; i < 8; ++i)
1405 minX = std::min(minX, fVertices[i].x());
1406 maxX = std::max(maxX, fVertices[i].x());
1407 minY = std::min(minY, fVertices[i].y());
1408 maxY = std::max(maxY, fVertices[i].y());
1415 if (minX >= maxX || minY >= maxY || -fDz >= fDz)
1417 std::ostringstream message;
1418 message <<
"Bad bounding box (min >= max) for solid: "
1420 <<
"\npMin = " << fMinBBox
1421 <<
"\npMax = " << fMaxBBox;
1422 G4Exception(
"G4GenericTrap::ComputeBoundingBox()",
"GeomSolids1001",
1432void G4GenericTrap::ComputeScratchLength()
1435 for (
auto i = 0; i < 4; ++i)
1437 if (fTwist[i] == 0.)
continue;
1441 G4ThreeVector p3(fVertices[i + 4].x(), fVertices[i + 4].y(), fDz);
1442 G4ThreeVector p4(fVertices[k + 4].x(), fVertices[k + 4].y(), fDz);
1446 pp[0] = p0 - norm * halfTolerance;
1447 pp[1] = p0 + norm * halfTolerance;
1449 vv[0] = (p4 - p1).unit();
1450 vv[1] = (p3 - p2).unit();
1452 for (
auto ip = 0; ip < 2; ++ip)
1457 for (
auto iv = 0; iv < 2; ++iv)
1463 G4double ABC = fSurf[i].A*vx + fSurf[i].B*vy + fSurf[i].C*vz;
1464 G4double ABCF = fSurf[i].A*px + fSurf[i].B*py + fSurf[i].C*pz + fSurf[i].F;
1466 G4double B = 0.5*(fSurf[i].D*vx + fSurf[i].E*vy + ABCF*vz + ABC*pz);
1467 G4double C = fSurf[i].D*px + fSurf[i].E*py + ABCF*pz + fSurf[i].G;
1469 if (
D < 0)
continue;
1471 scratch = std::min(leng, scratch);
1484 if ( (fpPolyhedron ==
nullptr)
1485 || fRebuildPolyhedron
1486 || (fpPolyhedron->GetNumberOfRotationStepsAtTimeOfCreation() !=
1487 fpPolyhedron->GetNumberOfRotationSteps()) )
1490 delete fpPolyhedron;
1492 fRebuildPolyhedron =
false;
1495 return fpPolyhedron;
1513 return { fMinBBox.x(), fMaxBBox.x(),
1514 fMinBBox.y(), fMaxBBox.y(),
1515 fMinBBox.z(), fMaxBBox.z() };
1526 G4int nVertices, nFacets;
1528 G4int subdivisions = 0;
1540 for(
G4int i = 0; i < 4; ++i)
1548 Dx = 0.5*(fMaxBBox.x() - fMinBBox.y());
1549 Dy = 0.5*(fMaxBBox.y() - fMinBBox.y());
1550 if (Dy > Dx) { Dx = Dy; }
1552 subdivisions = 8*
G4int(maxTwist/(Dx*Dx*Dx)*fDz);
1553 if (subdivisions < 4) { subdivisions = 4; }
1554 if (subdivisions > 30) { subdivisions = 30; }
1557 G4int sub4 = 4*subdivisions;
1558 nVertices = 8 + subdivisions*4;
1559 nFacets = 6 + subdivisions*4;
1560 G4double cf = 1./(subdivisions + 1);
1566 for (
G4int i = 0; i < 4; ++i)
1571 for (
G4int i = 0; i < subdivisions; ++i)
1573 for (
G4int j = 0; j < 4; ++j)
1575 G4TwoVector u = fVertices[j]+cf*(i+1)*(fVertices[j+4]-fVertices[j]);
1580 for (
G4int i = 4; i < 8; ++i)
1589 polyhedron->
SetFacet(++icur, 1, 4, 3, 2);
1590 for (
G4int i = 0; i < subdivisions + 1; ++i)
1593 polyhedron->
SetFacet(++icur, 5+is, 8+is, 4+is, 1+is);
1594 polyhedron->
SetFacet(++icur, 8+is, 7+is, 3+is, 4+is);
1595 polyhedron->
SetFacet(++icur, 7+is, 6+is, 2+is, 3+is);
1596 polyhedron->
SetFacet(++icur, 6+is, 5+is, 1+is, 2+is);
1598 polyhedron->
SetFacet(++icur, 5+sub4, 6+sub4, 7+sub4, 8+sub4);
1613 std::ostringstream message;
1614 message.precision(16);
1615 message << icase <<
" in " <<
GetName() <<
"\n"
1616 <<
" p" << p <<
"\n"
1617 <<
" v" << v <<
"\n"
1618 <<
" A = " <<
A <<
" (is "
1619 << ((
A < 0) ?
"negative, instead of positive)" :
"positive, instead of negative)") <<
" !?\n";
1629void G4GenericTrap::WarningSignB(
const G4String& method,
const G4String& icase,
1633 std::ostringstream message;
1634 message.precision(16);
1635 message << icase <<
" in " <<
GetName() <<
"\n"
1636 <<
" p" << p <<
"\n"
1637 <<
" v" << v <<
"\n"
1638 <<
" f = " << f <<
" B = " <<
B <<
" (is "
1639 << ((
B < 0) ?
"negative, instead of positive)" :
"positive, instead of negative)") <<
" !?\n";
1641 const G4String
function =
"G4GenericTrap::DistanceTo" + method +
"(p,v)";
1653 G4String check =
"";
1656 G4double tcheck = 0.5*(ttout[0] + ttin[1]);
1657 if (
Inside(p + v*tcheck) !=
kOutside) check =
"\n !!! check point 0.5*(ttout[0] + ttin[1]) is NOT outside !!!";
1660 auto position =
Inside(p);
1661 std::ostringstream message;
1662 message.precision(16);
1663 message << k <<
"_Unexpected sequence of intersections in solid: " <<
GetName() <<
" !?\n"
1664 <<
" position = " << ((position ==
kInside) ?
"kInside" : ((position ==
kOutside) ?
"kOutside" :
"kSurface")) <<
"\n"
1665 <<
" p" << p <<
"\n"
1666 <<
" v" << v <<
"\n"
1667 <<
" range : [" << tmin <<
", " << tmax <<
"]\n"
1669 << ((ttin[0] ==
DBL_MAX) ? kInfinity : ttin[0]) <<
", "
1670 << ((ttin[1] ==
DBL_MAX) ? kInfinity : ttin[1]) <<
"\n"
1672 << ((ttout[0] ==
DBL_MAX) ? kInfinity : ttout[0]) <<
", "
1673 << ((ttout[1] ==
DBL_MAX) ? kInfinity : ttout[1]) << check <<
"\n";
1675 G4Exception(
"G4GenericTrap::DistanceToIn(p,v)",
"GeomSolids1002",
1683void G4GenericTrap::WarningDistanceToOut(
const G4ThreeVector& p,
1687 auto position =
Inside(p);
1688 std::ostringstream message;
1689 message.precision(16);
1690 message <<
"Unexpected final tout = " << tout <<
" in solid: " <<
GetName() <<
" !?\n"
1691 <<
" position = " << ((position ==
kInside) ?
"kInside" : ((position ==
kOutside) ?
"kOutside" :
"kSurface")) <<
"\n"
1692 <<
" p" << p <<
"\n"
1693 <<
" v" << v <<
"\n";
1695 G4Exception(
"G4GenericTrap::DistanceToOut(p,v)",
"GeomSolids1002",
G4TemplateAutoLock< G4Mutex > G4AutoLock
std::vector< G4ThreeVector > G4ThreeVectorList
G4double(*)(G4double) function
G4double C(G4double temp)
G4double B(G4double temperature)
G4double D(G4double temp)
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
#define G4MUTEX_INITIALIZER
CLHEP::Hep3Vector G4ThreeVector
CLHEP::Hep2Vector G4TwoVector
Hep3Vector cross(const Hep3Vector &) const
double dot(const Hep3Vector &) const
G4bool BoundingBoxVsVoxelLimits(const EAxis pAxis, const G4VoxelLimits &pVoxelLimits, const G4Transform3D &pTransform3D, G4double &pMin, G4double &pMax) const
G4bool CalculateExtent(const EAxis pAxis, const G4VoxelLimits &pVoxelLimits, const G4Transform3D &pTransform3D, G4double &pMin, G4double &pMax) const
G4double DistanceToIn(const G4ThreeVector &p, const G4ThreeVector &v) const override
G4GenericTrap & operator=(const G4GenericTrap &rhs)
G4double DistanceToOut(const G4ThreeVector &p, const G4ThreeVector &v, const G4bool calcNorm=false, G4bool *validNorm=nullptr, G4ThreeVector *n=nullptr) const override
G4double GetZHalfLength() const
std::ostream & StreamInfo(std::ostream &os) const override
G4TwoVector GetVertex(G4int index) const
G4GeometryType GetEntityType() const override
G4Polyhedron * GetPolyhedron() const override
G4ThreeVector SurfaceNormal(const G4ThreeVector &p) const override
G4double GetTwistAngle(G4int index) const
G4double GetCubicVolume() override
G4VisExtent GetExtent() const override
G4VSolid * Clone() const override
G4int GetVisSubdivisions() const
~G4GenericTrap() override
EInside Inside(const G4ThreeVector &p) const override
G4Polyhedron * CreatePolyhedron() 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
G4bool IsFaceted() const override
G4GenericTrap(const G4String &name, G4double halfZ, const std::vector< G4TwoVector > &vertices)
G4double GetSurfaceArea() override
G4ThreeVector GetPointOnSurface() const override
void DescribeYourselfTo(G4VGraphicsScene &scene) const override
virtual void AddSolid(const G4Box &)=0
G4VSolid(const G4String &name)
G4VSolid & operator=(const G4VSolid &rhs)
void SetVertex(G4int index, const G4Point3D &v)
void SetFacet(G4int index, G4int iv1, G4int iv2, G4int iv3, G4int iv4=0)