27#ifndef imagingUtils_hpp
28#define imagingUtils_hpp
32#include "../math/constants.hpp"
33#include "../mxError.hpp"
52template <
typename realT>
58 return ( lambda / metersPerPixel ) * ( 1. / pixels );
70template <
class arrayT>
72 typename arrayT::Scalar eps = 0,
73 typename arrayT::Scalar rad = 0,
75 typename arrayT::Scalar overscan = 0
82 mxError(
"circularPupil", MXE_INVALIDARG,
"Central obscuration can not be < 0." );
88 mxError(
"circularPupil", MXE_INVALIDARG,
"Central obscuration can not be > 1." );
95 typename arrayT::Scalar r;
96 typename arrayT::Scalar xc = 0.5 * ( l0 - 1 );
97 typename arrayT::Scalar yc = 0.5 * ( l1 - 1 );
100 rad = 0.5 * std::min( l0 - 1, l1 - 1 );
102 for(
size_t i = 0; i < l0; i++ )
104 for(
size_t j = 0; j < l1; j++ )
106 r = std::sqrt( std::pow( i - xc, 2 ) + std::pow( j - yc, 2 ) );
108 if( r <= rad + ( overscan ) && r >= eps * rad )
123template <
class arrayT>
124void drawLine( arrayT &im,
125 typename arrayT::Scalar x0,
126 typename arrayT::Scalar y0,
127 typename arrayT::Scalar x1,
128 typename arrayT::Scalar y1,
129 typename arrayT::Scalar halfWidth
136 typename arrayT::Scalar xc, yc;
138 xc = 0.5 * ( im.rows() - 1 );
139 yc = 0.5 * ( im.cols() - 1 );
141 typename arrayT::Scalar m = ( y1 - y0 ) / ( x1 - x0 );
146 for(
int x = x0; x <= x1; ++x )
148 y = y0 + ( x - x0 ) * m;
150 for(
int i = 0; i <= halfWidth; ++i )
152 if( x + xc >= 0 && x + xc < d1 && y + yc + i >= 0 && y + yc + i < d2 )
153 im( x + xc, y + yc + i ) = 0;
154 if( x + xc >= 0 && x + xc < d1 && y + yc - i >= 0 && y + yc - i < d2 )
155 im( x + xc, y + yc - i ) = 0;
161 for(
int x = x1; x <= x0; ++x )
163 y = y0 + ( x - x0 ) * m;
165 for(
int i = 0; i <= halfWidth; ++i )
167 if( x + xc >= 0 && x + xc < d1 && y + yc + i >= 0 && y + yc + i < d2 )
168 im( x + xc, y + yc + i ) = 0;
169 if( x + xc >= 0 && x + xc < d1 && y + yc - i >= 0 && y + yc - i < d2 )
170 im( x + xc, y + yc - i ) = 0;
185template <
typename arrayOutT,
typename arrayInT>
186void makeComplexPupil( arrayOutT &complexPupil,
const arrayInT &realPupil,
int wavefrontSizePixels )
189 complexPupil.resize( wavefrontSizePixels, wavefrontSizePixels );
190 complexPupil.set(
typename arrayOutT::Scalar( 0, 0 ) );
193 int bl = 0.5 * ( complexPupil.rows() - 1 ) - 0.5 * ( realPupil.rows() - 1. );
195 for(
int i = 0; i < realPupil.cols(); ++i )
197 for(
int j = 0; j < realPupil.rows(); ++j )
199 complexPupil( bl + i, bl + j ) =
200 typename arrayOutT::Scalar( realPupil( i, j ), 0 );
216template <
typename arrayOutT,
typename arrayInT>
218 const arrayInT &realAmplitude,
219 const arrayInT &realPhase,
220 int wavefrontSizePixels )
223 complexWavefront.resize( wavefrontSizePixels, wavefrontSizePixels );
224 complexWavefront.set(
typename arrayOutT::Scalar( 0, 0 ) );
227 int bl = 0.5 * ( complexWavefront.rows() - 1 ) - 0.5 * ( realAmplitude.rows() - 1. );
229 for(
int i = 0; i < realAmplitude.cols(); ++i )
231 for(
int j = 0; j < realAmplitude.rows(); ++j )
233 complexWavefront( bl + j, bl + i ) =
234 realAmplitude( j, i ) * exp( (
typename arrayOutT::Scalar( 0, 1 ) ) * realPhase( j, i ) );
247template <
typename wavefrontT>
249 typename wavefrontT::Scalar::value_type xTilt,
250 typename wavefrontT::Scalar::value_type yTilt )
252 typedef typename wavefrontT::Scalar complexT;
253 typedef typename wavefrontT::Scalar::value_type realT;
257 int wfsSizeX = complexWavefront.cols();
258 int wfsSizeY = complexWavefront.rows();
260 realT xCen = 0.5 * ( wfsSizeX - 1 );
261 realT yCen = 0.5 * ( wfsSizeY - 1 );
263 realT argX = 2.0 * pi / ( wfsSizeX - 1.0 );
264 realT argY = 2.0 * pi / ( wfsSizeY - 1.0 );
266 for(
int ii = 0; ii < wfsSizeX; ++ii )
268 for(
int jj = 0; jj < wfsSizeY; ++jj )
270 complexWavefront( ii, jj ) =
271 complexWavefront( ii, jj ) *
272 exp( complexT( (realT)0., argX * xTilt * ( ii - xCen ) + argY * yTilt * ( jj - yCen ) ) );
278template <
typename imageT1,
289 int dest_cols = dest.cols();
291 int src_cols = src.cols();
293 typedef typename imageT1::Scalar dataT;
298 for(
int j = 0; j < imYsz; ++j )
300 dest_data = &dest.data()[imX0 + ( imY0 + j ) * dest_cols];
301 src_data = &src.data()[wfX0 + ( wfY0 + j ) * src_cols];
303 memcpy( dest_data, src_data,
sizeof( dataT ) * imXsz );
311template <
typename imageT1,
321 if( dest.rows() != src.rows() )
324 mx::err::sizeerr,
"mx::imagingUtils::extractMaskedPixels",
"dest and src do not have same size (rows)" );
327 if( dest.cols() != src.cols() )
330 mx::err::sizeerr,
"mx::imagingUtils::extractMaskedPixels",
"dest and src do not have same size (cols)" );
333 if( src.rows() != mask.rows() )
336 mx::err::sizeerr,
"mx::imagingUtils::extractMaskedPixels",
"src and mask do not have same size (rows)" );
339 if( src.cols() != mask.cols() )
342 mx::err::sizeerr,
"mx::imagingUtils::extractMaskedPixels",
"src and mask do not have same size (cols)" );
345 for(
int cc = 0; cc < dest.cols(); ++cc )
347 for(
int rr = 0; rr < dest.rows(); ++rr )
349 if( mask( rr, cc ) != 0 )
351 dest( rr, cc ) = src( rr, cc );
357template <
typename realImageT,
typename complexImageT>
358void extractIntensityImage(
359 realImageT &im,
int imX0,
int imXsz,
int imY0,
int imYsz, complexImageT &wf,
int wfX0,
int wfY0 )
361 int im_rows = im.cols();
363 int wf_rows = wf.cols();
365 typename realImageT::Scalar *im_data;
366 typename complexImageT::Scalar *wf_data;
368 for(
int j = 0; j < imXsz; ++j )
370 im_data = &im.data()[imX0 + ( imY0 + j ) * im_rows];
371 wf_data = &wf.data()[wfX0 + ( wfY0 + j ) * wf_rows];
372 for(
int i = 0; i < imYsz; ++i )
374 im_data[i] = norm( wf_data[i] );
379template <
typename realImageT,
typename complexImageT>
380void extractIntensityImageAccum(
381 realImageT &im,
int imX0,
int imXsz,
int imY0,
int imYsz, complexImageT &wf,
int wfX0,
int wfY0 )
383 int im_rows = im.cols();
385 int wf_rows = wf.cols();
387 typename realImageT::Scalar *im_data;
388 typename complexImageT::Scalar *wf_data;
390 for(
int j = 0; j < imXsz; ++j )
392 im_data = &im.data()[imX0 + ( imY0 + j ) * im_rows];
393 wf_data = &wf.data()[wfX0 + ( wfY0 + j ) * wf_rows];
394 for(
int i = 0; i < imYsz; ++i )
396 im_data[i] += norm( wf_data[i] );
mxException for a size error
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
void tiltWavefront(wavefrontT &complexWavefront, typename wavefrontT::Scalar::value_type xTilt, typename wavefrontT::Scalar::value_type yTilt)
Apply a tilt to a wavefront.
void makeComplexPupil(arrayOutT &complexPupil, const arrayInT &realPupil, int wavefrontSizePixels)
Create a complex pupil plane wavefront from a real amplitude mask.
realT fftPlateScale(size_t pixels, realT metersPerPixel, realT lambda)
Calculate the angular plate scale (radians per pixel) of an image after propagation by FFT.
int circularPupil(arrayT &m, typename arrayT::Scalar eps=0, typename arrayT::Scalar rad=0, typename arrayT::Scalar overscan=0)
Fill in an Eigen-like array with a circular pupil mask.
Declares and defines a class for managing images.
void extractMaskedPixels(imageT1 &dest, const imageT2 &src, const imageT3 &mask)
Extract a pixels from one image and insert them into a second based on a mask.
void extractBlock(imageT1 &dest, int imX0, int imXsz, int imY0, int imYsz, imageT2 &src, int wfX0, int wfY0)
Extract a block from one image and insert it into a second.