/* *
* javascript日历控件
* @author tugenhua
*/
function Calendar(options,callback){
var self = this ;
self.options = $.extend({},defaults,options || {});
self.targetCls = $(self.options.targetCls);
if (self.targetCls.length < 1) {return ;}
self.language = self.options.language;
// 显示日历面板
self.flag = false ;
self.callback = callback;
self._init();
self._bindEnv();
};
$.extend(Calendar.prototype,{
_init: function (){
var self = this ;
// 先渲染日历面板
self._renderCalendarPanel();
},
_renderCalendarPanel: function (){
var self = this ,
options = self.options;
// 如果input输入框有日期的话
if (self.targetCls.val().length > 0) {
self.date = self._dateParse(self.targetCls.val());
}
self.date = new Date(self.date);
if (isNaN(self.date.getFullYear())){
self.date = new Date();
}
var defYear = self.date.getFullYear(),
defMonth = self.date.getMonth() + 1;
// 定义每月的天数
self.month_day = new Array(31,28+self._leapYear(defYear),31,30,31,30,31,31,30,31,30,31);
// 定义每周的日期
self.date_name_week = self.language.weekList;
// 定义周末
var saturday = 6 - options.wdays,
sunday = (7-options.wdays >= 7) ? 0 : (7-options.wday);
// 创建日历面板dom节点
var date_pane = $('
'),
date_hd = $('
').appendTo(date_pane),
date_table = $(''
).appendTo(date_pane);
// type 为 yyyy-mm-dd 时候 就不显示时间
if (options.type == 'yyyy-mm-dd HH:MM' || options.type == 'yyyy-mm-dd HH:MM:SS'
) {
var date_time = $('
'
)
.appendTo(date_pane);
}
var date_button = $('
'
).appendTo(date_pane);
date_hd.html( "
< > "
);
var date_txt = $('
'
).appendTo(date_hd),
date_set = $('
'
).appendTo(date_hd),
html = ""
;
for (
var i = options.beginyear; i < options.endyear; i++
) {
html +="
"+i+" "
;
}
var year_list = $('
'
).html(html).appendTo(date_set).val(defYear);
date_set.append( " - "
);
html = ''
;
for (
var i = 0; i < 12; i++
) {
html += '
'+self.language.monthList[i]+' '
;
}
var month_list = $('
'
).html(html).appendTo(date_set).val(defMonth);
html = '
';
// 遍历一周7天
for (var i = 0; i < 7; i++) {
html += "";
html += (i+options.wday * 1 < 7) ? self.date_name_week[i+options.wday] : self.date_name_week[i+options.wday - 7];
html +=" ";
};
html +=" "
;
html +="
"
;
date_table.html(html);
// 面板及背景遮挡层插入到页面中
date_pane.appendTo("body"
);
// 创建遮罩层的目地是:只显示一个日历面板
var block_bg=$("
").appendTo("body"
);
// 赋值 全局
self.dateTxt =
date_txt;
self.yearList =
year_list;
self.monthList =
month_list;
self.dateTable =
date_table;
self.saturday =
saturday;
self.sunday =
sunday;
self.datePane =
date_pane;
self.blockBg =
block_bg;
self.dateSet =
date_set;
self.defYear =
defYear;
self.defMonth =
defMonth;
self.dateTime =
date_time;
// 根据当前的时间 渲染时分秒
self._renderTime(self.date);
// 根据年份 月份来渲染天
self._renderBody(defYear,defMonth);
},
// 返回 年月分的 天数
_dayNumOfMonth:
function (Year,Month){
var d =
new Date(Year,Month,0
);
return d.getDate();
},
/*
* 根据当前的时间,渲染时分秒
*/
_renderTime: function (date){
var self =
this ,
options =
self.options;
var hours =
new Date(date).getHours(),
minutes =
new Date(date).getMinutes(),
seconds =
new Date(date).getSeconds(),
curTime;
if (hours < 10
) {
hours = '0' +
hours;
}
if (minutes < 10
) {
minutes = '0' +
minutes;
}
if (seconds < 10
) {
seconds = '0' +
seconds;
}
if (options.type == 'yyyy-mm-dd HH:MM:SS'
) {
curTime = hours + ":" + minutes + ":" +
seconds;
} else if (options.type == 'yyyy-mm-dd HH:MM'
) {
curTime = hours + ":" +
minutes;
}
$(self.dateTime).find( '.spinner-text'
).val(curTime);
},
/*
* 渲染日历天数
* @param {y,m} 年 月
*/
_renderBody: function (y,m){
var self =
this ;
var options =
self.options;
if (m < 1
) {
y --
;
m = 12
;
} else if (m > 12
) {
y ++
;
m = 1
;
}
var tempM =
m,
cur_m =
m;
m --;
// 月份从0开始的
var prevMonth = tempM - 1,
// 上个月的月份
prevDay = self._dayNumOfMonth(y,tempM - 1),
// 上个月的天数
nextMonth = tempM + 1,
// 下个月的月份
nextDay = self._dayNumOfMonth(y,tempM + 1),
// 下个月的天数
curDay = self._dayNumOfMonth(y,tempM);
// 当前月份的天数
self.month_day[ 1]=28+self._leapYear(y);
// 闰年的话 29天 否则 28天
var temp_html = ""
,
temp_date =
new Date(y,m,1
);
var now_date =
new Date();
now_date.setHours( 0
);
now_date.setMinutes( 0
);
now_date.setSeconds( 0
);
// 如果输入框有值的话
if (self.targetCls.val().length > 0
) {
var val_date=
self._dateParse(self.targetCls.val())
}
val_date =
new Date(val_date);
if (isNaN(val_date.getFullYear())){
val_date =
null ;
};
// 获取当月的第一天
var firstDay = temp_date.getDay() - options.wday < 0 ?
temp_date.getDay() - options.wday + 7
:
temp_date.getDay() -
options.wday;
// 每月所需要的行数
var monthRows = Math.ceil((firstDay+self.month_day[m]) / 7
);
var td_num,
day_num,
diff_now,
diff_set;
var disabled;
for (
var i= 0; i < monthRows; i++
) {
temp_html += "
";
for (var j = 0; j < 7; j++) {
td_num =i*7+j;
day_num =td_num-firstDay+1;
if (day_num<=0) {
if (day_num == 0) {
day_num = prevDay - day_num
text_m = prevMonth
} else {
day_num = prevDay + day_num;
text_m = prevMonth
}
} else if (day_num > self.month_day[m]){
day_num = day_num - curDay;
text_m = nextMonth
} else {
text_m = cur_m;
}
temp_html +=";
if (typeof (day_num) == 'number') {
diff_now =null ;
diff_set =null ;
temp_date = new Date(y,m,day_num);
if (text_m == cur_m) {
diff_now =Date.parse(now_date)-Date.parse(temp_date);
diff_set =Date.parse(val_date)-Date.parse(temp_date);
}
if (cur_m > text_m || cur_m < text_m) {
disabled = 'disabled';
} else {
disabled = "";
}
temp_html +=(" title='"+y+options.separator+tempM+options.separator+day_num+"' class='num "+disabled+"");
// 高亮周末、今天、选中
if (diff_set==0){ // 选中的时候 增加select 类名
temp_html+=" selected";
} else if (diff_now==0){
temp_html +=" now"; // 当前时间增加now类名
}else if (j==self.saturday){
temp_html +=" sat"; // 周六增加sat类名
}else if (j==self.sunday){
temp_html +=" sun"; // 周日增加sun类名
};
temp_html +=("'");
};
temp_html +=(" date-day='"+day_num+"'>"+day_num+" ");
}
temp_html +=" "
;
}
$(self.dateTable).find( "tbody"
).html(temp_html);
$(self.dateTxt).html( "
"+y+" "+options.language.year+"
"+options.language.monthList[m]+" "+
options.language.month);
$(self.yearList).val(y);
$(self.monthList).val(m +1
);
return this ;
},
_dateParse: function (date){
var newdate =
date;
newdate =newdate.replace(/\./g,"/"
);
newdate =newdate.replace(/-/g,"/"
);
newdate =newdate.replace(/\//g,"/"
);
newdate =
Date.parse(newdate);
return newdate;
},
/*
* 判断是否是闰年
* @param y 年份
* 1.能被4整除且不能被100整除 2.能被100整除且能被400整除
*/
_leapYear: function (y) {
return ((y%4==0 && y%100!=0) || y%400==0) ? 1 : 0
;
},
_bindEnv: function (){
var self =
this ;
var options =
self.options;
$(self.targetCls).unbind( 'click').bind('click',
function (){
self.show();
// 渲染时分秒
self._renderTime(
new Date());
});
// 关闭面板事件
self.blockBg.unbind('click').bind("click",
function (){
self.hide();
});
// 点击上一页 下一页事件
self.datePane.delegate('a','click',
function (){
if (!
this .rel){
return };
var _rel =
this .rel;
if (_rel == 'prev'
) {
self._renderBody(self.yearList.val(),parseInt(self.monthList.val(), 10) -1
);
return ;
} else if (_rel == 'next'
) {
self._renderBody(self.yearList.val(),parseInt(self.monthList.val(), 10) +1
);
return ;
}
});
// 选择日期事件
self.datePane.delegate('td','click',
function (){
var _this = $(
this );
if (_this.hasClass('num') && !_this.hasClass('disabled'
)) {
self.dateTable.find( "td").removeClass("selected"
);
_this.addClass( "selected"
);
var day = _this.attr("date-day"
);
if (options.type == 'yyyy-mm-dd'
) {
self._selectDay(day);
}
}
});
// 显示年月选择
self.dateTxt.unbind('click').bind("click",
function (){
self.dateTxt.hide();
self.dateSet.show();
});
// 更改年月事件
self.yearList.unbind('change').bind("change",
function (){
self._renderBody(self.yearList.val(),self.monthList.val());
});
self.monthList.unbind( 'change').bind("change",
function (){
self._renderBody(self.yearList.val(),self.monthList.val());
});
// 点击确定按钮
self.datePane.delegate('.date-ok','click',
function (){
var nums = $(self.dateTable).find('.num'
),
flag =
false ;
for (
var i = 0, ilen = nums.length; i < ilen; i++
) {
if ($(nums[i]).hasClass("selected"
)) {
var day = $(nums[i]).attr("date-day"
);
flag =
true ;
break ;
} else {
if ($(nums[i]).hasClass("now"
)) {
var day = $(nums[i]).attr("date-day"
);
flag =
true ;
continue ;
}
}
}
if (flag) {
self._selectDay(day);
}
});
// 点击关闭按钮
self.datePane.delegate('.date-closed','click',
function (){
self.hide();
});
// 点击今天按钮
self.datePane.delegate('.date-current','click',
function (){
// 获取当前第几号
var date =
new Date(),
curDay =
date.getDate(),
curYear =
date.getFullYear(),
curMonth = date.getMonth() + 1
;
self._renderBody(curYear,curMonth);
self.dateTable.find( "td.num").removeClass("selected"
);
self.dateTable.find( "td.num").each(
function (){
var day = $(
this ).attr("date-day"
);
if (day ==
curDay) {
$( this ).addClass("selected"
);
}
});
});
// 点击箭头向上按钮
self.datePane.delegate('.spinner-arrow-up','click',
function (){
self._inputValChange( true );
});
// 点击箭头向下按钮
self.datePane.delegate('.spinner-arrow-down','click',
function (){
self._inputValChange( false );
});
},
/*
* 选择某一天的时候 把值存入输入框里 且面板隐藏
* @_selectDay {private}
*/
_selectDay: function (d) {
var self =
this ;
var year,
month;
month =
self.monthList.val();
day =
d;
var options =
self.options;
month ="0" +
self.monthList.val();
day = "0" +
d;
month =month.substr((month.length-2
),month.length);
day =day.substr((day.length-2
),day.length);
if (options.type == 'yyyy-mm-dd'
) {
self.targetCls.val(self.yearList.val() +options.separator+month+options.separator+
day);
self.hide();
self.callback && $.isFunction(self.callback) && self.callback(self.yearList.val()+options.separator+month+options.separator+
day);
} else if (options.type == 'yyyy-mm-dd HH:MM:SS' || options.type == 'yyyy-mm-dd HH:MM'
){
self.targetCls.val(self.yearList.val() +options.separator+month+options.separator+day+ " " +$(self.dateTime).find('.spinner-text'
).val());
self.hide();
self.callback && $.isFunction(self.callback) && self.callback(self.yearList.val()+options.separator+month+options.separator+day+ " " +$(self.dateTime).find('.spinner-text'
).val());
}
return this ;
},
/*
* 点击向上或者向下 时间输入框值改变
* @param {flag} boolean 布尔型。如果为true 则是往上加 否则是往下减
*/
_inputValChange: function (flag){
var self =
this ,
options =
self.options;
var inputVal = $(self.datePane).find('.spinner-text'
).val();
var arrs = inputVal.split(":"
),
hour = arrs[0] ? arrs[0] : ''
,
minute = arrs[1] ? arrs[1] : ''
,
second = arrs[2] ? arrs[2] : ''
;
if (flag) {
hour ++
;
if (hour >= 24
) {
return ;
} else if (hour < 10
) {
hour = "0" +
hour;
}
} else {
hour --
;
if (hour < 0
) {
return ;
} else if (hour < 10
) {
hour = '0' +
hour;
}
}
var val;
if (options.type == "yyyy-mm-dd HH:MM:SS"
) {
val = hour + ":" + minute + ":" +
second;
} else if (options.type == 'yyyy-mm-dd HH:MM'
) {
val = hour + ":" +
minute;
}
$( '.spinner-text'
,self.datePane).val(val);
},
/*
* 显示日历面板
* @method show {public}
*/
show: function (){
var self =
this ;
if (self.flag) {
return ;
}
var doc_w =
document.body.clientWidth,
doc_h =
document.body.clientHeight,
pane_top =
self.targetCls.offset().top,
pane_left =
self.targetCls.offset().left,
obj_w =
self.targetCls.outerWidth(),
obj_h =
self.targetCls.outerHeight();
pane_top = pane_top+
obj_h;
self.datePane.css({ "top":pane_top,"left"
:pane_left}).show();
self.blockBg.css({width:doc_w,height:doc_h}).show();
self.flag =
true ;
return this ;
},
/*
* 清除日期
* @method clear {public}
*/
clear: function (){
var self =
this ;
self.targetCls.val( ''
);
self._renderBody(self.defYear,self.defMonth);
self.hide();
return this ;
},
/*
* 获取当前选中的日期
* @method getValue {public}
* @return val
*/
getValue: function (){
var self =
this ;
return self.targetCls.val();
},
/*
* 隐藏日历面板
*/
hide: function (){
var self =
this ;
if (!self.flag) {
return ;}
self.datePane.hide();
self.blockBg.hide();
self.dateSet.hide();
self.dateTxt.show();
self.flag =
false ;
return this ;
}
});
var defaults =
{
targetCls : '',
// 渲染日历的class
beginyear : 1949,
// 开始年份
endyear : 2050,
// 结束年份
date :
new Date(),
// 默认日期
type : "yyyy-mm-dd",
// 日期格式
separator : "-",
// 日期链接符
wday : 0,
// 周第一天
language : {
year: "年"
,
month: "月"
,
monthList:[ "1","2","3","4","5","6","7","8","9","10","11","12"
],
weekList:[ "日","一","二","三","四","五","六"
]}
};