15
15
16
16
namespace csv {
17
17
namespace internals {
18
+ static int DECIMAL_PLACES = 5 ;
19
+
18
20
/* * to_string() for unsigned integers */
19
21
template <typename T,
20
22
csv::enable_if_t <std::is_unsigned<T>::value, int > = 0 >
@@ -48,25 +50,52 @@ namespace csv {
48
50
typename T,
49
51
csv::enable_if_t <std::is_floating_point<T>::value, int > = 0
50
52
>
51
- inline std::string to_string (T value) {
52
- std::string result;
53
+ inline std::string to_string (T value) {
54
+ std::string result;
53
55
54
- if (value < 0 ) result = " -" ;
55
-
56
- // Integral part
57
- size_t integral = (size_t )(std::abs (value));
58
- result += (integral == 0 ) ? " 0" : to_string (integral);
56
+ T integral_part;
57
+ T fractional_part = std::abs (std::modf (value, &integral_part));
58
+ integral_part = std::abs (integral_part);
59
+
60
+ // Integral part
61
+ if (value < 0 ) result = " -" ;
62
+
63
+ if (integral_part == 0 ) {
64
+ result = " 0" ;
65
+ }
66
+ else {
67
+ for (short n_digits = log (integral_part) / log (10 ); n_digits + 1 > 0 ; n_digits --) {
68
+ short digit = std::fmod (integral_part, pow10 (n_digits + 1 )) / pow10 (n_digits);
69
+ result += (char )(' 0' + digit);
70
+ }
71
+ }
59
72
60
- // Decimal part
61
- size_t decimal = ( size_t )((( double ) std::abs (value) - ( double )integral) * 100000 ) ;
73
+ // Decimal part
74
+ result += " . " ;
62
75
63
- result += " ." ;
64
- result += (decimal == 0 ) ? " 0" : to_string (decimal);
76
+ if (fractional_part > 0 ) {
77
+ fractional_part *= pow10 (DECIMAL_PLACES);
78
+ for (short n_digits = DECIMAL_PLACES; n_digits > 0 ; n_digits--) {
79
+ short digit = std::fmod (fractional_part, pow10 (n_digits)) / pow10 (n_digits - 1 );
80
+ result += (char )(' 0' + digit);
81
+ }
82
+ }
83
+ else {
84
+ result += " 0" ;
85
+ }
65
86
66
- return result;
87
+ return result;
67
88
}
68
89
}
69
90
91
+ /* * Sets how many places after the decimal will be written for floating point numbers
92
+ *
93
+ * @param precision Number of decimal places
94
+ */
95
+ inline static void set_decimal_places (int precision) {
96
+ internals::DECIMAL_PLACES = precision;
97
+ }
98
+
70
99
/* * @name CSV Writing */
71
100
// /@{
72
101
/* *
@@ -102,6 +131,7 @@ namespace csv {
102
131
* @param _out Stream to write to
103
132
* @param _quote_minimal Limit field quoting to only when necessary
104
133
*/
134
+
105
135
DelimWriter (OutputStream& _out, bool _quote_minimal = true )
106
136
: out(_out), quote_minimal(_quote_minimal) {};
107
137
@@ -213,7 +243,7 @@ namespace csv {
213
243
bool quote_escape = false ;
214
244
215
245
for (auto ch : in) {
216
- if (ch == Quote || ch == Delim) {
246
+ if (ch == Quote || ch == Delim || ch == ' \r ' || ch == ' \n ' ) {
217
247
quote_escape = true ;
218
248
break ;
219
249
}
@@ -222,9 +252,10 @@ namespace csv {
222
252
if (!quote_escape) {
223
253
if (quote_minimal) return std::string (in);
224
254
else {
225
- std::string ret (Quote, 1 );
255
+ std::string ret (1 , Quote );
226
256
ret += in.data ();
227
257
ret += Quote;
258
+ return ret;
228
259
}
229
260
}
230
261
0 commit comments