4 #include "INDRAGeometryBuilder.h"
8 #include "TGeoMedium.h"
10 #include "TGeoMatrix.h"
11 #include "TGeoManager.h"
12 #include "TGeoCompositeShape.h"
16 #include "KVDataSet.h"
35 for (
int i = 0; i < 8; i++) {
37 offY += coo[2 * i + 1];
41 for (
int i = 0; i < 8; i++) {
43 coo[2 * i + 1] -= offY;
131 Double_t planeDist = normal.Mag();
132 for (
int i = 0; i < 4; i++) {
133 Double_t rap = (planeDist + depth) / planeDist;
134 backcoords[i] = rap * frontcoords[i];
150 TString path = Form(
"%s.cao", detname);
152 infos.ReadFile(path, kEnvAll);
154 Double_t unit_converter = 0.1;
156 fInnerFront[0].SetX(infos.GetValue(
"INNER.A.X", 0.0)*unit_converter);
157 fInnerFront[0].SetY(infos.GetValue(
"INNER.A.Y", 0.0)*unit_converter);
158 fInnerFront[0].SetZ(infos.GetValue(
"INNER.A.Z", 0.0)*unit_converter);
159 fInnerFront[3].SetX(infos.GetValue(
"INNER.B.X", 0.0)*unit_converter);
160 fInnerFront[3].SetY(infos.GetValue(
"INNER.B.Y", 0.0)*unit_converter);
161 fInnerFront[3].SetZ(infos.GetValue(
"INNER.B.Z", 0.0)*unit_converter);
162 fInnerFront[2].SetX(infos.GetValue(
"INNER.C.X", 0.0)*unit_converter);
163 fInnerFront[2].SetY(infos.GetValue(
"INNER.C.Y", 0.0)*unit_converter);
164 fInnerFront[2].SetZ(infos.GetValue(
"INNER.C.Z", 0.0)*unit_converter);
165 fInnerFront[1].SetX(infos.GetValue(
"INNER.D.X", 0.0)*unit_converter);
166 fInnerFront[1].SetY(infos.GetValue(
"INNER.D.Y", 0.0)*unit_converter);
167 fInnerFront[1].SetZ(infos.GetValue(
"INNER.D.Z", 0.0)*unit_converter);
171 for (
int i = 0; i <= 3; i++) {
176 fOuterFront[0].SetX(infos.GetValue(
"OUTER.A.X", 0.0)*unit_converter);
177 fOuterFront[0].SetY(infos.GetValue(
"OUTER.A.Y", 0.0)*unit_converter);
178 fOuterFront[0].SetZ(infos.GetValue(
"OUTER.A.Z", 0.0)*unit_converter);
179 fOuterFront[3].SetX(infos.GetValue(
"OUTER.B.X", 0.0)*unit_converter);
180 fOuterFront[3].SetY(infos.GetValue(
"OUTER.B.Y", 0.0)*unit_converter);
181 fOuterFront[3].SetZ(infos.GetValue(
"OUTER.B.Z", 0.0)*unit_converter);
182 fOuterFront[2].SetX(infos.GetValue(
"OUTER.C.X", 0.0)*unit_converter);
183 fOuterFront[2].SetY(infos.GetValue(
"OUTER.C.Y", 0.0)*unit_converter);
184 fOuterFront[2].SetZ(infos.GetValue(
"OUTER.C.Z", 0.0)*unit_converter);
185 fOuterFront[1].SetX(infos.GetValue(
"OUTER.D.X", 0.0)*unit_converter);
186 fOuterFront[1].SetY(infos.GetValue(
"OUTER.D.Y", 0.0)*unit_converter);
187 fOuterFront[1].SetZ(infos.GetValue(
"OUTER.D.Z", 0.0)*unit_converter);
191 for (
int i = 0; i <= 3; i++) {
196 phi0 = infos.GetValue(
"PHIZERO", 30.0);
197 thetaMin = infos.GetValue(
"THETAMIN", 0.0);
198 thetaMax = infos.GetValue(
"THETAMAX", 0.0);
199 Ndets = infos.GetValue(
"NDETS", 1);
216 fInnerRing = infos.GetValue(
"INNER.RING", ring);
222 fModmax = infos.GetValue(
"MODMAX", 24);
232 printf(
"Detector name : %s\n",
fDetName.Data());
234 printf(
"Frame centre:\n");
236 printf(
"Inner centre:\n");
238 printf(
"Outer centre: \n");
240 printf(
"Coordinates of outer pads :\n");
242 printf(
"Coordinates of inner pads :\n");
244 printf(
"Material of outer frame : %s\n",
fFrameMat.GetTitle());
259 TVector3 fFrameBack[4];
260 TVector3 fFrameBackCentre;
271 Double_t vertices[16];
272 for (
int i = 0; i < 8; i++) {
273 vertices[2 * i] = corners[i].X();
274 vertices[2 * i + 1] = corners[i].Y();
279 vol_name.Form(
"STRUCT_%s_%02d", det_type.Data(), ring_num);
280 fFrameVolume = gGeoManager->MakeVolumeAssembly(vol_name);
281 fFrameVolume->SetMedium(gGeoManager->GetMedium(
"Vacuum"));
282 vol_name.Form(
"DEADZONE_%s_%02d", det_type.Data(), ring_num);
287 TGeoTranslation* offset =
new TGeoTranslation(offX, offY, -(0.5 *
fTotalThickness - (1.e-4 + dz)));
288 TGeoVolume* frame = gGeoManager->MakeArb8(vol_name.Data(), med, dz, vertices);
289 frame->SetLineColor(med->GetMaterial()->GetDefaultColor());
305 TRotation rot_to_frame;
306 rot_to_frame.SetYEulerAngles(-centre.Phi(), -centre.Theta(), 0.);
307 TVector3 displZ(0, 0, centre.Mag());
308 for (
int i = 0; i < 4; i++) {
309 ownframe[i] = rot_to_frame * orig[i] - displZ;
324 Double_t theta =
fFrameCentre.Theta() * TMath::RadToDeg();
325 TGeoRotation rot1, rot2;
326 rot2.SetAngles(phi + 90., theta, 0.);
327 rot1.SetAngles(-90., 0., 0.);
331 TGeoTranslation tran(0, 0, trans_z);
332 TGeoHMatrix h = rot2 * tran * rot1;
333 TGeoHMatrix* ph =
new TGeoHMatrix(h);
336 gGeoManager->GetTopVolume()->AddNode(
fFrameVolume, copy_no, ph);
360 if (!thisdetector || !thisdetector->
IsPresent()) {
389 if (!strcmp(det,
"PHOS"))
392 fDetName.Form(
"%s_%02d01", det, ring);
411 for (i = 1; i <=
Ndets; i++) {
412 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
420 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
425 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
433 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
440 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, outerMod);
446 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, outerMod);
456 Info(
"MakeRing",
"Detector type %s ring number %d : ABSENT", det, ring);
459 Info(
"MakeRing",
"Detector type %s ring number %d : %d/%d detectors present", det, ring, npresent, ntotal);
468 for (i = 1; i <=
Ndets; i++) {
476 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
484 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
489 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
497 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, innerMod);
504 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, outerMod);
510 if (!strcmp(det,
"PHOS"))
fDetName.Form(
"%s_%02d", det, outerMod);
518 if (!npresent) Info(
"MakeRing",
"\tBlock %d: ABSENT", i);
519 else if (npresent < ntotal) {
520 Info(
"MakeRing",
"\tBlock %d: %d/%d detectors present", i, npresent, ntotal);
521 Info(
"MakeRing",
"\tCASE NOT TREATED. ALL DETECTORS PRESENT.");
529 Info(
"MakeRing",
"DET:%s RING:%d Making new prototype group for i=%d/%d",
530 det, ring, i,
Ndets);
576 if (((ring > 9 && ring < 13) && i == 1) || (ring > 12 && i == 2)) {
599 Double_t vertices[16];
610 if (thick > 0.0) no_abs++;
612 multi_layer = (no_abs > 1);
615 fDetVolume = gGeoManager->MakeVolumeAssembly(Form(
"DET_%s", det));
616 fDetVolume->SetMedium(gGeoManager->GetMedium(
"Vacuum"));
618 TVector3 frontPlane[4], backPlane[4], frontCentre, backCentre;
620 for (
int i = 0; i < 4; i++) frontPlane[i] = som[i];
626 Double_t depth_in_det = 0.;
649 for (
int i = 0; i < 8; i++) {
650 vertices[2 * i] = corners[i].X();
651 vertices[2 * i + 1] = corners[i].Y();
654 Double_t dz = thick / 2.;
657 if (no_abs ==
fActiveLayer) vol_name = Form(
"ACTIVE_%s", det);
658 else vol_name = Form(
"%s_%d_%s", det, no_abs, abs->GetName());
661 vol_name = Form(
"DET_%s", det);
664 gGeoManager->MakeArb8(vol_name.Data(), med, dz, vertices);
665 vol->SetLineColor(med->GetMaterial()->GetDefaultColor());
669 TGeoTranslation* tr =
new TGeoTranslation(offX, offY, trans_z);
678 depth_in_det += thick;
682 for (
int i = 0; i < 4; i++) frontPlane[i] = backPlane[i];
683 frontCentre = backCentre;
708 TVector3 a10 = corners[1] - corners[0];
709 TVector3 a21 = corners[2] - corners[1];
710 TVector3 normal = a10.Cross(a21);
711 Double_t d = (corners[3] * normal) / normal.Mag2();
728 Double_t thetamax, Double_t phimin, Double_t phimax, TVector3* corners)
738 TVector3 a10 = plane[1] - plane[0];
739 TVector3 a21 = plane[2] - plane[1];
740 TVector3 normal = a10.Cross(a21);
741 Double_t d0 = plane[3] * normal;
744 l.SetMagThetaPhi(1., thetamin * TMath::DegToRad(), phimin * TMath::DegToRad());
745 corners[3] = l * (d0 / (l * normal));
746 l.SetMagThetaPhi(1., thetamax * TMath::DegToRad(), phimin * TMath::DegToRad());
747 corners[2] = l * (d0 / (l * normal));
748 l.SetMagThetaPhi(1., thetamax * TMath::DegToRad(), phimax * TMath::DegToRad());
749 corners[1] = l * (d0 / (l * normal));
750 l.SetMagThetaPhi(1., thetamin * TMath::DegToRad(), phimax * TMath::DegToRad());
751 corners[0] = l * (d0 / (l * normal));
760 gGeoManager->CloseGeometry();
762 gGeoManager->DefaultColors();
763 gGeoManager->GetTopVolume()->Draw(
"ogl");
772 for (
int i = 0; i < 4; i++) {
773 newpad[3 - i].SetMagThetaPhi(orig[i].Mag(), orig[i].Theta(), 2.*phicentre - orig[i].Phi());
789 new TGeoBBox(
"TARGET_FRAME", 3., 3., 0.1 / 2.);
790 new TGeoEltu(
"TARGET_HOLE", 2., 2., 0.1 / 2.);
791 TGeoCompositeShape* cs =
new TGeoCompositeShape(
"TARGET_FRAME",
"TARGET_FRAME - TARGET_HOLE");
792 TGeoVolume* target_frame =
new TGeoVolume(
"TARGET_FRAME", cs, target_holder_mat.
GetGeoMedium());
793 gGeoManager->GetTopVolume()->AddNode(target_frame, 1);
800 TGeoVolume* target = gGeoManager->MakeEltu(
"TARGET", targMat->
GetGeoMedium(), 2., 2., targMat->
GetThickness() / 2.);
801 gGeoManager->GetTopVolume()->AddNode(target, 1);
812 Error(
"Build",
"You must build the geometry with gDataSet->BuildMultiDetector() before calling this method");
817 Error(
"Build",
"You must defined gGeoManager with KVMultiDetArray::CreateGeoManager before calling this method");
826 for (
int ring = 2; ring <= 16; ring += 2) {
828 if (ring > 1 && ring < 10)
MakeRing(
"SI", ring);
835 for (
int ring = 10; ring <= 17; ring++) {
840 gGeoManager->DefaultColors();
841 gGeoManager->CloseGeometry();
889 Error(
"Build",
"You must build the geometry with gDataSet->BuildMultiDetector() before calling this method");
894 Error(
"Build",
"You must defined gGeoManager with KVMultiDetArray::CreateGeoManager before calling this method");
903 for (
int ring = 2; ring <= 16; ring += 2) {
905 if (ring > 1 && ring < 10) {
914 for (
int ring = 10; ring <= 17; ring++) {
926 void INDRAGeometryBuilder::BuildEtalonVolumes()
931 TGeoMedium* Silicon = mat_si.GetGeoMedium();
932 TGeoMedium* Lithium = mat_li.GetGeoMedium();
933 TGeoMedium* Alu = mat_al.GetGeoMedium();
935 Double_t sili_diameter_total = 2.54;
936 Double_t holder_thickness = 0.05;
937 Double_t holder_length = 1.0;
938 Double_t sili_diameter_active = 2.36;
939 Double_t sili_silicon_thickness = 0.22;
940 Double_t sili_lithium_thickness = 0.0044;
941 Double_t si75_thickness = 0.008;
942 Double_t si75_diameter_active = 2.2;
944 fEtalonVol =
new TGeoVolumeAssembly(
"STRUCT_ETALON");
945 fEtalonVol->SetMedium(gGeoManager->GetMedium(
"Vacuum"));
947 TGeoVolume* holder = gGeoManager->MakeTube(
"DEADZONE_ETALON_HOLDER", Alu,
948 (sili_diameter_total) / 2., (sili_diameter_total + 2 * holder_thickness) / 2., holder_length / 2.);
949 holder->SetLineColor(holder->GetMaterial()->GetDefaultColor());
952 Double_t w = sili_silicon_thickness + sili_lithium_thickness;
953 TGeoVolume* sili_dz = gGeoManager->MakeTube(
"DEADZONE_SILI", Alu,
954 sili_diameter_active / 2., (sili_diameter_total) / 2., w / 2.);
955 sili_dz->SetLineColor(sili_dz->GetMaterial()->GetDefaultColor());
957 fEtalonVol->AddNode(sili_dz, 1,
new TGeoTranslation(0, 0, holder_length / 4.));
959 TGeoVolumeAssembly* sili =
new TGeoVolumeAssembly(
"DET_SILI");
960 sili->SetMedium(gGeoManager->GetMedium(
"Vacuum"));
961 TGeoVolume* sili_si = gGeoManager->MakeTube(
"ACTIVE_SILI", Silicon,
962 0., (sili_diameter_active) / 2., (sili_silicon_thickness) / 2.);
963 TGeoVolume* sili_li = gGeoManager->MakeTube(
"SILI_LI", Lithium,
964 0., (sili_diameter_active) / 2., (sili_lithium_thickness) / 2.);
965 sili_si->SetLineColor(sili_si->GetMaterial()->GetDefaultColor());
966 sili_li->SetLineColor(sili_li->GetMaterial()->GetDefaultColor());
967 sili->AddNode(sili_si, 1,
new TGeoTranslation(0, 0, -(w / 2. - sili_silicon_thickness / 2.)));
968 sili->AddNode(sili_li, 1,
new TGeoTranslation(0, 0, w / 2. - sili_lithium_thickness / 2.));
969 fEtalonVol->AddNode(sili, 1,
new TGeoTranslation(0, 0, holder_length / 4.));
971 TGeoVolume* si75 = gGeoManager->MakeTube(
"DET_SI75", Silicon,
972 0., (si75_diameter_active) / 2., (si75_thickness) / 2.);
973 TGeoVolume* si75_dz = gGeoManager->MakeTube(
"DEADZONE_SI75", Alu,
974 (si75_diameter_active) / 2., (sili_diameter_total) / 2., (si75_thickness) / 2.);
993 Double_t dist[] = {17.4005, 17.4005, 17.4005, 16.7005, 16.7005, 16.7005, 17.4005, 17.4005,};
998 TGeoTranslation trans;
999 trans.SetDz(dist[RING - 10]);
1000 TGeoRotation rot1, rot2;
1002 TGeoHMatrix* ph = 0;
1007 rot1.SetAngles(-1.*
fEtalonPhi[RING - 10], 0., 0.);
1013 TGeoTranslation p(1.5, 2.5, 0);
1014 h = rot2 * trans * p * rot1;
1016 else if (RING == 11) {
1017 TGeoTranslation p(1.5, -1.5, 0);
1018 h = rot2 * trans * p * rot1;
1020 else if (RING == 12) {
1021 TGeoTranslation p(1.5, 0, 0);
1022 h = rot2 * trans * p * rot1;
1024 else if (RING == 13) {
1025 TGeoTranslation p(-4, -0.5, 0);
1026 h = rot2 * trans * p * rot1;
1028 else if (RING == 14) {
1029 TGeoTranslation p(-1.7, 1.75, 0);
1030 h = rot2 * trans * p * rot1;
1032 else if (RING == 15) {
1033 TGeoTranslation p(-1.7, -3, 0);
1034 h = rot2 * trans * p * rot1;
1036 else if (RING == 16) {
1037 TGeoTranslation p(0, 2.5, 0);
1038 h = rot2 * trans * p * rot1;
1041 TGeoTranslation p(0, -2.25, 0);
1042 h = rot2 * trans * p * rot1;
1044 ph =
new TGeoHMatrix(h);
1045 gGeoManager->GetTopVolume()->AddNode(
fEtalonVol, RING, ph);
Build INDRA geometry from Huguet CAO infos.
void CorrectCoordinates(Double_t *, Double_t &, Double_t &)
void MakeDetector(const Char_t *det, TVector3 *som, TVector3 cen)
make volume corresponding to the actual detector
INDRAGeometryBuilder()
Default constructor.
Int_t Ndets
number of detectors in ring
void Copy(TObject &) const
void CalculateBackPlaneCoordinates(TVector3 *frontcoords, TVector3 centre, Double_t depth, TVector3 *backcoords)
TGeoVolume * fDetVolume
geo volume representing frame
void CalculateCentre(TVector3 *corners, TVector3 ¢re)
Double_t fTotalThickness
sum of thicknesses of layers of current detector
Double_t fEtalonTheta[10]
void ReadDetCAO(const Char_t *detname, Int_t ring)
Int_t fActiveLayer
index of active layer of current detector
TString fDetName
name of detector
TVector3 fInnerCentre
centre of inner face
void PlaceFrame(Double_t phi, Int_t copy_no)
position frame (dead zone) volume in geometry
KVMaterial fFrameMat
material of outer frame
void ReflectPad(TVector3 *orig, Double_t phicentre, TVector3 *newpad)
Double_t phi0
theoretical geometry
TVector3 fInnerFront[4]
coords of inner front face
void MakeEtalon(int RING)
void CalculateCornersInPlane(TVector3 *plane, Double_t thetamin, Double_t thetamax, Double_t phimin, Double_t phimax, TVector3 *corners)
TVector3 fOuterFront[4]
coords of outer front face
TVector3 fOuterCentre
centre of outer face
void Print(Option_t *="") const
Bool_t CheckDetectorPresent(TString detname)
void TransformToOwnFrame(TVector3 *orig, TVector3 ¢re, TVector3 *ownframe)
KVDetector * fProtoDetector
detector used for last fLayers look up
TVector3 fFrameFront[4]
coords of outer front face
virtual ~INDRAGeometryBuilder()
Destructor.
void Build(Bool_t withTarget=kTRUE, Bool_t closeGeometry=kTRUE)
void MakeRing(const Char_t *det, int ring)
KVList * fLayers
list of materials making up layers of current detector
void MakeFrame(TString det_type, Int_t ring_num)
TVector3 fFrameCentre
centre of frame
TGeoVolume * fFrameVolume
geo volume representing frame
void PlaceDetector()
position detector inside frame
TGeoTranslation * fDetectorPosition
TGeoVolumeAssembly * fEtalonVol
Base class for KaliVeda framework.
static Bool_t SearchKVFile(const Char_t *name, TString &fullpath, const Char_t *kvsubdir="")
Base class for detector geometry description.
Bool_t HasSameStructureAs(const KVDetector *) const
Double_t GetTotalThicknessInCM()
KVMaterial * GetActiveLayer() const
KVList * GetListOfAbsorbers() const
virtual Bool_t IsPresent() const
KVDetector * GetDetector(const Char_t *name) const
Return detector in this structure with given name.
Description of physical materials used to construct detectors & targets; interface to range tables.
virtual Double_t GetThickness() const
virtual TGeoMedium * GetGeoMedium(const Char_t *="")
virtual void SetMaterial(const Char_t *type)
Handles lists of named parameters with different types, a list of KVNamedParameter objects.
Bool_t HasParameter(const Char_t *name) const
Strings used to represent a set of ranges of values.
Bool_t Contains(Int_t val) const
returns kTRUE if the value 'val' is contained in the ranges defined by the number list
virtual Int_t GetSize() const
virtual TObject * First() const
Calculation/correction of energy losses of particles through an experimental target.
KVList * GetLayers() const