//+ Jonas Raoni Soares Silva
//@ http://jsfromhell.com/forms/masked-input [v1.0]

/*
MaskInput(field: HTMLInputElement, mask: String): void
Adiciona máscara a um campo.
field 	campo que vai receber a máscara
mask	máscara que será aplicada

Regras Padrões
* 	a = A-Z e 0-9
* 	A = A-Z, acentos e 0-9
* 	9 = 0-9
* 	C = A-Z e acentos
* 	c = A-Z
* 	* = qualquer coisa

Regras Especiais
* 	E = (Except) exceção
* 	O = (Only) somente

Criação de Máscaras

Máscara simples:
nesse tipo de máscara o usuário pode digitar no máximo a mesma
quantidade de caracteres que a máscara contém.
	
Exemplo:	
Telefone = (99)9999-9999
Data = 99/99/9999

Máscara especial "regra^exceções":
esse tipo de máscara é composto por 2 partes, separadas por "^",
o lado esquerdo especifica a regra e o direito as exceções para a regra selecionada.
	
Exemplo:
9^abc = a regra é aceitar somente números "9" e a exceção são os caracteres a, b e c
c^123 = aceita somente caracteres de a-z e a exceção são os números 1, 2 e 3

Uso das regras especiais:
ela é semelhante a máscara especial, porém o lado esquerdo tem um significado diferente,
podendo ser "E" (qualquer coisa, exceto...) ou "O" (somente...)

Exemplo:
E^abc: aceita qualquer coisa, menos a, b e c
O^123: só permite os caracteres 1, 2 e 3
*/

MaskInput = function(f, m){
    function mask(e){
        var patterns = {"1": /[A-Z]/i, "2": /[0-9]/, "4": /[\xC0-\xFF]/i, "8": /./ },
            rules = { "a": 3, "A": 7, "9": 2, "C":5, "c": 1, "*": 8};
        function accept(c, rule){
            for(var i = 1, r = rules[rule] || 0; i <= r; i<<=1)
                if(r & i && patterns[i].test(c))
                    break;
                return i <= r || c == rule;
        }
        var k, mC, r, c = String.fromCharCode(k = e.key), l = f.value.length;
        (!k || k == 8 ? 1 : (r = /^(.)\^(.*)$/.exec(m)) && (r[0] = r[2].indexOf(c) + 1) + 1 ?
            r[1] == "O" ? r[0] : r[1] == "E" ? !r[0] : accept(c, r[1]) || r[0]
            : (l = (f.value += m.substr(l, (r = /[A|9|C|\*]/i.exec(m.substr(l))) ?
            r.index : l)).length) < m.length && accept(c, m.charAt(l))) || e.preventDefault();
    }
    for(var i in !/^(.)\^(.*)$/.test(m) && (f.maxLength = m.length), {keypress: 0, keyup: 1})
        addEvent(f, i, mask);
};

//+ Carlos R. L. Rodrigues
//@ http://www.jsfromhell.com/geral/event-listener [v1.4]

addEvent = function(o, e, f, s){
	var r = o[r = "_" + (e = "on" + e)] = o[r] || (o[e] ? [[o[e], o]] : []), a, c, d;
	r[r.length] = [f, s || o], o[e] = function(e){
		try{
			(e = e || event).preventDefault || (e.preventDefault = function(){e.returnValue = false;});
			e.stopPropagation || (e.stopPropagation = function(){e.cancelBubble = true;});
			e.target || (e.target = e.srcElement || null);
			e.key = (e.which + 1 || e.keyCode + 1) - 1 || 0;
		}catch(f){}
		for(d = 1, f = r.length; f; r[--f] && (a = r[f][0], o = r[f][1], a.call ? c = a.call(o, e) : (o._ = a, c = o._(e), o._ = null), d &= c !== false));
		return e = null, !!d;
    }
};

removeEvent = function(o, e, f, s){
	for(var i = (e = o["_on" + e] || []).length; i;)
		if(e[--i] && e[i][0] == f && (s || o) == e[i][1])
			return delete e[i];
	return false;
};