sync
[roojs1] / Roo / bootstrap / NumberField.js
1 /*
2  * - LGPL
3  *
4  * Input
5  * 
6  */
7
8 /**
9  * @class Roo.bootstrap.NumberField
10  * @extends Roo.bootstrap.Input
11  * Bootstrap NumberField class
12  * 
13  * 
14  * 
15  * 
16  * @constructor
17  * Create a new NumberField
18  * @param {Object} config The config object
19  */
20
21 Roo.bootstrap.NumberField = function(config){
22     Roo.bootstrap.NumberField.superclass.constructor.call(this, config);
23 };
24
25 Roo.extend(Roo.bootstrap.NumberField, Roo.bootstrap.Input, {
26     
27     /**
28      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
29      */
30     allowDecimals : true,
31     /**
32      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
33      */
34     decimalSeparator : ".",
35     /**
36      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
37      */
38     decimalPrecision : 2,
39     /**
40      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
41      */
42     allowNegative : true,
43     /**
44      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
45      */
46     minValue : Number.NEGATIVE_INFINITY,
47     /**
48      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
49      */
50     maxValue : Number.MAX_VALUE,
51     /**
52      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
53      */
54     minText : "The minimum value for this field is {0}",
55     /**
56      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
57      */
58     maxText : "The maximum value for this field is {0}",
59     /**
60      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
61      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
62      */
63     nanText : "{0} is not a valid number",
64     /**
65      * @cfg {Boolean} castInt (true|false) cast int if true (defalut true)
66      */
67     castInt : true,
68     /**
69      * @cfg {String} thousandsDelimiter Symbol of thousandsDelimiter
70      */
71     thousandsDelimiter : false,
72     /**
73      * @cfg {String} valueAlign alignment of value
74      */
75     valueAlign : "left",
76
77     getAutoCreate : function()
78     {
79         var hiddenInput = {
80             tag: 'input',
81             type: 'hidden',
82             id: Roo.id(),
83             cls: 'hidden-number-input'
84         };
85         
86         if (this.name) {
87             hiddenInput.name = this.name;
88         }
89         
90         this.name = '';
91         
92         var cfg = Roo.bootstrap.NumberField.superclass.getAutoCreate.call(this);
93         
94         this.name = hiddenInput.name;
95         
96         if(cfg.cn.length > 0) {
97             cfg.cn.push(hiddenInput);
98         }
99         
100         return cfg;
101     },
102
103     // private
104     initEvents : function()
105     {   
106         Roo.bootstrap.NumberField.superclass.initEvents.call(this);
107         
108         var allowed = "0123456789";
109         
110         if(this.allowDecimals){
111             allowed += this.decimalSeparator;
112         }
113         
114         if(this.allowNegative){
115             allowed += "-";
116         }
117         
118         if(this.thousandsDelimiter) {
119             allowed += ",";
120         }
121         
122         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
123         
124         var keyPress = function(e){
125             
126             var k = e.getKey();
127             
128             var c = e.getCharCode();
129             
130             if(
131                     (String.fromCharCode(c) == '.' || String.fromCharCode(c) == '-') &&
132                     allowed.indexOf(String.fromCharCode(c)) === -1
133             ){
134                 Roo.log('stoping??');
135                 e.stopEvent();
136                 return;
137             }
138             
139             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
140                 return;
141             }
142             
143             if(allowed.indexOf(String.fromCharCode(c)) === -1){
144                 Roo.log('stoping2??');
145                 e.stopEvent();
146             }
147         };
148         
149         this.el.on("keypress", keyPress, this);
150     },
151     
152     validateValue : function(value)
153     {
154         
155         if(!Roo.bootstrap.NumberField.superclass.validateValue.call(this, value)){
156             return false;
157         }
158         
159         var num = this.parseValue(value);
160         
161         if(isNaN(num)){
162             this.markInvalid(String.format(this.nanText, value));
163             return false;
164         }
165         
166         if(num < this.minValue){
167             this.markInvalid(String.format(this.minText, this.minValue));
168             return false;
169         }
170         
171         if(num > this.maxValue){
172             this.markInvalid(String.format(this.maxText, this.maxValue));
173             return false;
174         }
175         
176         return true;
177     },
178
179     getValue : function()
180     {
181         var v = this.hiddenEl().getValue();
182         
183         return this.fixPrecision(this.parseValue(v));
184     },
185
186     parseValue : function(value)
187     {
188         if(this.thousandsDelimiter) {
189             value += "";
190             r = new RegExp(",", "g");
191             value = value.replace(r, "");
192         }
193         
194         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
195         return isNaN(value) ? '' : value;
196     },
197
198     fixPrecision : function(value)
199     {
200         if(this.thousandsDelimiter) {
201             value += "";
202             r = new RegExp(",", "g");
203             value = value.replace(r, "");
204         }
205         
206         var nan = isNaN(value);
207         
208         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
209             return nan ? '' : value;
210         }
211         return parseFloat(value).toFixed(this.decimalPrecision);
212     },
213
214     setValue : function(v)
215     {
216         v = String(this.fixPrecision(v)).replace(".", this.decimalSeparator);
217         
218         this.value = v;
219         
220         if(this.rendered){
221             
222             this.hiddenEl().dom.value = (v === null || v === undefined ? '' : v);
223             
224             this.inputEl().dom.value = Roo.util.Format.number(v, this.decimalPrecision, 
225                 this.thousandsDelimiter || ''
226             );
227             
228             if(this.allowBlank && !v) {
229                 this.inputEl().dom.value = '';
230             }
231             
232             this.validate();
233         }
234     },
235
236     decimalPrecisionFcn : function(v)
237     {
238         return Math.floor(v);
239     },
240
241     beforeBlur : function()
242     {
243         if(!this.castInt){
244             return;
245         }
246         
247         var v = this.parseValue(this.getRawValue());
248         if(v){
249             this.setValue(v);
250         }
251     },
252     
253     hiddenEl : function()
254     {
255         return this.el.select('input.hidden-number-input',true).first();
256     }
257     
258 });
259
260