#include "TGeoManager.h"
#include "TGeoCache.h"
#include "TGeoMatrix.h"
#include "TGeoShape.h"
#include "TGeoVolume.h"
#include "TVirtualGeoPainter.h"
#include "TGeoPhysicalNode.h"
ClassImp(TGeoPhysicalNode)
TGeoPhysicalNode::TGeoPhysicalNode()
{
fLevel = 0;
fMatrices = 0;
fNodes = 0;
fMatrixOrig = 0;
SetVisibility(kTRUE);
SetVisibleFull(kFALSE);
SetIsVolAtt(kTRUE);
SetAligned(kFALSE);
}
TGeoPhysicalNode::TGeoPhysicalNode(const char *path)
{
if (!strlen(path)) {
Error("ctor", "path not valid");
return;
}
fLevel = 0;
fMatrices = new TObjArray(30);
fNodes = new TObjArray(30);
fMatrixOrig = 0;
SetPath(path);
SetVisibility(kTRUE);
SetVisibleFull(kFALSE);
SetIsVolAtt(kTRUE);
SetAligned(kFALSE);
}
TGeoPhysicalNode::TGeoPhysicalNode(const TGeoPhysicalNode& gpn) :
TObject(gpn),
TAttLine(gpn),
fLevel(gpn.fLevel),
fMatrices(gpn.fMatrices),
fNodes(gpn.fNodes),
fMatrixOrig(gpn.fMatrixOrig)
{
}
TGeoPhysicalNode& TGeoPhysicalNode::operator=(const TGeoPhysicalNode& gpn)
{
if(this!=&gpn) {
TObject::operator=(gpn);
TAttLine::operator=(gpn);
fLevel=gpn.fLevel;
fMatrices=gpn.fMatrices;
fNodes=gpn.fNodes;
fMatrixOrig=gpn.fMatrixOrig;
}
return *this;
}
TGeoPhysicalNode::~TGeoPhysicalNode()
{
if (fMatrices) {
fMatrices->Delete();
delete fMatrices;
}
if (fNodes) delete fNodes;
if (fMatrixOrig) delete fMatrixOrig;
}
void TGeoPhysicalNode::Align(TGeoMatrix *newmat, TGeoShape *newshape, Bool_t check)
{
if (!newmat && !newshape) return;
if (TGeoManager::IsLocked()) {
Error("Align", "Not performed. Geometry in LOCKED mode !");
return;
}
TGeoNode *node = GetNode();
if (node->IsOffset()) {
Error("Align", "Cannot align division nodes: %s\n",node->GetName());
return;
}
TGeoNode *nnode = 0;
TGeoVolume *vm = GetVolume(0);
TGeoVolume *vd = 0;
Int_t i,id;
if (!IsAligned()) {
for (i=0; i<fLevel; i++) {
node = GetNode(i+1);
id = vm->GetIndex(node);
if (id < 0) {
Error("Align","cannot align node %s",GetNode(i+1)->GetName());
return;
}
vd = node->GetVolume()->CloneVolume();
nnode = node->MakeCopyNode();
nnode->SetVolume(vd);
nnode->SetMotherVolume(vm);
vm->GetNodes()->RemoveAt(id);
vm->GetNodes()->AddAt(nnode,id);
fNodes->RemoveAt(i+1);
fNodes->AddAt(nnode,i+1);
vm = vd;
}
} else {
nnode = GetNode();
}
TGeoNodeMatrix *aligned = (TGeoNodeMatrix*)nnode;
vm = nnode->GetMotherVolume();
vd = nnode->GetVolume();
if (newmat) {
if (!newmat->IsRegistered()) newmat->RegisterYourself();
aligned->SetMatrix(newmat);
TGeoHMatrix *global = GetMatrix();
TGeoHMatrix *up = GetMatrix(fLevel-1);
*global = up;
global->Multiply(newmat);
}
if (newshape) vd->SetShape(newshape);
for (i=fLevel-1; i>0; i--) {
vd = GetVolume(i);
if (!vd->IsAssembly()) break;
vd->GetShape()->ComputeBBox();
if (vd->GetVoxels()) vd->GetVoxels()->SetNeedRebuild();
}
TGeoVoxelFinder *voxels = vm->GetVoxels();
if (voxels) voxels->SetNeedRebuild();
if (check) vm->CheckOverlaps();
gGeoManager->CdTop();
SetAligned(kTRUE);
}
void TGeoPhysicalNode::cd() const
{
}
void TGeoPhysicalNode::Draw(Option_t * )
{
}
TGeoNode *TGeoPhysicalNode::GetMother(Int_t levup) const
{
Int_t ind = fLevel-levup;
if (ind<0) return 0;
return (TGeoNode*)fNodes->UncheckedAt(ind);
}
TGeoHMatrix *TGeoPhysicalNode::GetMatrix(Int_t level) const
{
if (level<0) return (TGeoHMatrix*)fMatrices->UncheckedAt(fLevel);
if (level>fLevel) return 0;
return (TGeoHMatrix*)fMatrices->UncheckedAt(level);
}
const char *TGeoPhysicalNode::GetName() const
{
static char *pNodeName=0;
static Int_t maxNodes = 0;
if (!maxNodes) {
maxNodes= 500;
pNodeName = new char[maxNodes];
}
pNodeName[0] = 0;
Int_t n = 0;
for (Int_t level=0;level<=fLevel; level++) {
const char *name = GetNode(level)->GetName();
Int_t nch = strlen(name);
if (n+nch+2 > maxNodes) {
maxNodes = 2*(n+nch+2);
delete [] pNodeName;
pNodeName = new char[maxNodes];
return GetName();
}
sprintf(pNodeName+n,"/%s",name);
n += nch+1;
}
return pNodeName;
}
TGeoNode *TGeoPhysicalNode::GetNode(Int_t level) const
{
if (level<0) return (TGeoNode*)fNodes->UncheckedAt(fLevel);
if (level>fLevel) return 0;
return (TGeoNode*)fNodes->UncheckedAt(level);
}
TGeoVolume *TGeoPhysicalNode::GetVolume(Int_t level) const
{
TGeoNode *node = GetNode(level);
if (node) return node->GetVolume();
return 0;
}
TGeoShape *TGeoPhysicalNode::GetShape(Int_t level) const
{
TGeoVolume *vol = GetVolume(level);
if (vol) return vol->GetShape();
return 0;
}
void TGeoPhysicalNode::Paint(Option_t * )
{
TVirtualGeoPainter *painter = gGeoManager->GetGeomPainter();
if (!painter) return;
}
void TGeoPhysicalNode::SetBranchAsState()
{
TGeoNodeCache *cache = gGeoManager->GetCache();
if (!cache) {
Error("SetBranchAsState","no state available");
return;
}
if (!cache->IsDummy()) {
Error("SetBranchAsState", "not implemented for full cache");
return;
}
if (!fNodes) fNodes = new TObjArray(30);
if (!fMatrices) fMatrices = new TObjArray(30);
fLevel = gGeoManager->GetLevel();
TGeoHMatrix **matrices = (TGeoHMatrix **) cache->GetMatrices();
TGeoNode **branch = (TGeoNode **) cache->GetBranch();
for (Int_t i=0; i<=fLevel; i++) {
fNodes->AddAt(branch[i],i);
fMatrices->AddAt(new TGeoHMatrix(*matrices[i]),i);
}
TGeoNode *node = (TGeoNode*)fNodes->UncheckedAt(fLevel);
if (!fMatrixOrig) fMatrixOrig = new TGeoHMatrix();
*fMatrixOrig = node->GetMatrix();
}
Bool_t TGeoPhysicalNode::SetPath(const char *path)
{
if (!gGeoManager->cd(path)) {
Error("SetPath","wrong path -> maybe RestoreMasterVolume");
return kFALSE;
}
SetBranchAsState();
return kTRUE;
}
ClassImp(TGeoPNEntry)
TGeoPNEntry::TGeoPNEntry()
{
fNode = 0;
fMatrix = 0;
}
TGeoPNEntry::TGeoPNEntry(const char *name, const char *path)
:TNamed(name, path)
{
if (!gGeoManager || !gGeoManager->IsClosed() || !gGeoManager->CheckPath(path)) {
TString errmsg("Cannot define a physical node link without a closed geometry and a valid path !");
Error("ctor", errmsg.Data());
throw errmsg;
return;
}
fNode = 0;
fMatrix = 0;
}
void TGeoPNEntry::SetPhysicalNode(TGeoPhysicalNode *node)
{
if (fNode && node) {
Warning("SetPhysicalNode", "Physical node changed for entry %s", GetName());
Warning("SetPhysicalNode", "=== New path: %s", node->GetName());
}
fNode = node;
}
void TGeoPNEntry::SetMatrix(TGeoHMatrix *mat)
{
fMatrix = mat;
if (fMatrix) fMatrix->RegisterYourself();
}
ROOT page - Class index - Class Hierarchy - Top of the page
This page has been automatically generated. If you have any comments or suggestions about the page layout send a mail to ROOT support, or contact the developers with any questions or problems regarding ROOT.