Files
yii/framework/web/js/source/jquery.maskedinput.js
qiang.xue 662798d8a5
2008-09-28 12:03:53 +00:00

252 lines
7.3 KiB
JavaScript

/*
* Copyright (c) 2007-2008 Josh Bush (digitalbush.com)
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following
* conditions:
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/
/*
* Version: 1.1.3
* Release: 2008-04-16
*/
(function($) {
//Helper Function for Caret positioning
$.fn.caret=function(begin,end){
if(this.length==0) return;
if (typeof begin == 'number') {
end = (typeof end == 'number')?end:begin;
return this.each(function(){
if(this.setSelectionRange){
this.focus();
this.setSelectionRange(begin,end);
}else if (this.createTextRange){
var range = this.createTextRange();
range.collapse(true);
range.moveEnd('character', end);
range.moveStart('character', begin);
range.select();
}
});
} else {
if (this[0].setSelectionRange){
begin = this[0].selectionStart;
end = this[0].selectionEnd;
}else if (document.selection && document.selection.createRange){
var range = document.selection.createRange();
begin = 0 - range.duplicate().moveStart('character', -100000);
end = begin + range.text.length;
}
return {begin:begin,end:end};
}
};
//Predefined character definitions
var charMap={
'9':"[0-9]",
'a':"[A-Za-z]",
'*':"[A-Za-z0-9]"
};
//Helper method to inject character definitions
$.mask={
addPlaceholder : function(c,r){
charMap[c]=r;
}
};
$.fn.unmask=function(){
return this.trigger("unmask");
};
//Main Method
$.fn.mask = function(mask,settings) {
settings = $.extend({
placeholder: "_",
completed: null
}, settings);
//Build Regex for format validation
var re = new RegExp("^"+
$.map( mask.split(""), function(c,i){
return charMap[c]||((/[A-Za-z0-9]/.test(c)?"":"\\")+c);
}).join('')+
"$");
return this.each(function(){
var input=$(this);
var buffer=new Array(mask.length);
var locked=new Array(mask.length);
var valid=false;
var ignore=false; //Variable for ignoring control keys
var firstNonMaskPos=null;
//Build buffer layout from mask & determine the first non masked character
$.each( mask.split(""), function(i,c){
locked[i]=(charMap[c]==null);
buffer[i]=locked[i]?c:settings.placeholder;
if(!locked[i] && firstNonMaskPos==null)
firstNonMaskPos=i;
});
function focusEvent(){
checkVal();
writeBuffer();
setTimeout(function(){
$(input[0]).caret(valid?mask.length:firstNonMaskPos);
},0);
};
function keydownEvent(e){
var pos=$(this).caret();
var k = e.keyCode;
ignore=(k < 16 || (k > 16 && k < 32 ) || (k > 32 && k < 41));
//delete selection before proceeding
if((pos.begin-pos.end)!=0 && (!ignore || k==8 || k==46)){
clearBuffer(pos.begin,pos.end);
}
//backspace and delete get special treatment
if(k==8){//backspace
while(pos.begin-->=0){
if(!locked[pos.begin]){
buffer[pos.begin]=settings.placeholder;
if($.browser.opera){
//Opera won't let you cancel the backspace, so we'll let it backspace over a dummy character.
s=writeBuffer();
input.val(s.substring(0,pos.begin)+" "+s.substring(pos.begin));
$(this).caret(pos.begin+1);
}else{
writeBuffer();
$(this).caret(Math.max(firstNonMaskPos,pos.begin));
}
return false;
}
}
}else if(k==46){//delete
clearBuffer(pos.begin,pos.begin+1);
writeBuffer();
$(this).caret(Math.max(firstNonMaskPos,pos.begin));
return false;
}else if (k==27){//escape
clearBuffer(0,mask.length);
writeBuffer();
$(this).caret(firstNonMaskPos);
return false;
}
};
function keypressEvent(e){
if(ignore){
ignore=false;
//Fixes Mac FF bug on backspace
return (e.keyCode == 8)? false: null;
}
e=e||window.event;
var k=e.charCode||e.keyCode||e.which;
var pos=$(this).caret();
if(e.ctrlKey || e.altKey){//Ignore
return true;
}else if ((k>=41 && k<=122) ||k==32 || k>186){//typeable characters
var p=seekNext(pos.begin-1);
if(p<mask.length){
if(new RegExp(charMap[mask.charAt(p)]).test(String.fromCharCode(k))){
buffer[p]=String.fromCharCode(k);
writeBuffer();
var next=seekNext(p);
$(this).caret(next);
if(settings.completed && next == mask.length)
settings.completed.call(input);
}
}
}
return false;
};
function clearBuffer(start,end){
for(var i=start;i<end&&i<mask.length;i++){
if(!locked[i])
buffer[i]=settings.placeholder;
}
};
function writeBuffer(){
return input.val(buffer.join('')).val();
};
function checkVal(){
//try to place charcters where they belong
var test=input.val();
var pos=0;
for(var i=0;i<mask.length;i++){
if(!locked[i]){
buffer[i]=settings.placeholder;
while(pos++<test.length){
//Regex Test each char here.
var reChar=new RegExp(charMap[mask.charAt(i)]);
if(test.charAt(pos-1).match(reChar)){
buffer[i]=test.charAt(pos-1);
break;
}
}
}
}
var s=writeBuffer();
if(!s.match(re)){
input.val("");
clearBuffer(0,mask.length);
valid=false;
}else
valid=true;
};
function seekNext(pos){
while(++pos<mask.length){
if(!locked[pos])
return pos;
}
return mask.length;
};
input.one("unmask",function(){
input.unbind("focus",focusEvent);
input.unbind("blur",checkVal);
input.unbind("keydown",keydownEvent);
input.unbind("keypress",keypressEvent);
if ($.browser.msie)
this.onpaste= null;
else if ($.browser.mozilla)
this.removeEventListener('input',checkVal,false);
});
input.bind("focus",focusEvent);
input.bind("blur",checkVal);
input.bind("keydown",keydownEvent);
input.bind("keypress",keypressEvent);
//Paste events for IE and Mozilla thanks to Kristinn Sigmundsson
if ($.browser.msie)
this.onpaste= function(){setTimeout(checkVal,0);};
else if ($.browser.mozilla)
this.addEventListener('input',checkVal,false);
checkVal();//Perform initial check for existing values
});
};
})(jQuery);