338{
339 using enum Acts::AxisDirection;
340
341 ACTS_VERBOSE("Build layers: " << (type < 0 ? "NEGATIVE" : "POSITIVE")
342 << " ENDCAP");
343 std::vector<std::shared_ptr<const ActsDetectorElement>> elements =
345 std::map<std::tuple<int, int, int>, std::vector<const Acts::Surface *>>
346 initialLayers{};
347
348 for (const auto &element : elements) {
349
350 IdentityHelper id = element->identityHelper();
351 if (
id.
bec() * type <= 0) {
352 continue;
353 }
354
356 if(id.layer_disk() >= 3) continue;
357 }
359 if(id.layer_disk() <= 2) continue;
360 }
361
362 m_cfg.elementStore->push_back(element);
363
364 std::tuple<int, int, int>
key{
id.bec(),
id.layer_disk(),
id.eta_module()};
365
366 initialLayers[
key].push_back(&element->surface());
367 }
368
369 ACTS_VERBOSE("Found " << initialLayers.size() << " "
370 << (type < 0 ? "NEGATIVE" : "POSITIVE")
371 << " ENDCAP inital layers");
372
373 std::vector<Acts::ProtoLayer> protoLayers;
374 protoLayers.reserve(initialLayers.size());
375
376 for (const auto &[key, surfaces] : initialLayers) {
377 auto &pl = protoLayers.emplace_back(gctx, surfaces);
378 pl.envelope[AxisR] =
m_cfg.endcapEnvelopeR;
379 pl.envelope[AxisZ] =
m_cfg.endcapEnvelopeZ;
380 }
381
382
383 std::sort(protoLayers.begin(), protoLayers.end(),
384 [type](
const Acts::ProtoLayer &
a,
const Acts::ProtoLayer &b) {
385 double midA = (a.min(AxisZ) + a.max(AxisZ)) / 2.0;
386 double midB = (b.min(AxisZ) + b.max(AxisZ)) / 2.0;
387 if (type < 0) {
388 return midA < midB;
389 } else {
390 return midA > midB;
391 }
392 });
393
394 auto plPrintZ = [](const auto &pl) -> std::string {
395 std::stringstream
ss;
396 double zMid = (pl.min(AxisZ) + pl.max(AxisZ)) / 2.0;
397 ss <<
" < " << pl.min(AxisZ) <<
" | " << zMid <<
" | "
398 << pl.max(AxisZ) << " > ";
400 };
401 if (
logger().doPrint(Acts::Logging::VERBOSE)) {
402 for (const auto &pl : protoLayers) {
403
404 ACTS_VERBOSE(" -> at < zMin | zMid | zMax >: " << plPrintZ(pl));
405
406 ACTS_VERBOSE(" -> at rMin / rMax: " << pl.min(AxisR) << " / "
407 << pl.max(AxisR));
408 }
409 }
410
411 std::vector<Acts::ProtoLayer> mergedProtoLayers;
412
413 if (
m_cfg.doEndcapLayerMerging) {
414 mergedProtoLayers.push_back(protoLayers.front());
415 std::vector<const Acts::Surface *> surfaces;
416 for (
size_t i = 1;
i < protoLayers.size();
i++) {
417 auto &pl = protoLayers[
i];
418 auto &pl_prev = mergedProtoLayers.back();
419
420 ACTS_VERBOSE("Compare: " << plPrintZ(pl_prev) << " and " << plPrintZ(pl));
421 bool overlap = (pl.min(AxisZ) <= pl_prev.max(AxisZ) &&
422 pl.max(AxisZ) >= pl_prev.min(AxisZ));
423 ACTS_VERBOSE(" -> overlap? " << (overlap ? "yes" : "no"));
424 if (overlap) {
425 ACTS_VERBOSE(" ===> merging");
426 surfaces.clear();
427 surfaces.reserve(pl.surfaces().size() + pl_prev.surfaces().size());
428 surfaces.insert(surfaces.end(), pl.surfaces().begin(),
429 pl.surfaces().end());
430 surfaces.insert(surfaces.end(), pl_prev.surfaces().begin(),
431 pl_prev.surfaces().end());
432 mergedProtoLayers.pop_back();
433 auto &new_pl =
434 mergedProtoLayers.emplace_back(gctx, std::move(surfaces));
435 new_pl.envelope[AxisR] = pl.envelope[AxisR];
436 new_pl.envelope[AxisZ] = pl.envelope[AxisZ];
437 } else {
438 mergedProtoLayers.push_back(std::move(pl));
439 }
440 }
441
442 ACTS_VERBOSE("" << mergedProtoLayers.size() << " "
443 << (type < 0 ? "NEGATIVE" : "POSITIVE")
444 << " ENDCAP layers remain after merging");
445 } else {
446 mergedProtoLayers = protoLayers;
447 }
448
449 if (
m_cfg.objDebugOutput) {
450 for (
size_t i = 0;
i < mergedProtoLayers.size();
i++) {
451
452 std::stringstream
ss;
453 ss <<
"obj/" <<
m_cfg.mode <<
"_" << (
type < 0 ?
"neg" :
"pos")
454 <<
"_disk_" << std::setfill(
'0') << std::setw(2) <<
i <<
".obj";
455
456 std::ofstream ofs{
ss.str()};
457 Acts::ObjVisualization3D vis{};
458 Acts::ViewConfig vc = Acts::s_viewSensitive;
459 vc.quarterSegments = 200;
460 for (const auto &surface : mergedProtoLayers[i].surfaces()) {
461 Acts::GeometryView3D::drawSurface(vis, *surface, gctx,
462 Acts::Transform3::Identity(), vc);
463 }
464
465 vis.write(ofs);
466 }
467 }
468
469 std::vector<std::shared_ptr<const Surface>> ownedSurfaces;
470 for (const auto &pl : mergedProtoLayers) {
471
472 std::unique_ptr<Acts::ApproachDescriptor> approachDescriptor = nullptr;
473 std::shared_ptr<const Acts::ProtoSurfaceMaterial> materialProxy = nullptr;
474
475 double layerZ = pl.medium(AxisZ);
476 double layerHalfZ = 0.5 * pl.range(AxisZ);
477 double layerThickness = pl.range(AxisZ);
478
479 double layerZInner = layerZ - layerHalfZ;
480 double layerZOuter = layerZ + layerHalfZ;
481
482 if (std::abs(layerZInner) > std::abs(layerZOuter))
484
485 std::vector<std::shared_ptr<const Acts::Surface>> aSurfaces;
486
487 Acts::Transform3 transformNominal(Translation3(0., 0., layerZ));
488 Acts::Transform3 transformInner(Translation3(0., 0., layerZInner));
489 Acts::Transform3 transformOuter(Translation3(0., 0., layerZOuter));
490
491 std::shared_ptr<Acts::DiscSurface> innerBoundary =
492 Acts::Surface::makeShared<Acts::DiscSurface>(
493 transformInner, pl.min(AxisR), pl.max(AxisR));
494 aSurfaces.push_back(innerBoundary);
495
496 std::shared_ptr<Acts::DiscSurface> nominalSurface =
497 Acts::Surface::makeShared<Acts::DiscSurface>(
498 transformNominal, pl.min(AxisR), pl.max(AxisR));
499 aSurfaces.push_back(nominalSurface);
500
501 std::shared_ptr<Acts::DiscSurface> outerBoundary =
502 Acts::Surface::makeShared<Acts::DiscSurface>(
503 transformOuter, pl.min(AxisR), pl.max(AxisR));
504 aSurfaces.push_back(outerBoundary);
505
506 if(layerThickness > 2_mm) {
507 ACTS_VERBOSE("Wide disc layer ("<< layerThickness << ") => adding cylinder like approach surfaces");
508 Acts::Transform3
trf{Translation3{0, 0, layerZ}};
509 auto cylinderInner =
510 Acts::Surface::makeShared<Acts::CylinderSurface>(
511 trf, pl.min(AxisR), layerHalfZ);
512 aSurfaces.push_back(cylinderInner);
513
514 auto cylinderOuter =
515 Acts::Surface::makeShared<Acts::CylinderSurface>(
516 trf, pl.max(AxisR), layerHalfZ);
517 aSurfaces.push_back(cylinderOuter);
518 }
519
520
521 size_t matBinsPhi =
m_cfg.endcapMaterialBins.first;
522 size_t matBinsR =
m_cfg.endcapMaterialBins.second;
523
524 Acts::BinUtility materialBinUtil(matBinsPhi, -
M_PI,
M_PI, Acts::closed,
525 AxisPhi);
526 materialBinUtil +=
527 Acts::BinUtility(matBinsR, pl.min(AxisR), pl.max(AxisR),
528 Acts::open, AxisR, transformNominal);
529
530 materialProxy =
531 std::make_shared<const Acts::ProtoSurfaceMaterial>(materialBinUtil);
532
533 ACTS_VERBOSE("[L] Layer is marked to carry support material on Surface ( "
534 "inner=0 / center=1 / outer=2 ) : "
535 << "inner");
536 ACTS_VERBOSE("with binning: [" << matBinsPhi << ", " << matBinsR << "]");
537
538 ACTS_VERBOSE("Created ApproachSurfaces for disc layer at:");
539 ACTS_VERBOSE(" - inner: Z=" << layerZInner);
540 ACTS_VERBOSE(" - central: Z=" << layerZ);
541 ACTS_VERBOSE(" - outer: Z=" << layerZOuter);
542
543
544 innerBoundary->assignSurfaceMaterial(materialProxy);
545
546
547
551
552 std::map<int, std::set<int>> phiModuleByRing;
553
554
555 for (const auto &srf : pl.surfaces()) {
556 auto elm = dynamic_cast<const ActsDetectorElement *>(
557 srf->associatedDetectorElement());
558 if (elm) {
559 auto id = elm->identityHelper();
560 int ring_number;
562 ring_number = id.eta_module();
563 }
564 else {
565 ring_number = id.layer_disk();
566 }
567 phiModuleByRing[ring_number].insert(id.phi_module());
568 }
569 }
570 size_t nModPhi = std::numeric_limits<size_t>::max();
571 for(const auto& [ring, phiModules] : phiModuleByRing) {
572 nModPhi = std::min(nModPhi, phiModules.size());
573 }
574
575 size_t nModR = phiModuleByRing.size();
576
577 ACTS_VERBOSE("Identifier reports: " << nModPhi << " is lowest for " << nModR
578 << " r-rings");
579
580 size_t nBinsPhi = nModPhi *
m_cfg.numberOfBinsFactor;
581 size_t nBinsR = nModR *
m_cfg.numberOfBinsFactor;
582
583 ACTS_VERBOSE("Creating r x phi binned layer with " << nBinsR << " x "
584 << nBinsPhi << " bins");
585
586
587 approachDescriptor =
588 std::make_unique<Acts::GenericApproachDescriptor>(aSurfaces);
589
590
591 ownedSurfaces.clear();
592 ownedSurfaces.reserve(pl.surfaces().size());
593 std::transform(pl.surfaces().begin(), pl.surfaces().end(),
594 std::back_inserter(ownedSurfaces),
595 [](const auto &s) { return s->getSharedPtr(); });
596
597 auto layer =
m_cfg.layerCreator->discLayer(gctx, ownedSurfaces, nBinsR,
598 nBinsPhi, pl, Transform3::Identity(),
599 std::move(approachDescriptor));
600
601 layersOutput.push_back(layer);
602 }
603}
void sort(typename DataModel_detail::iterator< DVL > beg, typename DataModel_detail::iterator< DVL > end)
Specialization of sort for DataVector/List.
void swap(ElementLinkVector< DOBJ > &lhs, ElementLinkVector< DOBJ > &rhs)