The component works, but the news properties do not!
AssemblyInfo
[assembly: WebResource("TesteMascara.Scripts.FieldMoney.js", "text/javascript")]
JS
Ext.define('Ext.ux.form.FieldMoney', {
extend : 'Ext.form.field.Text',
alias : 'widget.field-money',
symbol : 'US$ ',
showSymbol : false,
symbolStay : false,
thousands : ',',
decimal : '.',
precision : 2,
defaultZero : true,
allowZero : false,
allowNegative : false,
onRender : function() {
this.callParent(arguments);
var name = this.name || this.inputEl.dom.name;
this.hiddenField = this.inputEl.insertSibling({
tag : 'input',
type : 'hidden',
name : name,
value : this._parseValue(this.value)
});
this.hiddenName = name;
this.inputEl.dom.removeAttribute('name');
this.inputEl.on({
keyup : {
scope : this,
fn : this._onUpdateHidden
},
blur : {
scope : this,
fn : this._onUpdateHidden
}
}, Ext.isIE ? 'after' : 'before');
this.setValue = Ext.Function.createSequence(this.setValue, this._onUpdateHidden, this);
},
_parseValue : function(v) {
var v = this.getValue();
v = v.replace(/\D/g, "");
if(v.length==0){
return 0;
}
if(this.precision > 0) {
return parseFloat(v.substr(0, (v.length - this.precision)) + "." + v.substr((v.length - this.precision)));
} else {
return parseFloat(v);
}
},
_onUpdateHidden : function() {
this.hiddenField.dom.value = this._parseValue(this.getValue());
},
initEvents : function() {
var input = this.inputEl;
input.on('keypress', this._onKeypress, this);
input.on('keydown', this._onKeydown, this);
input.on('blur', this._onBlur, this);
input.on('focus', this._onFocus, this);
this.callParent();
},
_onFocus : function(e) {
var input = this.inputEl;
var mask = this._getDefaultMask();
if(input.getValue() == mask) {
input.getValue('');
} else if(input.getValue() == '' && this.defaultZero) {
input.dom.value = (this._setSymbol(mask));
} else {
input.dom.value = (this._setSymbol(input.getValue()));
}
if(this.createTextRange) {
var textRange = this.createTextRange();
textRange.collapse(false);
// set the cursor at the end of the input
textRange.select();
}
},
_onBlur : function(e) {
if(Ext.isIE) {
this._onKeypress(e);
}
var input = this.inputEl;
if(input.getValue() == '' || input.getValue() == this._setSymbol(this._getDefaultMask()) || input.getValue() == this.symbol) {
if(!this.allowZero)
input.dom.value = '';
else if(!this.symbolStay)
input.dom.value = (this._getDefaultMask());
else
input.dom.value = (this._setSymbol(this._getDefaultMask()));
} else {
if(!this.symbolStay)
input.dom.value = (input.getValue().replace(this.symbol, ''));
else if(this.symbolStay && input.getValue() == this.symbol)
input.dom.value = (this._setSymbol(this._getDefaultMask()));
}
},
_onKeydown : function(e) {
e = e || window.event;
var k = e.getCharCode() || e.getKey() || e.which, input = this.inputEl;
if(k == undefined)
return false;
//needed to handle an IE "special" event
if(input.getAttribute('readonly') && (k != 13 && k != 9))
return false;
// don't allow editing of readonly fields but allow tab/enter
var x = input.dom;
var selection = this._getInputSelection(x);
var startPos = selection.start;
var endPos = selection.end;
if(k == 8) {// backspace key
e.preventDefault();
if(startPos == endPos) {
// Remove single character
x.value = x.value.substring(0, startPos - 1) + x.value.substring(endPos, x.value.length);
startPos = startPos - 1;
} else {
// Remove multiple characters
x.value = x.value.substring(0, startPos) + x.value.substring(endPos, x.value.length);
}
this._maskAndPosition(x, startPos);
return false;
} else if(k == 9) {// tab key
return true;
} else if(k == 46 || k == 63272) {// delete key (with special case for safari)
preventDefault(e);
if(x.selectionStart == x.selectionEnd) {
// Remove single character
x.value = x.value.substring(0, startPos) + x.value.substring(endPos + 1, x.value.length);
} else {
//Remove multiple characters
x.value = x.value.substring(0, startPos) + x.value.substring(endPos, x.value.length);
}
this._maskAndPosition(x, startPos);
return false;
} else {// any other key
return true;
}
},
_onKeypress : function(e) {
e = e || window.event;
var k = e.getCharCode() || e.getKey() || e.which, input = this.inputEl;
if(k == undefined)
return false;
//needed to handle an IE "special" event
if(input.getAttribute('readonly') && (k != 13 && k != 9))
return false;
// don't allow editing of readonly fields but allow tab/enter
if(k < 48 || k > 57) {// any key except the numbers 0-9
if(k == 45) {// -(minus) key
this.setValue(this._changeSign(input));
return false;
}
if(k == 43) {// +(plus) key
input.val(input.getValue().replace('-', ''));
return false;
} else if(k == 13 || k == 9) {// enter key or tab key
return true;
} else if(k == 37 || k == 39) {// left arrow key or right arrow key
return true;
} else {// any other key with keycode less than 48 and greater than 57
e.preventDefault();
return true;
}
} else if(input.getValue().length == input.getAttribute('maxlength')) {
return false;
} else {
e.preventDefault();
var key = String.fromCharCode(k);
var x = input.dom;
var selection = this._getInputSelection(x);
var startPos = selection.start;
var endPos = selection.end;
x.value = x.value.substring(0, startPos) + key + x.value.substring(endPos, x.value.length);
this._maskAndPosition(x, startPos + 1);
return false;
}
},
_getDefaultMask : function() {
var n = parseFloat('0') / Math.pow(10, this.precision);
return (n.toFixed(this.precision)).replace(new RegExp('\\.', 'g'), this.decimal);
},
_maskValue : function(v) {
var input = this.inputEl;
v = v.replace(this.symbol, '');
var strCheck = '0123456789';
var len = v.length;
var a = '', t = '', neg = '';
if(len != 0 && v.charAt(0) == '-') {
v = v.replace('-', '');
if(this.allowNegative) {
neg = '-';
}
}
if(len == 0) {
if(!this.defaultZero)
return t;
t = '0.00';
}
for(var i = 0; i < len; i++) {
if((v.charAt(i) != '0') && (v.charAt(i) != this.decimal))
break;
}
for(; i < len; i++) {
if(strCheck.indexOf(v.charAt(i)) != -1)
a += v.charAt(i);
}
var n = parseFloat(a);
n = isNaN(n) ? 0 : n / Math.pow(10, this.precision);
t = n.toFixed(this.precision);
i = this.precision == 0 ? 0 : 1;
var p, d = (t=t.split('.'))[i].substr(0, this.precision);
for( p = ( t = t[0]).length; (p -= 3) >= 1; ) {
t = t.substr(0, p) + this.thousands + t.substr(p);
}
return (this.precision > 0) ? this._setSymbol(neg + t + this.decimal + d + Array((this.precision + 1) - d.length).join(0)) : this._setSymbol(neg + t);
},
_setSymbol : function(v) {
if(this.showSymbol) {
if(v.substr(0, this.symbol.length) != this.symbol)
return this.symbol + v;
}
return v;
},
_maskAndPosition : function(x, startPos) {
var input = this.inputEl, originalLen = input.getValue().length;
input.dom.value = this._maskValue(x.value);
var newLen = input.getValue().length;
startPos = startPos - (originalLen - newLen);
this._setCursorPosition(startPos);
},
_changeSign : function(i) {
if(this.allowNegative) {
var vic = i.getValue();
if(i.getValue() != '' && i.getValue().charAt(0) == '-') {
return i.getValue().replace('-', '');
} else {
return '-' + i.getValue();
}
} else {
return i.getValue();
}
},
_getInputSelection : function(el) {
var start = 0, end = 0, normalizedValue, range, textInputRange, len, endRange;
if( typeof el.selectionStart == "number" && typeof el.selectionEnd == "number") {
start = el.selectionStart;
end = el.selectionEnd;
} else {
range = document.selection.createRange();
if(range && range.parentElement() == el) {
len = el.value.length;
normalizedValue = el.value.replace(/\r\n/g, "\n");
// Create a working TextRange that lives only in the input
textInputRange = el.createTextRange();
textInputRange.moveToBookmark(range.getBookmark());
// Check if the start and end of the selection are at the very end
// of the input, since moveStart/moveEnd doesn't return what we want
// in those cases
endRange = el.createTextRange();
endRange.collapse(false);
if(textInputRange.compareEndPoints("StartToEnd", endRange) > -1) {
start = end = len;
} else {
start = -textInputRange.moveStart("character", -len);
start += normalizedValue.slice(0, start).split("\n").length - 1;
if(textInputRange.compareEndPoints("EndToEnd", endRange) > -1) {
end = len;
} else {
end = -textInputRange.moveEnd("character", -len);
end += normalizedValue.slice(0, end).split("\n").length - 1;
}
}
}
}
return {
start : start,
end : end
};
},
_setCursorPosition : function(pos) {
var elem = this.inputEl;
if(elem.setSelectionRange) {
elem.focus();
elem.setSelectionRange(pos, pos);
} else if(elem.createTextRange) {
var range = elem.createTextRange();
range.collapse(true);
range.moveEnd('character', pos);
range.moveStart('character', pos);
range.select();
}
}
});
Ext.ComponentMgr.registerType("field-money", Ext.ux.form.FieldMoney);
My control:
using System.Collections.Generic;
using System.ComponentModel;
using System.Xml.Serialization;
using Ext.Net;
using Newtonsoft.Json;
namespace TesteMascara
{
public class TextFieldMoney : TextField
{
public override string XType
{
get
{
return "field-money";
}
}
public override string InstanceOf
{
get
{
return "Ext.ux.form.FieldMoney";
}
}
protected override List<ResourceItem> Resources
{
get
{
List<ResourceItem> resources = base.Resources;
resources.Capacity += 1;
resources.Add(new ClientScriptItem(typeof(TextFieldMoney), "TesteMascara.Scripts.FieldMoney.js", ""));
return resources;
}
}
[Browsable(false)]
[EditorBrowsable(EditorBrowsableState.Never)]
[DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
[XmlIgnore]
[JsonIgnore]
public override ConfigOptionsCollection ConfigOptions
{
get
{
ConfigOptionsCollection list = base.ConfigOptions;
list.Add("precision", new ConfigOption("precision", new SerializationOptions("precision", JsonMode.Object), null, this.Precision));
list.Add("thousands", new ConfigOption("thousands", new SerializationOptions("thousands", JsonMode.Object), null, this.Thousands));
list.Add("decimal", new ConfigOption("decimal", new SerializationOptions("decimal", JsonMode.Object), null, this.Decimal));
return list;
}
}
[Meta]
[ConfigOption]
[Category("5. Field")]
[DefaultValue(2)]
[Description("")]
public virtual int Precision
{
get
{
return this.State.Get<int>("precision", 2);
}
set
{
this.State.Set("precision", value);
}
}
[Meta]
[ConfigOption]
[Category("5. Field")]
[DefaultValue('.')]
[Description("")]
public virtual char Thousands
{
get
{
return this.State.Get<char>("thousands", '.');
}
set
{
this.State.Set("thousands", value);
}
}
[Meta]
[ConfigOption]
[Category("5. Field")]
[DefaultValue(',')]
[Description("")]
public virtual char Decimal
{
get
{
return this.State.Get<char>("decimal", ',');
}
set
{
this.State.Set("decimal", value);
}
}
}
}
My test:
<%@ Page Title="" Language="C#" MasterPageFile="~/Site.Master" AutoEventWireup="true" CodeBehind="TesteMascara.aspx.cs" Inherits="TesteMascara.TesteMascara" %>
<%@ Register Assembly="TesteMascara" Namespace="TesteMascara" TagPrefix="cc" %>
<asp:Content ID="Content1" ContentPlaceHolderID="head" runat="server">
</asp:Content>
<asp:Content ID="Content2" ContentPlaceHolderID="ContentPlaceHolder1" runat="server">
<cc:TextFieldMoney ID="TextFieldMoney" runat="server" FieldLabel="Mascara" Decimal="," Thousands="." />
</asp:Content>
Thanks!