mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
fitMoffat.hpp
Go to the documentation of this file.
1/** \file fitMoffat.hpp
2 * \author Jared R. Males
3 * \brief Tools for fitting Moffat functions to data.
4 * \ingroup fitting_files
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015, 2016, 2017 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 fitMoffat_hpp
28#define fitMoffat_hpp
29
30#include "levmarInterface.hpp"
31#include "../func/moffat.hpp"
32
33#include "../../improc/eigenImage.hpp"
34
35namespace mx
36{
37namespace math
38{
39namespace fit
40{
41
42/// Wrapper for a native array to pass to \ref levmarInterface, with Moffat details.
43/** \ingroup moffat_peak_fit
44 */
45template <typename realT>
47{
48 realT *data{ nullptr }; ///< Pointer to the array of y values
49 realT *coords{ nullptr }; ///< Pointer to the array of x values (optional)
50 size_t nx{ 0 }; ///< X dimension of the array
51
52 realT m_I0{ 0 };
53 realT m_I{ 0 };
54 realT m_x0{ 0 };
55 realT m_alpha{ 0 };
56 realT m_beta{ 0 };
57
58 int m_I0_idx{ 0 };
59 int m_I_idx{ 1 };
60 int m_x0_idx{ 2 };
61 int m_alpha_idx{ 4 };
62 int m_beta_idx{ 5 };
63
64 int m_nparams = 5;
65
66 /// Set whether each parameter is fixed.
67 /** Sets the parameter indices appropriately.
68 */
69 void setFixed( bool I0, bool I, bool x0, bool alpha, bool beta )
70 {
71 int idx = 0;
72
73 if( I0 )
74 m_I0_idx = -1;
75 else
76 m_I0_idx = idx++;
77
78 if( I )
79 m_I_idx = -1;
80 else
81 m_I_idx = idx++;
82
83 if( x0 )
84 m_x0_idx = -1;
85 else
86 m_x0_idx = idx++;
87
88 if( alpha )
89 m_alpha_idx = -1;
90 else
91 m_alpha_idx = idx++;
92
93 if( beta )
94 m_beta_idx = -1;
95 else
96 m_beta_idx = idx++;
97
98 m_nparams = idx;
99 }
100
101 realT I0( realT *p )
102 {
103 if( m_I0_idx < 0 )
104 {
105 return m_I0;
106 }
107 else
108 {
109 return p[m_I0_idx];
110 }
111 }
112
113 void I0( realT *p, realT nI0 )
114 {
115 if( m_I0_idx < 0 )
116 {
117 m_I0 = nI0;
118 }
119 else
120 {
121 p[m_I0_idx] = nI0;
122 }
123 }
124
125 realT I( realT *p )
126 {
127 if( m_I_idx < 0 )
128 {
129 return m_I;
130 }
131 else
132 {
133 return p[m_I_idx];
134 }
135 }
136
137 void I( realT *p, realT nI )
138 {
139 if( m_I_idx < 0 )
140 {
141 m_I = nI;
142 }
143 else
144 {
145 p[m_I_idx] = nI;
146 }
147 }
148
149 realT x0( realT *p )
150 {
151 if( m_x0_idx < 0 )
152 {
153 return m_x0;
154 }
155 else
156 {
157 return p[m_x0_idx];
158 }
159 }
160
161 void x0( realT *p, realT nx0 )
162 {
163 if( m_x0_idx < 0 )
164 {
165 m_x0 = nx0;
166 }
167 else
168 {
169 p[m_x0_idx] = nx0;
170 }
171 }
172
173 realT alpha( realT *p )
174 {
175 if( m_alpha_idx < 0 )
176 {
177 return m_alpha;
178 }
179 else
180 {
181 return p[m_alpha_idx];
182 }
183 }
184
185 void alpha( realT *p, realT nalpha )
186 {
187 if( m_alpha_idx < 0 )
188 {
189 m_alpha = nalpha;
190 }
191 else
192 {
193 p[m_alpha_idx] = nalpha;
194 }
195 }
196
197 realT beta( realT *p )
198 {
199 if( m_beta_idx < 0 )
200 {
201 return m_beta;
202 }
203 else
204 {
205 return p[m_beta_idx];
206 }
207 }
208
209 void beta( realT *p, realT nbeta )
210 {
211 if( m_beta_idx < 0 )
212 {
213 m_beta = nbeta;
214 }
215 else
216 {
217 p[m_beta_idx] = nbeta;
218 }
219 }
220
221 int nparams()
222 {
223 return m_nparams;
224 }
225};
226
227/** \defgroup moffat_peak_fit Moffat Functions
228 * \brief Fitting Moffat functions to data.
229 *
230 * The Moffat Function\cite moffat_1969, a.k.a. the Moffat Profile, a.k.a. the Moffat Distribution, has the form
231 * \f[
232 I(x) = I_{pk}\left[ 1 + \frac{x^2}{\alpha^2}\right]^{-\beta}
233 * \f]
234 * With \f$\beta=1\f$ it is the
235 * Lorentzian or Cauchy distribution. See also https://en.wikipedia.org/wiki/Moffat_distribution and
236 * https://en.wikipedia.org/wiki/Cauchy_distribution.
237 *
238 * For utilities for working with Moffat functions see \ref gen_math_moffats.
239 *
240 * \ingroup peak_fit
241 */
242
243///\ref levmarInterface fitter structure for the 1D Moffat.
244/** \ingroup moffat_peak_fit
245 *
246 */
247template <typename _realT>
249{
250 typedef _realT realT;
251
252 static const int nparams = 6;
253
254 static void func( realT *p, realT *hx, int m, int n, void *adata )
255 {
257
258 size_t idx_mat, idx_dat;
259
260 idx_dat = 0;
261
262 realT I0 = arr->I0( p );
263 realT I = arr->I( p );
264 realT x0 = arr->x0( p );
265 realT alpha = arr->alpha( p );
266 realT beta = arr->beta( p );
267
268 if( arr->coords )
269 {
270 for( int i = 0; i < arr->nx; ++i )
271 {
272 hx[i] = func::moffat<realT>( arr->coords[i], I0, I, x0, alpha, beta ) - arr->data[i];
273 }
274 }
275 else
276 {
277 for( int i = 0; i < arr->nx; ++i )
278 {
279 hx[i] = func::moffat<realT>( i, I0, I, x0, alpha, beta ) - arr->data[i];
280 }
281 }
282 }
283};
284
285/// Class to manage fitting a 1D Moffat to data via the \ref levmarInterface
286/** Fits \ref gen_math_moffats to a 1-dimensional array of data.
287 *
288 * This class allows for treating any of the parameters as fixed.
289 *
290 * \tparam fitterT a type meeting the requirements specified in \ref levmarInterface.
291 *
292 * \ingroup moffat_peak_fit
293 *
294 */
295template <typename _realT>
296class fitMoffat1D : public levmarInterface<moffat1D_fitter<_realT>>
297{
298
299 public:
300 typedef _realT realT;
302
303 protected:
305
306 void initialize()
307 {
308 this->allocate_params( arr.nparams() );
309 this->adata = &arr;
310 }
311
312 public:
314 {
315 initialize();
316 }
317
319 {
320 }
321
322 /// Set whether each parameter is fixed.
323 /** Sets the parameter indices appropriately.
324 */
325 void setFixed( bool I0, ///< [in] if true, then I0 will be not be part of the fit
326 bool I, ///< [in] if true, then I will be not be part of the fit
327 bool x0, ///< [in] if true, then x0 will be not be part of the fit
328 bool alpha, ///< [in] if true, then alpha will be not be part of the fit
329 bool beta ///< [in] if true, then beta will be not be part of the fit
330 )
331 {
332 arr.setFixed( I0, I, x0, alpha, beta );
333 this->allocate_params( arr.nparams() );
334 }
335
336 /// Set the initial guess for a symmetric Moffat.
337 void setGuess( realT I0, ///< [in] the constant background level
338 realT I, ///< [in] the peak scaling
339 realT x0, ///< [in] the center x-coordinate
340 realT alpha, ///< [in] the width parameter
341 realT beta ///< [in] the shape parameter
342 )
343 {
344 arr.I0( this->p, I0 );
345 arr.I( this->p, I );
346 arr.x0( this->p, x0 );
347 arr.alpha( this->p, alpha );
348 arr.beta( this->p, beta );
349 }
350
351 /// Set the data aray.
352 void setArray( realT *data, ///< [in] pointer to an nx sized array of the y-values to be fit
353 int nx ///< [in] the number of pixels in the x direction of the data array
354 )
355 {
356 arr.data = data;
357 arr.nx = nx;
358
359 this->n = nx;
360 }
361
362 /// Set the data aray and the coordinates.
363 void setArray( realT *data, ///< [in] pointer to an nx sized array of the y-values to be fit
364 realT *coords, ///< [in] point to an nx sized array of the x-values of the data
365 int nx ///< [in] the number of pixels in the x direction of the data array
366 )
367 {
368 arr.data = data;
369 arr.coords = coords;
370 arr.nx = nx;
371
372 this->n = nx;
373 }
374
375 /// Get the current value of I0, the constant.
376 /**
377 * \returns the current value of I0
378 */
379 realT I0()
380 {
381 return arr.I0( this->p );
382 }
383
384 /// Get the current value of I, the peak scaling.
385 /**
386 * \returns the current value of I
387 */
388 realT I()
389 {
390 return arr.I( this->p );
391 }
392
393 /// Get the center x-
394 /**
395 * \returns the current value of x0
396 */
397 realT x0()
398 {
399 return arr.x0( this->p );
400 }
401
402 /// Return the width parameter
403 /**
404 * \returns the current value of alpha
405 */
406 realT alpha()
407 {
408 return arr.alpha( this->p );
409 }
410
411 /// Return the shape parameter
412 /**
413 * \returns the current value of beta
414 */
415 realT beta()
416 {
417 return arr.beta( this->p );
418 }
419
420 /// Return the full-width at half maximum
421 /**
422 * \returns the FWHM calculated from alpha and beta
423 */
424 realT fwhm()
425 {
426 return func::moffatFWHM( alpha(), beta() );
427 }
428};
429
430// forward
431template <typename realT>
432struct array2FitMoffat;
433
434// forward
435template <typename _realT>
436struct moffat2D_sym_fitter;
437
438/// Class to manage fitting a 2D Moffat to data via the \ref levmarInterface
439/** Fits \ref gen_math_moffats to a 2-dimensional array of data.
440 *
441 * This class allows for treating any of the parameters as fixed.
442 *
443 * \tparam fitterT a type meeting the requirements specified in \ref levmarInterface.
444 *
445 * \ingroup moffat_peak_fit
446 *
447 */
448template <typename fitterT>
449class fitMoffat2D : public levmarInterface<fitterT>
450{
451
452 public:
453 typedef typename fitterT::realT realT;
454
455 protected:
457
458 void initialize()
459 {
460 this->allocate_params( arr.nparams() );
461 this->adata = &arr;
462 }
463
464 public:
466 {
467 initialize();
468 }
469
471 {
472 }
473
474 /// Set whether each parameter is fixed.
475 /** Sets the parameter indices appropriately.
476 */
477 void setFixed( bool I0, ///< [in] if true, then I0 will be not be part of the fit
478 bool I, ///< [in] if true, then I will be not be part of the fit
479 bool x0, ///< [in] if true, then x0 will be not be part of the fit
480 bool y0, ///< [in] if true, then y0 will be not be part of the fit
481 bool alpha, ///< [in] if true, then alpha will be not be part of the fit
482 bool beta ///< [in] if true, then beta will be not be part of the fit
483 )
484 {
485 arr.setFixed( I0, I, x0, y0, alpha, beta );
486 this->allocate_params( arr.nparams() );
487 }
488
489 /// Set the initial guess for a symmetric Moffat.
490 void setGuess( realT I0, ///< [in] the constant background level
491 realT I, ///< [in] the peak scaling
492 realT x0, ///< [in] the center x-coordinate
493 realT y0, ///< [in] the center y-coordinate
494 realT alpha, ///< [in] the width parameter
495 realT beta ///< [in] the shape parameter
496 )
497 {
498 arr.I0( this->p, I0 );
499 arr.I( this->p, I );
500 arr.x0( this->p, x0 );
501 arr.y0( this->p, y0 );
502 arr.alpha( this->p, alpha );
503 arr.beta( this->p, beta );
504 }
505
506 /// Set the data aray.
507 void setArray( realT *data, ///< [in] pointer to an nx X ny array of data to be fit
508 int nx, ///< [in] the number of pixels in the x direction of the data array
509 int ny ///< [in] the number of pixels in the y direction of the data array
510 )
511 {
512 arr.data = data;
513 arr.nx = nx;
514 arr.ny = ny;
515
516 this->n = nx * ny;
517 }
518
519 /// Get the current value of I0, the constant.
520 /**
521 * \returns the current value of I0
522 */
523 realT I0()
524 {
525 return arr.I0( this->p );
526 }
527
528 /// Get the current value of I, the peak scaling.
529 /**
530 * \returns the current value of I
531 */
532 realT I()
533 {
534 return arr.I( this->p );
535 }
536
537 /// Get the center x-
538 /**
539 * \returns the current value of x0
540 */
541 realT x0()
542 {
543 return arr.x0( this->p );
544 }
545
546 /// Get the center y-coordinate
547 /**
548 * \returns the current value of y0
549 */
550 realT y0()
551 {
552 return arr.y0( this->p );
553 }
554
555 /// Return the width parameter
556 /**
557 * \returns the current value of alpha
558 */
559 realT alpha()
560 {
561 return arr.alpha( this->p );
562 }
563
564 /// Return the shape parameter
565 /**
566 * \returns the current value of beta
567 */
568 realT beta()
569 {
570 return arr.beta( this->p );
571 }
572
573 /// Return the full-width at half maximum
574 /**
575 * \returns the FWHM calculated from alpha and beta
576 */
577 realT fwhm()
578 {
579 return func::moffatFWHM( alpha(), beta() );
580 }
581};
582
583/// Wrapper for a native array to pass to \ref levmarInterface, with Moffat details.
584/** \ingroup moffat_peak_fit
585 */
586template <typename realT>
588{
589 realT *data{ nullptr }; ///< Pointer to the array
590 size_t nx{ 0 }; ///< X dimension of the array
591 size_t ny{ 0 }; ///< Y dimension of the array
592
593 realT m_I0{ 0 };
594 realT m_I{ 0 };
595 realT m_x0{ 0 };
596 realT m_y0{ 0 };
597 realT m_alpha{ 0 };
598 realT m_beta{ 0 };
599
600 int m_I0_idx{ 0 };
601 int m_I_idx{ 1 };
602 int m_x0_idx{ 2 };
603 int m_y0_idx{ 3 };
604 int m_alpha_idx{ 4 };
605 int m_beta_idx{ 5 };
606
607 int m_nparams = 6;
608
609 /// Set whether each parameter is fixed.
610 /** Sets the parameter indices appropriately.
611 */
612 void setFixed( bool I0, bool I, bool x0, bool y0, bool alpha, bool beta )
613 {
614 int idx = 0;
615
616 if( I0 )
617 m_I0_idx = -1;
618 else
619 m_I0_idx = idx++;
620
621 if( I )
622 m_I_idx = -1;
623 else
624 m_I_idx = idx++;
625
626 if( x0 )
627 m_x0_idx = -1;
628 else
629 m_x0_idx = idx++;
630
631 if( y0 )
632 m_y0_idx = -1;
633 else
634 m_y0_idx = idx++;
635
636 if( alpha )
637 m_alpha_idx = -1;
638 else
639 m_alpha_idx = idx++;
640
641 if( beta )
642 m_beta_idx = -1;
643 else
644 m_beta_idx = idx++;
645
646 m_nparams = idx;
647 }
648
649 realT I0( realT *p )
650 {
651 if( m_I0_idx < 0 )
652 {
653 return m_I0;
654 }
655 else
656 {
657 return p[m_I0_idx];
658 }
659 }
660
661 void I0( realT *p, realT nI0 )
662 {
663 if( m_I0_idx < 0 )
664 {
665 m_I0 = nI0;
666 }
667 else
668 {
669 p[m_I0_idx] = nI0;
670 }
671 }
672
673 realT I( realT *p )
674 {
675 if( m_I_idx < 0 )
676 {
677 return m_I;
678 }
679 else
680 {
681 return p[m_I_idx];
682 }
683 }
684
685 void I( realT *p, realT nI )
686 {
687 if( m_I_idx < 0 )
688 {
689 m_I = nI;
690 }
691 else
692 {
693 p[m_I_idx] = nI;
694 }
695 }
696
697 realT x0( realT *p )
698 {
699 if( m_x0_idx < 0 )
700 {
701 return m_x0;
702 }
703 else
704 {
705 return p[m_x0_idx];
706 }
707 }
708
709 void x0( realT *p, realT nx0 )
710 {
711 if( m_x0_idx < 0 )
712 {
713 m_x0 = nx0;
714 }
715 else
716 {
717 p[m_x0_idx] = nx0;
718 }
719 }
720
721 realT y0( realT *p )
722 {
723 if( m_y0_idx < 0 )
724 {
725 return m_y0;
726 }
727 else
728 {
729 return p[m_y0_idx];
730 }
731 }
732
733 void y0( realT *p, realT ny0 )
734 {
735 if( m_y0_idx < 0 )
736 {
737 m_y0 = ny0;
738 }
739 else
740 {
741 p[m_y0_idx] = ny0;
742 }
743 }
744
745 realT alpha( realT *p )
746 {
747 if( m_alpha_idx < 0 )
748 {
749 return m_alpha;
750 }
751 else
752 {
753 return p[m_alpha_idx];
754 }
755 }
756
757 void alpha( realT *p, realT nalpha )
758 {
759 if( m_alpha_idx < 0 )
760 {
761 m_alpha = nalpha;
762 }
763 else
764 {
765 p[m_alpha_idx] = nalpha;
766 }
767 }
768
769 realT beta( realT *p )
770 {
771 if( m_beta_idx < 0 )
772 {
773 return m_beta;
774 }
775 else
776 {
777 return p[m_beta_idx];
778 }
779 }
780
781 void beta( realT *p, realT nbeta )
782 {
783 if( m_beta_idx < 0 )
784 {
785 m_beta = nbeta;
786 }
787 else
788 {
789 p[m_beta_idx] = nbeta;
790 }
791 }
792
793 int nparams()
794 {
795 return m_nparams;
796 }
797};
798
799///\ref levmarInterface fitter structure for the symmetric Moffat.
800/** \ingroup moffat_peak_fit
801 *
802 */
803template <typename _realT>
805{
806 typedef _realT realT;
807
808 static const int nparams = 6;
809
810 static void func( realT *p, realT *hx, int m, int n, void *adata )
811 {
813
814 size_t idx_mat, idx_dat;
815
816 idx_dat = 0;
817
818 realT I0 = arr->I0( p );
819 realT I = arr->I( p );
820 realT x0 = arr->x0( p );
821 realT y0 = arr->y0( p );
822 realT alpha = arr->alpha( p );
823 realT beta = arr->beta( p );
824
825 for( int i = 0; i < arr->nx; ++i )
826 {
827 for( int j = 0; j < arr->ny; ++j )
828 {
829 idx_mat = i + j * arr->nx;
830
831 hx[idx_dat] = func::moffat2D<realT>( i, j, I0, I, x0, y0, alpha, beta ) - arr->data[idx_mat];
832
833 ++idx_dat;
834 }
835 }
836 }
837};
838
839/// Alias for the fitMoffat2D type fitting the symmetric Moffat profile.
840/** \ingroup moffat_peak_fit
841 */
842template <typename realT>
844
845} // namespace fit
846} // namespace math
847
848} // namespace mx
849
850#endif // fitMoffat_hpp
Class to manage fitting a 1D Moffat to data via the levmarInterface.
realT beta()
Return the shape parameter.
void setGuess(realT I0, realT I, realT x0, realT alpha, realT beta)
Set the initial guess for a symmetric Moffat.
void setArray(realT *data, realT *coords, int nx)
Set the data aray and the coordinates.
void setArray(realT *data, int nx)
Set the data aray.
realT x0()
Get the center x-.
realT alpha()
Return the width parameter.
realT I0()
Get the current value of I0, the constant.
realT fwhm()
Return the full-width at half maximum.
realT I()
Get the current value of I, the peak scaling.
void setFixed(bool I0, bool I, bool x0, bool alpha, bool beta)
Set whether each parameter is fixed.
Class to manage fitting a 2D Moffat to data via the levmarInterface.
realT y0()
Get the center y-coordinate.
void setFixed(bool I0, bool I, bool x0, bool y0, bool alpha, bool beta)
Set whether each parameter is fixed.
void setArray(realT *data, int nx, int ny)
Set the data aray.
realT I()
Get the current value of I, the peak scaling.
realT I0()
Get the current value of I0, the constant.
realT beta()
Return the shape parameter.
realT x0()
Get the center x-.
void setGuess(realT I0, realT I, realT x0, realT y0, realT alpha, realT beta)
Set the initial guess for a symmetric Moffat.
realT fwhm()
Return the full-width at half maximum.
realT alpha()
Return the width parameter.
A templatized interface to the levmar package.
void allocate_params()
Allocate parameters array based on previous call to nParams.
void * adata
Pointer to possibly additional data, passed uninterpreted to func & jacf.
realT moffatFWHM(realT alpha, realT beta)
Compute the full-width at half-maximum of a Moffat profile.
Definition moffat.hpp:193
constexpr floatT six_fifths()
Return 6/5 in the specified precision.
A c++ interface to the templatized levmar minimization routines..
The mxlib c++ namespace.
Definition mxError.hpp:106
Wrapper for a native array to pass to levmarInterface, with Moffat details.
Definition fitMoffat.hpp:47
realT * data
Pointer to the array of y values.
Definition fitMoffat.hpp:48
void setFixed(bool I0, bool I, bool x0, bool alpha, bool beta)
Set whether each parameter is fixed.
Definition fitMoffat.hpp:69
size_t nx
X dimension of the array.
Definition fitMoffat.hpp:50
realT * coords
Pointer to the array of x values (optional)
Definition fitMoffat.hpp:49
Wrapper for a native array to pass to levmarInterface, with Moffat details.
void setFixed(bool I0, bool I, bool x0, bool y0, bool alpha, bool beta)
Set whether each parameter is fixed.
size_t nx
X dimension of the array.
size_t ny
Y dimension of the array.
realT * data
Pointer to the array.
levmarInterface fitter structure for the 1D Moffat.
levmarInterface fitter structure for the symmetric Moffat.