Geant4 11.3.0
Toolkit for the simulation of the passage of particles through matter
Loading...
Searching...
No Matches
G4PhysicalVolumeModel.cc
Go to the documentation of this file.
1//
2// ********************************************************************
3// * License and Disclaimer *
4// * *
5// * The Geant4 software is copyright of the Copyright Holders of *
6// * the Geant4 Collaboration. It is provided under the terms and *
7// * conditions of the Geant4 Software License, included in the file *
8// * LICENSE and available at http://cern.ch/geant4/license . These *
9// * include a list of copyright holders. *
10// * *
11// * Neither the authors of this software system, nor their employing *
12// * institutes,nor the agencies providing financial support for this *
13// * work make any representation or warranty, express or implied, *
14// * regarding this software system or assume any liability for its *
15// * use. Please see the license in the file LICENSE and URL above *
16// * for the full disclaimer and the limitation of liability. *
17// * *
18// * This code implementation is the result of the scientific and *
19// * technical work of the GEANT4 collaboration. *
20// * By using, copying, modifying or distributing the software (or *
21// * any work based on the software) you agree to acknowledge its *
22// * use in resulting scientific publications, and indicate your *
23// * acceptance of all terms of the Geant4 Software license. *
24// ********************************************************************
25//
26//
27//
28//
29// John Allison 31st December 1997.
30// Model for physical volumes.
31
33
34#include "G4VGraphicsScene.hh"
35#include "G4VPhysicalVolume.hh"
38#include "G4LogicalVolume.hh"
39#include "G4VSolid.hh"
40#include "G4SubtractionSolid.hh"
42#include "G4Material.hh"
43#include "G4VisAttributes.hh"
46#include "G4Polyhedron.hh"
48#include "G4AttDefStore.hh"
49#include "G4AttDef.hh"
50#include "G4AttValue.hh"
51#include "G4UnitsTable.hh"
52#include "G4Vector3D.hh"
53#include "G4Mesh.hh"
54
55#include <sstream>
56#include <iomanip>
57
58#define G4warn G4cout
59
62 , G4int requestedDepth
63 , const G4Transform3D& modelTransform
64 , const G4ModelingParameters* pMP
65 , G4bool useFullExtent
66 , const std::vector<G4PhysicalVolumeNodeID>& baseFullPVPath)
67: G4VModel (pMP)
68, fpTopPV (pVPV)
69, fTopPVCopyNo (pVPV? pVPV->GetCopyNo(): 0)
70, fRequestedDepth (requestedDepth)
71, fUseFullExtent (useFullExtent)
72, fTransform (modelTransform)
73, fCurrentDepth (0)
75, fCurrentPVCopyNo (fpTopPV? fpTopPV->GetCopyNo(): 0)
76, fpCurrentLV (fpTopPV? fpTopPV->GetLogicalVolume(): 0)
77, fpCurrentMaterial (fpCurrentLV? fpCurrentLV->GetMaterial(): 0)
78, fCurrentTransform (modelTransform)
79, fBaseFullPVPath (baseFullPVPath)
81, fAbort (false)
82, fCurtailDescent (false)
85, fNClippers (0)
87{
88 fType = "G4PhysicalVolumeModel";
89
90 if (!fpTopPV) {
91
92 // In some circumstances creating an "empty" G4PhysicalVolumeModel is
93 // allowed, so I have supressed the G4Exception below. If it proves to
94 // be a problem we might have to re-instate it, but it is unlikley to
95 // be used except by visualisation experts. See, for example, /vis/list,
96 // where it is used simply to get a list of G4AttDefs.
97 // G4Exception
98 // ("G4PhysicalVolumeModel::G4PhysicalVolumeModel",
99 // "modeling0010", FatalException, "Null G4PhysicalVolumeModel pointer.");
100
101 fTopPVName = "NULL";
102 fGlobalTag = "Empty";
103 fGlobalDescription = "G4PhysicalVolumeModel " + fGlobalTag;
104
105 } else {
106
107 fTopPVName = fpTopPV -> GetName ();
108 std::ostringstream oss;
109 oss << fpTopPV->GetName() << ':' << fpTopPV->GetCopyNo()
110 << " BasePath:" << fBaseFullPVPath;
111 fGlobalTag = oss.str();
112 fGlobalDescription = "G4PhysicalVolumeModel " + fGlobalTag;
114 }
115}
116
121
123(const std::vector<G4PhysicalVolumeNodeID>& path)
124{
126 for (const auto& node: path) {
127 PVNameCopyNoPath.push_back
129 (node.GetPhysicalVolume()->GetName(),node.GetCopyNo()));
130 }
131 return PVNameCopyNoPath;
132}
133
135(const std::vector<G4PhysicalVolumeNodeID>& path)
136// Converts to path string, e.g., " World 0 Envelope 0 Shape1 0".
137// Note leading space character.
138{
139 std::ostringstream oss;
140 oss << path;
141 return oss.str();
142}
143
145{
146 // To handle paramaterisations, set copy number and compute dimensions
147 // to get extent right
148 G4VPVParameterisation* pP = fpTopPV -> GetParameterisation ();
149 if (pP) {
150 fpTopPV -> SetCopyNo (fTopPVCopyNo);
151 G4VSolid* solid = pP -> ComputeSolid (fTopPVCopyNo, fpTopPV);
152 solid -> ComputeDimensions (pP, fTopPVCopyNo, fpTopPV);
153 }
154 if (fUseFullExtent) {
155 fExtent = fpTopPV -> GetLogicalVolume () -> GetSolid () -> GetExtent ();
156 } else {
157 // Calculate extent of *drawn* volumes, i.e., ignoring culled, e.g.,
158 // invisible volumes, by traversing the whole geometry hierarchy below
159 // this physical volume.
160 G4BoundingExtentScene beScene(this);
161 const G4int tempRequestedDepth = fRequestedDepth;
162 const G4Transform3D tempTransform = fTransform;
163 const G4ModelingParameters* tempMP = fpMP;
164 fRequestedDepth = -1; // Always search to all depths to define extent.
165 fTransform = G4Transform3D(); // Extent is in local cooridinates
167 (0, // No default vis attributes needed.
168 G4ModelingParameters::wf, // wireframe (not relevant for this).
169 true, // Global culling.
170 true, // Cull invisible volumes.
171 false, // Density culling.
172 0., // Density (not relevant if density culling false).
173 true, // Cull daughters of opaque mothers.
174 24); // No of sides (not relevant for this operation).
175 mParams.SetSpecialMeshRendering(true); // Avoids traversing parameterisations
176 fpMP = &mParams;
177 DescribeYourselfTo (beScene);
178 fExtent = beScene.GetBoundingExtent();
179 fpMP = tempMP;
180 fTransform = tempTransform;
181 fRequestedDepth = tempRequestedDepth;
182 }
183 G4double radius = fExtent.GetExtentRadius();
184 if (radius < 0.) { // Nothing in the scene - revert to top extent
185 fExtent = fpTopPV -> GetLogicalVolume () -> GetSolid () -> GetExtent ();
186 }
187 fExtent.Transform(fTransform);
188}
189
191(G4VGraphicsScene& sceneHandler)
192{
193 if (!fpTopPV) {
195 ("G4PhysicalVolumeModel::DescribeYourselfTo",
196 "modeling0012", FatalException, "No model.");
197 return; // Should never reach here, but keeps Coverity happy.
198 }
199
200 if (!fpMP) {
202 ("G4PhysicalVolumeModel::DescribeYourselfTo",
203 "modeling0013", FatalException, "No modeling parameters.");
204 return; // Should never reach here, but keeps Coverity happy.
205 }
206
207 fNClippers = 0;
208 G4DisplacedSolid* pSectionSolid = fpMP->GetSectionSolid();
209 G4DisplacedSolid* pCutawaySolid = fpMP->GetCutawaySolid();
211 if (pSectionSolid) fNClippers++;
212 if (pCutawaySolid) fNClippers++;
213 if (fNClippers > 1) {
215 ed << "More than one solid cutter/clipper:";
216 if (fpClippingSolid) ed << "\nclipper in force";
217 if (pSectionSolid) ed << "\nsectioner in force";
218 if (pCutawaySolid) ed << "\ncutaway in force";
219 G4Exception("G4PhysicalVolumeModel::DescribeSolid", "modeling0016", JustWarning, ed);
220 }
221
222 G4Transform3D startingTransformation = fTransform;
223
224 fNTouchables.clear(); // Keeps count of touchable drawn at each depth
225
227 (fpTopPV,
229 startingTransformation,
230 sceneHandler);
231
233 for (const auto& entry : fNTouchables) {
234 fTotalTouchables += entry.second;
235 }
236
237 // Reset or clear data...
238 fCurrentDepth = 0;
240 fCurrentPVCopyNo = fpTopPV->GetCopyNo();
241 fpCurrentLV = fpTopPV->GetLogicalVolume();
242 fpCurrentMaterial = fpCurrentLV? fpCurrentLV->GetMaterial(): 0;
244 fDrawnPVPath.clear();
245 fAbort = false;
246 fCurtailDescent = false;
247}
248
250{
251 if (fpCurrentPV) {
252 std::ostringstream o;
253 o << fpCurrentPV -> GetCopyNo ();
254 return fpCurrentPV -> GetName () + ":" + o.str();
255 }
256 else {
257 return "WARNING: NO CURRENT VOLUME - global tag is " + fGlobalTag;
258 }
259}
260
262{
263 return "G4PhysicalVolumeModel " + GetCurrentTag ();
264}
265
267(G4VPhysicalVolume* pVPV,
268 G4int requestedDepth,
269 const G4Transform3D& theAT,
270 G4VGraphicsScene& sceneHandler)
271{
272 // Visits geometry structure to a given depth (requestedDepth), starting
273 // at given physical volume with given starting transformation and
274 // describes volumes to the scene handler.
275 // requestedDepth < 0 (default) implies full visit.
276 // theAT is the Accumulated Transformation.
277
278 // Find corresponding logical volume and (later) solid, storing in
279 // local variables to preserve re-entrancy.
280 G4LogicalVolume* pLV = pVPV -> GetLogicalVolume ();
281 G4VSolid* pSol = nullptr;
282 G4Material* pMaterial = nullptr;
283
284 if (!(pVPV -> IsReplicated ())) {
285 // Non-replicated physical volume.
286 pSol = pLV -> GetSolid ();
287 pMaterial = pLV -> GetMaterial ();
288 DescribeAndDescend (pVPV, requestedDepth, pLV, pSol, pMaterial,
289 theAT, sceneHandler);
290 }
291 else {
292 // Replicated or parametrised physical volume.
293 EAxis axis;
294 G4int nReplicas;
295 G4double width;
297 G4bool consuming;
298 pVPV -> GetReplicationData (axis, nReplicas, width, offset, consuming);
299 G4int nBegin = 0;
300 G4int nEnd = nReplicas;
301 if (fCurrentDepth == 0) { // i.e., top volume
302 nBegin = fTopPVCopyNo; // Describe only one volume, namely the one
303 nEnd = nBegin + 1; // specified by the given copy number.
304 }
305 G4VPVParameterisation* pP = pVPV -> GetParameterisation ();
306 if (pP) { // Parametrised volume.
307 for (int n = nBegin; n < nEnd; n++) {
308 pSol = pP -> ComputeSolid (n, pVPV);
309 pP -> ComputeTransformation (n, pVPV);
310 pSol -> ComputeDimensions (pP, n, pVPV);
311 pVPV -> SetCopyNo (n);
313 // Create a touchable of current parent for ComputeMaterial.
314 // fFullPVPath has not been updated yet so at this point it
315 // corresponds to the parent.
317 pMaterial = pP -> ComputeMaterial (n, pVPV, &parentTouchable);
318 DescribeAndDescend (pVPV, requestedDepth, pLV, pSol, pMaterial,
319 theAT, sceneHandler);
320 }
321 }
322 else { // Plain replicated volume. From geometry_guide.txt...
323 // The replica's positions are claculated by means of a linear formula.
324 // Replication may occur along:
325 //
326 // o Cartesian axes (kXAxis,kYAxis,kZAxis)
327 //
328 // The replications, of specified width have coordinates of
329 // form (-width*(nReplicas-1)*0.5+n*width,0,0) where n=0.. nReplicas-1
330 // for the case of kXAxis, and are unrotated.
331 //
332 // o Radial axis (cylindrical polar) (kRho)
333 //
334 // The replications are cons/tubs sections, centred on the origin
335 // and are unrotated.
336 // They have radii of width*n+offset to width*(n+1)+offset
337 // where n=0..nReplicas-1
338 //
339 // o Phi axis (cylindrical polar) (kPhi)
340 // The replications are `phi sections' or wedges, and of cons/tubs form
341 // They have phi of offset+n*width to offset+(n+1)*width where
342 // n=0..nReplicas-1
343 //
344 pSol = pLV -> GetSolid ();
345 pMaterial = pLV -> GetMaterial ();
346 G4ThreeVector originalTranslation = pVPV -> GetTranslation ();
347 G4RotationMatrix* pOriginalRotation = pVPV -> GetRotation ();
348 G4double originalRMin = 0., originalRMax = 0.;
349 if (axis == kRho && pSol->GetEntityType() == "G4Tubs") {
350 originalRMin = ((G4Tubs*)pSol)->GetInnerRadius();
351 originalRMax = ((G4Tubs*)pSol)->GetOuterRadius();
352 }
353 G4bool visualisable = true;
354 for (int n = nBegin; n < nEnd; n++) {
355 G4ThreeVector translation; // Identity.
356 G4RotationMatrix rotation; // Identity - life enough for visualizing.
357 G4RotationMatrix* pRotation = 0;
358 switch (axis) {
359 default:
360 case kXAxis:
361 translation = G4ThreeVector (-width*(nReplicas-1)*0.5+n*width,0,0);
362 break;
363 case kYAxis:
364 translation = G4ThreeVector (0,-width*(nReplicas-1)*0.5+n*width,0);
365 break;
366 case kZAxis:
367 translation = G4ThreeVector (0,0,-width*(nReplicas-1)*0.5+n*width);
368 break;
369 case kRho:
370 if (pSol->GetEntityType() == "G4Tubs") {
371 ((G4Tubs*)pSol)->SetInnerRadius(width*n+offset);
372 ((G4Tubs*)pSol)->SetOuterRadius(width*(n+1)+offset);
373 } else {
374 if (fpMP->IsWarning())
375 G4warn <<
376 "G4PhysicalVolumeModel::VisitGeometryAndGetVisReps: WARNING:"
377 "\n built-in replicated volumes replicated in radius for "
378 << pSol->GetEntityType() <<
379 "-type\n solids (your solid \""
380 << pSol->GetName() <<
381 "\") are not visualisable."
382 << G4endl;
383 visualisable = false;
384 }
385 break;
386 case kPhi:
387 rotation.rotateZ (-(offset+(n+0.5)*width));
388 // Minus Sign because for the physical volume we need the
389 // coordinate system rotation.
390 pRotation = &rotation;
391 break;
392 }
393 pVPV -> SetTranslation (translation);
394 pVPV -> SetRotation (pRotation);
395 pVPV -> SetCopyNo (n);
397 if (visualisable) {
398 DescribeAndDescend (pVPV, requestedDepth, pLV, pSol, pMaterial,
399 theAT, sceneHandler);
400 }
401 }
402 // Restore originals...
403 pVPV -> SetTranslation (originalTranslation);
404 pVPV -> SetRotation (pOriginalRotation);
405 if (axis == kRho && pSol->GetEntityType() == "G4Tubs") {
406 ((G4Tubs*)pSol)->SetInnerRadius(originalRMin);
407 ((G4Tubs*)pSol)->SetOuterRadius(originalRMax);
408 }
409 }
410 }
411}
412
414(G4VPhysicalVolume* pVPV,
415 G4int requestedDepth,
416 G4LogicalVolume* pLV,
417 G4VSolid* pSol,
418 G4Material* pMaterial,
419 const G4Transform3D& theAT,
420 G4VGraphicsScene& sceneHandler)
421{
422 // Maintain useful data members...
423 fpCurrentPV = pVPV;
424 fCurrentPVCopyNo = pVPV->GetCopyNo();
425 fpCurrentLV = pLV;
426 fpCurrentMaterial = pMaterial;
427
428 // Create a nodeID for use below - note the "drawn" flag is true
429 G4int copyNo = fpCurrentPV->GetCopyNo();
430 auto nodeID = G4PhysicalVolumeNodeID
432
433 // Update full path of physical volumes...
434 fFullPVPath.push_back(nodeID);
435
436 const G4RotationMatrix objectRotation = pVPV -> GetObjectRotationValue ();
437 const G4ThreeVector& translation = pVPV -> GetTranslation ();
438 G4Transform3D theLT (G4Transform3D (objectRotation, translation));
439
440 // Compute the accumulated transformation...
441 // Note that top volume's transformation relative to the world
442 // coordinate system is specified in theAT == startingTransformation
443 // = fTransform (see DescribeYourselfTo), so first time through the
444 // volume's own transformation, which is only relative to its
445 // mother, i.e., not relative to the world coordinate system, should
446 // not be accumulated.
447 G4Transform3D theNewAT (theAT);
448 if (fCurrentDepth != 0) theNewAT = theAT * theLT;
449 fCurrentTransform = theNewAT;
450
451 const G4VisAttributes* pVisAttribs = pLV->GetVisAttributes();
452 // If the volume does not have any vis attributes, create it.
453 G4VisAttributes* tempVisAtts = nullptr;
454 if (!pVisAttribs) {
455 if (fpMP->GetDefaultVisAttributes()) {
456 tempVisAtts = new G4VisAttributes(*fpMP->GetDefaultVisAttributes());
457 } else {
458 tempVisAtts = new G4VisAttributes;
459 }
460 // The user may request /vis/viewer/set/colourByDensity.
461 if (fpMP->GetCBDAlgorithmNumber() == 1) {
462 // Algorithm 1: 3 parameters: Simple rainbow mapping.
463 if (fpMP->GetCBDParameters().size() != 3) {
464 G4Exception("G4PhysicalVolumeModelTouchable::DescribeAndDescend",
465 "modeling0014",
467 "Algorithm-parameter mismatch for Colour By Density");
468 } else {
469 const G4double d = pMaterial? pMaterial->GetDensity(): 0.;
470 const G4double d0 = fpMP->GetCBDParameters()[0]; // Invisible d < d0.
471 const G4double d1 = fpMP->GetCBDParameters()[1]; // Rainbow d0->d1->d2.
472 const G4double d2 = fpMP->GetCBDParameters()[2]; // Blue d > d2.
473 if (d < d0) { // Density < d0 is invisible.
474 tempVisAtts->SetVisibility(false);
475 } else { // Intermediate densities are on a spectrum.
476 G4double red, green, blue;
477 if (d < d1) {
478 red = (d1-d)/(d1-d0); green = (d-d0)/(d1-d0); blue = 0.;
479 } else if (d < d2) {
480 red = 0.; green = (d2-d)/(d2-d1); blue = (d-d1)/(d2-d1);
481 } else { // Density >= d2 is blue.
482 red = 0.; green = 0.; blue = 1.;
483 }
484 tempVisAtts->SetColour(G4Colour(red,green,blue));
485 }
486 }
487 } else if (fpMP->GetCBDAlgorithmNumber() == 2) {
488 // Algorithm 2
489 // ...etc.
490 }
491 pVisAttribs = tempVisAtts;
492 }
493 // From here, can assume pVisAttribs is a valid pointer. This is necessary
494 // because PreAddSolid needs a vis attributes object.
495
496 // Check if vis attributes are to be modified by a /vis/touchable/set/ command.
497 const auto& vams = fpMP->GetVisAttributesModifiers();
498 if (vams.size()) {
499 // OK, we have some VAMs (Vis Attributes Modifiers).
500 for (const auto& vam: vams) {
501 const auto& vamPath = vam.GetPVNameCopyNoPath();
502 if (vamPath.size() == fFullPVPath.size()) {
503 // OK, we have a size match.
504 // Check the volume name/copy number path.
505 auto iVAMNameCopyNo = vamPath.begin();
506 auto iPVNodeId = fFullPVPath.begin();
507 for (; iVAMNameCopyNo != vamPath.end(); ++iVAMNameCopyNo, ++iPVNodeId) {
508 if (!(
509 iVAMNameCopyNo->GetName() ==
510 iPVNodeId->GetPhysicalVolume()->GetName() &&
511 iVAMNameCopyNo->GetCopyNo() ==
512 iPVNodeId->GetPhysicalVolume()->GetCopyNo()
513 )) {
514 // This path element does NOT match.
515 break;
516 }
517 }
518 if (iVAMNameCopyNo == vamPath.end()) {
519 // OK, the paths match (the above loop terminated normally).
520 // Create a vis atts object for the modified vis atts.
521 // It is static so that we may return a reliable pointer to it.
522 static G4VisAttributes modifiedVisAtts;
523 // Initialise it with the current vis atts and reset the pointer.
524 modifiedVisAtts = *pVisAttribs;
525 pVisAttribs = &modifiedVisAtts;
526 const G4VisAttributes& transVisAtts = vam.GetVisAttributes();
527 switch (vam.GetVisAttributesSignifier()) {
529 modifiedVisAtts.SetVisibility(transVisAtts.IsVisible());
530 break;
532 modifiedVisAtts.SetDaughtersInvisible
533 (transVisAtts.IsDaughtersInvisible());
534 break;
536 modifiedVisAtts.SetColour(transVisAtts.GetColour());
537 break;
539 modifiedVisAtts.SetLineStyle(transVisAtts.GetLineStyle());
540 break;
542 modifiedVisAtts.SetLineWidth(transVisAtts.GetLineWidth());
543 break;
545 if (transVisAtts.IsForceDrawingStyle()) {
546 if (transVisAtts.GetForcedDrawingStyle() ==
548 modifiedVisAtts.SetForceWireframe(true);
549 }
550 }
551 break;
553 if (transVisAtts.IsForceDrawingStyle()) {
554 if (transVisAtts.GetForcedDrawingStyle() ==
556 modifiedVisAtts.SetForceSolid(true);
557 }
558 }
559 break;
561 if (transVisAtts.IsForceDrawingStyle()) {
562 if (transVisAtts.GetForcedDrawingStyle() ==
564 modifiedVisAtts.SetForceCloud(true);
565 }
566 }
567 break;
569 modifiedVisAtts.SetForceNumberOfCloudPoints
570 (transVisAtts.GetForcedNumberOfCloudPoints());
571 break;
573 if (transVisAtts.IsForceAuxEdgeVisible()) {
574 modifiedVisAtts.SetForceAuxEdgeVisible
575 (transVisAtts.IsForcedAuxEdgeVisible());
576 }
577 break;
579 modifiedVisAtts.SetForceLineSegmentsPerCircle
580 (transVisAtts.GetForcedLineSegmentsPerCircle());
581 break;
582 }
583 }
584 }
585 }
586 }
587
588 // Check for special mesh rendering
589 if (fpMP->IsSpecialMeshRendering()) {
590 G4bool potentialG4Mesh = false;
591 if (fpMP->GetSpecialMeshVolumes().empty()) {
592 // No volumes specified - all are potentially possible
593 potentialG4Mesh = true;
594 } else {
595 // Name and (optionally) copy number of container volume is specified
596 for (const auto& pvNameCopyNo: fpMP->GetSpecialMeshVolumes()) {
597 if (pVPV->GetName() == pvNameCopyNo.GetName()) {
598 // We have a name match
599 if (pvNameCopyNo.GetCopyNo() < 0) {
600 // Any copy number is OK
601 potentialG4Mesh = true;
602 } else {
603 if (pVPV->GetCopyNo() == pvNameCopyNo.GetCopyNo()) {
604 // We have a name and copy number match
605 potentialG4Mesh = true;
606 }
607 }
608 }
609 }
610 }
611 if (potentialG4Mesh) {
612 // Create - or at least attempt to create - a mesh. If it cannot be created
613 // out of this pVPV the type will be "invalid".
614 G4Mesh mesh(pVPV,theNewAT);
615 if (mesh.GetMeshType() != G4Mesh::invalid) {
616 // Create "artificial" nodeID to represent the replaced volumes
617 G4int artCopyNo = 0;
618 auto artPV = mesh.GetParameterisedVolume();
619 auto artDepth = fCurrentDepth + 1;
620 auto artNodeID = G4PhysicalVolumeNodeID(artPV,artCopyNo,artDepth);
621 fFullPVPath.push_back(artNodeID);
622 fDrawnPVPath.push_back(artNodeID);
623 sceneHandler.AddCompound(mesh);
624 fFullPVPath.pop_back();
625 fDrawnPVPath.pop_back();
626 delete tempVisAtts; // Needs cleaning up (Coverity warning!!)
627 return; // Mesh found and processed - nothing more to do.
628 } // else continue processing
629 }
630 }
631
632 // Make decision to draw...
633 G4bool thisToBeDrawn = true;
634
635 // There are various reasons why this volume
636 // might not be drawn...
637 G4bool culling = fpMP->IsCulling();
638 G4bool cullingInvisible = fpMP->IsCullingInvisible();
639 G4bool markedVisible
640 = pVisAttribs->IsVisible() && pVisAttribs->GetColour().GetAlpha() > 0;
641 G4bool cullingLowDensity = fpMP->IsDensityCulling();
642 G4double density = pMaterial? pMaterial->GetDensity(): 0;
643 G4double densityCut = fpMP -> GetVisibleDensity ();
644
645 // 1) Global culling is on....
646 if (culling) {
647 // 2) Culling of invisible volumes is on...
648 if (cullingInvisible) {
649 // 3) ...and the volume is marked not visible...
650 if (!markedVisible) thisToBeDrawn = false;
651 }
652 // 4) Or culling of low density volumes is on...
653 if (cullingLowDensity) {
654 // 5) ...and density is less than cut value...
655 if (density < densityCut) thisToBeDrawn = false;
656 }
657 }
658 // 6) The user has asked for all further traversing to be aborted...
659 if (fAbort) thisToBeDrawn = false;
660
661 // Set "drawn" flag (it was true by default) - thisToBeDrawn may be false
662 nodeID.SetDrawn(thisToBeDrawn);
663
664 if (thisToBeDrawn) {
665
666 // Update path of drawn physical volumes...
667 fDrawnPVPath.push_back(nodeID);
668
669 if (fpMP->IsExplode() && fDrawnPVPath.size() == 1) {
670 // For top-level drawn volumes, explode along radius...
671 G4Transform3D centering = G4Translate3D(fpMP->GetExplodeCentre());
672 G4Transform3D centred = centering.inverse() * theNewAT;
673 G4Scale3D oldScale;
674 G4Rotate3D oldRotation;
675 G4Translate3D oldTranslation;
676 centred.getDecomposition(oldScale, oldRotation, oldTranslation);
677 G4double explodeFactor = fpMP->GetExplodeFactor();
678 G4Translate3D newTranslation =
679 G4Translate3D(explodeFactor * oldTranslation.dx(),
680 explodeFactor * oldTranslation.dy(),
681 explodeFactor * oldTranslation.dz());
682 theNewAT = centering * newTranslation * oldRotation * oldScale;
683 }
684
685 auto fullDepth = fCurrentDepth + (G4int)fBaseFullPVPath.size();
686 fNTouchables[fullDepth]++; // Increment for every touchable drawn at each depth
687
688 DescribeSolid (theNewAT, pSol, pVisAttribs, sceneHandler);
689
690 }
691
692 // Make decision to draw daughters, if any. There are various
693 // reasons why daughters might not be drawn...
694
695 // First, reasons that do not depend on culling policy...
696 G4int nDaughters = (G4int)pLV->GetNoDaughters();
697 G4bool daughtersToBeDrawn = true;
698 // 1) There are no daughters...
699 if (!nDaughters) daughtersToBeDrawn = false;
700 // 2) We are at the limit if requested depth...
701 else if (requestedDepth == 0) daughtersToBeDrawn = false;
702 // 3) The user has asked for all further traversing to be aborted...
703 else if (fAbort) daughtersToBeDrawn = false;
704 // 4) The user has asked that the descent be curtailed...
705 else if (fCurtailDescent) daughtersToBeDrawn = false;
706
707 // Now, reasons that depend on culling policy...
708 else {
709 G4bool daughtersInvisible = pVisAttribs->IsDaughtersInvisible();
710 // Culling of covered daughters request. This is computed in
711 // G4VSceneHandler::CreateModelingParameters() depending on view
712 // parameters...
713 G4bool cullingCovered = fpMP->IsCullingCovered();
714 G4bool surfaceDrawing =
715 fpMP->GetDrawingStyle() == G4ModelingParameters::hsr ||
716 fpMP->GetDrawingStyle() == G4ModelingParameters::hlhsr;
717 if (pVisAttribs->IsForceDrawingStyle()) {
718 switch (pVisAttribs->GetForcedDrawingStyle()) {
719 default:
720 case G4VisAttributes::wireframe: surfaceDrawing = false; break;
721 case G4VisAttributes::solid: surfaceDrawing = true; break;
722 }
723 }
724 G4bool opaque = pVisAttribs->GetColour().GetAlpha() >= 1.;
725 // 5) Global culling is on....
726 if (culling) {
727 // 6) ..and culling of invisible volumes is on...
728 if (cullingInvisible) {
729 // 7) ...and the mother requests daughters invisible
730 if (daughtersInvisible) daughtersToBeDrawn = false;
731 }
732 // 8) Or culling of covered daughters is requested...
733 if (cullingCovered) {
734 // 9) ...and surface drawing is operating...
735 if (surfaceDrawing) {
736 // 10) ...but only if mother is visible...
737 if (thisToBeDrawn) {
738 // 11) ...and opaque...
739 if (opaque) daughtersToBeDrawn = false;
740 }
741 }
742 }
743 }
744 }
745
746 if (daughtersToBeDrawn) {
747 for (G4int iDaughter = 0; iDaughter < nDaughters; iDaughter++) {
748 // Store daughter pVPV in local variable ready for recursion...
749 G4VPhysicalVolume* pDaughterVPV = pLV -> GetDaughter (iDaughter);
750 // Descend the geometry structure recursively...
753 (pDaughterVPV, requestedDepth - 1, theNewAT, sceneHandler);
755 }
756 }
757
758 // Clean up
759 delete tempVisAtts;
760
761 // Reset for normal descending of next volume at this level...
762 fCurtailDescent = false;
763
764 // Pop item from paths physical volumes...
765 fFullPVPath.pop_back();
766 if (thisToBeDrawn) {
767 fDrawnPVPath.pop_back();
768 }
769}
770
771namespace
772{
773 G4bool SubtractionBoundingLimits(const G4VSolid* target)
774 {
775 // Algorithm from G4SubtractionSolid::BoundingLimits
776 // Since it is unclear how the shape of the first solid will be changed
777 // after subtraction, just return its original bounding box.
778 G4ThreeVector pMin, pMax;
779 const auto& pSolA = target;
780 pSolA->BoundingLimits(pMin,pMax);
781 // Check correctness of the bounding box
782 if (pMin.x() >= pMax.x() || pMin.y() >= pMax.y() || pMin.z() >= pMax.z()) {
783 // Bad bounding box (min >= max)
784 // This signifies a subtraction of non-intersecting volumes
785 return false;
786 }
787 return true;
788 }
789
790 G4bool IntersectionBoundingLimits(const G4VSolid* target, const G4DisplacedSolid* intersector)
791 {
792 // Algorithm from G4IntersectionSolid::BoundingLimits
793 G4ThreeVector pMin, pMax;
794 G4ThreeVector minA,maxA, minB,maxB;
795 const auto& pSolA = target;
796 const auto& pSolB = intersector;
797 pSolA->BoundingLimits(minA,maxA);
798 pSolB->BoundingLimits(minB,maxB);
799 pMin.set(std::max(minA.x(),minB.x()),
800 std::max(minA.y(),minB.y()),
801 std::max(minA.z(),minB.z()));
802 pMax.set(std::min(maxA.x(),maxB.x()),
803 std::min(maxA.y(),maxB.y()),
804 std::min(maxA.z(),maxB.z()));
805 if (pMin.x() >= pMax.x() || pMin.y() >= pMax.y() || pMin.z() >= pMax.z()) {
806 // Bad bounding box (min >= max)
807 // This signifies a subtraction of non-intersecting volumes
808 return false;
809 }
810 return true;
811 }
812}
813
815(const G4Transform3D& theAT,
816 G4VSolid* pSol,
817 const G4VisAttributes* pVisAttribs,
818 G4VGraphicsScene& sceneHandler)
819{
820 G4DisplacedSolid* pSectionSolid = fpMP->GetSectionSolid();
821 G4DisplacedSolid* pCutawaySolid = fpMP->GetCutawaySolid();
822
824
825 // Normal case - no clipping, etc. - or, illegally, more than one of those
826 sceneHandler.PreAddSolid (theAT, *pVisAttribs);
827 pSol -> DescribeYourselfTo (sceneHandler); // Standard treatment.
828 sceneHandler.PostAddSolid ();
829
830 } else { // fNClippers == 1
831
832 G4VSolid* pResultantSolid = nullptr;
833 G4DisplacedSolid* pDisplacedSolid = nullptr;
834
835 if (fpClippingSolid) {
836 pDisplacedSolid = new G4DisplacedSolid("clipper", fpClippingSolid, theAT.inverse());
837 switch (fClippingMode) {
838 case subtraction:
839 if (SubtractionBoundingLimits(pSol)) {
840 pResultantSolid = new G4SubtractionSolid
841 ("subtracted_clipped_solid", pSol, pDisplacedSolid);
842 }
843 break;
844 case intersection:
845 if (IntersectionBoundingLimits(pSol, pDisplacedSolid)) {
846 pResultantSolid = new G4IntersectionSolid
847 ("intersected_clipped_solid", pSol, pDisplacedSolid);
848 }
849 break;
850 }
851
852 } else if (pSectionSolid) {
853 pDisplacedSolid = new G4DisplacedSolid("intersector", pSectionSolid, theAT.inverse());
854 if (IntersectionBoundingLimits(pSol, pDisplacedSolid)) {
855 pResultantSolid = new G4IntersectionSolid("sectioned_solid", pSol, pDisplacedSolid);
856 }
857
858 } else if (pCutawaySolid) {
859 pDisplacedSolid = new G4DisplacedSolid("cutaway", pCutawaySolid, theAT.inverse());
860 switch (fpMP->GetCutawayMode()) {
862 if (SubtractionBoundingLimits(pSol)) {
863 pResultantSolid = new G4SubtractionSolid("cutaway_solid", pSol, pDisplacedSolid);
864 }
865 break;
867 if (IntersectionBoundingLimits(pSol, pDisplacedSolid)) {
868 pResultantSolid = new G4IntersectionSolid("cutaway_solid", pSol, pDisplacedSolid);
869 }
870 break;
871 }
872 }
873
874 if (pResultantSolid) {
875 sceneHandler.PreAddSolid (theAT, *pVisAttribs);
876 pResultantSolid -> DescribeYourselfTo (sceneHandler);
877 sceneHandler.PostAddSolid ();
878 }
879
880 delete pResultantSolid;
881 delete pDisplacedSolid;
882 }
883}
884
886{
887// Not easy to see how to validate this sort of model. Previously there was
888// a check that a volume of the same name (fTopPVName) existed somewhere in
889// the geometry tree but under some circumstances this consumed lots of CPU
890// time. Instead, let us simply check that the volume (fpTopPV) exists in the
891// physical volume store.
892 const auto& pvStore = G4PhysicalVolumeStore::GetInstance();
893 auto iterator = find(pvStore->begin(),pvStore->end(),fpTopPV);
894 if (iterator == pvStore->end()) {
895 if (warn) {
897 ed << "Attempt to validate a volume that is no longer in the physical volume store.";
898 G4Exception("G4PhysicalVolumeModel::Validate", "modeling0015", JustWarning, ed);
899 }
900 return false;
901 } else {
902 return true;
903 }
904}
905
906const std::map<G4String,G4AttDef>* G4PhysicalVolumeModel::GetAttDefs() const
907{
908 G4bool isNew;
909 std::map<G4String,G4AttDef>* store
910 = G4AttDefStore::GetInstance("G4PhysicalVolumeModel", isNew);
911 if (isNew) {
912 (*store)["PVPath"] =
913 G4AttDef("PVPath","Physical Volume Path","Physics","","G4String");
914 (*store)["BasePVPath"] =
915 G4AttDef("BasePVPath","Base Physical Volume Path","Physics","","G4String");
916 (*store)["LVol"] =
917 G4AttDef("LVol","Logical Volume","Physics","","G4String");
918 (*store)["Solid"] =
919 G4AttDef("Solid","Solid Name","Physics","","G4String");
920 (*store)["EType"] =
921 G4AttDef("EType","Entity Type","Physics","","G4String");
922 (*store)["DmpSol"] =
923 G4AttDef("DmpSol","Dump of Solid properties","Physics","","G4String");
924 (*store)["LocalTrans"] =
925 G4AttDef("LocalTrans","Local transformation of volume","Physics","","G4String");
926 (*store)["LocalExtent"] =
927 G4AttDef("LocalExtent","Local extent of volume","Physics","","G4String");
928 (*store)["GlobalTrans"] =
929 G4AttDef("GlobalTrans","Global transformation of volume","Physics","","G4String");
930 (*store)["GlobalExtent"] =
931 G4AttDef("GlobalExtent","Global extent of volume","Physics","","G4String");
932 (*store)["Material"] =
933 G4AttDef("Material","Material Name","Physics","","G4String");
934 (*store)["Density"] =
935 G4AttDef("Density","Material Density","Physics","G4BestUnit","G4double");
936 (*store)["State"] =
937 G4AttDef("State","Material State (enum undefined,solid,liquid,gas)","Physics","","G4String");
938 (*store)["Radlen"] =
939 G4AttDef("Radlen","Material Radiation Length","Physics","G4BestUnit","G4double");
940 (*store)["Region"] =
941 G4AttDef("Region","Cuts Region","Physics","","G4String");
942 (*store)["RootRegion"] =
943 G4AttDef("RootRegion","Root Region (0/1 = false/true)","Physics","","G4bool");
944 }
945 return store;
946}
947
948static std::ostream& operator<< (std::ostream& o, const G4Transform3D t)
949{
950 using namespace std;
951
952 G4Scale3D sc;
953 G4Rotate3D r;
954 G4Translate3D tl;
955 t.getDecomposition(sc, r, tl);
956
957 const int w = 10;
958
959 // Transformation itself
960 o << setw(w) << t.xx() << setw(w) << t.xy() << setw(w) << t.xz() << setw(w) << t.dx() << endl;
961 o << setw(w) << t.yx() << setw(w) << t.yy() << setw(w) << t.yz() << setw(w) << t.dy() << endl;
962 o << setw(w) << t.zx() << setw(w) << t.zy() << setw(w) << t.zz() << setw(w) << t.dz() << endl;
963
964 // Translation
965 o << "= translation:" << endl;
966 o << setw(w) << tl.dx() << setw(w) << tl.dy() << setw(w) << tl.dz() << endl;
967
968 // Rotation
969 o << "* rotation:" << endl;
970 o << setw(w) << r.xx() << setw(w) << r.xy() << setw(w) << r.xz() << endl;
971 o << setw(w) << r.yx() << setw(w) << r.yy() << setw(w) << r.yz() << endl;
972 o << setw(w) << r.zx() << setw(w) << r.zy() << setw(w) << r.zz() << endl;
973
974 // Scale
975 o << "* scale:" << endl;
976 o << setw(w) << sc.xx() << setw(w) << sc.yy() << setw(w) << sc.zz() << endl;
977
978 // Transformed axes
979 o << "Transformed axes:" << endl;
980 o << "x': " << r * G4Vector3D(1., 0., 0.) << endl;
981 o << "y': " << r * G4Vector3D(0., 1., 0.) << endl;
982 o << "z': " << r * G4Vector3D(0., 0., 1.) << endl;
983
984 return o;
985}
986
987std::vector<G4AttValue>* G4PhysicalVolumeModel::CreateCurrentAttValues() const
988{
989 std::vector<G4AttValue>* values = new std::vector<G4AttValue>;
990
991 if (!fpCurrentLV) {
993 ("G4PhysicalVolumeModel::CreateCurrentAttValues",
994 "modeling0004",
996 "Current logical volume not defined.");
997 return values;
998 }
999
1000 std::ostringstream oss; oss << fFullPVPath;
1001 values->push_back(G4AttValue("PVPath", oss.str(),""));
1002
1003 oss.str(""); oss << fBaseFullPVPath;
1004 values->push_back(G4AttValue("BasePVPath", oss.str(),""));
1005
1006 values->push_back(G4AttValue("LVol", fpCurrentLV->GetName(),""));
1007 G4VSolid* pSol = fpCurrentLV->GetSolid();
1008
1009 values->push_back(G4AttValue("Solid", pSol->GetName(),""));
1010
1011 values->push_back(G4AttValue("EType", pSol->GetEntityType(),""));
1012
1013 oss.str(""); oss << '\n' << *pSol;
1014 values->push_back(G4AttValue("DmpSol", oss.str(),""));
1015
1016 const G4RotationMatrix localRotation = fpCurrentPV->GetObjectRotationValue();
1017 const G4ThreeVector& localTranslation = fpCurrentPV->GetTranslation();
1018 oss.str(""); oss << '\n' << G4Transform3D(localRotation,localTranslation);
1019 values->push_back(G4AttValue("LocalTrans", oss.str(),""));
1020
1021 oss.str(""); oss << '\n' << pSol->GetExtent() << std::endl;
1022 values->push_back(G4AttValue("LocalExtent", oss.str(),""));
1023
1024 oss.str(""); oss << '\n' << fCurrentTransform;
1025 values->push_back(G4AttValue("GlobalTrans", oss.str(),""));
1026
1027 oss.str(""); oss << '\n' << (pSol->GetExtent()).Transform(fCurrentTransform) << std::endl;
1028 values->push_back(G4AttValue("GlobalExtent", oss.str(),""));
1029
1030 G4String matName = fpCurrentMaterial? fpCurrentMaterial->GetName(): G4String("No material");
1031 values->push_back(G4AttValue("Material", matName,""));
1032
1033 G4double matDensity = fpCurrentMaterial? fpCurrentMaterial->GetDensity(): 0.;
1034 values->push_back(G4AttValue("Density", G4BestUnit(matDensity,"Volumic Mass"),""));
1035
1037 oss.str(""); oss << matState;
1038 values->push_back(G4AttValue("State", oss.str(),""));
1039
1040 G4double matRadlen = fpCurrentMaterial? fpCurrentMaterial->GetRadlen(): 0.;
1041 values->push_back(G4AttValue("Radlen", G4BestUnit(matRadlen,"Length"),""));
1042
1043 G4Region* region = fpCurrentLV->GetRegion();
1044 G4String regionName = region? region->GetName(): G4String("No region");
1045 values->push_back(G4AttValue("Region", regionName,""));
1046
1047 oss.str(""); oss << fpCurrentLV->IsRootRegion();
1048 values->push_back(G4AttValue("RootRegion", oss.str(),""));
1049
1050 return values;
1051}
1052
1053G4bool G4PhysicalVolumeModel::G4PhysicalVolumeNodeID::operator<
1055{
1056 if (fpPV < right.fpPV) return true;
1057 if (fpPV == right.fpPV) {
1058 if (fCopyNo < right.fCopyNo) return true;
1059 if (fCopyNo == right.fCopyNo)
1060 return fNonCulledDepth < right.fNonCulledDepth;
1061 }
1062 return false;
1063}
1064
1065G4bool G4PhysicalVolumeModel::G4PhysicalVolumeNodeID::operator!=
1067{
1068 if (fpPV != right.fpPV ||
1069 fCopyNo != right.fCopyNo ||
1070 fNonCulledDepth != right.fNonCulledDepth ||
1071 fTransform != right.fTransform ||
1072 fDrawn != right.fDrawn) return true;
1073 return false;
1074}
1075
1076std::ostream& operator<<
1077 (std::ostream& os, const G4PhysicalVolumeModel::G4PhysicalVolumeNodeID& node)
1078{
1079 G4VPhysicalVolume* pPV = node.GetPhysicalVolume();
1080 if (pPV) {
1081 os << pPV->GetName()
1082 << ' ' << node.GetCopyNo()
1083// << '[' << node.GetNonCulledDepth() << ']'
1084// << ':' << node.GetTransform()
1085 ;
1086// os << " (";
1087// if (!node.GetDrawn()) os << "not ";
1088// os << "drawn)";
1089 } else {
1090 os << " (Null PV node)";
1091 }
1092 return os;
1093}
1094
1095std::ostream& operator<<
1096(std::ostream& os, const std::vector<G4PhysicalVolumeModel::G4PhysicalVolumeNodeID>& path)
1097{
1098 if (path.empty()) {
1099 os << " TOP";
1100 } else {
1101 for (const auto& nodeID: path) {
1102 os << ' ' << nodeID;
1103 }
1104 }
1105 return os;
1106}
1107
1109(const std::vector<G4PhysicalVolumeNodeID>& fullPVPath):
1110 fFullPVPath(fullPVPath) {}
1111
1113{
1114 size_t i = fFullPVPath.size() - depth - 1;
1115 if (i >= fFullPVPath.size()) {
1116 G4Exception("G4PhysicalVolumeModelTouchable::GetTranslation",
1117 "modeling0005",
1119 "Index out of range. Asking for non-existent depth");
1120 }
1121 static G4ThreeVector tempTranslation;
1122 tempTranslation = fFullPVPath[i].GetTransform().getTranslation();
1123 return tempTranslation;
1124}
1125
1127{
1128 size_t i = fFullPVPath.size() - depth - 1;
1129 if (i >= fFullPVPath.size()) {
1130 G4Exception("G4PhysicalVolumeModelTouchable::GetRotation",
1131 "modeling0006",
1133 "Index out of range. Asking for non-existent depth");
1134 }
1135 static G4RotationMatrix tempRotation;
1136 tempRotation = fFullPVPath[i].GetTransform().getRotation();
1137 return &tempRotation;
1138}
1139
1141{
1142 size_t i = fFullPVPath.size() - depth - 1;
1143 if (i >= fFullPVPath.size()) {
1144 G4Exception("G4PhysicalVolumeModelTouchable::GetVolume",
1145 "modeling0007",
1147 "Index out of range. Asking for non-existent depth");
1148 }
1149 return fFullPVPath[i].GetPhysicalVolume();
1150}
1151
1153{
1154 size_t i = fFullPVPath.size() - depth - 1;
1155 if (i >= fFullPVPath.size()) {
1156 G4Exception("G4PhysicalVolumeModelTouchable::GetSolid",
1157 "modeling0008",
1159 "Index out of range. Asking for non-existent depth");
1160 }
1161 return fFullPVPath[i].GetPhysicalVolume()->GetLogicalVolume()->GetSolid();
1162}
1163
1165{
1166 size_t i = fFullPVPath.size() - depth - 1;
1167 if (i >= fFullPVPath.size()) {
1168 G4Exception("G4PhysicalVolumeModelTouchable::GetReplicaNumber",
1169 "modeling0009",
1171 "Index out of range. Asking for non-existent depth");
1172 }
1173 return fFullPVPath[i].GetCopyNo();
1174}
@ JustWarning
@ FatalException
@ FatalErrorInArgument
void G4Exception(const char *originOfException, const char *exceptionCode, G4ExceptionSeverity severity, const char *description)
std::ostringstream G4ExceptionDescription
G4ThreadLocal T * G4GeomSplitter< T >::offset
G4State
@ kStateUndefined
CLHEP::HepRotation G4RotationMatrix
#define G4warn
Definition G4Scene.cc:41
#define G4BestUnit(a, b)
CLHEP::Hep3Vector G4ThreeVector
HepGeom::Transform3D G4Transform3D
HepGeom::Translate3D G4Translate3D
HepGeom::Scale3D G4Scale3D
HepGeom::Rotate3D G4Rotate3D
double G4double
Definition G4Types.hh:83
bool G4bool
Definition G4Types.hh:86
int G4int
Definition G4Types.hh:85
HepGeom::Vector3D< G4double > G4Vector3D
Definition G4Vector3D.hh:34
#define G4endl
Definition G4ios.hh:67
double z() const
double x() const
double y() const
void set(double x, double y, double z)
HepRotation & rotateZ(double delta)
Definition Rotation.cc:87
const G4VisExtent & GetBoundingExtent() const
G4double GetAlpha() const
Definition G4Colour.hh:173
void BoundingLimits(G4ThreeVector &pMin, G4ThreeVector &pMax) const override
const G4VisAttributes * GetVisAttributes() const
std::size_t GetNoDaughters() const
G4double GetDensity() const
G4VPhysicalVolume * GetParameterisedVolume() const
Definition G4Mesh.hh:74
MeshType GetMeshType() const
Definition G4Mesh.hh:75
@ invalid
Definition G4Mesh.hh:52
std::vector< PVNameCopyNo > PVNameCopyNoPath
void SetSpecialMeshRendering(G4bool)
const G4RotationMatrix * GetRotation(G4int depth) const
G4PhysicalVolumeModelTouchable(const std::vector< G4PhysicalVolumeNodeID > &fullPVPath)
const G4ThreeVector & GetTranslation(G4int depth) const
void DescribeAndDescend(G4VPhysicalVolume *, G4int requestedDepth, G4LogicalVolume *, G4VSolid *, G4Material *, const G4Transform3D &, G4VGraphicsScene &)
std::vector< G4PhysicalVolumeNodeID > fFullPVPath
void VisitGeometryAndGetVisReps(G4VPhysicalVolume *, G4int requestedDepth, const G4Transform3D &, G4VGraphicsScene &)
std::vector< G4AttValue > * CreateCurrentAttValues() const
G4PhysicalVolumeModel(G4VPhysicalVolume *=0, G4int requestedDepth=UNLIMITED, const G4Transform3D &modelTransformation=G4Transform3D(), const G4ModelingParameters *=0, G4bool useFullExtent=false, const std::vector< G4PhysicalVolumeNodeID > &baseFullPVPath=std::vector< G4PhysicalVolumeNodeID >())
std::vector< G4PhysicalVolumeNodeID > fDrawnPVPath
virtual void DescribeSolid(const G4Transform3D &theAT, G4VSolid *pSol, const G4VisAttributes *pVisAttribs, G4VGraphicsScene &sceneHandler)
static G4String GetPVNamePathString(const std::vector< G4PhysicalVolumeNodeID > &)
G4String GetCurrentDescription() const
void DescribeYourselfTo(G4VGraphicsScene &)
std::vector< G4PhysicalVolumeNodeID > fBaseFullPVPath
G4VPhysicalVolume * fpCurrentPV
std::map< G4int, G4int > fNTouchables
const std::map< G4String, G4AttDef > * GetAttDefs() const
static G4ModelingParameters::PVNameCopyNoPath GetPVNameCopyNoPath(const std::vector< G4PhysicalVolumeNodeID > &)
static G4PhysicalVolumeStore * GetInstance()
const G4String & GetName() const
virtual void AddCompound(const G4VTrajectory &)=0
virtual void PostAddSolid()=0
virtual void PreAddSolid(const G4Transform3D &objectTransformation, const G4VisAttributes &visAttribs)=0
G4VisExtent fExtent
Definition G4VModel.hh:97
G4String fGlobalDescription
Definition G4VModel.hh:96
const G4VisExtent & GetExtent() const
G4String fType
Definition G4VModel.hh:94
const G4ModelingParameters * fpMP
Definition G4VModel.hh:98
G4VModel(const G4ModelingParameters *=0)
Definition G4VModel.cc:37
G4String fGlobalTag
Definition G4VModel.hh:95
virtual G4int GetCopyNo() const =0
const G4String & GetName() const
G4String GetName() const
virtual G4VisExtent GetExtent() const
Definition G4VSolid.cc:697
virtual void BoundingLimits(G4ThreeVector &pMin, G4ThreeVector &pMax) const
Definition G4VSolid.cc:680
virtual G4GeometryType GetEntityType() const =0
G4int GetForcedNumberOfCloudPoints() const
G4double GetLineWidth() const
G4bool IsDaughtersInvisible() const
void SetColour(const G4Colour &)
void SetVisibility(G4bool=true)
void SetForceAuxEdgeVisible(G4bool=true)
void SetForceCloud(G4bool=true)
G4int GetForcedLineSegmentsPerCircle() const
void SetForceWireframe(G4bool=true)
void SetLineWidth(G4double)
LineStyle GetLineStyle() const
const G4Colour & GetColour() const
G4bool IsVisible() const
G4bool IsForceAuxEdgeVisible() const
G4bool IsForcedAuxEdgeVisible() const
ForcedDrawingStyle GetForcedDrawingStyle() const
void SetForceSolid(G4bool=true)
void SetLineStyle(LineStyle)
void SetForceLineSegmentsPerCircle(G4int nSegments)
void SetDaughtersInvisible(G4bool=true)
void SetForceNumberOfCloudPoints(G4int nPoints)
G4bool IsForceDrawingStyle() const
double dy() const
double zz() const
double yz() const
double dz() const
double dx() const
void getDecomposition(Scale3D &scale, Rotate3D &rotation, Translate3D &translation) const
double xy() const
double zx() const
double yx() const
Transform3D inverse() const
double zy() const
double xx() const
double yy() const
double xz() const
EAxis
Definition geomdefs.hh:54
@ kPhi
Definition geomdefs.hh:60
@ kYAxis
Definition geomdefs.hh:56
@ kXAxis
Definition geomdefs.hh:55
@ kZAxis
Definition geomdefs.hh:57
@ kRho
Definition geomdefs.hh:58
std::map< G4String, G4AttDef > * GetInstance(const G4String &storeKey, G4bool &isNew)