1
/* -*- mode: c++; c-basic-offset: 2; indent-tabs-mode: nil; -*-
2
* vim:expandtab:shiftwidth=2:tabstop=2:smarttab:
4
* Copyright (C) 2008 Sun Microsystems
6
* This program is free software; you can redistribute it and/or modify
7
* it under the terms of the GNU General Public License as published by
8
* the Free Software Foundation; version 2 of the License.
10
* This program is distributed in the hope that it will be useful,
11
* but WITHOUT ANY WARRANTY; without even the implied warranty of
12
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13
* GNU General Public License for more details.
15
* You should have received a copy of the GNU General Public License
16
* along with this program; if not, write to the Free Software
17
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
20
#include <drizzled/server_includes.h>
23
#include <drizzled/functions/str/format.h>
26
Change a number to format '3,333,333,333.000'.
28
This should be 'internationalized' sometimes.
31
const int FORMAT_MAX_DECIMALS= 30;
33
Item_func_format::Item_func_format(Item *org, Item *dec)
34
: Item_str_func(org, dec)
38
void Item_func_format::fix_length_and_dec()
40
collation.set(default_charset());
41
uint32_t char_length= args[0]->max_length/args[0]->collation.collation->mbmaxlen;
42
max_length= ((char_length + (char_length-args[0]->decimals)/3) *
43
collation.collation->mbmaxlen);
49
This needs to be fixed for multi-byte character set where numbers
50
are stored in more than one byte
53
String *Item_func_format::val_str(String *str)
57
/* Number of decimal digits */
59
/* Number of characters used to represent the decimals, including '.' */
64
dec= (int) args[1]->val_int();
65
if (args[1]->null_value)
71
dec= set_zone(dec, 0, FORMAT_MAX_DECIMALS);
72
dec_length= dec ? dec+1 : 0;
75
if (args[0]->result_type() == DECIMAL_RESULT ||
76
args[0]->result_type() == INT_RESULT)
78
my_decimal dec_val, rnd_dec, *res;
79
res= args[0]->val_decimal(&dec_val);
80
if ((null_value=args[0]->null_value))
81
return 0; /* purecov: inspected */
82
my_decimal_round(E_DEC_FATAL_ERROR, res, dec, false, &rnd_dec);
83
my_decimal2string(E_DEC_FATAL_ERROR, &rnd_dec, 0, 0, 0, str);
84
str_length= str->length();
90
double nr= args[0]->val_real();
91
if ((null_value=args[0]->null_value))
92
return 0; /* purecov: inspected */
93
nr= my_double_round(nr, (int64_t) dec, false, false);
94
/* Here default_charset() is right as this is not an automatic conversion */
95
str->set_real(nr, dec, default_charset());
98
str_length=str->length();
100
str_length--; // Don't count sign
102
/* We need this test to handle 'nan' values */
103
if (str_length >= dec_length+4)
106
length= str->length()+(diff=((int)(str_length- dec_length-1))/3);
107
str= copy_if_not_alloced(&tmp_str,str,length);
109
tmp= (char*) str->ptr()+length - dec_length-1;
110
for (pos= (char*) str->ptr()+length-1; pos != tmp; pos--)
129
void Item_func_format::print(String *str, enum_query_type query_type)
131
str->append(STRING_WITH_LEN("format("));
132
args[0]->print(str, query_type);
134
args[1]->print(str, query_type);