Fixed #5431 -- Added Argentinean localflavor. Thanks, Ramiro Morales.
[django.git] / django / contrib / localflavor / ar / forms.py
blob6d86e4e67667c10a326ead275c14015c99e709fd
1 # -*- coding: utf-8 -*-
2 """
3 AR-specific Form helpers.
4 """
6 from django.newforms import ValidationError
7 from django.newforms.fields import RegexField, CharField, Select, EMPTY_VALUES
8 from django.utils.encoding import smart_unicode
9 from django.utils.translation import ugettext
10 import re
12 class ARProvinceSelect(Select):
13 """
14 A Select widget that uses a list of Argentinean provinces/autonomous cities
15 as its choices.
16 """
17 def __init__(self, attrs=None):
18 from ar_provinces import PROVINCE_CHOICES
19 super(ARProvinceSelect, self).__init__(attrs, choices=PROVINCE_CHOICES)
21 class ARPostalCodeField(RegexField):
22 """
23 A field that accepts a `classic´ NNNN Postal Code or a CPA.
25 See http://www.correoargentino.com.ar/consulta_cpa/home.php
26 """
27 def __init__(self, *args, **kwargs):
28 super(ARPostalCodeField, self).__init__(r'^\d{4}$|^[A-HJ-NP-Za-hj-np-z]\d{4}\D{3}$',
29 min_length=4, max_length=8,
30 error_message=ugettext("Enter a postal code in the format NNNN or ANNNNAAA."),
31 *args, **kwargs)
33 def clean(self, value):
34 value = super(ARPostalCodeField, self).clean(value)
35 if value in EMPTY_VALUES:
36 return u''
37 if len(value) not in (4, 8):
38 raise ValidationError(ugettext("Enter a postal code in the format NNNN or ANNNNAAA."))
39 if len(value) == 8:
40 return u'%s%s%s' % (value[0].upper(), value[1:5], value[5:].upper())
41 return value
43 class ARDNIField(CharField):
44 """
45 A field that validates `Documento Nacional de Identidad´ (DNI) numbers.
46 """
47 def __init__(self, *args, **kwargs):
48 super(ARDNIField, self).__init__(max_length=10, min_length=7, *args,
49 **kwargs)
51 def clean(self, value):
52 """
53 Value can be a string either in the [X]X.XXX.XXX or [X]XXXXXXX formats.
54 """
55 value = super(ARDNIField, self).clean(value)
56 if value in EMPTY_VALUES:
57 return u''
58 if not value.isdigit():
59 value = value.replace('.', '')
60 if not value.isdigit():
61 raise ValidationError(ugettext("This field requires only numbers."))
62 if len(value) not in (7, 8):
63 raise ValidationError(
64 ugettext("This field requires 7 or 8 digits."))
66 return value
68 class ARCUITField(RegexField):
69 """
70 This field validates a CUIT (Código Único de Identificación Tributaria). A
71 CUIT is of the form XX-XXXXXXXX-V. The last digit is a check digit.
72 """
73 def __init__(self, *args, **kwargs):
74 super(ARCUITField, self).__init__(r'^\d{2}-?\d{8}-?\d$',
75 error_message=ugettext('Enter a valid CUIT in XX-XXXXXXXX-X or XXXXXXXXXXXX format.'),
76 *args, **kwargs)
78 def clean(self, value):
79 """
80 Value can be either a string in the format XX-XXXXXXXX-X or an
81 11-digit number.
82 """
83 value = super(ARCUITField, self).clean(value)
84 if value in EMPTY_VALUES:
85 return u''
86 value, cd = self._canon(value)
87 if self._calc_cd(value) != cd:
88 raise ValidationError(ugettext("Invalid CUIT."))
89 return self._format(value, cd)
91 def _canon(self, cuit):
92 cuit = cuit.replace('-', '')
93 return cuit[:-1], cuit[-1]
95 def _calc_cd(self, cuit):
96 mults = (5, 4, 3, 2, 7, 6, 5, 4, 3, 2)
97 tmp = sum([m * int(cuit[idx]) for idx, m in enumerate(mults)])
98 return str(11 - tmp % 11)
100 def _format(self, cuit, check_digit=None):
101 if check_digit == None:
102 check_digit = cuit[-1]
103 cuit = cuit[:-1]
104 return u'%s-%s-%s' % (cuit[:2], cuit[2:], check_digit)