mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
fitsHeaderCard.cpp
Go to the documentation of this file.
1/** \file fitsHeaderCard.cpp
2 * \brief Definitiions for a class to work with a FITS header card
3 * \ingroup fits_processing_files
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015-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
28
29namespace mx
30{
31namespace fits
32{
33
34fitsHeaderCard::fitsHeaderCard( const std::string &k, const std::string &v, const std::string &c )
35{
36 m_keyword = k;
37 m_valueStr.str( v );
38 m_valueGood = false;
39 m_valueStrGood = true;
40 m_type = fitsType<std::string>();
41 m_comment = c;
42}
43
44fitsHeaderCard::fitsHeaderCard( const std::string &k, const std::string &v, const int &type, const std::string &c )
45{
46 m_keyword = k;
47 m_valueStr.str( v );
48 m_valueGood = false;
49 m_valueStrGood = true;
50 m_type = type;
51 m_comment = c;
52}
53
54fitsHeaderCard::fitsHeaderCard( const std::string &k, fitsCommentType v, const std::string &c )
55{
56 m_keyword = k;
57 m_valueGood = false;
58 m_valueStrGood = false;
59 m_type = fitsType<fitsCommentType>();
60 m_comment = c;
61}
62
63fitsHeaderCard::fitsHeaderCard( const std::string &k, fitsHistoryType v, const std::string &c )
64{
65 m_keyword = k;
66 m_valueGood = false;
67 m_valueStrGood = false;
68 m_type = fitsType<fitsHistoryType>();
69 m_comment = c;
70}
71
72fitsHeaderCard::fitsHeaderCard( const std::string &k )
73{
74 m_keyword = k;
75}
76
77fitsHeaderCard::fitsHeaderCard( const std::string &k, const int type )
78{
79 m_keyword = k;
80 m_type = type;
81}
82
84{
85 m_keyword = card.m_keyword;
86 m_type = card.m_type;
87 memcpy( &m_value, &card.m_value, sizeof( values ) );
88 m_valueStr.str( card.m_valueStr.str() );
91 m_comment = card.m_comment;
92}
93
95{
96 m_keyword = card.m_keyword;
97 m_type = card.m_type;
98 memcpy( &m_value, &card.m_value, sizeof( values ) );
99 m_valueStr.str( card.m_valueStr.str() );
102 m_comment = card.m_comment;
103
104 return *this;
105}
106
108{
109 if( !m_valueGood )
110 {
111 mxThrowException( err::paramnotset,
112 "fitsHeaderCard::convertToString()",
113 "no value to convert for " + m_keyword );
114 return;
115 }
116
117 if( m_type == fitsType<char *>() || m_type == fitsType<std::string>() )
118 {
119 m_valueStrGood = true; // It should be hard to get here, but just in case.
120 return;
121 }
122
123 m_valueStr.str( "" );
124 m_valueStr.precision( 10 );
125
126 switch( m_type )
127 {
128 case fitsType<char>():
129 m_valueStr << static_cast<int>( m_value.Char );
130 break;
131 case fitsType<unsigned char>():
132 m_valueStr << static_cast<int>( m_value.UChar );
133 break;
134 case fitsType<short>():
135 m_valueStr << m_value.Short;
136 break;
137 case fitsType<unsigned short>():
138 m_valueStr << m_value.UShort;
139 break;
140 case fitsType<int>():
141 m_valueStr << m_value.Int;
142 break;
143 case fitsType<unsigned int>():
144 m_valueStr << m_value.UInt;
145 break;
146 case fitsType<long>():
147 m_valueStr << m_value.Long;
148 break;
149 case fitsType<unsigned long>():
150 m_valueStr << m_value.ULong;
151 break;
152 case fitsType<long long>():
153 m_valueStr << m_value.LongLong;
154 break;
155 case fitsType<unsigned long long>():
156 m_valueStr << m_value.ULongLong;
157 break;
158 case fitsType<float>():
159 m_valueStr << m_value.Float;
160 break;
161 case fitsType<std::complex<float>>():
162 m_valueStr << m_value.complexFloat;
163 break;
164 case fitsType<double>():
165 m_valueStr << m_value.Double;
166 break;
167 case fitsType<std::complex<double>>():
168 m_valueStr << m_value.complexDouble;
169 break;
170 case fitsType<fitsCommentType>():
171 return;
172 case fitsType<fitsHistoryType>():
173 return;
174 default:
175 mxThrowException( err::invalidarg, "fitsHeaderCard::convertToString()", "Unknown FITS type" );
176 }
177
178 m_valueStrGood = true;
179 return;
180}
181
182template <>
183void fitsHeaderCard::convertFromString<char>()
184{
185 m_type = fitsType<char>();
186 try
187 {
188 m_value.Char = std::stoi( m_valueStr.str() );
189 }
190 catch( const std::exception &e )
191 {
192 std::string msg = "exception from std::stoi. no value for ";
193 msg += m_keyword;
194 m_valueGood = false;
195
196 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<char>", msg );
197 }
198 m_valueGood = true;
199}
200
201template <>
202void fitsHeaderCard::convertFromString<unsigned char>()
203{
204 m_type = fitsType<unsigned char>();
205 try
206 {
207 m_value.UChar = std::stoi( m_valueStr.str() );
208 }
209 catch( const std::exception &e )
210 {
211 std::string msg = "exception from std::stoi. no value for ";
212 msg += m_keyword;
213 m_valueGood = false;
214
215 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<unsigned char>", msg );
216 }
217 m_valueGood = true;
218}
219
220template <>
221void fitsHeaderCard::convertFromString<short>()
222{
223 m_type = fitsType<short>();
224 try
225 {
226 m_value.Short = std::stoi( m_valueStr.str() );
227 }
228 catch( const std::exception &e )
229 {
230 std::string msg = "exception from std::stoi. no value for ";
231 msg += m_keyword;
232 m_valueGood = false;
233
234 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<short>", msg );
235 }
236 m_valueGood = true;
237}
238
239template <>
240void fitsHeaderCard::convertFromString<unsigned short>()
241{
242 m_type = fitsType<unsigned short>();
243 try
244 {
245 m_value.UShort = std::stoi( m_valueStr.str() );
246 }
247 catch( const std::exception &e )
248 {
249 std::string msg = "exception from std::stoi. no value for ";
250 msg += m_keyword;
251 m_valueGood = false;
252
253 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<unsigned short>", msg );
254 }
255 m_valueGood = true;
256}
257
258template <>
259void fitsHeaderCard::convertFromString<int>()
260{
261 m_type = fitsType<int>();
262 try
263 {
264 m_value.Int = std::stoi( m_valueStr.str() );
265 }
266 catch( const std::exception &e )
267 {
268 std::string msg = "exception from std::stoi. no value for ";
269 msg += m_keyword;
270 m_valueGood = false;
271
272 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<int>", msg );
273 }
274 m_valueGood = true;
275}
276
277template <>
278void fitsHeaderCard::convertFromString<unsigned int>()
279{
280 m_type = fitsType<unsigned int>();
281 try
282 {
283 m_value.UInt = std::stol( m_valueStr.str() );
284 }
285 catch( const std::exception &e )
286 {
287 std::string msg = "exception from std::stoi. no value for ";
288 msg += m_keyword;
289 m_valueGood = false;
290
291 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<unsigned int>", msg );
292 }
293 m_valueGood = true;
294}
295
296template <>
297void fitsHeaderCard::convertFromString<long>()
298{
299 m_type = fitsType<long>();
300 try
301 {
302 m_value.Long = std::stol( m_valueStr.str() );
303 }
304 catch( const std::exception &e )
305 {
306 std::string msg = "exception from std::stoi. no value for ";
307 msg += m_keyword;
308 m_valueGood = false;
309
310 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<long>", msg );
311 }
312 m_valueGood = true;
313}
314
315template <>
316void fitsHeaderCard::convertFromString<unsigned long>()
317{
318 m_type = fitsType<unsigned long>();
319 try
320 {
321 m_value.ULong = std::stoul( m_valueStr.str() );
322 }
323 catch( const std::exception &e )
324 {
325 std::string msg = "exception from std::stoi. no value for ";
326 msg += m_keyword;
327 m_valueGood = false;
328
329 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<unsigned long>", msg );
330 }
331 m_valueGood = true;
332}
333
334template <>
335void fitsHeaderCard::convertFromString<long long>()
336{
337 m_type = fitsType<long long>();
338 try
339 {
340 m_value.LongLong = std::stoll( m_valueStr.str() );
341 }
342 catch( const std::exception &e )
343 {
344 std::string msg = "exception from std::stoi. no value for ";
345 msg += m_keyword;
346 m_valueGood = false;
347
348 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<long long>", msg );
349 }
350 m_valueGood = true;
351}
352
353template <>
354void fitsHeaderCard::convertFromString<unsigned long long>()
355{
356 m_type = fitsType<unsigned long long>();
357 try
358 {
359 m_value.ULongLong = std::stoull( m_valueStr.str() );
360 }
361 catch( const std::exception &e )
362 {
363 std::string msg = "exception from std::stoi. no value for ";
364 msg += m_keyword;
365 m_valueGood = false;
366
367 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<unsigned long long>", msg );
368 }
369 m_valueGood = true;
370}
371
372template <>
373void fitsHeaderCard::convertFromString<float>()
374{
375 m_type = fitsType<float>();
376 try
377 {
378 m_value.Float = std::stof( m_valueStr.str() );
379 }
380 catch( const std::exception &e )
381 {
382 std::string msg = "exception from std::stof. no value for ";
383 msg += m_keyword;
384 m_valueGood = false;
385
386 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<float>", msg );
387 }
388 m_valueGood = true;
389}
390
391template <>
392void fitsHeaderCard::convertFromString<std::complex<float>>()
393{
394 mxThrowException( err::notimpl,
395 "fitsHeaderCard::convertFromString<std::complex<float>>",
396 "no conversion from string to std::complex<float>" );
397 m_type = fitsType<std::complex<float>>();
398 m_value.complexFloat = std::stof( m_valueStr.str() );
399 m_valueGood = true;
400}
401
402template <>
403void fitsHeaderCard::convertFromString<double>()
404{
405
406 m_type = fitsType<double>();
407
408 try
409 {
410 m_value.Double = std::stod( m_valueStr.str() );
411 }
412 catch( const std::exception &e )
413 {
414 std::string msg = "exception from std::stod. no value for ";
415 msg += m_keyword;
416 m_valueGood = false;
417
418 mxThrowException( err::invalidarg, "fitsHeaderCard::convertFromString<float>", msg );
419 }
420 m_valueGood = true;
421}
422
423template <>
424void fitsHeaderCard::convertFromString<std::complex<double>>()
425{
426 mxThrowException( err::notimpl,
427 "fitsHeaderCard::convertFromString<std::complex<double>>",
428 "no conversion from string to std::complex<double>" );
429 m_type = fitsType<std::complex<double>>();
430 m_value.complexDouble = std::stof( m_valueStr.str() );
431 m_valueGood = true;
432}
433
434template <typename typeT>
436{
437 switch( m_type )
438 {
439 case fitsType<unsigned char>():
440 {
441 return m_value.UChar;
442 }
443 case fitsType<char>():
444 {
445 return m_value.Char;
446 }
447 case fitsType<short>():
448 {
449 return m_value.Short;
450 }
451 case fitsType<unsigned short>():
452 {
453 return m_value.UShort;
454 }
455 case fitsType<int>():
456 {
457 return m_value.Int;
458 }
459 case fitsType<unsigned int>():
460 {
461 return m_value.UInt;
462 }
463 case fitsType<long>():
464 {
465 return m_value.Long;
466 }
467 case fitsType<unsigned long>():
468 {
469 return m_value.ULong;
470 }
471 case fitsType<long long>():
472 {
473 return m_value.LongLong;
474 }
475 case fitsType<unsigned long long>():
476 {
477 return m_value.ULongLong;
478 }
479 case fitsType<float>():
480 {
481 return m_value.Float;
482 }
483 case fitsType<std::complex<float>>():
484 {
485 mxThrowException( err::notimpl,
486 "fitsHeaderCard::convertedValue<typeT>",
487 std::string( "conversions no supported for complex types in " ) + m_keyword );
488 }
489 case fitsType<double>():
490 {
491 return m_value.Double;
492 }
493 case fitsType<std::complex<double>>():
494 {
495 mxThrowException( err::notimpl,
496 "fitsHeaderCard::convertedValue<typeT>",
497 std::string( "conversions no supported for complex types in " ) + m_keyword );
498 }
499 case fitsType<fitsCommentType>():
500 {
501 mxThrowException( err::invalidarg,
502 "fitsHeaderCard::convertedValue<typeT>",
503 "cannot convert comment to numeric type" );
504 }
505 case fitsType<fitsHistoryType>():
506 {
507 mxThrowException( err::invalidarg,
508 "fitsHeaderCard::convertedValue<typeT>",
509 "cannot convert history to numeric type" );
510 }
511 case TSTRING:
512 {
513 mxThrowException( err::invalidarg,
514 "fitsHeaderCard::convertedValue<typeT>",
515 std::string( "cannot convert string to numeric type in " ) + m_keyword );
516 }
517 default:
518 {
519 mxThrowException( err::invalidarg,
520 "fitsHeaderCard::convertedValue<typeT>",
521 std::string( "invalid FITS type for conversion in " ) + m_keyword );
522 }
523 }
524}
525
527{
528 if( !m_valueGood )
529 {
530 m_type = newtype;
531 return;
532 }
533
534 switch( newtype )
535 {
536 case fitsType<unsigned char>():
537 {
538 m_value.UChar = convertedValue<unsigned char>();
539 break;
540 }
541 case fitsType<char>():
542 {
543 m_value.Char = convertedValue<char>();
544 break;
545 }
546 case fitsType<short>():
547 {
548 m_value.Short = convertedValue<short>();
549 break;
550 }
551 case fitsType<unsigned short>():
552 {
553 m_value.UShort = convertedValue<unsigned short>();
554 break;
555 }
556 case fitsType<int>():
557 {
558 m_value.Int = convertedValue<int>();
559 break;
560 }
561 case fitsType<unsigned int>():
562 {
563 m_value.UInt = convertedValue<unsigned int>();
564 break;
565 }
566 case fitsType<long>():
567 {
568 m_value.Long = convertedValue<long>();
569 break;
570 }
571 case fitsType<unsigned long>():
572 {
573 m_value.ULong = convertedValue<unsigned long>();
574 break;
575 }
576 case fitsType<long long>():
577 {
578 m_value.LongLong = convertedValue<long long>();
579 break;
580 }
581 case fitsType<unsigned long long>():
582 {
583 m_value.ULongLong = convertedValue<unsigned long long>();
584 break;
585 }
586 case fitsType<float>():
587 {
588 m_value.Float = convertedValue<float>();
589 break;
590 }
591 case fitsType<std::complex<float>>():
592 {
593 mxThrowException( err::notimpl,
594 "fitsHeaderCard::convertValue",
595 std::string( "conversions not supported for complex types in " ) + m_keyword );
596 }
597 case fitsType<double>():
598 {
599 m_value.Double = convertedValue<double>();
600 break;
601 }
602 case fitsType<std::complex<double>>():
603 {
604 mxThrowException( err::notimpl,
605 "fitsHeaderCard::convertValue",
606 std::string( "conversions not supported for complex types in " ) + m_keyword );
607 }
608 case fitsType<fitsCommentType>():
609 {
610 mxThrowException( err::invalidarg, "fitsHeaderCard::convertValue", "cannot convert comment to numeric type" );
611 }
612 case fitsType<fitsHistoryType>():
613 {
614 mxThrowException( err::invalidarg, "fitsHeaderCard::convertValue", "cannot convert history to numeric type" );
615 }
616 case TSTRING:
617 {
619 m_type = newtype;
620 m_valueGood = false;
621 return;
622 }
623 default:
624 {
625 mxThrowException( err::invalidarg,
626 "fitsHeaderCard::convertValue",
627 std::string( "invalid FITS type for conversion in " ) + m_keyword );
628 }
629 }
630
631 m_type = newtype;
632 m_valueGood = true;
633}
634
635const std::string &fitsHeaderCard::keyword() const
636{
637 return m_keyword;
638}
639
640void fitsHeaderCard::keyword( const std::string &kw )
641{
642 m_keyword = kw;
643}
644
645template <>
646std::string fitsHeaderCard::value<std::string>()
647{
648 if( m_valueStrGood == false )
649 {
650 convertToString();
651 }
652
653 // Strip ' from beginning and end if present
654 std::string str = m_valueStr.str();
655
656 if( str[0] == '\'' )
657 {
658 str.erase( 0, 1 );
659 }
660
661 if( str[str.size() - 1] == '\'' )
662 {
663 str.erase( str.size() - 1, 1 );
664 }
665
666 return str;
667}
668
669template <>
670char fitsHeaderCard::value<char>()
671{
672 if( m_valueGood == false )
673 {
674 convertFromString<char>();
675 }
676
677 if( m_type != fitsType<char>() )
678 {
679 return convertedValue<char>();
680 }
681
682 return m_value.Char;
683}
684
685template <>
686unsigned char fitsHeaderCard::value<unsigned char>()
687{
688 if( m_valueGood == false )
689 {
690 convertFromString<unsigned char>();
691 }
692
693 if( m_type != fitsType<unsigned char>() )
694 {
695 return convertedValue<unsigned char>();
696 }
697
698 return m_value.UChar;
699}
700
701template <>
702short fitsHeaderCard::value<short>()
703{
704 if( m_valueGood == false )
705 {
706 convertFromString<short>();
707 }
708
709 if( m_type != fitsType<short>() )
710 {
711 return convertedValue<short>();
712 }
713
714 return m_value.Short;
715}
716
717template <>
718unsigned short fitsHeaderCard::value<unsigned short>()
719{
720 if( m_valueGood == false )
721 {
722 convertFromString<unsigned short>();
723 }
724
725 if( m_type != fitsType<unsigned short>() )
726 {
727 return convertedValue<unsigned short>();
728 }
729
730 return m_value.UShort;
731}
732
733template <>
734int fitsHeaderCard::value<int>()
735{
736 if( m_valueGood == false )
737 {
738 convertFromString<int>();
739 }
740
741 if( m_type != fitsType<int>() )
742 {
743 return convertedValue<int>();
744 }
745
746 return m_value.Int;
747}
748
749template <>
750unsigned int fitsHeaderCard::value<unsigned int>()
751{
752 if( m_valueGood == false )
753 {
754 convertFromString<unsigned int>();
755 }
756
757 if( m_type != fitsType<unsigned int>() )
758 {
759 return convertedValue<unsigned int>();
760 }
761
762 return m_value.UInt;
763}
764
765template <>
766long fitsHeaderCard::value<long>()
767{
768 if( m_valueGood == false )
769 {
770 convertFromString<long>();
771 }
772
773 if( m_type != fitsType<long>() )
774 {
775 return convertedValue<long>();
776 }
777
778 return m_value.Long;
779}
780
781template <>
782unsigned long fitsHeaderCard::value<unsigned long>()
783{
784 if( m_valueGood == false )
785 {
786 convertFromString<unsigned long>();
787 }
788
789 if( m_type != fitsType<unsigned long>() )
790 {
791 return convertedValue<unsigned long>();
792 }
793
794 return m_value.ULong;
795}
796
797template <>
798long long fitsHeaderCard::value<long long>()
799{
800 if( m_valueGood == false )
801 {
802 convertFromString<long long>();
803 }
804
805 if( m_type != fitsType<long long>() )
806 {
807 return convertedValue<long long>();
808 }
809
810 return m_value.LongLong;
811}
812
813template <>
814unsigned long long fitsHeaderCard::value<unsigned long long>()
815{
816 if( m_valueGood == false )
817 {
818 convertFromString<unsigned long long>();
819 }
820
821 if( m_type != fitsType<unsigned long long>() )
822 {
823 return convertedValue<unsigned long long>();
824 }
825
826 return m_value.ULongLong;
827}
828
829template <>
830float fitsHeaderCard::value<float>()
831{
832 if( m_valueGood == false )
833 {
834 convertFromString<float>();
835 }
836
837 if( m_type != fitsType<float>() )
838 {
839 return convertedValue<float>();
840 }
841
842 return m_value.Float;
843}
844
845template <>
846std::complex<float> fitsHeaderCard::value<std::complex<float>>()
847{
848 if( m_valueGood == false )
849 {
850 convertFromString<std::complex<float>>();
851 }
852
853 if( m_type != fitsType<std::complex<float>>() )
854 {
855 return convertedValue<std::complex<float>>();
856 }
857
858 return m_value.complexFloat;
859}
860
861template <>
862double fitsHeaderCard::value<double>()
863{
864 if( m_valueGood == false )
865 {
866 convertFromString<double>();
867 }
868
869 if( m_type != fitsType<double>() )
870 {
871 return convertedValue<double>();
872 }
873
874 return m_value.Double;
875}
876
877template <>
878std::complex<double> fitsHeaderCard::value<std::complex<double>>()
879{
880 if( m_valueGood == false )
881 {
882 convertFromString<std::complex<double>>();
883 }
884
885 if( m_type != fitsType<std::complex<double>>() )
886 {
887 return convertedValue<std::complex<double>>();
888 }
889
890 return m_value.complexDouble;
891}
892
894{
895 return value<std::string>();
896}
897
899{
900 return value<char>();
901}
902
904{
905 return value<unsigned char>();
906}
907
909{
910 return value<short>();
911}
912
914{
915 return value<unsigned short>();
916}
917
919{
920 return value<int>();
921}
922
924{
925 return value<unsigned int>();
926}
927
929{
930 return value<long>();
931}
932
934{
935 return value<unsigned long>();
936}
937
939{
940 return value<long long>();
941}
942
943unsigned long long fitsHeaderCard::ULongLong()
944{
945 return value<unsigned long long>();
946}
947
949{
950 return value<float>();
951}
952
953std::complex<float> fitsHeaderCard::complexFloat()
954{
955 return value<std::complex<float>>();
956}
957
959{
960 return value<double>();
961}
962
963std::complex<double> fitsHeaderCard::complexDouble()
964{
965 return value<std::complex<double>>();
966}
967
968void fitsHeaderCard::value( const char *v )
969{
970 // Strip ' from beginning and end if present
971 std::string str = v;
972
973 if( str[0] == '\'' )
974 {
975 str.erase( 0, 1 );
976 }
977
978 if( str[str.size() - 1] == '\'' )
979 {
980 str.erase( str.size() - 1, 1 );
981 }
982
983 m_valueStr.str( str );
984
985 m_valueGood = false;
986 m_valueStrGood = true;
987 m_type = fitsType<char *>();
988}
989
990void fitsHeaderCard::value( const std::string &v )
991{
992 // Strip ' from beginning and end if present
993 std::string str = v;
994
995 if( str[0] == '\'' )
996 {
997 str.erase( 0, 1 );
998 }
999
1000 if( str[str.size() - 1] == '\'' )
1001 {
1002 str.erase( str.size() - 1, 1 );
1003 }
1004
1005 m_valueStr.str( v );
1006 m_valueGood = false;
1007 m_valueStrGood = true;
1008 m_type = fitsType<std::string>();
1009}
1010
1011void fitsHeaderCard::value( const char &v )
1012{
1013 m_value.Char = v;
1014 m_valueGood = true;
1015 m_valueStrGood = false;
1016 m_type = fitsType<char>();
1017}
1018
1019void fitsHeaderCard::value( const unsigned char &v )
1020{
1021 m_value.UChar = v;
1022 m_valueGood = true;
1023 m_valueStrGood = false;
1024 m_type = fitsType<unsigned char>();
1025}
1026
1027void fitsHeaderCard::value( const short int &v )
1028{
1029 m_value.Short = v;
1030 m_valueGood = true;
1031 m_valueStrGood = false;
1032 m_type = fitsType<short>();
1033}
1034
1035void fitsHeaderCard::value( const unsigned short &v )
1036{
1037 m_value.UShort = v;
1038 m_valueGood = true;
1039 m_valueStrGood = false;
1040 m_type = fitsType<unsigned short>();
1041}
1042
1043void fitsHeaderCard::value( const int &v )
1044{
1045 m_value.Int = v;
1046 m_valueGood = true;
1047 m_valueStrGood = false;
1048 m_type = fitsType<int>();
1049}
1050
1051void fitsHeaderCard::value( const unsigned int &v )
1052{
1053 m_value.UInt = v;
1054 m_valueGood = true;
1055 m_valueStrGood = false;
1056 m_type = fitsType<unsigned int>();
1057}
1058
1059void fitsHeaderCard::value( const long &v )
1060{
1061 m_value.Long = v;
1062 m_valueGood = true;
1063 m_valueStrGood = false;
1064 m_type = fitsType<long>();
1065}
1066
1067void fitsHeaderCard::value( const unsigned long int &v )
1068{
1069 m_value.ULong = v;
1070 m_valueGood = true;
1071 m_valueStrGood = false;
1072 m_type = fitsType<unsigned long>();
1073}
1074
1075void fitsHeaderCard::value( const long long &v )
1076{
1077 m_value.LongLong = v;
1078 m_valueGood = true;
1079 m_valueStrGood = false;
1080 m_type = fitsType<long long>();
1081}
1082
1083void fitsHeaderCard::value( const unsigned long long int &v )
1084{
1085 m_value.ULongLong = v;
1086 m_valueGood = true;
1087 m_valueStrGood = false;
1088 m_type = fitsType<unsigned long long>();
1089}
1090
1091void fitsHeaderCard::value( const float &v )
1092{
1093 m_value.Float = v;
1094 m_valueGood = true;
1095 m_valueStrGood = false;
1096 m_type = fitsType<float>();
1097}
1098
1099void fitsHeaderCard::value( const std::complex<float> &v )
1100{
1101 m_value.complexFloat = v;
1102 m_valueGood = true;
1103 m_valueStrGood = false;
1104 m_type = fitsType<std::complex<float>>();
1105}
1106
1107void fitsHeaderCard::value( const double &v )
1108{
1109 m_value.Double = v;
1110 m_valueGood = true;
1111 m_valueStrGood = false;
1112 m_type = fitsType<double>();
1113}
1114
1115void fitsHeaderCard::value( const std::complex<double> &v )
1116{
1117 m_value.complexDouble = v;
1118 m_valueGood = true;
1119 m_valueStrGood = false;
1120 m_type = fitsType<std::complex<double>>();
1121}
1122
1124{
1125 return m_type;
1126}
1127
1128void fitsHeaderCard::type( const int &t )
1129{
1130 if( t == m_type )
1131 return;
1132
1133 if( m_valueGood )
1134 {
1135 convertValue( t );
1136 }
1137 else
1138 m_type = t;
1139
1140 // Need to reconvert, always favor the actual value.
1142 {
1143 m_valueStrGood = false;
1144 }
1145}
1146
1147std::string fitsHeaderCard::valueStr()
1148{
1149 if( !m_valueStrGood )
1150 {
1152 }
1153
1154 std::string s = m_valueStr.str();
1155 return s;
1156}
1157
1158bool fitsHeaderCard::valueGood()
1159{
1160 return m_valueGood;
1161}
1162
1163bool fitsHeaderCard::valueStrGood()
1164{
1165 return m_valueStrGood;
1166}
1167
1168const std::string &fitsHeaderCard::comment()
1169{
1170 return m_comment;
1171}
1172
1173void fitsHeaderCard::comment( const std::string &c )
1174{
1175 m_comment = c;
1176}
1177
1178int fitsHeaderCard::write( fitsfile *fptr )
1179{
1180 if( m_type == fitsType<char *>() || m_type == fitsType<std::string>() )
1181 {
1182 return fits_write_key<char *>( fptr,
1183 (char *)m_keyword.c_str(),
1184 (void *)m_valueStr.str().c_str(),
1185 (char *)m_comment.c_str() );
1186 }
1187
1188 // If the string is good, meaning already converted.
1189 if( m_valueStrGood == true )
1190 {
1191 // This populates the card directly.
1192 return fits_write_key<fitsUnknownType>( fptr,
1193 (char *)m_keyword.c_str(),
1194 (void *)m_valueStr.str().c_str(),
1195 (char *)m_comment.c_str() );
1196 }
1197
1198 // Ok, now we write the type directly using fitsio routines because it hasn't been converted.
1199 switch( m_type )
1200 {
1201 case fitsType<unsigned char>():
1202 {
1203 return fits_write_key<unsigned char>( fptr,
1204 (char *)m_keyword.c_str(),
1205 &m_value.UChar,
1206 (char *)m_comment.c_str() );
1207 }
1208 case fitsType<char>():
1209 {
1210 return fits_write_key<char>( fptr, (char *)m_keyword.c_str(), &m_value.Char, (char *)m_comment.c_str() );
1211 }
1212 case fitsType<short>():
1213 {
1214 return fits_write_key<short>( fptr, (char *)m_keyword.c_str(), &m_value.Short, (char *)m_comment.c_str() );
1215 }
1216 case fitsType<unsigned short>():
1217 {
1218 return fits_write_key<unsigned short>( fptr,
1219 (char *)m_keyword.c_str(),
1220 &m_value.UShort,
1221 (char *)m_comment.c_str() );
1222 }
1223 case fitsType<int>():
1224 {
1225 return fits_write_key<int>( fptr, (char *)m_keyword.c_str(), &m_value.Int, (char *)m_comment.c_str() );
1226 }
1227 case fitsType<unsigned int>():
1228 {
1229 return fits_write_key<unsigned int>( fptr,
1230 (char *)m_keyword.c_str(),
1231 &m_value.UInt,
1232 (char *)m_comment.c_str() );
1233 }
1234 case fitsType<long>():
1235 {
1236 return fits_write_key<long>( fptr, (char *)m_keyword.c_str(), &m_value.Long, (char *)m_comment.c_str() );
1237 }
1238 case fitsType<unsigned long>():
1239 {
1240 return fits_write_key<unsigned long>( fptr,
1241 (char *)m_keyword.c_str(),
1242 &m_value.ULong,
1243 (char *)m_comment.c_str() );
1244 }
1245 case fitsType<long long>():
1246 {
1247 return fits_write_key<long long>( fptr,
1248 (char *)m_keyword.c_str(),
1249 &m_value.LongLong,
1250 (char *)m_comment.c_str() );
1251 }
1252 case fitsType<unsigned long long>():
1253 {
1254 return fits_write_key<unsigned long long>( fptr,
1255 (char *)m_keyword.c_str(),
1256 &m_value.ULongLong,
1257 (char *)m_comment.c_str() );
1258 }
1259 case fitsType<float>():
1260 {
1261 return fits_write_key<float>( fptr, (char *)m_keyword.c_str(), &m_value.Float, (char *)m_comment.c_str() );
1262 }
1263 case fitsType<double>():
1264 {
1265 return fits_write_key<double>( fptr, (char *)m_keyword.c_str(), &m_value.Double, (char *)m_comment.c_str() );
1266 }
1267 case fitsType<fitsCommentType>():
1268 {
1269 return fits_write_comment( fptr, (char *)m_comment.c_str() );
1270 }
1271 case fitsType<fitsHistoryType>():
1272 {
1273 return fits_write_history( fptr, (char *)m_comment.c_str() );
1274 }
1275 default:
1276 {
1277 mxThrowException( err::invalidarg,
1278 "fitsHeaderCard::write",
1279 std::string( "invalid FITS type for " ) + m_keyword );
1280 }
1281 }
1282}
1283
1284} // namespace fits
1285} // namespace mx
mxException for invalid arguments
mxException for not implemented features
mxException for parameters which aren't set
Class to manage the three components of a FITS header card.
std::complex< float > complexFloat()
Get the value as a std::complex<float>
const std::string & keyword() const
Get the keyword.
fitsHeaderCard & operator=(const fitsHeaderCard &card)
Assignment.
int type() const
Get the type.
int m_type
The FITS type of the value, and indicates which member of m_values to access.
std::stringstream m_valueStr
The value in string form.
char Char()
Get the value as a char.
std::string m_keyword
The keyword.
bool m_valueGood
Flag indicating if the value is valid.
const std::string & comment()
Get the comment.
typeT value()
Get the value.
double Double()
Get the value as a double.
long Long()
Get the value as a long.
int Int()
Get the value as a int.
unsigned char UChar()
Get the value as an unsigned char.
typeT convertedValue()
Get the value from its type converted to a different type.
unsigned short UShort()
Get the value as an unsigned short.
unsigned long ULong()
Get the value as an unsigned long.
unsigned int UInt()
Get the value as an unsigned int.
std::complex< double > complexDouble()
Get the value as a std::complex<double>
short Short()
Get the value as a short.
std::string m_comment
The comment.
bool m_valueStrGood
Flag indicating if the value string is valid.
long long LongLong()
Get the value as a long long.
void convertValue(int newtype)
Convert the value from its type to a different type.
unsigned long long ULongLong()
Get the value as an unsigned long long.
void convertToString()
Convert from the type to a string.
std::string String()
Get the value as a string.
int write(fitsfile *fptr)
float Float()
Get the value as a float.
A class to work with a FITS header card.
constexpr int fitsType()
Return the cfitsio constant for a given data type.
The mxlib c++ namespace.
Definition mxError.hpp:106
The native type is held in a union.
unsigned int UInt
the unsigned int value
unsigned long long ULongLong
the unsigned long long value
std::complex< double > complexDouble
the std::complex<double> value
std::complex< float > complexFloat
the std::complex<float> value
unsigned short UShort
the unsigned short value
unsigned long ULong
the unsigned long value
unsigned char UChar
the unsigned char value
long long LongLong
the long long value