annotate babel/numbers.py @ 28:695884591af6

* Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime. * Move locale data loading from `babel.core` into a separate `babel.localedata` module. * Add curency names and symbols to locale data.
author cmlenz
date Sun, 03 Jun 2007 15:27:27 +0000
parents 6c2c9fc7d787
children 8b6804eac9e5
rev   line source
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
1 # -*- coding: utf-8 -*-
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
2 #
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
3 # Copyright (C) 2007 Edgewall Software
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
4 # All rights reserved.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
5 #
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
6 # This software is licensed as described in the file COPYING, which
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
7 # you should have received as part of this distribution. The terms
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
8 # are also available at http://babel.edgewall.org/wiki/License.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
9 #
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
10 # This software consists of voluntary contributions made by many
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
11 # individuals. For the exact contribution history, see the revision
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
12 # history and logs, available at http://babel.edgewall.org/log/.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
13
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
14 """Locale dependent formatting and parsing of numeric data.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
15
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
16 The default locale for the functions in this module is determined by the
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
17 following environment variables, in that order:
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
18
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
19 * ``LC_NUMERIC``,
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
20 * ``LC_ALL``, and
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
21 * ``LANG``
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
22 """
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
23 # TODO: percent and scientific formatting
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
24
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
25 import re
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
26
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
27 from babel.core import Locale
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
28 from babel.util import default_locale
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
29
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
30 __all__ = ['format_number', 'format_decimal', 'format_currency',
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
31 'format_percent', 'format_scientific', 'parse_number',
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
32 'parse_decimal']
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
33 __docformat__ = 'restructuredtext en'
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
34
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
35 LC_NUMERIC = default_locale('LC_NUMERIC')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
36
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
37 def get_decimal_symbol(locale=LC_NUMERIC):
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
38 """Return the symbol used by the locale to separate decimal fractions.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
39
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
40 >>> get_decimal_symbol('en_US')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
41 u'.'
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
42
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
43 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
44 :return: the decimal symbol
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
45 :rtype: `unicode`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
46 """
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
47 return Locale.parse(locale).number_symbols.get('decimal', u'.')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
48
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
49 def get_group_symbol(locale=LC_NUMERIC):
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
50 """Return the symbol used by the locale to separate groups of thousands.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
51
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
52 >>> get_group_symbol('en_US')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
53 u','
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
54
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
55 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
56 :return: the group symbol
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
57 :rtype: `unicode`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
58 """
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
59 return Locale.parse(locale).number_symbols.get('group', u',')
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
60
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
61 def format_number(number, locale=LC_NUMERIC):
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
62 """Returns the given number formatted for a specific locale.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
63
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
64 >>> format_number(1099, locale='en_US')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
65 u'1,099'
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
66
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
67 :param number: the number to format
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
68 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
69 :return: the formatted number
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
70 :rtype: `unicode`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
71 """
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
72 # Do we really need this one?
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
73 return format_decimal(number, locale=locale)
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
74
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
75 def format_decimal(number, format=None, locale=LC_NUMERIC):
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
76 """Returns the given decimal number formatted for a specific locale.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
77
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
78 >>> format_decimal(1, locale='en_US')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
79 u'1'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
80 >>> format_decimal(1.2345, locale='en_US')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
81 u'1.234'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
82 >>> format_decimal(1.2345, locale='sv_SE')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
83 u'1,234'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
84 >>> format_decimal(12345, locale='de_DE')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
85 u'12.345'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
86 >>> format_decimal(-1.2345, format='#,##0.##;-#', locale='sv_SE')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
87 u'-1,23'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
88 >>> format_decimal(-1.2345, format='#,##0.##;(#)', locale='sv_SE')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
89 u'(1,23)'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
90
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
91 The appropriate thousands grouping and the decimal separator are used for
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
92 each locale:
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
93
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
94 >>> format_decimal(12345, locale='en_US')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
95 u'12,345'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
96
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
97 :param number: the number to format
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
98 :param format:
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
99 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
100 :return: the formatted decimal number
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
101 :rtype: `unicode`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
102 """
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
103 locale = Locale.parse(locale)
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
104 pattern = locale.decimal_formats.get(format)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
105 if not pattern:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
106 pattern = parse_pattern(format)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
107 return pattern.apply(number, locale)
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
108
28
695884591af6 * Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime.
cmlenz
parents: 24
diff changeset
109 def format_currency(number, locale=LC_NUMERIC):
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
110 """Returns formatted currency value.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
111
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
112 >>> format_currency(1099.98, locale='en_US')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
113 u'1,099.98'
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
114
28
695884591af6 * Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime.
cmlenz
parents: 24
diff changeset
115 :param number: the number to format
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
116 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
117 :return: the formatted currency value
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
118 :rtype: `unicode`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
119 """
28
695884591af6 * Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime.
cmlenz
parents: 24
diff changeset
120 return format_decimal(number, locale=locale)
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
121
28
695884591af6 * Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime.
cmlenz
parents: 24
diff changeset
122 def format_percent(number, format=None, locale=LC_NUMERIC):
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
123 """Returns formatted percent value for a specific locale.
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
124
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
125 >>> format_percent(0.34, locale='en_US')
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
126 u'34%'
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
127 >>> format_percent(25.1234, locale='en_US')
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
128 u'2,512%'
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
129 >>> format_percent(25.1234, locale='sv_SE')
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
130 u'2\\xa0512 %'
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
131
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
132 :param number: the percent number to format
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
133 :param format:
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
134 :param locale: the `Locale` object or locale identifier
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
135 :return: the formatted percent number
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
136 :rtype: `unicode`
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
137 """
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
138 locale = Locale.parse(locale)
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
139 pattern = locale.percent_formats.get(format)
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
140 if not pattern:
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
141 pattern = parse_pattern(format)
28
695884591af6 * Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime.
cmlenz
parents: 24
diff changeset
142 return pattern.apply(number, locale)
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
143
28
695884591af6 * Reduce size of locale data pickles by only storing the data provided by each locale itself, and merging inherited data at runtime.
cmlenz
parents: 24
diff changeset
144 def format_scientific(number, locale=LC_NUMERIC):
3
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
145 raise NotImplementedError
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
146
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
147 def parse_number(string, locale=LC_NUMERIC):
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
148 """Parse localized number string into a long integer.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
149
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
150 >>> parse_number('1,099', locale='en_US')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
151 1099L
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
152 >>> parse_number('1.099', locale='de_DE')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
153 1099L
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
154
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
155 :param string: the string to parse
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
156 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
157 :return: the parsed number
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
158 :rtype: `long`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
159 :raise `ValueError`: if the string can not be converted to a number
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
160 """
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
161 return long(string.replace(get_group_symbol(locale), ''))
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
162
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
163 def parse_decimal(string, locale=LC_NUMERIC):
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
164 """Parse localized decimal string into a float.
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
165
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
166 >>> parse_decimal('1,099.98', locale='en_US')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
167 1099.98
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
168 >>> parse_decimal('1.099,98', locale='de_DE')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
169 1099.98
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
170
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
171 :param string: the string to parse
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
172 :param locale: the `Locale` object or locale identifier
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
173 :return: the parsed decimal number
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
174 :rtype: `float`
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
175 :raise `ValueError`: if the string can not be converted to a decimal number
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
176 """
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
177 locale = Locale.parse(locale)
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
178 string = string.replace(get_group_symbol(locale), '') \
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
179 .replace(get_decimal_symbol(locale), '.')
e9eaddab598e Import of initial code base.
cmlenz
parents:
diff changeset
180 return float(string)
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
181
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
182
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
183 PREFIX_END = r'[^0-9@#.,]'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
184 NUMBER_TOKEN = r'[0-9@#.\-,E]'
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
185
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
186 PREFIX_PATTERN = r"(?P<prefix>(?:'[^']*'|%s)*)" % PREFIX_END
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
187 NUMBER_PATTERN = r"(?P<number>%s+)" % NUMBER_TOKEN
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
188 SUFFIX_PATTERN = r"(?P<suffix>.*)"
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
189
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
190 number_re = re.compile(r"%s%s%s" % (PREFIX_PATTERN, NUMBER_PATTERN,
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
191 SUFFIX_PATTERN))
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
192
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
193 # TODO:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
194 # Filling
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
195 # Rounding
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
196 # Scientific notation
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
197 # Significant Digits
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
198 def parse_pattern(pattern):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
199 """Parse number format patterns"""
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
200 if isinstance(pattern, NumberPattern):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
201 return pattern
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
202
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
203 # Do we have a negative subpattern?
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
204 if ';' in pattern:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
205 pattern, neg_pattern = pattern.split(';', 1)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
206 pos_prefix, number, pos_suffix = number_re.search(pattern).groups()
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
207 neg_prefix, _, neg_suffix = number_re.search(neg_pattern).groups()
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
208 else:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
209 pos_prefix, number, pos_suffix = number_re.search(pattern).groups()
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
210 neg_prefix = '-' + pos_prefix
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
211 neg_suffix = pos_suffix
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
212 if '.' in number:
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
213 integer, fraction = number.rsplit('.', 1)
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
214 else:
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
215 integer = number
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
216 fraction = ''
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
217 min_frac = max_frac = 0
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
218
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
219 def parse_precision(p):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
220 """Calculate the min and max allowed digits"""
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
221 min = max = 0
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
222 for c in p:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
223 if c == '0':
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
224 min += 1
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
225 max += 1
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
226 elif c == '#':
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
227 max += 1
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
228 else:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
229 break
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
230 return min, max
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
231
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
232 def parse_grouping(p):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
233 """Parse primary and secondary digit grouping
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
234
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
235 >>> parse_grouping('##')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
236 0, 0
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
237 >>> parse_grouping('#,###')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
238 3, 3
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
239 >>> parse_grouping('#,####,###')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
240 3, 4
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
241 """
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
242 width = len(p)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
243 g1 = p.rfind(',')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
244 if g1 == -1:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
245 return 1000, 1000
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
246 g1 = width - g1 - 1
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
247 g2 = p[:-g1 - 1].rfind(',')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
248 if g2 == -1:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
249 return g1, g1
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
250 g2 = width - g1 - g2 - 2
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
251 return g1, g2
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
252
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
253 int_precision = parse_precision(integer)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
254 frac_precision = parse_precision(fraction)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
255 grouping = parse_grouping(integer)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
256 int_precision = (int_precision[0], 1000) # Unlimited
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
257 return NumberPattern(pattern, (pos_prefix, neg_prefix),
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
258 (pos_suffix, neg_suffix), grouping,
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
259 int_precision, frac_precision)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
260
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
261
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
262 class NumberPattern(object):
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
263
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
264 def __init__(self, pattern, prefix, suffix, grouping,
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
265 int_precision, frac_precision):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
266 self.pattern = pattern
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
267 self.prefix = prefix
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
268 self.suffix = suffix
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
269 self.grouping = grouping
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
270 self.int_precision = int_precision
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
271 self.frac_precision = frac_precision
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
272 if '%' in ''.join(self.prefix + self.suffix):
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
273 self.scale = 100.0
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
274 elif u'‰' in ''.join(self.prefix + self.suffix):
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
275 self.scale = 1000.0
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
276 else:
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
277 self.scale = 1.0
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
278
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
279 def __repr__(self):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
280 return '<%s %r>' % (type(self).__name__, self.pattern)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
281
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
282 def apply(self, value, locale):
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
283 value *= self.scale
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
284 negative = int(value < 0)
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
285 a, b = str(value).split('.')
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
286 a = a.lstrip('-')
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
287 return '%s%s%s%s' % (self.prefix[negative],
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
288 self._format_int(a, locale),
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
289 self._format_frac(b, locale),
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
290 self.suffix[negative])
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
291
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
292 def _format_int(self, value, locale):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
293 min, max = self.int_precision
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
294 width = len(value)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
295 if width < min:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
296 value += '0' * (min - width)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
297 gsize = self.grouping[0]
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
298 ret = ''
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
299 symbol = get_group_symbol(locale)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
300 while len(value) > gsize:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
301 ret = symbol + value[-gsize:] + ret
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
302 value = value[:-gsize]
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
303 gsize = self.grouping[1]
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
304 return value + ret
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
305
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
306 def _format_frac(self, value, locale):
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
307 min, max = self.frac_precision
24
6c2c9fc7d787 Implemented babel.numbers.format_percent
jonas
parents: 11
diff changeset
308 if max == 0 or (min == 0 and int(value) == 0):
11
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
309 return ''
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
310 width = len(value)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
311 if width < min:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
312 value += '0' * (min - width)
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
313 if width > max:
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
314 value = value[:max] # FIXME: Rounding?!?
11f64b232b04 Add basic support for number format patterns.
jonas
parents: 3
diff changeset
315 return get_decimal_symbol(locale) + value
Copyright (C) 2012-2017 Edgewall Software