27#ifndef fourierModes_hpp
28#define fourierModes_hpp
33#include "../mxError.hpp"
35#include "../math/constants.hpp"
36#include "../math/geo.hpp"
55#define MX_FOURIER_BASIC 0
60#define MX_FOURIER_MODIFIED 1
79 typename typeN::Scalar m,
80 typename typeN::Scalar n
83 typedef typename typeN::Scalar realT;
88 realT uc, vc, u, v, D;
90 uc = 0.5 * ( dim1 - 1.0 );
91 vc = 0.5 * ( dim2 - 1.0 );
93 D = std::max( dim1, dim2 );
95 for(
int i = 0; i < dim1; ++i )
98 for(
int j = 0; j < dim2; ++j )
127template <
class typeN>
130 typedef typename typeN::Scalar realT;
132 int dim1 = im.cols();
133 int dim2 = im.rows();
135 realT uc, vc, u, v, D;
137 uc = 0.5 * ( dim1 - 1.0 );
138 vc = 0.5 * ( dim2 - 1.0 );
140 D = std::max( dim1, dim2 );
142 for(
int i = 0; i < dim1; ++i )
145 for(
int j = 0; j < dim2; ++j )
168template <
class typeN>
169int makeFourierMode( typeN im,
typename typeN::Scalar m,
typename typeN::Scalar n,
int p )
177 mxError(
"makeFourierMode", MXE_INVALIDARG,
"p must be +1 (cosine) or -1 (sine)." );
199template <
class typeN>
201 typeN im,
typename typeN::Scalar m,
typename typeN::Scalar n,
int p,
typename typeN::Scalar ang = 0 )
203 typedef typename typeN::Scalar realT;
205 if( p != 1 && p != -1 )
207 mxError(
"makeModifiedFourierMode", MXE_INVALIDARG,
"p must be +1 or -1." );
210 int dim1 = im.cols();
211 int dim2 = im.rows();
213 realT xc, yc, x, y, arg, D;
215 xc = 0.5 * ( dim1 - 1.0 );
216 yc = 0.5 * ( dim2 - 1.0 );
218 D = std::max( dim1, dim2 );
227 for(
int i = 0; i < dim1; ++i )
230 for(
int j = 0; j < dim2; ++j )
237 realT x0 = x * c_ang - y * s_ang;
238 realT y0 = x * s_ang + y * c_ang;
246 im( i, j ) = cos( arg ) + p * sin( arg );
262template <
class typeN>
264 typename typeN::Scalar m,
265 typename typeN::Scalar n,
266 typename typeN::Scalar ang = 0 )
280template <
class typeN>
282 typename typeN::Scalar m,
283 typename typeN::Scalar n,
284 typename typeN::Scalar ang = 0 )
315 double fr1 = spf1.
m * spf1.
m + spf1.
n * spf1.
n;
316 double fr2 = spf2.
m * spf2.
m + spf2.
n * spf2.
n;
320 if( spf1.
n == spf2.
n )
322 if( spf1.
m == spf2.
m )
323 return ( spf1.
p < spf2.
p );
325 return ( spf1.
m < spf2.
m );
327 return ( spf1.
n < spf2.
n );
331 return ( fr1 < fr2 );
344 spf.resize( Nmodes );
348 for(
int ll = 0; ll <= N; ++ll )
350 for(
int l = ll, k = 0; l >= 0; --l, ++k )
352 if( modeCount >= Nmodes )
354 mxError(
"makeFourierModeFreqs_PandV", MXE_SIZEERR,
"mode count exceeded expected number of modes" );
358 if( k == 0 && l > .5 * N )
361 if( k > .5 * N && ( l == 0 || l >= N - k ) )
375 spf[modeCount].m = _k;
376 spf[modeCount].n = _l;
377 spf[modeCount].p = 1;
382 if( !( ( k == 0 && l == 0 ) || ( k == 0.5 * N && l == 0 ) || ( k == 0 && l == 0.5 * N ) ||
383 ( k == 0.5 * N && l == 0.5 * N ) ) )
385 spf[modeCount].m = _k;
386 spf[modeCount].n = _l;
387 spf[modeCount].p = -1;
410 double k1 = spf1.
m * spf1.
m + spf1.
n * spf1.
n;
411 double k2 = spf2.
m * spf2.
m + spf2.
n * spf2.
n;
416 double a1 = atan2( spf1.
n, spf1.
m );
418 a1 += math::two_pi<double>();
420 double a2 = atan2( spf2.
n, spf2.
m );
422 a2 += math::two_pi<double>();
425 return ( spf1.
p < spf2.
p );
445 krad = floor( 0.5 * N );
447 Nmodes = 0.25 * math::pi<double>() * N * N * 1.1;
449 spf.resize( Nmodes );
453 for(
int m = -krad; m <= krad; ++m )
455 int nmax = sqrt( krad * krad - m * m );
457 for(
int n = 0; n <= nmax; ++n )
459 if( n == 0 && m <= 0 )
462 for(
int p = -1; p <= 1; p += 2 )
464 if( modeCount >= Nmodes )
466 mxError(
"makeFourierModeFreqs_Circ", MXE_SIZEERR,
"mode count exceeded expected number of modes" );
470 spf[modeCount].m = m;
471 spf[modeCount].n = n;
472 spf[modeCount].p = p;
479 spf.erase( spf.begin() + modeCount, spf.end() );
500 if( p != -1 && p != 1 )
502 mxError(
"fourierModeCoordinates", MXE_INVALIDARG,
"p must +/-1" );
506 if( m == 0 && n == 0 )
508 mxError(
"fourierModeCoordinates", MXE_INVALIDARG,
"piston is ignored" );
512 int m0 = std::max( abs( m ), abs( n ) );
514 int i0 = 2 * ( m0 * m0 - m0 );
533 return 2 * ( i0 + di ) + ( 1 + p ) / 2;
549 mxError(
"fourierModeCoordinates", MXE_INVALIDARG,
"i must be >= 0" );
560 while( 2 * ( m0 * m0 - m0 ) <= ii )
564 int di = ii - 2 * ( m0 * m0 - m0 );
571 else if( di <= 3 * m0 )
582 p = -1 + 2 * ( i % 2 );
594 int Ndx = floor( 0.5 * ( N ) );
596 int Nmodes = 2 * ( ( 2 * Ndx + 1 ) * ( Ndx + 1 ) - ( Ndx + 1 ) );
598 spf.resize( Nmodes );
602 for(
int i = 0; i < Nmodes; ++i )
624template <
typename cubeT>
626 cubeT &cube,
int dim, std::vector<fourierModeDef> spf,
int basisType,
typename cubeT::Scalar ang = 0 )
628 int Nmodes = spf.size();
630 cube.resize( dim, dim, Nmodes );
632 for(
int j = 0; j < Nmodes; ++j )
634 if( basisType == MX_FOURIER_MODIFIED )
641 if(
makeFourierMode( cube.image( j ), spf[j].m, spf[j].n, spf[j].p ) < 0 )
661template <
typename cubeT>
664 std::vector<fourierModeDef> spf;
685template <
typename cubeT>
688 std::vector<fourierModeDef> spf;
709template <
typename cubeT>
712 std::vector<fourierModeDef> spf;
int makeFourierModeS(typeN im, typename typeN::Scalar m, typename typeN::Scalar n)
Populate a single sine mode.
int makeFourierBasis_Rect(cubeT &cube, int dim, int N, int basisType, typename cubeT::Scalar ang=0)
Generate a rectangular Fourier basis.
int fourierModeCoordinates(int &m, int &n, int &p, int i)
Calculate the (m,n,p) coordinates of a Fourier mode given its index.
int makeModifiedFourierModeP(typeN im, typename typeN::Scalar m, typename typeN::Scalar n, typename typeN::Scalar ang=0)
Populate a single modified Fourier +1 mode.
int makeFourierBasis_Circ(cubeT &cube, int dim, int N, int basisType)
Generate a circular Fourier basis.
int makeModifiedFourierModeM(typeN im, typename typeN::Scalar m, typename typeN::Scalar n, typename typeN::Scalar ang=0)
Populate a single modified Fourier -1 mode.
int makeFourierModeFreqs_Rect(std::vector< fourierModeDef > &spf, int N)
Generate a rectangular spatial frequency grid.
int makeFourierBasis_PandV(cubeT &cube, int dim, int N, int basisType)
Generate a rectangular modified Fourier basis with the Poyneer and Veran ordering.
int makeModifiedFourierMode(typeN im, typename typeN::Scalar m, typename typeN::Scalar n, int p, typename typeN::Scalar ang=0)
Populate a single modified Fourier mode.
bool comp_fourierModeDef(const fourierModeDef &spf1, const fourierModeDef &spf2)
Sorting functor for modes according to the mx::AO standard.
int fourierModeNumber(int m, int n, int p)
Calculate the index of a Fourier mode given its (m,n,p) coordinates.
int fillFourierBasis(cubeT &cube, int dim, std::vector< fourierModeDef > spf, int basisType, typename cubeT::Scalar ang=0)
Fill in a cube with a Fourier basis.
int makeFourierModeFreqs_Circ(std::vector< fourierModeDef > &spf, int N)
Generate a circular spatial frequency grid.
int makeFourierMode(typeN im, typename typeN::Scalar m, typename typeN::Scalar n, int p)
Populate a single basic Fourier mode.
int makeFourierModeFreqs_PandV(std::vector< fourierModeDef > &spf, int N)
Generate a Poyneer and Veran spatial frequency grid.
int makeFourierModeC(typeN im, typename typeN::Scalar m, typename typeN::Scalar n)
Populate a single cosine mode.
bool comp_fourierModeDef_PandV(const fourierModeDef &spf1, const fourierModeDef &spf2)
Sorting functor for modes according to the Poyner and Veran (2005) standard.
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
realT dtor(realT q)
Convert from degrees to radians.
Structure to contain the parameters of a Fourier mode.
int m
Spatial frequency m index.
int n
Spatial frequency n index.
int p
+/- 1 for modified Fourier modes, -1==>sine, +1==>cosine for basic Fourier modes.
fourierModeDef()
Constructor.