10DBScanClusterBuilder::DBScanClusterBuilder() {
12 cluster_hit_dist_ = 100;
14 min_cluster_hit_mult_ = 2;
17DBScanClusterBuilder::DBScanClusterBuilder(
float minHitEnergy,
20 float minClusterHitMult) {
21 min_hit_energy_ = minHitEnergy;
22 cluster_hit_dist_ = clusterHitDist;
23 cluster_z_bias_ = clusterZBias;
24 min_cluster_hit_mult_ = minClusterHitMult;
27std::vector<std::vector<const ldmx::CalorimeterHit *> >
28DBScanClusterBuilder::runDBSCAN(
29 const std::vector<const ldmx::CalorimeterHit *> &hits_) {
30 const int n = hits_.size();
31 std::vector<std::vector<const ldmx::CalorimeterHit *> > idx_clusters;
32 std::vector<unsigned int> tried;
34 std::vector<unsigned int> used;
36 for (
unsigned int i = 0; i < n; i++) {
37 if (isIn(i, tried))
continue;
39 ldmx_log(debug) <<
"trying " << i;
40 if (hits_[i]->getEnergy() < min_hit_energy_)
continue;
41 std::set<unsigned int> neighbors;
42 unsigned int n_nearby = 1;
44 for (
unsigned int j = 0; j < n; j++) {
46 dist(hits_[i], hits_[j]) < cluster_hit_dist_) {
48 if (hits_[j]->getEnergy() >= min_hit_energy_) n_nearby++;
51 if (n_nearby >= min_cluster_hit_mult_) {
52 std::vector<const ldmx::CalorimeterHit *> idx_cluster{
55 ldmx_log(debug) <<
"- starting a cluster from " << i;
56 for (
unsigned int j : neighbors) {
57 if (!isIn(j, tried)) {
59 ldmx_log(debug) <<
"== tried " << j;
60 std::vector<unsigned int> neighbors2;
61 for (
unsigned int k = 0; k < n; k++) {
62 if (dist(hits_[k], hits_[j]) < cluster_hit_dist_) {
63 neighbors2.push_back(k);
66 for (
unsigned int k : neighbors2) neighbors.insert(k);
69 ldmx_log(debug) <<
"== used " << j;
71 idx_cluster.push_back(hits_[j]);
74 idx_clusters.push_back(idx_cluster);
77 ldmx_log(debug) <<
"done. writing this many clusters out: "
78 << idx_clusters.size();
82void DBScanClusterBuilder::fillClusterInfoFromHits(
84 bool logEnergyWeight) {
85 float e(0), x(0), y(0), z(0), xx(0), yy(0), zz(0), n(0);
88 std::vector<float> raw_xvals{};
89 std::vector<float> raw_yvals{};
90 std::vector<float> raw_zvals{};
91 std::vector<float> raw_evals{};
92 std::vector<const ldmx::CalorimeterHit *> constituent_hits;
95 if (h->getEnergy() < min_hit_energy_)
continue;
96 if (logEnergyWeight) w = log(h->getEnergy()) - log(min_hit_energy_);
98 x += w * h->getXPos();
99 y += w * h->getYPos();
100 z += w * h->getZPos();
101 xx += w * h->getXPos() * h->getXPos();
102 yy += w * h->getYPos() * h->getYPos();
103 zz += w * h->getZPos() * h->getZPos();
106 raw_xvals.push_back(h->getXPos());
107 raw_yvals.push_back(h->getYPos());
108 raw_zvals.push_back(h->getZPos());
109 raw_evals.push_back(h->getEnergy());
110 constituent_hits.emplace_back(h);
118 xx = sqrt(xx - x * x);
119 yy = sqrt(yy - y * y);
120 zz = sqrt(zz - z * z);
124 cl->setRMSXYZ(xx, yy, zz);
125 cl->setHitValsX(raw_xvals);
126 cl->setHitValsY(raw_yvals);
127 cl->setHitValsZ(raw_zvals);
128 cl->setHitValsE(raw_evals);
131 if (raw_xvals.size() > 2) {
133 std::vector<float> sorted_z = raw_zvals;
134 std::sort(sorted_z.begin(), sorted_z.end());
135 if ((sorted_z.size() > 2) and (sorted_z.back() - sorted_z.front() > 1e3)) {
136 for (
int i = 0; i < raw_xvals.size(); i++) {
137 raw_xvals[i] = raw_xvals[i] - x;
138 raw_yvals[i] = raw_yvals[i] - y;
139 raw_zvals[i] = raw_zvals[i] - z;
142 TGraph gxz(raw_zvals.size(), raw_zvals.data(), raw_xvals.data());
143 auto r_xz = gxz.Fit(
"pol1",
"SQ");
144 cl->setDXDZ(r_xz->Value(1));
145 cl->setEDXDZ(r_xz->ParError(1));
147 TGraph gyz(raw_zvals.size(), raw_zvals.data(), raw_yvals.data());
148 auto r_yz = gyz.Fit(
"pol1",
"SQ");
149 cl->setDYDZ(r_yz->Value(1));
150 cl->setEDYDZ(r_yz->ParError(1));
Implementation of DBSCAN clustering algo.
Stores cluster information from the ECal.
void setNHits(int nHits)
Sets total number of hits in the cluster.
void addHits(const std::vector< const ldmx::CalorimeterHit * > hitsVec)
Take in the hits that make up the cluster.
void setCentroidXYZ(double centroid_x, double centroid_y, double centroid_z)
Sets the three coordinates of the cluster centroid.
void setEnergy(double energy)
Sets total energy for the cluster.
Represents a reconstructed hit in a calorimeter cell within the detector.