mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
aperturePhotometer.hpp
Go to the documentation of this file.
1/** \file aperturePhotometer.hpp
2 * \brief Class for conducting aperture photometry on an image.
3 * \ingroup image_processing_files
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2018-2022 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#ifndef improc_aperturePhotometer_hpp
28#define improc_aperturePhotometer_hpp
29
30#include <map>
31#include <mx/improc/eigenImage.hpp>
32#include <mx/improc/imageMasks.hpp>
33
34namespace mx
35{
36namespace improc
37{
38
39/// Class for performing aperture photometry on images.
40/** Designed to very efficiently calculate the cumulative flux as a function of radius,
41 * with minimal cost for performing the measurement on multiple images.
42 *
43 * This initial version assumes the source is at 0.5*(rows-1) and 0.5*(cols-1).
44 *
45 * First call resize with the desired size. This organizes the radius vector and
46 * the index image which maps pixels to position in the radius vector.
47 *
48 * Then call cumPhot with the image, which will sum the flux contained within each radius.
49 *
50 * \ingroup improc
51 *
52 * \tparam realT the real floating point type of the data
53 */
54template <typename realT>
56{
57
58 protected:
59 int m_sizeX{ 0 }; ///< The size of the image in X (rows).
60 int m_sizeY{ 0 }; ///< The size of the image in Y (columns).
61
62 realT m_xcen{ 0 }; ///< The x coordinate of the center of the image. Radii are caculated relative to this point.
63 ///< Default is 0.5*(sizeX-1).
64 realT m_ycen{ 0 }; ///< The y coordinate of the center of the image. Radii are caculated relative to this point.
65 ///< Default is 0.5*(sizeY-1).
66
67 std::vector<realT> m_radius; ///< Holds the ordered unique radii of the pixels in the image.
68
69 eigenImage<size_t> m_indexIm; ///< Maps a pixel in the image to the position in the radius vector.
70
71 bool m_useNanMask{ false };
72
73 eigenImage<realT> *m_nanMask{ nullptr };
74
75 realT m_bgMin{ 0 };
76 realT m_bgMax{ 0 };
77
78 std::vector<realT> m_bgx;
79 std::vector<realT> m_bgy;
80
81 public:
82 /// Resize the photometer, recalculating the radius vector and index image.
83 /** If size and center are unchanged, nothing is done.
84 *
85 * \note Source must be at geometric center of image.
86 *
87 * \returns 0 on success
88 * \returns -1 on error
89 */
90 int resize( int sizeX, ///< [in] The new size in rows
91 int sizeY, ///< [in] The new size in columns
92 realT xcen, ///< The x coordinate of the center of the image. Radii are caculated relative to this
93 ///< point. Default is 0.5*(sizeX-1).
94 realT ycen ///< The y coordinate of the center of the image. Radii are caculated relative to this
95 ///< point. Default is 0.5*(sizeY-1).
96 );
97
98 /// Resize the photometer for a centered image, recalculating the radius vector and index image.
99 /** If size and center are unchanged, nothing is done.
100 *
101 * This version uses the geometric center of the image.
102 *
103 * \overload
104 *
105 * \returns 0 on success
106 * \returns -1 on error f
107 */
108 int resize( int sizeX, ///< [in] The new size in rows
109 int sizeY ///< [in] The new size in columns
110 );
111
112 /// Get the cumulative photometry of an image as a function of radius.
113 /**
114 *
115 * \returns 0 on success
116 * \returns -1 on error
117 */
118 int cumPhot(
119 std::vector<realT> &cumPhot, ///< [out] the cumulative photometry at each point in the radius vector. Resized.
120 eigenImage<realT> &im, ///< [in] the image on which to perform the calculation
121 realT maxr =
122 0 ///< [in] [optional] the maximum radius to which to calculate. If <= 0 then max possible is used.
123 );
124
125 /// Get the cumulative photometry of an image as a function of radius.
126 /**
127 * \overload
128 *
129 * \returns 0 on success
130 * \returns -1 on error
131 */
132 int cumPhot(
133 std::vector<realT> &cumPhot, ///< [out] the cumulative photometry at each point in the radius vector. Resized.
134 std::vector<realT> &
135 deltaPhot, ///< [out] [optional] the photometry at each point in the radius vector, not-cumulative. Resized.
136 eigenImage<realT> &im, ///< [in] the image on which to perform the calculation
137 realT maxr =
138 0 ///< [in] [optional] the maximum radius to which to calculate. If <= 0 then max possible is used.
139 );
140
141 protected:
142 /// Get the cumulative photometry of an image as a function of radius.
143 /** This is called by the public cumPhot functions.
144 *
145 *
146 * \returns 0 on success
147 * \returns -1 on error
148 */
149 int cumPhotWork(
150 std::vector<realT> &cumPhot, ///< [out] the cumulative photometry at each point in the radius vector. Resized.
151 std::vector<realT> *deltaPhot, ///< [out] if not `nullptr`, this vector is filled with the sum at each radius.
152 eigenImage<realT> &im, ///< [in] the image on which to perform the calculation
153 realT maxr =
154 0 ///< [in] [optional] the maximum radius to which to calculate. If <= 0 then max possible is used.
155 );
156
157 public:
158 /// Get the radius value at an index of the vector.
159 realT radius( size_t i /**< [in] the index of the radius vector */ )
160 {
161 return m_radius[i];
162 }
163
164 void bgMin( realT bgm )
165 {
166 m_bgMin = bgm;
167 }
168
169 void bgMax( realT bgx )
170 {
171 m_bgMax = bgx;
172 }
173
174 void bg( realT bgm, realT bgx )
175 {
176 m_bgMin = bgm;
177 m_bgMax = bgx;
178 }
179};
180
181template <typename realT>
182int aperturePhotometer<realT>::resize( int sizeX, int sizeY, realT xcen, realT ycen )
183{
184 // Don't bother if we don't have to.
185 if( sizeX == m_sizeX && sizeY == m_sizeY && xcen == m_xcen && ycen == m_ycen )
186 return 0;
187
188 m_sizeX = sizeX;
189 m_sizeY = sizeY;
190 m_xcen = xcen;
191 m_ycen = ycen;
192
193 m_indexIm.resize( m_sizeX, m_sizeY );
194
195 // Get radius of each pixel up front, since we need it twice.
196 eigenImage<realT> radIm;
197 radIm.resize( m_sizeX, m_sizeY );
198 radiusImage( radIm, m_xcen, m_ycen ); // mx function to fill in the radius array
199
200 std::map<realT, size_t> unrads; // Map of unique radii to their indices in the radius vector.
201
202 // First populate the map
203 for( int i = 0; i < radIm.cols(); ++i )
204 {
205 for( int j = 0; j < radIm.rows(); ++j )
206 {
207 // unrads[radIm(j,i)] = 0;
208 unrads.insert( std::pair<realT, size_t>( radIm( j, i ), 0 ) );
209 }
210 }
211
212 // Now fill in the radius vector, and set map values to the indices
213 m_radius.resize( unrads.size() );
214 typename std::map<realT, size_t>::iterator it;
215 int n = 0;
216 for( it = unrads.begin(); it != unrads.end(); ++it )
217 {
218 m_radius[n] = it->first;
219 it->second = n;
220 ++n;
221 }
222 // Finally, fill in the index image with index of the radius vector for that pixel.
223 for( int i = 0; i < radIm.cols(); ++i )
224 {
225 for( int j = 0; j < radIm.rows(); ++j )
226 {
227 it = unrads.find( radIm( j, i ) );
228 m_indexIm( j, i ) = it->second;
229 }
230 }
231
232 if( m_bgMax > m_bgMin && m_bgMin > 0 )
233 {
234 m_bgx.clear();
235 m_bgy.clear();
236
237 for( int cc = 0; cc < radIm.cols(); ++cc )
238 {
239 for( int rr = 0; rr < radIm.rows(); ++rr )
240 {
241 if( radIm( rr, cc ) >= m_bgMin && radIm( rr, cc ) <= m_bgMax )
242 {
243 m_bgx.push_back( rr );
244 m_bgy.push_back( cc );
245 }
246 }
247 }
248 }
249
250 return 0;
251}
252
253template <typename realT>
254int aperturePhotometer<realT>::resize( int sizeX, int sizeY )
255{
256 realT xcen = 0.5 * ( 1.0 * sizeX - 1.0 );
257 realT ycen = 0.5 * ( 1.0 * sizeY - 1.0 );
258
259 return resize( sizeX, sizeY, xcen, ycen );
260}
261
262template <typename realT>
263int aperturePhotometer<realT>::cumPhot( std::vector<realT> &cumPhot, eigenImage<realT> &im, realT maxr )
264{
265 return cumPhotWork( cumPhot, nullptr, im, maxr );
266}
267
268template <typename realT>
269int aperturePhotometer<realT>::cumPhot( std::vector<realT> &cumPhot,
270 std::vector<realT> &deltaPhot,
272 realT maxr )
273{
274 return cumPhotWork( cumPhot, &deltaPhot, im, maxr );
275}
276
277template <typename realT>
278int aperturePhotometer<realT>::cumPhotWork( std::vector<realT> &cumPhot,
279 std::vector<realT> *deltaPhot,
281 realT maxr )
282{
283 realT xcen = m_xcen; // 0.5*(m_indexIm.rows()-1);
284 realT ycen = m_ycen; // 0.5*(m_indexIm.cols()-1);
285
286 if( maxr <= 0 )
287 maxr = m_radius.back();
288
289 cumPhot.resize( m_radius.size(), 0 );
290
291 // Make sure we stay in bounds
292 int x0 = xcen - maxr;
293 if( x0 < 0 )
294 x0 = 0;
295 int x1 = xcen + maxr;
296 if( x1 > m_indexIm.rows() - 1 )
297 x1 = m_indexIm.rows() - 1;
298
299 int y0 = ycen - maxr;
300 if( y0 < 0 )
301 y0 = 0;
302 int y1 = ycen + maxr;
303 if( y1 > m_indexIm.cols() - 1 )
304 y1 = m_indexIm.rows() - 1;
305
306 realT bg = 0;
307 // If using background annulus, first calc background;
308 if( m_bgx.size() > 0 )
309 {
310 std::vector<realT> bgann( m_bgx.size() );
311 for( size_t n = 0; n < m_bgx.size(); ++n )
312 {
313 bgann[n] = im( (int)m_bgx[n], (int)m_bgy[n] );
314 }
315
316 bg = math::vectorMedian( bgann );
317 std::cerr << "background: " << bg << "\n";
318 }
319
320 // First get sum at each unique radius
321 // Note: switched order of cols/rows indices for speeed
322 if( maxr == m_radius.back() )
323 {
324 for( int i = y0; i <= y1; ++i )
325 {
326 for( int j = x0; j <= x1; ++j )
327 {
328 cumPhot[m_indexIm( j, i )] += im( j, i ) - bg;
329 }
330 }
331 }
332 else
333 {
334 for( int i = y0; i <= y1; ++i )
335 {
336 for( int j = x0; j <= x1; ++j )
337 {
338 if( m_radius[m_indexIm( j, i )] <= maxr )
339 cumPhot[m_indexIm( j, i )] += im( j, i );
340 }
341 }
342 }
343
344 // If deltaPhot is desired, output it here.
345 if( deltaPhot != nullptr )
346 {
347 deltaPhot->assign( cumPhot.begin(), cumPhot.end() );
348 }
349
350 // Now perform cumulative sum
351 for( int i = 1; i < cumPhot.size(); ++i )
352 {
353 cumPhot[i] += cumPhot[i - 1];
354 }
355
356 return 0;
357}
358
359} // namespace improc
360} // namespace mx
361
362#endif // improc_aperturePhotometer_hpp
Class for performing aperture photometry on images.
eigenImage< size_t > m_indexIm
Maps a pixel in the image to the position in the radius vector.
realT radius(size_t i)
Get the radius value at an index of the vector.
int resize(int sizeX, int sizeY, realT xcen, realT ycen)
Resize the photometer, recalculating the radius vector and index image.
int cumPhot(std::vector< realT > &cumPhot, eigenImage< realT > &im, realT maxr=0)
Get the cumulative photometry of an image as a function of radius.
std::vector< realT > m_radius
Holds the ordered unique radii of the pixels in the image.
int cumPhotWork(std::vector< realT > &cumPhot, std::vector< realT > *deltaPhot, eigenImage< realT > &im, realT maxr=0)
Get the cumulative photometry of an image as a function of radius.
int m_sizeY
The size of the image in Y (columns).
int m_sizeX
The size of the image in X (rows).
Eigen::Array< scalarT, -1, -1 > eigenImage
Definition of the eigenImage type, which is an alias for Eigen::Array.
void radiusImage(eigenT &m, typename eigenT::Scalar xc, typename eigenT::Scalar yc, typename eigenT::Scalar scale=1)
Fills in the cells of an Eigen 2D Array with their radius from the center.
vectorT::value_type vectorMedian(const vectorT &vec, vectorT *work=0)
Calculate median of a vector, leaving the vector unaltered.
The mxlib c++ namespace.
Definition mxError.hpp:106