mirror of
https://github.com/yiisoft/yii.git
synced 2026-03-08 09:06:53 +01:00
252 lines
7.3 KiB
JavaScript
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); |