5 #include "TDecompSVD.h" 39 using namespace detail;
46 if (
slices.size() < 2)
return;
55 std::string fcnLabel =
"SP";
58 bool printHeader =
true;
59 for (
size_t isl = 0; isl <
slices.size(); ++isl) {
62 if (slc.pfps.empty())
continue;
63 for (
auto& pfp : slc.pfps)
64 PrintP(myprt, pfp, printHeader);
69 std::vector<std::vector<int>> stLists;
70 for (std::size_t sl1 = 0; sl1 <
slices.size() - 1; ++sl1) {
72 for (std::size_t sl2 = sl1 + 1; sl2 <
slices.size(); ++sl2) {
75 if (slc1.ID != slc2.ID)
continue;
76 for (
auto& p1 : slc1.pfps) {
77 if (p1.ID <= 0)
continue;
79 if (p1.PDGCode == 1111)
continue;
80 for (
auto& p2 : slc2.pfps) {
81 if (p2.ID <= 0)
continue;
83 if (p2.PDGCode == 1111)
continue;
87 for (
unsigned short e1 = 0; e1 < 2; ++e1) {
90 if (
InsideFV(slc1, p1, e1))
continue;
92 for (
unsigned short e2 = 0; e2 < 2; ++e2) {
95 if (
InsideFV(slc2, p2, e2))
continue;
97 float sep =
PosSep2(pos1, pos2);
98 if (sep > maxSep2)
continue;
100 if (cth < maxCth)
continue;
106 if (!gotit)
continue;
109 myprt <<
"Stitch slice " << slc1.ID <<
" P" << p1.UID <<
" TPC " << p1.TPCID.TPC;
110 myprt <<
" and P" << p2.UID <<
" TPC " << p2.TPCID.TPC;
111 myprt <<
" sep " << sqrt(maxSep2) <<
" maxCth " << maxCth;
115 for (
auto& pm : stLists) {
116 bool p1InList = (std::find(pm.begin(), pm.end(), p1.UID) != pm.end());
117 bool p2InList = (std::find(pm.begin(), pm.end(), p2.UID) != pm.end());
118 if (p1InList || p2InList) {
119 if (p1InList) pm.push_back(p2.UID);
120 if (p2InList) pm.push_back(p1.UID);
126 std::vector<int>
tmp(2);
129 stLists.push_back(tmp);
135 if (stLists.empty())
return;
137 for (
auto& stl : stLists) {
140 std::pair<unsigned short, unsigned short> minZIndx;
141 unsigned short minZEnd = 2;
142 for (
auto puid : stl) {
144 if (slcIndex.first == USHRT_MAX)
continue;
145 auto& pfp =
slices[slcIndex.first].pfps[slcIndex.second];
146 for (
unsigned short end = 0;
end < 2; ++
end) {
155 if (minZEnd > 1)
continue;
157 auto& pfp =
slices[minZIndx.first].pfps[minZIndx.second];
160 for (
auto puid : stl) {
161 if (puid == pfp.UID)
continue;
163 if (sIndx.first == USHRT_MAX)
continue;
164 auto& opfp =
slices[sIndx.first].pfps[sIndx.second];
166 pfp.TjUIDs.insert(pfp.TjUIDs.end(), opfp.TjUIDs.begin(), opfp.TjUIDs.end());
169 if (opfp.ParentUID > 0) {
171 if (pSlcIndx.first <
slices.size()) {
172 auto& parpfp =
slices[pSlcIndx.first].pfps[pSlcIndx.second];
173 std::replace(parpfp.DtrUIDs.begin(), parpfp.DtrUIDs.begin(), opfp.UID, pfp.UID);
176 for (
auto dtruid : opfp.DtrUIDs) {
178 if (dSlcIndx.first <
slices.size()) {
179 auto& dtrpfp =
slices[dSlcIndx.first].pfps[dSlcIndx.second];
180 dtrpfp.ParentUID = pfp.UID;
201 for (
auto& tj : slc.
tjs) {
202 for (
auto& tp : tj.Pts)
209 std::vector<MatchStruct> matVec;
216 unsigned short maxNit = 2;
217 if (slc.
nPlanes == 2) maxNit = 1;
221 for (
unsigned short nit = 0; nit < maxNit; ++nit) {
223 if (slc.
nPlanes == 3 && nit == 0) {
231 if (matVec.empty())
continue;
234 myprt <<
"nit " << nit <<
" MVI Count Tjs\n";
235 for (
unsigned int indx = 0; indx < matVec.size(); ++indx) {
236 auto& ms = matVec[indx];
237 myprt << std::setw(5) << indx << std::setw(6) << (int)ms.Count;
238 for (
auto tid : ms.TjIDs)
239 myprt <<
" T" << tid;
242 for (
auto tid : ms.TjIDs) {
243 auto& tj = slc.
tjs[tid - 1];
246 float frac = ms.Count / tpCnt;
247 myprt <<
" matFrac " << std::fixed << std::setprecision(3) << frac;
256 for (
auto& pfp : slc.
pfps)
258 PrintTP3Ds(clockData, detProp,
"FPFP", slc, pfp, -1);
269 std::vector<MatchStruct> matVec,
270 unsigned short matVec_Iter)
273 if (matVec.empty())
return;
278 for (std::size_t indx = 0; indx < matVec.size(); ++indx) {
281 if (foundMVI) prt =
true;
282 auto& ms = matVec[indx];
284 std::cout <<
"found MVI " << indx <<
" in MakePFParticles ms.Count = " << ms.Count <<
"\n";
287 if (ms.Count == 0)
continue;
290 for (std::size_t itj = 0; itj < ms.TjIDs.size(); ++itj) {
291 auto& tj = slc.
tjs[ms.TjIDs[itj] - 1];
292 for (
unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt)
293 if (tj.Pts[ipt].InPFP == 0) ++npts;
296 std::vector<PFPStruct> pfpVec(1);
299 pfpVec[0].TjIDs = ms.TjIDs;
300 pfpVec[0].MVI = indx;
303 if (!
MakeTP3Ds(detProp, slc, pfpVec[0], foundMVI)) {
304 if (foundMVI)
mf::LogVerbatim(
"TC") <<
" MakeTP3Ds failed. Too many points already used ";
308 if (!
FitSection(detProp, slc, pfpVec[0], 0))
continue;
309 if (pfpVec[0].SectionFits[0].ChiDOF > 500 && !pfpVec[0].Flags[
kSmallAngle]) {
312 << pfpVec[0].SectionFits[0].ChiDOF <<
"\n";
313 Recover(detProp, slc, pfpVec[0], foundMVI);
321 npts = pfpVec[0].TP3Ds.size();
322 pfpVec[0].AlgMod[
kJunk3D] = (npts < 20 &&
MCSMom(slc, pfpVec[0].TjIDs) < 50) || (npts < 10);
324 auto& pfp = pfpVec[0];
326 myprt <<
" indx " << matVec_Iter <<
"/" << indx <<
" Count " << std::setw(5)
328 myprt <<
" P" << pfpVec[0].ID;
330 for (
auto& tjid : pfp.TjIDs)
331 myprt <<
" T" << tjid;
332 myprt <<
" projInPlane";
333 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
334 CTP_t inCTP =
EncodeCTP(pfp.TPCID.Cryostat, pfp.TPCID.TPC, plane);
335 auto tp =
MakeBareTP(detProp, pfp.SectionFits[0].Pos, pfp.SectionFits[0].Dir, inCTP);
336 myprt <<
" " << std::setprecision(2) << tp.Delta;
338 myprt <<
" maxTjLen " << (int)
MaxTjLen(slc, pfp.TjIDs);
339 myprt <<
" MCSMom " <<
MCSMom(slc, pfp.TjIDs);
340 myprt <<
" PDGCodeVote " <<
PDGCodeVote(clockData, detProp, slc, pfp);
341 myprt <<
" nTP3Ds " << pfp.TP3Ds.size();
342 myprt <<
" Reco3DRange " 345 if (foundMVI) {
PrintTP3Ds(clockData, detProp,
"FF", slc, pfpVec[0], -1); }
346 for (
unsigned short ip = 0; ip < pfpVec.size(); ++ip) {
347 auto& pfp = pfpVec[ip];
350 for (
unsigned short end = 0;
end < 2; ++
end) {
352 pfp.EndFlag[
end].reset();
358 pfp.EndFlag[0][
kAtKink] =
true;
361 vx3.
X = pfp.TP3Ds[0].Pos[0];
362 vx3.
Y = pfp.TP3Ds[0].Pos[1];
363 vx3.
Z = pfp.TP3Ds[0].Pos[2];
372 slc.
vtx3s.push_back(vx3);
373 pfp.Vx3ID[0] = vx3.
ID;
374 auto& prevPFP = slc.
pfps[slc.
pfps.size() - 1];
375 prevPFP.Vx3ID[1] = vx3.
ID;
380 if (pfp.TjIDs.size() == 2 && slc.
nPlanes == 3 && pfp.TP3Ds.size() > 20 &&
395 if (!
ReSection(detProp, slc, pfp, foundMVI))
continue;
397 if (foundMVI) {
PrintTP3Ds(clockData, detProp,
"RS", slc, pfp, -1); }
402 FillGaps3D(clockData, detProp, slc, pfp, foundMVI);
408 for (
auto& tp3d : pfp.TP3Ds) {
410 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
413 FilldEdx(clockData, detProp, slc, pfp);
414 pfp.PDGCode =
PDGCodeVote(clockData, detProp, slc, pfp);
416 PrintTP3Ds(clockData, detProp,
"STORE", slc, pfp, -1);
433 if (pfp.
TjIDs.empty())
return false;
434 if (pfp.
TP3Ds.empty())
return false;
435 if (pfp.
ID <= 0)
return false;
438 std::vector<std::pair<int, float>> tjTPCnt;
439 for (
auto& tp3d : pfp.
TP3Ds) {
441 if (tp3d.TjID <= 0)
return false;
443 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
444 if (tp.InPFP > 0 && tp.InPFP != pfp.
ID)
return false;
446 unsigned short indx = 0;
447 for (indx = 0; indx < tjTPCnt.size(); ++indx)
448 if (tjTPCnt[indx].first == tp3d.TjID)
break;
449 if (indx == tjTPCnt.size()) tjTPCnt.push_back(std::make_pair(tp3d.TjID, 0));
450 ++tjTPCnt[indx].second;
455 std::vector<int> nTjIDs;
456 for (
auto& tjtpcnt : tjTPCnt) {
457 auto& tj = slc.
tjs[tjtpcnt.first - 1];
459 if (tjtpcnt.second > 0.8 * npwc) nTjIDs.push_back(tjtpcnt.first);
461 if (prt) {
mf::LogVerbatim(
"TC") <<
"RTPs3D: P" << pfp.
ID <<
" nTjIDs " << nTjIDs.size(); }
463 if (nTjIDs.size() < 2) {
return false; }
482 std::vector<int> TinP;
483 for (
auto& pfp : slc.
pfps) {
484 if (pfp.ID <= 0)
continue;
486 for (std::size_t ipt = 0; ipt < pfp.TP3Ds.size(); ++ipt) {
487 auto& tp3d = pfp.TP3Ds[ipt];
488 if (tp3d.TjID <= 0)
continue;
489 if (std::find(TinP.begin(), TinP.end(), tp3d.TjID) == TinP.end()) TinP.push_back(tp3d.TjID);
490 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
516 std::vector<int> killme;
517 for (
auto& pfp : slc.
pfps) {
518 if (pfp.ID <= 0)
continue;
519 for (
auto& tp3d : pfp.TP3Ds) {
520 if (tp3d.TjID <= 0)
continue;
522 if (std::find(killme.begin(), killme.end(), tp3d.TjID) == killme.end())
523 killme.push_back(tp3d.TjID);
529 for (
auto tid : killme)
534 std::vector<Trajectory> ptjs(slc.
nPlanes);
536 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
537 ptjs[plane].Pass = 0;
540 ptjs[plane].StepDir = 0;
544 ptjs[plane].AlgMod[
kMat3D] =
true;
548 for (
auto& pfp : slc.
pfps) {
549 if (pfp.ID <= 0)
continue;
551 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
552 ptjs[plane].Pts.clear();
560 for (
auto& tp3d : pfp.TP3Ds) {
561 if (tp3d.TjID <= 0)
continue;
564 auto tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
565 if (tp.InPFP > 0 && tp.InPFP != pfp.ID)
continue;
571 ptjs[plane].Pts.push_back(tp);
575 std::vector<int> tids(ptjs.size(), 0);
576 for (
unsigned short plane = 0; plane < ptjs.size(); ++plane) {
577 auto& tj = ptjs[plane];
578 if (tj.Pts.size() < 2)
continue;
579 tj.PDGCode = pfp.PDGCode;
580 tj.MCSMom =
MCSMom(slc, tj);
583 auto& newTj = slc.
tjs.back();
584 pfp.TjIDs.push_back(newTj.ID);
585 tids[plane] = newTj.ID;
588 for (
unsigned short end = 0;
end < 2; ++
end) {
589 if (pfp.Vx3ID[
end] <= 0)
continue;
590 auto& vx3 = slc.
vtx3s[pfp.Vx3ID[
end] - 1];
591 for (
unsigned short plane = 0; plane < ptjs.size(); ++plane) {
592 if (tids[plane] == 0)
continue;
593 if (vx3.Vx2ID[plane] <= 0)
continue;
594 auto& vx2 = slc.
vtxs[vx3.Vx2ID[plane] - 1];
595 auto& tj = slc.
tjs[tids[plane] - 1];
597 tj.VtxID[tend] = vx2.ID;
599 mf::LogVerbatim(
"TC") <<
"MPFPTjs: 3V" << vx3.ID <<
" -> 2V" << vx2.ID <<
" -> T" 600 << tj.ID <<
"_" << tend <<
" in plane " << plane;
619 unsigned int maxWire = slc.
nWires[0];
620 for (
auto nw : slc.
nWires)
621 if (nw < maxWire) maxWire = nw;
623 unsigned int firstWire = maxWire / 2;
626 std::vector<std::pair<unsigned short, unsigned short>> pln1pln2;
627 for (
unsigned short pln1 = 0; pln1 < slc.
nPlanes - 1; ++pln1) {
628 for (
unsigned short pln2 = pln1 + 1; pln2 < slc.
nPlanes; ++pln2) {
629 auto p1p2 = std::make_pair(pln1, pln2);
630 if (std::find(pln1pln2.begin(), pln1pln2.end(), p1p2) != pln1pln2.end())
continue;
632 for (
unsigned int wire = firstWire; wire < maxWire; ++wire) {
635 if (!intersection00)
continue;
640 if (!intersection10)
continue;
645 if (!intersection01)
continue;
653 tcwi.
y = intersection00->y;
654 tcwi.
z = intersection00->z;
655 tcwi.
dydw1 = (intersection10->y - intersection00->y) / 10;
656 tcwi.
dzdw1 = (intersection10->z - intersection00->z) / 10;
657 tcwi.
dydw2 = (intersection01->y - intersection00->y) / 10;
658 tcwi.
dzdw2 = (intersection01->z - intersection00->z) / 10;
677 if (pln1 == pln2)
return false;
680 std::swap(pln1, pln2);
681 std::swap(wir1, wir2);
685 if (wi.pln1 != pln1)
continue;
686 if (wi.pln2 != pln2)
continue;
688 double dw1 = wir1 - wi.wir1;
689 double dw2 = wir2 - wi.wir2;
690 y = (float)(wi.y + dw1 * wi.dydw1 + dw2 * wi.dydw2);
691 z = (float)(wi.z + dw1 * wi.dzdw1 + dw2 * wi.dzdw2);
704 std::vector<int> inTraj((*
evt.
allHits).size(), 0);
705 for (
auto& tch : slc.
slHits)
706 inTraj[tch.allHitsIndex] = tch.InTraj;
709 std::array<int, 3> tIDs;
711 std::vector<std::array<int, 3>> mtIDs;
713 std::vector<unsigned short> mCnt;
715 unsigned short maxCnt = USHRT_MAX;
718 std::vector<unsigned short> tMaxed;
723 if (sptHits.size() != 3)
continue;
725 if (!
SptInTPC(sptHits, tpc))
continue;
726 unsigned short cnt = 0;
727 for (
unsigned short plane = 0; plane < 3; ++plane) {
728 unsigned int iht = sptHits[plane];
729 if (iht == UINT_MAX)
continue;
730 if (inTraj[iht] <= 0)
continue;
731 tIDs[plane] = inTraj[iht];
734 if (cnt != 3)
continue;
736 unsigned short indx = 0;
737 for (indx = 0; indx < mtIDs.size(); ++indx)
738 if (tIDs == mtIDs[indx])
break;
739 if (indx == mtIDs.size()) {
741 mtIDs.push_back(tIDs);
745 if (mCnt[indx] == maxCnt) {
747 tMaxed.insert(tMaxed.end(), tIDs[0]);
748 tMaxed.insert(tMaxed.end(), tIDs[1]);
749 tMaxed.insert(tMaxed.end(), tIDs[2]);
755 std::vector<SortEntry> sortVec;
756 for (
unsigned short indx = 0; indx < mCnt.size(); ++indx) {
757 auto& tIDs = mtIDs[indx];
759 float minTPCnt = USHRT_MAX;
760 for (
auto tid : tIDs) {
761 auto& tj = slc.
tjs[tid - 1];
763 if (tpcnt < minTPCnt) minTPCnt = tpcnt;
765 float frac = (float)mCnt[indx] / minTPCnt;
767 if (frac < 0.05)
continue;
771 sortVec.push_back(se);
773 if (sortVec.size() > 1) std::sort(sortVec.begin(), sortVec.end(),
valsDecreasing);
775 matVec.resize(sortVec.size());
777 for (
unsigned short ii = 0; ii < sortVec.size(); ++ii) {
778 unsigned short indx = sortVec[ii].index;
779 auto& ms = matVec[ii];
780 ms.Count = mCnt[indx];
782 for (
unsigned short plane = 0; plane < 3; ++plane)
783 ms.TjIDs[plane] = mtIDs[indx][plane];
789 bool SptInTPC(
const std::array<unsigned int, 3>& sptHits,
unsigned int tpc)
794 unsigned int ahi = UINT_MAX;
795 for (
auto ii : sptHits)
796 if (ii != UINT_MAX) {
800 if (ahi >= (*
evt.
allHits).size())
return false;
803 if (
hit.WireID().TPC == tpc)
return true;
826 std::array<unsigned short, 3> tIDs;
828 std::vector<std::array<unsigned short, 3>> mtIDs;
830 std::vector<unsigned short> mCnt;
832 unsigned short maxCnt = USHRT_MAX;
835 std::vector<unsigned short> tMaxed;
837 for (std::size_t ipt = 0; ipt < slc.
mallTraj.size() - 1; ++ipt) {
840 if (std::find(tMaxed.begin(), tMaxed.end(), iTjPt.id) != tMaxed.end())
continue;
841 auto& itp = slc.
tjs[iTjPt.id - 1].Pts[iTjPt.ipt];
842 unsigned int iPlane = iTjPt.plane;
843 unsigned int iWire = std::nearbyint(itp.Pos[0]);
844 tIDs[iPlane] = iTjPt.id;
845 bool hitMaxCnt =
false;
846 for (std::size_t jpt = ipt + 1; jpt < slc.
mallTraj.size() - 1; ++jpt) {
849 if (jTjPt.plane == iTjPt.plane)
continue;
851 if (jTjPt.xlo > iTjPt.xhi)
continue;
853 if (jTjPt.xlo > iTjPt.xhi + xcut)
break;
855 if (std::find(tMaxed.begin(), tMaxed.end(), jTjPt.id) != tMaxed.end())
continue;
856 auto& jtp = slc.
tjs[jTjPt.id - 1].Pts[jTjPt.ipt];
857 unsigned short jPlane = jTjPt.plane;
858 unsigned int jWire = jtp.Pos[0];
861 tIDs[jPlane] = jTjPt.id;
862 for (std::size_t kpt = jpt + 1; kpt < slc.
mallTraj.size(); ++kpt) {
865 if (kTjPt.plane == iTjPt.plane || kTjPt.plane == jTjPt.plane)
continue;
866 if (kTjPt.xlo > iTjPt.xhi)
continue;
868 if (kTjPt.xlo > iTjPt.xhi + xcut)
break;
870 if (std::find(tMaxed.begin(), tMaxed.end(), kTjPt.id) != tMaxed.end())
continue;
871 auto& ktp = slc.
tjs[kTjPt.id - 1].Pts[kTjPt.ipt];
872 unsigned short kPlane = kTjPt.plane;
873 unsigned int kWire = ktp.Pos[0];
876 if (
std::abs(ijPos[0] - ikPos[0]) > yzcut)
continue;
877 if (
std::abs(ijPos[1] - ikPos[1]) > yzcut)
continue;
881 unsigned int indx = 0;
882 for (indx = 0; indx < mtIDs.size(); ++indx)
883 if (tIDs == mtIDs[indx])
break;
884 if (indx == mtIDs.size()) {
886 mtIDs.push_back(tIDs);
890 if (mCnt[indx] == maxCnt) {
892 tMaxed.insert(tMaxed.end(), tIDs[0]);
893 tMaxed.insert(tMaxed.end(), tIDs[1]);
894 tMaxed.insert(tMaxed.end(), tIDs[2]);
899 if (hitMaxCnt)
break;
903 if (mCnt.empty())
return;
905 std::vector<SortEntry> sortVec;
906 for (std::size_t indx = 0; indx < mCnt.size(); ++indx) {
907 auto& tIDs = mtIDs[indx];
910 for (
auto tid : tIDs) {
911 auto& tj = slc.
tjs[tid - 1];
914 float frac = mCnt[indx] / tpCnt;
917 if (frac < 0.05)
continue;
921 sortVec.push_back(se);
923 if (sortVec.size() > 1) std::sort(sortVec.begin(), sortVec.end(),
valsDecreasing);
925 matVec.resize(sortVec.size());
927 for (std::size_t ii = 0; ii < sortVec.size(); ++ii) {
928 unsigned short indx = sortVec[ii].index;
929 auto& ms = matVec[ii];
930 ms.Count = mCnt[indx];
932 for (
unsigned short plane = 0; plane < 3; ++plane)
933 ms.TjIDs[plane] = (
int)mtIDs[indx][plane];
952 std::array<unsigned short, 2> tIDs;
954 std::vector<std::array<unsigned short, 2>> mtIDs;
956 std::vector<unsigned short> mCnt;
958 unsigned short maxCnt = USHRT_MAX;
961 std::vector<unsigned short> tMaxed;
963 for (std::size_t ipt = 0; ipt < slc.
mallTraj.size() - 1; ++ipt) {
966 if (std::find(tMaxed.begin(), tMaxed.end(), iTjPt.id) != tMaxed.end())
continue;
967 auto& itp = slc.
tjs[iTjPt.id - 1].Pts[iTjPt.ipt];
968 unsigned short iPlane = iTjPt.plane;
969 unsigned int iWire = itp.Pos[0];
970 bool hitMaxCnt =
false;
971 for (std::size_t jpt = ipt + 1; jpt < slc.
mallTraj.size() - 1; ++jpt) {
974 if (jTjPt.plane == iTjPt.plane)
continue;
976 if (jTjPt.xlo > iTjPt.xhi)
continue;
978 if (jTjPt.xlo > iTjPt.xhi + xcut)
break;
980 if (std::find(tMaxed.begin(), tMaxed.end(), jTjPt.id) != tMaxed.end())
continue;
981 auto& jtp = slc.
tjs[jTjPt.id - 1].Pts[jTjPt.ipt];
982 unsigned short jPlane = jTjPt.plane;
983 unsigned int jWire = jtp.Pos[0];
990 if (tIDs[0] > tIDs[1]) std::swap(tIDs[0], tIDs[1]);
992 std::size_t indx = 0;
993 for (indx = 0; indx < mtIDs.size(); ++indx)
994 if (tIDs == mtIDs[indx])
break;
995 if (indx == mtIDs.size()) {
997 mtIDs.push_back(tIDs);
1001 if (mCnt[indx] == maxCnt) {
1003 tMaxed.insert(tMaxed.end(), tIDs[0]);
1004 tMaxed.insert(tMaxed.end(), tIDs[1]);
1008 if (hitMaxCnt)
break;
1012 if (mCnt.empty())
return;
1014 std::vector<SortEntry> sortVec;
1015 for (std::size_t indx = 0; indx < mCnt.size(); ++indx) {
1016 auto& tIDs = mtIDs[indx];
1019 for (
auto tid : tIDs) {
1020 auto& tj = slc.
tjs[tid - 1];
1023 float frac = mCnt[indx] / tpCnt;
1026 if (frac < 0.05)
continue;
1029 se.
val = mCnt[indx];
1030 sortVec.push_back(se);
1032 if (sortVec.size() > 1) std::sort(sortVec.begin(), sortVec.end(),
valsDecreasing);
1034 matVec.resize(sortVec.size());
1036 for (std::size_t ii = 0; ii < sortVec.size(); ++ii) {
1037 unsigned short indx = sortVec[ii].index;
1038 auto& ms = matVec[ii];
1039 ms.Count = mCnt[indx];
1041 for (
unsigned short plane = 0; plane < 2; ++plane)
1042 ms.TjIDs[plane] = (
int)mtIDs[indx][plane];
1056 for (
unsigned short sfi = 0; sfi < pfp.
SectionFits.size(); ++sfi) {
1058 if (!sf.NeedsUpdate)
continue;
1062 for (
unsigned short ipt = 0; ipt < pfp.
TP3Ds.size(); ++ipt) {
1063 auto& tp3d = pfp.
TP3Ds[ipt];
1064 if (tp3d.SFIndex < sfi)
continue;
1065 if (tp3d.SFIndex > sfi)
break;
1067 double delta = tp3d.Pos[0] - tp3d.TPX;
1068 sf.ChiDOF += delta * delta / tp3d.TPXErr2;
1070 if (sf.NPts < 5) { sf.ChiDOF = 0; }
1072 sf.ChiDOF /= (float)(sf.NPts - 4);
1074 sf.NeedsUpdate =
false;
1080 for (std::size_t sfi = 0; sfi < pfp.
SectionFits.size(); ++sfi) {
1082 if (!sf.NeedsUpdate)
continue;
1083 if (!
FitSection(detProp, slc, pfp, sfi))
return false;
1085 sf.NeedsUpdate =
false;
1089 for (
auto& tp3d : pfp.
TP3Ds) {
1111 if (pfp.
TP3Ds.size() < 12) {
1127 for (
auto& tp3d : pfp.
TP3Ds) {
1132 if (!
FitSection(detProp, slc, pfp, 0)) {
return false; }
1138 unsigned short min2DPts = 3;
1139 unsigned short fromPt = 0;
1141 for (
auto& tp3d : pfp.
TP3Ds)
1142 tp3d.SFIndex = USHRT_MAX;
1144 unsigned short nPtsToAdd = pfp.
TP3Ds.size() / 4;
1146 unsigned short nPts = nPtsToAdd;
1148 unsigned short nPtsMin =
Find3DRecoRange(slc, pfp, fromPt, min2DPts, 1) - fromPt + 1;
1149 if (nPtsMin >= pfp.
TP3Ds.size()) {
1154 if (nPts < nPtsMin) nPts = nPtsMin;
1156 if (pfp.
TP3Ds.size() > 100) {
1157 unsigned short nhalf = pfp.
TP3Ds.size() / 2;
1158 FitTP3Ds(detProp, slc, pfp, fromPt, nhalf, USHRT_MAX);
1161 bool lastSection =
false;
1162 for (
unsigned short sfIndex = 0; sfIndex < 20; ++sfIndex) {
1164 float chiDOFPrev = 0;
1166 for (
unsigned short nit = 0; nit < 10; ++nit) {
1168 unsigned short nPtsNext = nPts;
1169 if (!
FitTP3Ds(detProp, slc, pfp, fromPt, nPts, USHRT_MAX)) { nPtsNext += 1.5 * nPtsToAdd; }
1170 else if (chiDOF < chiLo) {
1177 nPtsNext += nPtsToAdd;
1181 else if (chiDOF > chiHi) {
1190 short npnext = (short)nPts - nHiChi * 5;
1193 if (npnext > nPtsMin) nPtsNext = npnext;
1201 if (fromPt + nPtsNext >= pfp.
TP3Ds.size()) {
1202 nPtsNext = pfp.
TP3Ds.size() - fromPt;
1207 myprt <<
" RS: P" << pfp.
ID <<
" sfi/nit/npts " << sfIndex <<
"/" << nit <<
"/" << nPts;
1208 myprt << std::fixed << std::setprecision(1) <<
" chiDOF " << chiDOF;
1209 myprt <<
" fromPt " << fromPt;
1210 myprt <<
" nPtsNext " << nPtsNext;
1211 myprt <<
" nHiChi " << nHiChi;
1212 myprt <<
" lastSection? " << lastSection;
1214 if (nPtsNext == 0)
break;
1216 if (lastSection)
break;
1217 if (chiDOF == chiDOFPrev) {
1222 chiDOFPrev = chiDOF;
1225 unsigned short toPt = fromPt + nPts;
1226 if (toPt > pfp.
TP3Ds.size()) toPt = pfp.
TP3Ds.size();
1227 for (
unsigned short ipt = fromPt; ipt < toPt; ++ipt)
1228 pfp.
TP3Ds[ipt].SFIndex = sfIndex;
1233 unsigned short nextFromPt = fromPt + nPts;
1235 unsigned short nextToPtMin =
Find3DRecoRange(slc, pfp, nextFromPt, min2DPts, 1);
1236 if (nextToPtMin == USHRT_MAX) {
1240 for (std::size_t ipt = nextFromPt; ipt < pfp.
TP3Ds.size(); ++ipt)
1241 pfp.
TP3Ds[ipt].SFIndex = sfIndex;
1247 if (lastSection)
break;
1249 fromPt = fromPt + nPts;
1251 nPtsMin =
Find3DRecoRange(slc, pfp, fromPt, min2DPts, 1) - fromPt + 1;
1252 if (nPtsMin >= pfp.
TP3Ds.size())
break;
1259 unsigned short badSFI = pfp.
SectionFits.size() - 1;
1262 for (std::size_t ipt = pfp.
TP3Ds.size() - 1; ipt > 0; --ipt) {
1263 auto& tp3d = pfp.
TP3Ds[ipt];
1264 if (tp3d.SFIndex < badSFI)
break;
1271 for (std::size_t ipt = pfp.
TP3Ds.size() - 1; ipt > 0; --ipt) {
1272 auto& tp3d = pfp.
TP3Ds[ipt];
1279 Update(detProp, slc, pfp);
1292 unsigned short fromPt,
1293 unsigned short toPt,
1294 unsigned short& nBadPts,
1295 unsigned short& firstBadPt)
1298 firstBadPt = USHRT_MAX;
1300 if (fromPt > pfp.
TP3Ds.size() - 1) {
1301 nBadPts = USHRT_MAX;
1304 if (toPt > pfp.
TP3Ds.size()) toPt = pfp.
TP3Ds.size();
1306 for (
unsigned short ipt = fromPt; ipt < toPt; ++ipt) {
1307 auto& tp3d = pfp.
TP3Ds[ipt];
1311 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
1329 if (pfp.
TP3Ds.size() < 12)
return false;
1331 if (toPt > pfp.
TP3Ds.size())
return false;
1333 if (nextToPt > pfp.
TP3Ds.size())
return false;
1340 unsigned short fromPt,
1341 unsigned short min2DPts,
1347 if (fromPt > pfp.
TP3Ds.size() - 1)
return USHRT_MAX;
1348 if (pfp.
TP3Ds.size() < 2 * min2DPts)
return USHRT_MAX;
1349 if (dir == 0)
return USHRT_MAX;
1351 std::vector<unsigned short> cntInPln(slc.
nPlanes);
1352 for (std::size_t ii = 0; ii < pfp.
TP3Ds.size(); ++ii) {
1353 unsigned short ipt = fromPt + ii;
1354 if (dir < 0) ipt = fromPt - ii;
1355 if (ipt >= pfp.
TP3Ds.size())
break;
1356 auto& tp3d = pfp.
TP3Ds[ipt];
1360 unsigned short enufInPlane = 0;
1361 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane)
1362 if (cntInPln[plane] >= min2DPts) ++enufInPlane;
1363 if (enufInPlane > 1)
return ipt + 1;
1364 if (dir < 0 && ipt == 0)
break;
1371 unsigned short sfIndex,
1372 unsigned short& fromPt,
1373 unsigned short& npts)
1377 if (pfp.
TP3Ds.empty())
return;
1381 for (std::size_t ipt = 0; ipt < pfp.
TP3Ds.size(); ++ipt) {
1382 auto& tp3d = pfp.
TP3Ds[ipt];
1383 if (tp3d.SFIndex < sfIndex)
continue;
1384 if (tp3d.SFIndex > sfIndex)
break;
1385 if (fromPt == USHRT_MAX) fromPt = ipt;
1394 unsigned short sfIndex)
1398 if (pfp.
TP3Ds.size() < 4)
return false;
1399 if (sfIndex >= pfp.
SectionFits.size())
return false;
1403 unsigned short fromPt = USHRT_MAX;
1404 unsigned short npts = 0;
1405 GetRange(pfp, sfIndex, fromPt, npts);
1406 if (fromPt == USHRT_MAX)
return false;
1407 if (npts < 4)
return false;
1410 for (
unsigned short ipt = fromPt; ipt < fromPt + npts; ++ipt) {
1411 auto& tp3d = pfp.
TP3Ds[ipt];
1412 if (tp3d.SFIndex != sfIndex)
return false;
1416 return FitTP3Ds(detProp, slc, pfp, fromPt, npts, sfIndex);
1423 const std::vector<TP3D>& tp3ds,
1424 unsigned short fromPt,
1426 unsigned short nPtsFit)
1433 if (nPtsFit < 5)
return sf;
1434 if (!(fitDir == -1 || fitDir == 1))
return sf;
1435 if (fitDir == 1 && fromPt + nPtsFit > tp3ds.size())
return sf;
1436 if (fitDir == -1 && fromPt < 3)
return sf;
1439 std::vector<std::array<double, 3>> ocs(slc.
nPlanes);
1440 for (
unsigned short p = 0; p < slc.
nPlanes; ++p) {
1444 ocs[p][0] = plane.WireCoordinate(
geo::Point_t{0, 0, 0});
1446 ocs[p][1] = plane.WireCoordinate(
geo::Point_t{0, 1, 0}) - ocs[p][0];
1448 ocs[p][2] = plane.WireCoordinate(
geo::Point_t{0, 0, 1}) - ocs[p][0];
1451 const unsigned int nvars = 4;
1452 unsigned int npts = 0;
1455 std::vector<unsigned short> cntInPln(slc.
nPlanes, 0);
1458 for (
short ii = 0; ii < nPtsFit; ++ii) {
1459 short ipt = fromPt + fitDir * ii;
1460 if (ipt < 0 || ipt >= (
short)tp3ds.size())
break;
1461 auto& tp3d = tp3ds[ipt];
1463 if (tp3d.TPXErr2 < 0.0001)
return sf;
1469 if (npts < 6)
return sf;
1471 unsigned short enufInPlane = 0;
1472 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane)
1473 if (cntInPln[plane] > 2) ++enufInPlane;
1474 if (enufInPlane < 2)
return sf;
1478 TMatrixD A(npts, nvars);
1481 unsigned short cnt = 0;
1483 for (
short ii = 0; ii < nPtsFit; ++ii) {
1484 short ipt = fromPt + fitDir * ii;
1485 auto& tp3d = tp3ds[ipt];
1488 double x = tp3d.TPX - x0;
1489 A[cnt][0] = weight * ocs[plane][1];
1490 A[cnt][1] = weight * ocs[plane][2];
1491 A[cnt][2] = weight * ocs[plane][1] *
x;
1492 A[cnt][3] = weight * ocs[plane][2] *
x;
1493 w[cnt] = weight * (tp3d.Wire - ocs[plane][0]);
1499 TVectorD tVec = svd.Solve(w, ok);
1501 double norm = sqrt(1 + tVec[2] * tVec[2] + tVec[3] * tVec[3]);
1509 sf.
Pos[1] = tVec[0];
1510 sf.
Pos[2] = tVec[1];
1515 TMatrixD AT(nvars, npts);
1517 TMatrixD ATA = AT * A;
1531 std::vector<TrajPoint> plnTP(slc.
nPlanes);
1532 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
1539 for (
short ii = 0; ii < nPtsFit; ++ii) {
1540 short ipt = fromPt + fitDir * ii;
1541 auto& tp3d = tp3ds[ipt];
1544 double dw = tp3d.Wire - plnTP[plane].Pos[0];
1546 double t = dw * plnTP[plane].DeltaRMS;
1547 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
1548 pos[xyz] = sf.
Pos[xyz] + t * sf.
Dir[xyz];
1552 double delta = pos[0] - tp3d.TPX;
1553 sf.
ChiDOF += delta * delta / tp3d.TPXErr2;
1555 double dangErr = delta / dw;
1556 sf.
DirErr[0] += dangErr * dangErr;
1559 sf.
ChiDOF /= (float)(npts - 4);
1568 unsigned short fromPt,
1569 unsigned short nPtsFit,
1570 unsigned short sfIndex)
1576 if (nPtsFit < 5)
return false;
1577 if (fromPt + nPtsFit > pfp.
TP3Ds.size())
return false;
1579 auto sf =
FitTP3Ds(detProp, slc, pfp.
TP3Ds, fromPt, 1, nPtsFit);
1580 if (sf.ChiDOF > 900)
return false;
1583 if (sfIndex >= pfp.
SectionFits.size())
return true;
1589 std::vector<TrajPoint> plnTP(slc.
nPlanes);
1590 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
1592 plnTP[plane] =
MakeBareTP(detProp, sf.Pos, sf.Dir, inCTP);
1595 bool needsSort =
false;
1596 double prevAlong = 0;
1597 for (
unsigned short ipt = fromPt; ipt < fromPt + nPtsFit; ++ipt) {
1598 auto& tp3d = pfp.
TP3Ds[ipt];
1600 double dw = tp3d.Wire - plnTP[plane].Pos[0];
1602 double t = dw * plnTP[plane].DeltaRMS;
1603 if (ipt == fromPt) { prevAlong = t; }
1605 if (t < prevAlong) needsSort =
true;
1608 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
1609 pos[xyz] = sf.Pos[xyz] + t * sf.Dir[xyz];
1613 double delta = pos[0] - tp3d.TPX;
1617 if (tp3d.Flags[
kTP3DGood]) sf.ChiDOF += delta * delta / tp3d.TPXErr2;
1637 if (pfp.
TP3Ds.empty())
return;
1642 std::vector<int> tjList;
1643 for (
auto& tp3d : pfp.
TP3Ds) {
1646 if (tp3d.TjID <= 0)
continue;
1647 if (std::find(tjList.begin(), tjList.end(), tp3d.TjID) == tjList.end())
1648 tjList.push_back(tp3d.TjID);
1652 std::vector<int> vx2List, vx3List;
1653 for (
auto tid : tjList) {
1654 auto& tj = slc.
tjs[tid - 1];
1655 for (
unsigned short end = 0;
end < 2; ++
end) {
1656 if (tj.VtxID[
end] <= 0)
continue;
1657 auto& vx2 = slc.
vtxs[tj.VtxID[
end] - 1];
1658 if (vx2.Vx3ID > 0) {
1659 if (std::find(vx3List.begin(), vx3List.end(), vx2.Vx3ID) == vx3List.end())
1660 vx3List.push_back(vx2.Vx3ID);
1665 if (std::find(vx2List.begin(), vx2List.end(), tj.VtxID[
end]) == vx2List.end())
1666 vx2List.push_back(tj.VtxID[
end]);
1671 if (vx2List.empty() && vx3List.empty())
return;
1674 myprt <<
"RV: P" << pfp.
ID <<
" ->";
1675 for (
auto tid : tjList)
1676 myprt <<
" T" << tid;
1678 for (
auto vid : vx3List)
1679 myprt <<
" 3V" << vid;
1680 if (!vx2List.empty()) {
1682 for (
auto vid : vx2List)
1683 myprt <<
" 2V" << vid;
1691 for (
auto vid : vx2List) {
1692 auto& vx2 = slc.
vtxs[vid - 1];
1701 if (!slc.
pfps.empty()) {
1702 auto& npfp = slc.
pfps[0];
1703 bool neutrinoPFP = (npfp.PDGCode == 12 || npfp.PDGCode == 14);
1704 if (neutrinoPFP) neutrinoVx = npfp.Vx3ID[0];
1706 unsigned short neutrinoVxEnd = 2;
1707 for (
unsigned short end = 0;
end < 2; ++
end) {
1710 if (pfp.
Vx3ID[
end] == neutrinoVx) neutrinoVxEnd =
end;
1712 if (std::find(vx3List.begin(), vx3List.end(), pfp.
Vx3ID[
end]) != vx3List.end())
continue;
1714 if (neutrinoVxEnd < 2 && neutrinoVxEnd != 0)
Reverse(pfp);
1731 if (pfp.
ID <= 0)
return;
1732 if (pfp.
TP3Ds.empty())
return;
1744 unsigned short nPtsAdded = 0;
1745 unsigned short fromPt = 0;
1746 unsigned short toPt = pWork.
TP3Ds.size();
1747 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
1748 CTP_t inCTP =
EncodeCTP(pWork.TPCID.Cryostat, pWork.TPCID.TPC, plane);
1749 unsigned short nWires, nAdd;
1764 if (prt)
mf::LogVerbatim(
"TC") <<
"FG3D P" << pWork.ID <<
" added " << nPtsAdded <<
" points";
1778 if (pfp.
TjIDs.size() != 2)
return false;
1779 if (slc.
nPlanes != 3)
return false;
1780 if (pfp.
TP3Ds.empty())
return false;
1783 std::vector<unsigned short> planes;
1784 for (
auto tid : pfp.
TjIDs)
1786 unsigned short thirdPlane = 3 - planes[0] - planes[1];
1790 unsigned int wire0 = 0;
1791 if (tp.Pos[0] > 0) wire0 = std::nearbyint(tp.Pos[0]);
1792 if (wire0 > slc.
nWires[thirdPlane]) wire0 = slc.
nWires[thirdPlane];
1794 unsigned short lastPt = pfp.
TP3Ds.size() - 1;
1796 unsigned int wire1 = 0;
1797 if (tp.Pos[0] > 0) wire1 = std::nearbyint(tp.Pos[0]);
1798 if (wire1 > slc.
nWires[thirdPlane]) wire1 = slc.
nWires[thirdPlane];
1799 if (wire0 == wire1)
return !
evt.
goodWire[thirdPlane][wire0];
1800 if (wire1 < wire0) std::swap(wire0, wire1);
1803 int wires = wire1 - wire0;
1804 for (
unsigned int wire = wire0; wire < wire1; ++wire)
1807 return (dead > 0.8 * wires);
1815 unsigned short fromPt,
1816 unsigned short toPt,
1819 unsigned short& nWires,
1820 unsigned short& nAdd,
1829 if (fromPt > toPt)
return;
1830 if (toPt >= pfp.
TP3Ds.size()) toPt = pfp.
TP3Ds.size() - 1;
1835 Average_dEdX(clockData, detProp, slc, pfp, dEdXAve, dEdXRms);
1836 float dEdxMin = 0.5, dEdxMax = 50.;
1837 if (dEdXAve > 0.5) {
1838 dEdxMin = dEdXAve - maxPull * dEdXRms;
1839 if (dEdxMin < 0.5) dEdxMin = 0.5;
1840 dEdxMax = dEdXAve + maxPull * dEdXRms;
1841 if (dEdxMax > 50.) dEdxMax = 50.;
1846 std::vector<TrajPoint> sfTPs;
1848 std::vector<std::pair<int, unsigned short>> tpUsed;
1849 for (
auto& tp3d : pfp.
TP3Ds) {
1850 if (tp3d.CTP != inCTP)
continue;
1851 tpUsed.push_back(std::make_pair(tp3d.TjID, tp3d.TPIndex));
1853 unsigned int toWire = 0;
1854 unsigned int fromWire = UINT_MAX;
1856 unsigned short inSF = USHRT_MAX;
1858 for (
unsigned short ipt = 0; ipt < pfp.
TP3Ds.size(); ++ipt) {
1859 auto& tp3d = pfp.
TP3Ds[ipt];
1861 if (ipt < pfp.
TP3Ds.size() - 1 && tp3d.SFIndex == inSF)
continue;
1863 if (tp3d.CTP == inCTP) {
1865 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
1866 sfTPs.push_back(tp);
1867 wire = std::nearbyint(tp.Pos[0]);
1868 if (wire >= slc.
nWires[pln])
break;
1873 auto tp =
MakeBareTP(detProp, tp3d.Pos, tp3d.Dir, inCTP);
1874 wire = std::nearbyint(tp.Pos[0]);
1875 if (wire >= slc.
nWires[pln])
break;
1876 sfTPs.push_back(tp);
1878 if (wire < fromWire) fromWire = wire;
1879 if (wire > toWire) toWire = wire;
1880 inSF = tp3d.SFIndex;
1882 if (sfTPs.empty())
return;
1884 if (sfTPs.size() > 1 && sfTPs[0].Pos[0] > sfTPs.back().Pos[0]) {
1885 std::reverse(sfTPs.begin(), sfTPs.end());
1892 mf::LogVerbatim(
"TC") <<
"APIR: inCTP " << inCTP <<
" fromWire " << fromWire <<
" toWire " 1896 for (
unsigned short subr = 0; subr < sfTPs.size(); ++subr) {
1897 auto& fromTP = sfTPs[subr];
1898 unsigned int toWireInSF = toWire;
1899 if (subr < sfTPs.size() - 1) toWireInSF = std::nearbyint(sfTPs[subr + 1].Pos[0]);
1901 unsigned int fromWire = std::nearbyint(fromTP.Pos[0]);
1902 if (fromWire > toWire)
continue;
1904 mf::LogVerbatim(
"TC") <<
" inCTP " << inCTP <<
" subr " << subr <<
" fromWire " << fromWire
1905 <<
" toWireInSF " << toWireInSF;
1906 for (
unsigned int wire = fromWire; wire <= toWireInSF; ++wire) {
1910 float bestPull = maxPull;
1912 for (
auto iht : fromTP.Hits) {
1913 if (slc.
slHits[iht].InTraj <= 0)
continue;
1915 auto& utj = slc.
tjs[slc.
slHits[iht].InTraj - 1];
1916 unsigned short tpIndex = 0;
1917 for (tpIndex = utj.EndPt[0]; tpIndex <= utj.EndPt[1]; ++tpIndex) {
1918 auto& utp = utj.Pts[tpIndex];
1919 if (utp.Chg <= 0)
continue;
1921 if (std::find(utp.Hits.begin(), utp.Hits.end(), iht) != utp.Hits.end())
break;
1923 if (tpIndex > utj.EndPt[1])
continue;
1925 std::pair<int, unsigned short> tppr = std::make_pair(utj.ID, tpIndex);
1926 if (std::find(tpUsed.begin(), tpUsed.end(), tppr) != tpUsed.end())
continue;
1927 tpUsed.push_back(tppr);
1928 auto& utp = utj.Pts[tpIndex];
1930 if (utp.InPFP > 0)
continue;
1933 auto newTP3D =
CreateTP3D(detProp, slc, utj.
ID, tpIndex);
1934 if (!
SetSection(detProp, pfp, newTP3D))
continue;
1936 newTP3D.Dir = pfp.
SectionFits[newTP3D.SFIndex].Dir;
1938 float dedx =
dEdx(clockData, detProp, slc, newTP3D);
1940 bool useIt = (pull < bestPull && dedx > dEdxMin && dedx < dEdxMax);
1941 if (!useIt)
continue;
1945 if (bestPull < maxPull) {
1946 if (prt && bestPull < 10) {
1949 myprt <<
"APIR: P" << pfp.
ID <<
" added TP " <<
PrintPos(tp);
1950 myprt <<
" pull " << std::fixed << std::setprecision(2) << bestPull;
1951 myprt <<
" dx " << bestTP3D.
TPX - bestTP3D.
Pos[0] <<
" in section " << bestTP3D.
SFIndex;
1953 if (
InsertTP3D(pfp, bestTP3D) == USHRT_MAX)
continue;
1966 std::size_t ipt = 0;
1967 for (ipt = 0; ipt < pfp.
TP3Ds.size(); ++ipt)
1969 if (ipt == pfp.
TP3Ds.size())
return USHRT_MAX;
1971 auto lastTP3D = pfp.
TP3Ds.back();
1972 if (ipt == 0 && tp3d.
along < pfp.
TP3Ds[0].along) {
1975 else if (tp3d.
SFIndex == lastTP3D.SFIndex && tp3d.
along > lastTP3D.along) {
1977 pfp.
TP3Ds.push_back(tp3d);
1980 return pfp.
TP3Ds.size() - 1;
1983 for (std::size_t iipt = ipt; iipt < pfp.
TP3Ds.size() - 1; ++iipt) {
1985 if (pfp.
TP3Ds[iipt + 1].SFIndex != tp3d.
SFIndex)
break;
1992 pfp.
TP3Ds.insert(pfp.
TP3Ds.begin() + ipt, tp3d);
2003 if (sfIndex > pfp.
SectionFits.size() - 1)
return false;
2005 if (sf.Pos[0] == 0.0 && sf.Pos[1] == 0.0 && sf.Pos[2] == 0.0)
return false;
2008 std::vector<TP3D> temp;
2010 std::vector<unsigned short> indx;
2012 float prevAlong = 0;
2014 bool needsSort =
false;
2015 for (std::size_t ii = 0; ii < pfp.
TP3Ds.size(); ++ii) {
2016 auto& tp3d = pfp.
TP3Ds[ii];
2017 if (tp3d.SFIndex != sfIndex)
continue;
2020 prevAlong = tp3d.along;
2023 if (tp3d.along < prevAlong) needsSort =
true;
2024 prevAlong = tp3d.along;
2026 temp.push_back(tp3d);
2029 if (temp.empty())
return false;
2031 if (temp.size() == 1)
return true;
2033 sf.NeedsUpdate =
false;
2037 bool contiguous =
true;
2038 for (std::size_t ipt = 1; ipt < indx.size(); ++ipt) {
2039 if (indx[ipt] != indx[ipt - 1] + 1) contiguous =
false;
2041 if (!contiguous) {
return false; }
2043 std::vector<SortEntry> sortVec(temp.size());
2044 for (std::size_t ii = 0; ii < temp.size(); ++ii) {
2045 sortVec[ii].index = ii;
2046 sortVec[ii].val = temp[ii].along;
2049 for (std::size_t ii = 0; ii < temp.size(); ++ii) {
2051 auto& tp3d = pfp.
TP3Ds[indx[ii]];
2052 tp3d = temp[sortVec[ii].index];
2054 sf.NeedsUpdate =
false;
2067 if (pfp.
TP3Ds.size() < 20)
return;
2074 unsigned short halfPt = p2.TP3Ds.size() / 2;
2075 for (
unsigned short ipt = halfPt; ipt < p2.TP3Ds.size(); ++ipt)
2076 p2.TP3Ds[ipt].SFIndex = 1;
2079 if (toPt > p2.TP3Ds.size())
return;
2081 if (toPt > p2.TP3Ds.size())
return;
2085 myprt <<
"Recover failed MVI " << p2.MVI <<
" in TPC " << p2.TPCID.TPC;
2086 for (
auto tid : p2.TjIDs)
2087 myprt <<
" T" << tid;
2118 unsigned short nJunk = 0;
2119 unsigned short nSA = 0;
2120 for (
unsigned short itj = 0; itj < pfp.
TjIDs.size(); ++itj) {
2121 auto& tj = slc.
tjs[pfp.
TjIDs[itj] - 1];
2122 if (tj.AlgMod[
kJunkTj]) ++nJunk;
2124 unsigned short iptMin = USHRT_MAX;
2125 float posMax = -1E6;
2126 unsigned short iptMax = USHRT_MAX;
2129 for (
unsigned short ipt = tj.EndPt[0]; ipt < tj.EndPt[1]; ++ipt) {
2130 auto& tp = tj.Pts[ipt];
2131 if (tp.Chg <= 0)
continue;
2133 if (tp.InPFP > 0)
continue;
2135 if (tp.Pos[1] > posMax) {
2139 if (tp.Pos[1] < posMin) {
2146 if (npwc == 0)
continue;
2148 if (
std::abs(aveAng) < 0.05) ++nSA;
2150 if (iptMin > tj.EndPt[0] + 4 && iptMin < tj.EndPt[1] - 4) pfp.
TjUIDs[itj] = iptMin;
2151 if (iptMax > tj.EndPt[0] + 4 && iptMax < tj.EndPt[1] - 4) pfp.
TjUIDs[itj] = iptMax;
2153 if (avail < 0.8 * cnt)
return false;
2163 for (
auto tid : pfp.
TjIDs) {
2164 auto& tj = slc.
tjs[tid - 1];
2166 if (nJunk == 1 && tj.AlgMod[
kJunkTj])
continue;
2169 bool isJunk = tj.AlgMod[
kJunkTj];
2170 for (
unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2171 auto& tp = tj.Pts[ipt];
2172 if (tp.Chg <= 0)
continue;
2173 if (tp.InPFP > 0)
continue;
2175 auto tp3d =
CreateTP3D(detProp, slc, tid, ipt);
2176 if (tp3d.Flags[
kTP3DBad])
continue;
2178 if (isJunk) tp3d.TPXErr2 *= 4;
2181 pfp.
TP3Ds.push_back(tp3d);
2200 if (pfp.
TjIDs.size() < 2)
return false;
2202 std::vector<SortEntry> sortVec(pfp.
TjIDs.size());
2203 unsigned short sbCnt = 0;
2204 for (
unsigned short itj = 0; itj < pfp.
TjIDs.size(); ++itj) {
2205 sortVec[itj].index = itj;
2206 auto& tj = slc.
tjs[pfp.
TjIDs[itj] - 1];
2208 if (pfp.
TjUIDs[itj] > 0) ++sbCnt;
2214 unsigned short tlIndex = sortVec[0].index;
2215 unsigned short nlIndex = sortVec[1].index;
2216 auto& tlong = slc.
tjs[pfp.
TjIDs[tlIndex] - 1];
2217 auto& nlong = slc.
tjs[pfp.
TjIDs[nlIndex] - 1];
2218 bool twoSections = (sbCnt > 1 && pfp.
TjUIDs[tlIndex] > 0 && pfp.
TjUIDs[nlIndex] > 0);
2219 unsigned short tStartPt = tlong.EndPt[0];
2220 unsigned short tEndPt = tlong.EndPt[1];
2221 unsigned short nStartPt = nlong.EndPt[0];
2222 unsigned short nEndPt = nlong.EndPt[1];
2225 tEndPt = pfp.
TjUIDs[tlIndex];
2226 nEndPt = pfp.
TjUIDs[nlIndex];
2229 myprt <<
"MakeSmallAnglePFP: creating two sections using points";
2230 myprt <<
" T" << tlong.ID <<
"_" << tEndPt;
2231 myprt <<
" T" << nlong.ID <<
"_" << nEndPt;
2234 std::vector<Point3_t> sfEndPos;
2235 for (
unsigned short isf = 0; isf < pfp.
SectionFits.size(); ++isf) {
2237 auto& ltp0 = tlong.Pts[tStartPt];
2238 auto& ltp1 = tlong.Pts[tEndPt];
2239 auto& ntp0 = nlong.Pts[nStartPt];
2240 auto& ntp1 = nlong.Pts[nEndPt];
2242 auto start =
MakeTP3D(detProp, ltp0, ntp0);
2245 std::cout <<
" Start/end fail in section " << isf <<
". Add recovery code\n";
2249 mf::LogVerbatim(
"TC") <<
" Start is outside the TPC " << start.Pos[0] <<
" " << start.Pos[1]
2250 <<
" " << start.Pos[2];
2254 <<
" " <<
end.Pos[2];
2256 if (isf == 0) sfEndPos.push_back(start.Pos);
2257 sfEndPos.push_back(
end.Pos);
2260 for (
unsigned short xyz = 0; xyz < 3; ++xyz) {
2261 sf.Dir[xyz] =
end.Pos[xyz] - start.Pos[xyz];
2262 sf.Pos[xyz] = (
end.Pos[xyz] + start.Pos[xyz]) / 2.;
2268 tStartPt = tEndPt + 1;
2269 tEndPt = tlong.EndPt[1];
2270 nStartPt = nEndPt + 1;
2271 nEndPt = nlong.EndPt[1];
2275 std::vector<TP3D> sf2pts;
2276 for (
unsigned short itj = 0; itj < sortVec.size(); ++itj) {
2277 int tid = pfp.
TjIDs[sortVec[itj].index];
2280 if (twoSections && pfp.
TjUIDs[sortVec[itj].index] < 0)
continue;
2281 auto& tj = slc.
tjs[tid - 1];
2282 unsigned short sb = tj.EndPt[1];
2283 if (twoSections && pfp.
TjUIDs[sortVec[itj].index] > 0) sb = pfp.
TjUIDs[sortVec[itj].index];
2285 std::vector<double> npwc(pfp.
SectionFits.size(), 0);
2286 for (
unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2287 auto& tp = tj.Pts[ipt];
2288 if (tp.Chg <= 0)
continue;
2289 if (ipt > sb) { ++npwc[1]; }
2294 double length =
PosSep(sfEndPos[0], sfEndPos[1]);
2295 double step = length / npwc[0];
2296 double along = -length / 2;
2297 unsigned short sfi = 0;
2298 for (
unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2299 auto& tp = tj.Pts[ipt];
2300 if (tp.Chg <= 0)
continue;
2301 auto tp3d =
CreateTP3D(detProp, slc, tid, ipt);
2302 if (tp3d.Flags[
kTP3DBad])
continue;
2303 if (ipt == sb + 1) {
2305 length =
PosSep(sfEndPos[1], sfEndPos[2]);
2306 step = length / npwc[1];
2307 along = -length / 2;
2313 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
2314 tp3d.Pos[xyz] = sf.Pos[xyz] + along * sf.Dir[xyz];
2317 double delta = tp3d.Pos[0] - tp3d.TPX;
2318 sf.ChiDOF += delta * delta / tp3d.TPXErr2;
2321 if (sfi == 0) { pfp.
TP3Ds.push_back(tp3d); }
2323 sf2pts.push_back(tp3d);
2327 if (pfp.
TP3Ds.size() < 4)
return false;
2329 if (sf.NPts < 5)
return false;
2330 sf.ChiDOF /= (float)(sf.NPts - 4);
2333 if (!sf2pts.empty()) {
2335 pfp.
TP3Ds.insert(pfp.
TP3Ds.end(), sf2pts.begin(), sf2pts.end());
2342 <<
" points in " << pfp.
SectionFits.size() <<
" sections\n";
2351 std::reverse(pfp.
TP3Ds.begin(), pfp.
TP3Ds.end());
2353 for (std::size_t sfi = 0; sfi < pfp.
SectionFits.size(); ++sfi) {
2356 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
2360 for (
auto& tp3d : pfp.
TP3Ds)
2362 std::swap(pfp.
dEdx[0], pfp.
dEdx[1]);
2380 unsigned short cnt = 0;
2386 for (
auto& tj : slc.
tjs) {
2389 if (tj.AlgMod[
kMat3D])
continue;
2391 if ((
int)planeID.
Cryostat != cstat)
continue;
2392 if ((
int)planeID.
TPC != tpc)
continue;
2393 int plane = planeID.
Plane;
2394 if (tj.ID <= 0)
continue;
2395 unsigned short tjID = tj.ID;
2396 for (
unsigned short ipt = tj.EndPt[0]; ipt <= tj.EndPt[1]; ++ipt) {
2397 auto& tp = tj.Pts[ipt];
2398 if (tp.Chg <= 0)
continue;
2399 if (tp.Pos[0] < -0.4)
continue;
2401 if (tp.InPFP > 0)
continue;
2402 if (muFuzzCut && tp.Environment[
kEnvNearMuon])
continue;
2403 tj2pt.
wire = std::nearbyint(tp.Pos[0]);
2408 tj2pt.
xlo = xpos - rms;
2409 tj2pt.
xhi = xpos + rms;
2410 tj2pt.
plane = plane;
2413 tj2pt.
npts = tj.EndPt[1] - tj.EndPt[0] + 1;
2419 std::vector<SortEntry> sortVec(slc.
mallTraj.size());
2420 for (std::size_t ipt = 0; ipt < slc.
mallTraj.size(); ++ipt) {
2422 sortVec[ipt].index = ipt;
2423 sortVec[ipt].val = slc.
mallTraj[ipt].xlo;
2429 for (std::size_t ii = 0; ii < sortVec.size(); ++ii)
2430 slc.
mallTraj[ii] = tallTraj[sortVec[ii].index];
2447 tp3d.
Dir = {{0.0, 0.0, 1.0}};
2448 tp3d.
Pos = {{999.0, 999.0, 999.0}};
2451 if (iPlnID == jPlnID)
return tp3d;
2458 if (dx > 20)
return tp3d;
2459 tp3d.
Pos[0] = (ix + jx) / 2;
2470 double ics = iplane.WireCoordinate(
geo::Point_t{0, 1, 0}) - iw0;
2472 double isn = iplane.WireCoordinate(
geo::Point_t{0, 0, 1}) - iw0;
2473 double jw0 = jplane.WireCoordinate(
geo::Point_t{0, 0, 0});
2474 double jcs = jplane.WireCoordinate(
geo::Point_t{0, 1, 0}) - jw0;
2475 double jsn = jplane.WireCoordinate(
geo::Point_t{0, 0, 1}) - jw0;
2476 double den = isn * jcs - ics * jsn;
2477 if (den == 0)
return tp3d;
2478 double iPos0 = itp.
Pos[0];
2479 double jPos0 = jtp.
Pos[0];
2481 tp3d.
Pos[2] = (jcs * (iPos0 - iw0) - ics * (jPos0 - jw0)) / den;
2484 if (useI) { tp3d.
Pos[1] = (iPos0 - iw0 - isn * tp3d.
Pos[2]) / ics; }
2486 tp3d.
Pos[1] = (jPos0 - jw0 - jsn * tp3d.
Pos[2]) / jcs;
2490 if (jtp.
Dir[1] == 0) {
2492 if (jtp.
Dir[0] > 0) { tp3d.
Dir[0] = 1; }
2504 double itp2_0 = itp.
Pos[0] + 100;
2505 double itp2_1 = itp.
Pos[1];
2514 double jtp2Pos0 = (jtp2Pos1 - jtp.
Pos[1]) * (jtp.
Dir[0] / jtp.
Dir[1]) + jtp.
Pos[0];
2516 pos2[2] = (jcs * (itp2_0 - iw0) - ics * (jtp2Pos0 - jw0)) / den;
2517 if (useI) { pos2[1] = (itp2_0 - iw0 - isn * pos2[2]) / ics; }
2519 pos2[1] = (jtp2Pos0 - jw0 - jsn * pos2[2]) / jcs;
2522 if (sep == 0)
return tp3d;
2523 for (
unsigned short ixyz = 0; ixyz < 3; ++ixyz)
2524 tp3d.
Dir[ixyz] = (pos2[ixyz] - tp3d.
Pos[ixyz]) / sep;
2533 if (v1[0] == v2[0] && v1[1] == v2[1] && v1[2] == v2[2])
return 0;
2542 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
2543 dir[xyz] = p2[xyz] - p1[xyz];
2544 if (dir[0] == 0 && dir[1] == 0 && dir[2] == 0)
return dir;
2556 return sqrt(
PosSep2(pos1, pos2));
2563 double d0 = pos1[0] - pos2[0];
2564 double d1 = pos1[1] - pos2[1];
2565 double d2 = pos1[2] - pos2[2];
2566 return d0 * d0 + d1 * d1 + d2 * d2;
2572 double den = v1[0] * v1[0] + v1[1] * v1[1] + v1[2] * v1[2];
2573 if (den == 0)
return false;
2591 unsigned short numEnds = 2;
2592 if (pfp.
PDGCode == 1111) numEnds = 1;
2595 for (
unsigned short end = 0;
end < numEnds; ++
end) {
2596 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane)
2604 for (
unsigned short end = 0;
end < numEnds; ++
end) {
2605 std::vector<float> cnt(slc.
nPlanes);
2608 for (std::size_t ii = 0; ii < pfp.
TP3Ds.size(); ++ii) {
2610 if (dir > 0) { ipt = ii; }
2612 ipt = pfp.
TP3Ds.size() - ii - 1;
2614 if (ipt >= pfp.
TP3Ds.size())
break;
2615 auto& tp3d = pfp.
TP3Ds[ipt];
2616 if (tp3d.Flags[
kTP3DBad])
continue;
2617 if (
PosSep2(tp3d.Pos, endPos) > maxSep2)
break;
2620 float dedx =
dEdx(clockData, detProp, slc, tp3d);
2621 if (dedx < 0.5)
continue;
2626 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
2627 if (cnt[plane] == 0)
continue;
2628 pfp.
dEdx[
end][plane] /= cnt[plane];
2650 for (
auto& tp3d : pfp.
TP3Ds) {
2652 double dedx =
dEdx(clockData, detProp, slc, tp3d);
2653 if (dedx < 0.5 || dedx > 80.)
continue;
2655 sum2 += dedx * dedx;
2658 if (cnt < 3)
return;
2659 dEdXAve = sum / cnt;
2661 dEdXRms = 0.3 * dEdXAve;
2662 double arg = sum2 - cnt * dEdXAve * dEdXAve;
2663 if (arg < 0)
return;
2664 dEdXRms = sqrt(arg) / (cnt - 1);
2666 double minRms = 0.05 * dEdXAve;
2667 if (dEdXRms < minRms) dEdXRms = minRms;
2677 if (tp3d.
TjID > (
int)slc.
slHits.size())
return 0;
2678 if (tp3d.
TjID <= 0)
return 0;
2685 for (
unsigned short ii = 0; ii < tp.Hits.size(); ++ii) {
2686 if (!tp.UseHit[ii])
continue;
2688 dQ +=
hit.Integral();
2692 if (dQ == 0)
return 0;
2695 std::abs(std::sin(angleToVert) * tp3d.
Dir[1] + std::cos(angleToVert) * tp3d.
Dir[2]);
2696 if (cosgamma < 1.
E-5)
return 0;
2698 double dQdx = dQ / dx;
2701 if (std::isinf(dedx)) dedx = 0;
2709 unsigned short tpIndex)
2717 if (tjID <= 0 || tjID > (
int)slc.
tjs.size())
return tp3d;
2718 auto& tj = slc.
tjs[tjID - 1];
2719 if (tpIndex < tj.EndPt[0] || tpIndex > tj.EndPt[1])
return tp3d;
2722 auto& tp2 = tj.Pts[tp3d.
TPIndex];
2730 if (tp2.AngleCode == 1) rms *= 2;
2732 if (tp2.AngleCode > 1) {
2733 std::vector<unsigned int> hitMultiplet;
2734 for (std::size_t ii = 0; ii < tp2.Hits.size(); ++ii) {
2735 if (!tp2.UseHit[ii])
continue;
2737 if (hitMultiplet.size() > 1)
break;
2744 tp3d.
Wire = tp2.Pos[0];
2757 if (tp3d.
Wire < 0)
return false;
2759 if (pfp.
SectionFits[0].Pos[0] == -10.0)
return false;
2768 for (std::size_t sfi = 0; sfi < pfp.
SectionFits.size(); ++sfi) {
2782 double dw = tp3d.
Wire - plnTP.Pos[0];
2784 double t = dw * plnTP.DeltaRMS;
2786 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
2787 tp3d.
Pos[xyz] = sf.Pos[xyz] + t * sf.Dir[xyz];
2806 pfp.
ID = slc.
pfps.size() + 1;
2827 if (slc.
pfps.empty())
return;
2829 for (
auto& pfp : slc.
pfps) {
2830 if (pfp.ID == 0)
continue;
2831 if (pfp.Vx3ID[0] > 0)
continue;
2832 if (pfp.SectionFits.empty())
continue;
2839 if (pfp.TP3Ds.empty()) {
2841 startPos = pfp.SectionFits[0].Pos;
2843 else if (!pfp.TP3Ds.empty()) {
2845 startPos = pfp.TP3Ds[0].Pos;
2847 vx3.
X = startPos[0];
2848 vx3.
Y = startPos[1];
2849 vx3.
Z = startPos[2];
2850 vx3.
ID = slc.
vtx3s.size() + 1;
2854 slc.
vtx3s.push_back(vx3);
2855 pfp.Vx3ID[0] = vx3.
ID;
2896 if (slc.
pfps.empty())
return;
2899 int neutrinoPFPID = 0;
2900 for (
auto& pfp : slc.
pfps) {
2901 if (pfp.ID == 0)
continue;
2902 if (!
tcc.
modes[
kTestBeam] && neutrinoPFPID == 0 && (pfp.PDGCode == 12 || pfp.PDGCode == 14)) {
2903 neutrinoPFPID = pfp.ID;
2909 constexpr
unsigned short end1 = 1;
2910 for (
auto& pfp : slc.
pfps) {
2911 if (pfp.ID == 0)
continue;
2913 if (pfp.Vx3ID[end1] > 0)
continue;
2917 unsigned short cnt3 = 0;
2918 unsigned short vx3id = 0;
2920 std::vector<int> vx2ids;
2921 for (
auto tjid : pfp.TjIDs) {
2922 auto& tj = slc.
tjs[tjid - 1];
2923 if (tj.VtxID[end1] == 0)
continue;
2924 auto& vx2 = slc.
vtxs[tj.VtxID[end1] - 1];
2925 if (vx2.Vx3ID == 0) {
2926 if (vx2.Topo == 1 && vx2.NTraj == 2) vx2ids.push_back(vx2.ID);
2929 if (vx3id == 0) vx3id = vx2.Vx3ID;
2930 if (vx2.Vx3ID == vx3id) ++cnt3;
2934 if (pfp.Vx3ID[1 - end1] == vx3id)
continue;
2935 pfp.Vx3ID[end1] = vx3id;
2940 for (
auto& pfp : slc.
pfps) {
2941 if (pfp.ID == 0)
continue;
2943 if (pfp.PDGCode == 12 || pfp.PDGCode == 14 || pfp.PDGCode == 22)
continue;
2946 int pfpParentID = INT_MAX;
2947 unsigned short nParent = 0;
2948 for (
auto tjid : pfp.TjIDs) {
2949 auto& tj = slc.
tjs[tjid - 1];
2950 if (tj.ParentID <= 0)
continue;
2951 auto parPFP =
GetAssns(slc,
"T", tj.ParentID,
"P");
2952 if (parPFP.empty())
continue;
2953 if (pfpParentID == INT_MAX) pfpParentID = parPFP[0];
2954 if (parPFP[0] == pfpParentID) ++nParent;
2957 auto& ppfp = slc.
pfps[pfpParentID - 1];
2959 pfp.ParentUID = ppfp.UID;
2961 ppfp.DtrUIDs.push_back(pfp.UID);
2965 if (neutrinoPFPID > 0) {
2966 auto& neutrinoPFP = slc.
pfps[neutrinoPFPID - 1];
2967 int vx3id = neutrinoPFP.Vx3ID[1];
2968 for (
auto& pfp : slc.
pfps) {
2969 if (pfp.ID == 0 || pfp.ID == neutrinoPFPID)
continue;
2970 if (pfp.Vx3ID[0] != vx3id)
continue;
2971 pfp.ParentUID = (size_t)neutrinoPFPID;
2972 neutrinoPFP.DtrUIDs.push_back(pfp.ID);
2973 if (pfp.PDGCode == 111) neutrinoPFP.PDGCode = 12;
2984 if (pfp.
TjIDs.empty())
return false;
2985 if (pfp.
PDGCode != 1111 && pfp.
TP3Ds.size() < 2)
return false;
2990 for (
auto& tp3d : pfp.
TP3Ds) {
2991 if (tp3d.TPIndex != USHRT_MAX) slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex].InPFP = pfp.
ID;
2996 unsigned short nNotSet = 0;
2997 for (
auto& tp3d : pfp.
TP3Ds) {
2998 if (tp3d.Flags[
kTP3DBad])
continue;
2999 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
3000 if (tp.InPFP != pfp.
ID) ++nNotSet;
3002 if (nNotSet > 0)
return false;
3004 if (pfp.
ID != (
int)slc.
pfps.size() + 1) pfp.
ID = slc.
pfps.size() + 1;
3009 for (
auto tjid : pfp.
TjIDs) {
3010 auto& tj = slc.
tjs[tjid - 1];
3011 tj.AlgMod[
kMat3D] =
true;
3014 slc.
pfps.push_back(pfp);
3022 if (pfp.
ID <= 0)
return false;
3023 if (end > 1)
return false;
3032 if (neutrinoPFP) { pos = pfp.
SectionFits[0].Pos; }
3033 else if (end == 0) {
3034 pos = pfp.
TP3Ds[0].Pos;
3039 return (pos[0] > slc.
xLo + abit && pos[0] < slc.
xHi - abit && pos[1] > slc.
yLo + abit &&
3040 pos[1] < slc.
yHi - abit && pos[2] > slc.
zLo + abit && pos[2] < slc.
zHi - abit);
3051 auto const world = TPC.GetCenter();
3053 if (pos[0] < world.X() - TPC.HalfWidth() + abit)
continue;
3054 if (pos[0] > world.X() + TPC.HalfWidth() - abit)
continue;
3055 if (pos[1] < world.Y() - TPC.HalfHeight() + abit)
continue;
3056 if (pos[1] > world.Y() + TPC.HalfHeight() - abit)
continue;
3057 if (pos[2] < world.Z() - TPC.Length() / 2 + abit)
continue;
3058 if (pos[2] > world.Z() + TPC.Length() / 2 - abit)
continue;
3071 if (pos1[0] == pos2[0] && pos1[1] == pos2[1] && pos1[2] == pos2[2])
return;
3074 double costh =
DotProd(dir1, ptDir);
3075 if (costh > 1) costh = 1;
3076 double sep =
PosSep(pos1, pos2);
3077 alongTrans[0] = costh * sep;
3078 double sinth = sqrt(1 - costh * costh);
3079 alongTrans[1] = sinth * sep;
3092 for (
unsigned short xyz = 0; xyz < 3; ++xyz) {
3093 p1End[xyz] = p1[xyz] + 10 * p1Dir[xyz];
3094 p2End[xyz] = p2[xyz] + 10 * p2Dir[xyz];
3117 double d1343, d4321, d1321, d4343, d2121;
3118 double numer, denom;
3119 constexpr
double EPS = std::numeric_limits<double>::min();
3121 p13[0] = p1[0] - p3[0];
3122 p13[1] = p1[1] - p3[1];
3123 p13[2] = p1[2] - p3[2];
3124 p43[0] = p4[0] - p3[0];
3125 p43[1] = p4[1] - p3[1];
3126 p43[2] = p4[2] - p3[2];
3128 p21[0] = p2[0] - p1[0];
3129 p21[1] = p2[1] - p1[1];
3130 p21[2] = p2[2] - p1[2];
3133 d1343 = p13[0] * p43[0] + p13[1] * p43[1] + p13[2] * p43[2];
3134 d4321 = p43[0] * p21[0] + p43[1] * p21[1] + p43[2] * p21[2];
3135 d1321 = p13[0] * p21[0] + p13[1] * p21[1] + p13[2] * p21[2];
3136 d4343 = p43[0] * p43[0] + p43[1] * p43[1] + p43[2] * p43[2];
3137 d2121 = p21[0] * p21[0] + p21[1] * p21[1] + p21[2] * p21[2];
3139 denom = d2121 * d4343 - d4321 * d4321;
3140 if (
std::abs(denom) < EPS)
return (
false);
3141 numer = d1343 * d4321 - d1321 * d4343;
3143 double mua = numer / denom;
3144 double mub = (d1343 + d4321 * mua) / d4343;
3146 intersect[0] = p1[0] + mua * p21[0];
3147 intersect[1] = p1[1] + mua * p21[1];
3148 intersect[2] = p1[2] + mua * p21[2];
3150 pb[0] = p3[0] + mub * p43[0];
3151 pb[1] = p3[1] + mub * p43[1];
3152 pb[2] = p3[2] + mub * p43[2];
3153 doca =
PosSep(intersect, pb);
3155 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
3156 intersect[xyz] += pb[xyz];
3157 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
3158 intersect[xyz] /= 2;
3172 float sep =
PosSep(pos1, pos2);
3173 if (sep == 0)
return -1;
3179 for (
unsigned short step = 0; step < nstep; ++step) {
3180 for (
unsigned short xyz = 0; xyz < 3; ++xyz)
3182 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
3192 if (cnt == 0)
return -1;
3204 if (pfp.
ID == 0)
return 0;
3205 if (pfp.
TjIDs.empty())
return 0;
3206 if (end > 1)
return 0;
3216 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
3218 std::vector<int> tjids(1);
3219 for (
auto tjid : pfp.
TjIDs) {
3220 auto& tj = slc.
tjs[tjid - 1];
3221 if (tj.CTP != inCTP)
continue;
3227 if (pos2[0] < -0.4)
continue;
3229 unsigned int wire = std::nearbyint(pos2[0]);
3230 if (wire > slc.
nWires[plane])
continue;
3231 if (slc.
wireHitRange[plane][wire].first == UINT_MAX)
continue;
3234 if (cf < lo) lo = cf;
3235 if (cf > hi) hi = cf;
3240 if (cnt == 0)
return 0;
3241 if (cnt > 1 && lo < 0.3 && hi > 0.8) {
3251 if (end > 1 || pfp.
SectionFits.empty())
return {{0., 0., 0.}};
3259 if (end > 1 || pfp.
SectionFits.empty())
return {{0., 0., 0.}};
3262 if (end == 0)
return pfp.
TP3Ds[0].Pos;
3269 if (pfp.
TP3Ds.empty())
return 0;
3275 unsigned short sfIndex,
3276 unsigned short& startPt,
3277 unsigned short& endPt)
3280 startPt = USHRT_MAX;
3282 if (sfIndex >= pfp.
SectionFits.size())
return false;
3285 for (std::size_t ipt = 0; ipt < pfp.
TP3Ds.size(); ++ipt) {
3286 auto& tp3d = pfp.
TP3Ds[ipt];
3287 if (tp3d.SFIndex < sfIndex)
continue;
3292 if (tp3d.SFIndex > sfIndex)
break;
3303 if (pfp.
ID == 0)
return 0;
3304 if (pfp.
TP3Ds.empty())
return 0;
3305 auto& pos0 = pfp.
TP3Ds[0].Pos;
3306 auto& pos1 = pfp.
TP3Ds[pfp.
TP3Ds.size() - 1].Pos;
3319 if (pfp.
TP3Ds.empty())
return -1;
3324 Average_dEdX(clockData, detProp, slc, pfp, dEdXAve, dEdXRms);
3325 if (dEdXAve < 0)
return 0;
3328 float length =
Length(pfp);
3332 for (
auto tjid : pfp.
TjIDs) {
3333 auto& tj = slc.
tjs[tjid - 1];
3335 if (el <= 0)
continue;
3336 mcsmom +=
MCSMom(slc, tj);
3337 chgrms += tj.ChgRMS;
3340 if (cnt < 2)
return 0;
3345 if (length > 150) vote = 13;
3347 if (vote == 0 && length > 50 && dEdXAve < 2.5 && mcsmom > 500) vote = 13;
3349 if (vote == 0 && dEdXAve > 3.0 && mcsmom > 200 && chgrms < 0.4) vote = 2212;
3351 if (vote == 0 && mcsmom < 50 && chgrms > 0.4) vote = 11;
3358 std::string someText,
3363 if (pfp.
TP3Ds.empty())
return;
3365 myprt << someText <<
" pfp P" << pfp.
ID <<
" MVI " << pfp.
MVI;
3366 for (
auto tid : pfp.
TjIDs)
3367 myprt <<
" T" << tid;
3371 for (
unsigned short ib = 0; ib <
pAlgModSize; ++ib) {
3377 <<
" SFI ________Pos________ ________Dir_______ _____EndPos________ ChiDOF NPts " 3379 for (std::size_t sfi = 0; sfi < pfp.
SectionFits.size(); ++sfi) {
3380 myprt << someText << std::setw(4) << sfi;
3382 myprt << std::fixed << std::setprecision(1);
3383 unsigned short startPt = 0, endPt = 0;
3385 auto& start = pfp.
TP3Ds[startPt].Pos;
3386 myprt << std::setw(7) << start[0] << std::setw(7) << start[1] << std::setw(7) << start[2];
3389 myprt <<
" Invalid";
3391 myprt << std::fixed << std::setprecision(2);
3392 myprt << std::setw(7) << sf.Dir[0] << std::setw(7) << sf.Dir[1] << std::setw(7)
3394 myprt << std::fixed << std::setprecision(1);
3395 if (endPt < pfp.
TP3Ds.size()) {
3397 myprt << std::setw(7) <<
end[0] << std::setw(7) <<
end[1] << std::setw(7) <<
end[2];
3400 myprt <<
" Invalid";
3402 myprt << std::setprecision(1) << std::setw(6) << sf.ChiDOF;
3403 myprt << std::setw(6) << sf.NPts;
3404 myprt << std::setw(6) << sf.NeedsUpdate;
3410 myprt << someText <<
" Note: GBH = TP3D Flags. G = Good, B = Bad, H = High dE/dx \n";
3413 <<
" ipt SFI ________Pos________ Delta Pull GBH Path along dE/dx S? T_ipt_P:W:T\n";
3415 unsigned short fromPt = 0;
3416 unsigned short toPt = pfp.
TP3Ds.size() - 1;
3417 if (printPts >= 0) fromPt = toPt;
3419 std::vector<float> dang(pfp.
TP3Ds.size(), -1);
3420 for (
unsigned short ipt = fromPt; ipt <= toPt; ++ipt) {
3421 auto tp3d = pfp.
TP3Ds[ipt];
3422 myprt << someText << std::setw(4) << ipt;
3423 myprt << std::setw(4) << tp3d.SFIndex;
3424 myprt << std::fixed << std::setprecision(1);
3425 myprt << std::setw(7) << tp3d.Pos[0] << std::setw(7) << tp3d.Pos[1] << std::setw(7)
3427 myprt << std::setprecision(1) << std::setw(6) << (tp3d.Pos[0] - tp3d.TPX);
3429 myprt << std::setprecision(1) << std::setw(6) << pull;
3431 myprt << std::setw(7) << std::setprecision(1) <<
PosSep(tp3d.Pos, pfp.
TP3Ds[0].Pos);
3432 myprt << std::setw(7) << std::setprecision(1) << tp3d.along;
3433 myprt << std::setw(6) << std::setprecision(2) <<
dEdx(clockData, detProp, slc, tp3d);
3436 for (
unsigned short plane = 0; plane < slc.
nPlanes; ++plane) {
3438 auto tp =
MakeBareTP(detProp, tp3d.Pos, inCTP);
3441 if (tp3d.TjID > 0) {
3442 auto& tp = slc.
tjs[tp3d.TjID - 1].Pts[tp3d.TPIndex];
3443 myprt <<
" T" << tp3d.TjID <<
"_" << tp3d.TPIndex <<
"_" <<
PrintPos(tp) <<
" " 3447 myprt <<
" UNDEFINED";
Expect tracks entering from the front face. Don't create neutrino PFParticles.
calo::CalorimetryAlg * caloAlg
code to link reconstructed objects back to the MC truth information
short MCSMom(const TCSlice &slc, const std::vector< int > &tjIDs)
MaybeLogger_< ELseverityLevel::ELsev_info, true > LogVerbatim
std::vector< Trajectory > tjs
vector of all trajectories in each plane
float Length(const PFPStruct &pfp)
bool dbgStitch
debug PFParticle stitching
void Average_dEdX(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp, float &dEdXAve, float &dEdXRms)
void MakePFParticles(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, std::vector< MatchStruct > matVec, unsigned short matVec_Iter)
std::array< std::vector< float >, 2 > dEdxErr
bool MakeVertexObsolete(std::string fcnLabel, TCSlice &slc, VtxStore &vx2, bool forceKill)
int TjID
ID of the trajectory -> TP3D assn.
bool ReconcileTPs(TCSlice &slc, PFPStruct &pfp, bool prt)
bool InsideFV(const TCSlice &slc, const PFPStruct &pfp, unsigned short end)
static unsigned int kWire
const std::vector< std::string > AlgBitNames
CTP_t CTP
Cryostat, TPC, Plane code.
bool InsideTPC(const Point3_t &pos, geo::TPCID &inTPCID)
const geo::WireReadoutGeom * wireReadoutGeom
std::array< double, 3 > Point3_t
bool SignalAtTp(TrajPoint &tp)
std::vector< float > vtx3DCuts
2D vtx -> 3D vtx matching cuts
double WireCoordinate(Point_t const &point) const
Returns the coordinate of the point on the plane, in wire units.
bool SortSection(PFPStruct &pfp, unsigned short sfIndex)
unsigned int NTPC(CryostatID const &cryoid=details::cryostat_zero) const
Returns the total number of TPCs in the specified cryostat.
Declaration of signal hit object.
void FindAlongTrans(Point3_t pos1, Vector3_t dir1, Point3_t pos2, Point2_t &alongTrans)
bool HasWire(WireID const &wireid) const
Returns whether we have the specified wire.
std::bitset< pAlgModSize > AlgMod
std::vector< std::vector< std::pair< unsigned int, unsigned int > > > wireHitRange
The data type to uniquely identify a Plane.
Geometry information for a single TPC.
Point3_t Pos
center position of this section
bool StoreTraj(TCSlice &slc, Trajectory &tj)
bool LineLineIntersect(Point3_t p1, Point3_t p2, Point3_t p3, Point3_t p4, Point3_t &intersect, float &doca)
bool IsShowerLike(TCSlice const &slc, std::vector< int > const &TjIDs)
constexpr auto abs(T v)
Returns the absolute value of the argument.
CryostatID_t Cryostat
Index of cryostat.
for(Int_t i=0;i< nentries;i++)
static unsigned int kPlane
void SetAngleCode(TrajPoint &tp)
PFPStruct CreatePFP(const TCSlice &slc)
Point3_t PosAtEnd(const PFPStruct &pfp, unsigned short end)
double DeltaAngle(const Vector3_t v1, const Vector3_t v2)
bool dbgSlc
debug only in the user-defined slice? default is all slices
void Match2Planes(TCSlice &slc, std::vector< MatchStruct > &matVec)
bool CanSection(const TCSlice &slc, const PFPStruct &pfp)
std::array< int, 2 > Vx3ID
float TPHitsRMSTime(const TCSlice &slc, const TrajPoint &tp, HitStatus_t hitRequest)
unsigned int MVI
MatchVec Index for detailed 3D matching.
void FilldEdx(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp)
void ReconcileVertices(TCSlice &slc, PFPStruct &pfp, bool prt)
std::string TPEnvString(const TrajPoint &tp)
std::string PrintPos(const TrajPoint &tp)
std::vector< std::array< unsigned int, 3 > > sptHits
SpacePoint -> Hits assns by plane.
float MaxTjLen(const TCSlice &slc, std::vector< int > &tjIDs)
bool Update(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp)
void MakePFPTjs(TCSlice &slc)
unsigned short Find3DRecoRange(const TCSlice &slc, const PFPStruct &pfp, unsigned short fromPt, unsigned short min2DPts, short dir)
bool MakeSmallAnglePFP(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
Vector3_t DirErr
and direction error
double ThetaZ() const
Angle of the wires from positive z axis; .
void AddPointsInRange(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, unsigned short fromPt, unsigned short toPt, CTP_t inCTP, float maxPull, unsigned short &nWires, unsigned short &nAdd, bool prt)
decltype(auto) constexpr end(T &&obj)
ADL-aware version of std::end.
float HitsRMSTime(const TCSlice &slc, const std::vector< unsigned int > &hitsInMultiplet, HitStatus_t hitRequest)
Access the description of the physical detector geometry.
std::vector< int > TjUIDs
struct of temporary 3D vertices
void PFPVertexCheck(TCSlice &slc)
Vector3_t Dir
and direction
std::vector< Tj2Pt > mallTraj
vector of trajectory points ordered by increasing X
void FillGaps3D(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
std::array< float, 2 > Point2_t
int PDGCodeVote(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp)
float unitsPerTick
scale factor from Tick to WSE equivalent units
Vector3_t PointDirection(const Point3_t p1, const Point3_t p2)
void FillWireIntersections(TCSlice &slc)
bool MakeTP3Ds(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
unsigned short TPIndex
and the TP index
void Match3Planes(TCSlice &slc, std::vector< MatchStruct > &matVec)
void GetHitMultiplet(const TCSlice &slc, unsigned int theHit, std::vector< unsigned int > &hitsInMultiplet, bool useLongPulseHits)
bool TCIntersectionPoint(unsigned int wir1, unsigned int wir2, unsigned int pln1, unsigned int pln2, float &y, float &z)
float ElectronLikelihood(const TCSlice &slc, const Trajectory &tj)
bool FitSection(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp, unsigned short sfIndex)
std::vector< std::vector< bool > > goodWire
Point3_t Pos
position of the trajectory
float ChgFracNearPos(const TCSlice &slc, const Point2_t &pos, const std::vector< int > &tjIDs)
std::vector< unsigned int > FindCloseHits(const TCSlice &slc, std::array< int, 2 > const &wireWindow, Point2_t const &timeWindow, const unsigned short plane, HitStatus_t hitRequest, bool usePeakTime, bool &hitsNear)
float ChgFracBetween(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, Point3_t pos1, Point3_t pos2)
TrajPoint MakeBareTP(detinfo::DetectorPropertiesData const &detProp, const Point3_t &pos, CTP_t inCTP)
std::vector< VtxStore > vtxs
2D vertices
tick_as<> tick
Tick number, represented by std::ptrdiff_t.
double ConvertXToTicks(double X, int p, int t, int c) const
void CountBadPoints(const TCSlice &slc, const PFPStruct &pfp, unsigned short fromPt, unsigned short toPt, unsigned short &nBadPts, unsigned short &firstBadPt)
bool ReSection(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, PFPStruct &pfp, bool prt)
std::vector< float > match3DCuts
3D matching cuts
std::vector< SectionFit > SectionFits
void MakeTrajectoryObsolete(TCSlice &slc, unsigned int itj)
double TPXErr2
(X position error)^2
double PosSep2(const Point3_t &pos1, const Point3_t &pos2)
const geo::GeometryCore * geom
bool SectionStartEnd(const PFPStruct &pfp, unsigned short sfIndex, unsigned short &startPt, unsigned short &endPt)
bool PointDirIntersect(Point3_t p1, Vector3_t p1Dir, Point3_t p2, Vector3_t p2Dir, Point3_t &intersect, float &doca)
The data type to uniquely identify a TPC.
PlaneID_t Plane
Index of the plane within its TPC.
constexpr unsigned int pAlgModSize
void PrintP(mf::LogVerbatim &myprt, PFPStruct const &pfp, bool &printHeader)
bool valsIncreasing(const SortEntry &c1, const SortEntry &c2)
bool SetMag(Vector3_t &v1, double mag)
Definition of data types for geometry description.
unsigned short FarEnd(const PFPStruct &pfp, const Point3_t &pos)
float dEdx(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, TP3D &tp3d)
std::vector< TCSlice > slices
Detector simulation of raw signals on wires.
void GetRange(const PFPStruct &pfp, unsigned short sfIndex, unsigned short &fromPt, unsigned short &npts)
bool WireIDsIntersect(WireID const &wid1, WireID const &wid2, Point_t &intersection) const
Computes the intersection between two wires.
unsigned short InsertTP3D(PFPStruct &pfp, TP3D &tp3d)
SectionFit FitTP3Ds(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, const std::vector< TP3D > &tp3ds, unsigned short fromPt, short fitDir, unsigned short nPtsFit)
double ConvertTicksToX(double ticks, int p, int t, int c) const
std::bitset< 16 > modes
number of points to find AveChg
std::vector< TCHit > slHits
void Recover(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc, PFPStruct &pfp, bool prt)
TP3D CreateTP3D(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, int tjID, unsigned short tpIndex)
ROOT::Math::PositionVector3D< ROOT::Math::Cartesian3D< double >, ROOT::Math::GlobalCoordinateSystemTag > Point_t
Type for representation of position in physical 3D space.
std::vector< int > GetAssns(TCSlice const &slc, std::string type1Name, int id, std::string type2Name)
bool ValidTwoPlaneMatch(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, const PFPStruct &pfp)
double DotProd(const Vector3_t &v1, const Vector3_t &v2)
std::vector< Vtx3Store > vtx3s
3D vertices
void FillmAllTraj(detinfo::DetectorPropertiesData const &detProp, TCSlice &slc)
std::bitset< 128 > useAlg
Max hit separation for making junk trajectories. < 0 to turn off.
Encapsulate the construction of a single detector plane .
Contains all timing reference information for the detector.
std::vector< float > pfpStitchCuts
cuts for stitching between TPCs
TP3D MakeTP3D(detinfo::DetectorPropertiesData const &detProp, const TrajPoint &itp, const TrajPoint &jtp)
geo::PlaneID DecodeCTP(CTP_t CTP)
float along
distance from the start Pos of the section
double PosSep(const Point3_t &pos1, const Point3_t &pos2)
range_type< T > Iterate() const
float PointPull(const TP3D &tp3d)
std::vector< TCWireIntersection > wireIntersections
std::vector< recob::Hit > const * allHits
bool valsDecreasing(const SortEntry &c1, const SortEntry &c2)
CTP_t EncodeCTP(unsigned int cryo, unsigned int tpc, unsigned int plane)
double dEdx_AREA(detinfo::DetectorClocksData const &clock_data, detinfo::DetectorPropertiesData const &det_prop, recob::Hit const &hit, double pitch, double T0=0) const
std::pair< unsigned short, unsigned short > GetSliceIndex(std::string typeName, int uID)
unsigned short MVI_Iter
MVI iteration - see FindPFParticles.
std::vector< unsigned int > nWires
std::array< double, 3 > Vector3_t
std::vector< TP3D > TP3Ds
int ID
ID of the recob::Slice (not the sub-slice)
std::array< std::vector< float >, 2 > dEdx
void Match3PlanesSpt(TCSlice &slc, std::vector< MatchStruct > &matVec)
void FindPFParticles(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, TCSlice &slc)
std::vector< PFPStruct > pfps
void MoveTPToWire(TrajPoint &tp, float wire)
unsigned short NumPtsWithCharge(const TCSlice &slc, const Trajectory &tj, bool includeDeadWires)
unsigned short CloseEnd(const Trajectory &tj, const Point2_t &pos)
PlaneGeo const & Plane(TPCID const &tpcid, View_t view) const
Returns the specified wire.
double TPX
X position of the TP or the single hit.
TPCID_t TPC
Index of the TPC within its cryostat.
bool StorePFP(TCSlice &slc, PFPStruct &pfp)
Collection of Physical constants used in LArSoft.
float ChgFracNearEnd(detinfo::DetectorPropertiesData const &detProp, const TCSlice &slc, const PFPStruct &pfp, unsigned short end)
void PrintTP3Ds(detinfo::DetectorClocksData const &clockData, detinfo::DetectorPropertiesData const &detProp, std::string someText, const TCSlice &slc, const PFPStruct &pfp, short printPts)
Vector3_t DirAtEnd(const PFPStruct &pfp, unsigned short end)
bool SetSection(detinfo::DetectorPropertiesData const &detProp, PFPStruct &pfp, TP3D &tp3d)
void DefinePFPParents(TCSlice &slc)
bool AttachToAnyVertex(TCSlice &slc, PFPStruct &pfp, float maxSep, bool prt)
Interface to geometry for wire readouts .
constexpr TPCID()=default
Default constructor: an invalid TPC ID.
double WirePitch() const
Return the wire pitch (in centimeters). It is assumed constant.
Encapsulate the construction of a single detector plane .
std::array< std::bitset< 8 >, 2 > EndFlag
bool SptInTPC(const std::array< unsigned int, 3 > &sptHits, unsigned int tpc)
void Reverse(PFPStruct &pfp)
unsigned short SFIndex
and the section fit index