JavaScript blog式日历控件

http://www.cnblogs.com/cloudgamer/archive/2008/08/23/1274459.html


近来要做一个记事本系统,想找一个合适的日历控件,但网上的都是那种日历选择控件。
于是到qq的记事本系统找了一个,但里面的算法有点落后,所以用了它的样式自己写了个。

效果:

<script type="text/javascript"> var $ = function (id) { return "string" == typeof id ? document.getElementById(id) : id; }; var Class = { create: function() { return function() { this.initialize.apply(this, arguments); } } } Object.extend = function(destination, source) { for (var property in source) { destination[property] = source[property]; } return destination; } var Calendar = Class.create(); Calendar.prototype = { initialize: function(container, options) { this.Container = $(container);//容器(table结构) this.Days = [];//日期对象列表 this.SetOptions(options); this.Year = this.options.Year; this.Month = this.options.Month; this.SelectDay = this.options.SelectDay ? new Date(this.options.SelectDay) : null; this.onSelectDay = this.options.onSelectDay; this.onToday = this.options.onToday; this.onFinish = this.options.onFinish; this.Draw(); }, //设置默认属性 SetOptions: function(options) { this.options = {//默认值 Year: new Date().getFullYear(),//显示年 Month: new Date().getMonth() + 1,//显示月 SelectDay: null,//选择日期 onSelectDay: function(){},//在选择日期触发 onToday: function(){},//在当天日期触发 onFinish: function(){}//日历画完后触发 }; Object.extend(this.options, options || {}); }, //上一个月 PreMonth: function() { //先取得上一个月的日期对象 var d = new Date(this.Year, this.Month - 2, 1); //再设置属性 this.Year = d.getFullYear(); this.Month = d.getMonth() + 1; //重新画日历 this.Draw(); }, //下一个月 NextMonth: function() { var d = new Date(this.Year, this.Month, 1); this.Year = d.getFullYear(); this.Month = d.getMonth() + 1; this.Draw(); }, //画日历 Draw: function() { //用来保存日期列表 var arr = []; //用当月第一天在一周中的日期值作为当月离第一天的天数 for(var i = 1, firstDay = new Date(this.Year, this.Month - 1, 1).getDay(); i <= firstDay; i++){ arr.push(" "); } //用当月最后一天在一个月中的日期值作为当月的天数 for(var i = 1, monthDay = new Date(this.Year, this.Month, 0).getDate(); i <= monthDay; i++){ arr.push(i); } var frag = document.createDocumentFragment(); this.Days = []; while(arr.length > 0){ //每个星期插入一个tr var row = document.createElement("tr"); //每个星期有7天 for(var i = 1; i <= 7; i++){ var cell = document.createElement("td"); cell.innerHTML = " "; if(arr.length > 0){ var d = arr.shift(); cell.innerHTML = d; if(d > 0){ this.Days[d] = cell; //判断是否今日 if(this.IsSame(new Date(this.Year, this.Month - 1, d), new Date())){ this.onToday(cell); } //判断是否选择日期 if(this.SelectDay && this.IsSame(new Date(this.Year, this.Month - 1, d), this.SelectDay)){ this.onSelectDay(cell); } } } row.appendChild(cell); } frag.appendChild(row); } //先清空内容再插入(ie的table不能用innerHTML) while(this.Container.hasChildNodes()){ this.Container.removeChild(this.Container.firstChild); } this.Container.appendChild(frag); this.onFinish(); }, //判断是否同一日 IsSame: function(d1, d2) { return (d1.getFullYear() == d2.getFullYear() && d1.getMonth() == d2.getMonth() && d1.getDate() == d2.getDate()); } }; </script>
<<
>>
20089
 123456
78910111213
14151617181920
21222324252627
282930    
<script language="JavaScript"> var cale = new Calendar("idCalendar", { SelectDay: new Date().setDate(10), onSelectDay: function(o){ o.className = "onSelect"; }, onToday: function(o){ o.className = "onToday"; }, onFinish: function(){ $("idCalendarYear").innerHTML = this.Year; $("idCalendarMonth").innerHTML = this.Month; var flag = [10,15,20]; for(var i = 0, len = flag.length; i < len; i++){ this.Days[flag[i]].innerHTML = " 使用说明:

程序比较简单,代码中都有说明,这里说说怎么使用。
首先是实例化一个Calendar,并设置参数。
参数说明:
Year:要显示的年份
Month:要显示的月份
SelectDay:选择日期
onSelectDay:在选择日期触发
onToday:在当天日期触发
onFinish:日历画完后触发

一般SelectDay设置成选择了的日期,并在onSelectDay中设置一个函数用来设置这个日期的样式,
例如实例里SelectDay设置成今个月10号并在那天样式设为onSelect:

SelectDay:  new  Date().setDate( 10 ),
onSelectDay: 
function (o){ o.className  =   " onSelect " ; },

而onToday就用来设置今日日期的样式,
例如实例里面把今天的日期的样式设为onToday:

onToday:  function (o){ o.className  =   " onToday " ; },

在onFinish中可以放需要设置日历的程序。
可以通过this.Year和this.Month获取当前日历显示的年份和月份。
对有数据的日期的也在这里设置,例如实例中是有一个当前月份的有数据的日期列表,然后根据这个列表对相应的日期进行设置:


var flag = [10,15,20];
for(var i = 0, len = flag.length; i < len; i++){
    
this.Days[flag[i]].innerHTML = "<a href='javascript:void(0);' οnclick=/"alert('日期是:"+this.Year+"/"+this.Month+"/"+flag[i]+"');return false;/">" + flag[i] + "</a>";
}

实例中是固定了这个日期列表,实际应用中可以根据年份月份获取对应的日期列表,
个人推荐用年份月份通过ajax获取。

程序中还有两个有用的方法PreMonth(显示上一个月)和NextMonth(显示下一个月)。

 

测试代码: 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" />
<title>blog式日历控件</title>
</head>
<body>
<script type="text/javascript">
var $ = function (id) {
    
return "string" == typeof id ? document.getElementById(id) : id;
};

var Class = {
  create: 
function() {
    
return function() {
      
this.initialize.apply(this, arguments);
    }
  }
}

Object.extend 
= function(destination, source) {
    
for (var property in source) {
        destination[property] 
= source[property];
    }
    
return destination;
}


var Calendar = Class.create();
Calendar.prototype 
= {
  initialize: 
function(container, options) {
    
this.Container = $(container);//容器(table结构)
    this.Days = [];//日期对象列表
    
    
this.SetOptions(options);
    
    
this.Year = this.options.Year;
    
this.Month = this.options.Month;
    
this.SelectDay = this.options.SelectDay ? new Date(this.options.SelectDay) : null;
    
this.onSelectDay = this.options.onSelectDay;
    
this.onToday = this.options.onToday;
    
this.onFinish = this.options.onFinish;    
    
    
this.Draw();
  },
  
//设置默认属性
  SetOptions: function(options) {
    
this.options = {//默认值
        Year:            new Date().getFullYear(),//显示年
        Month:            new Date().getMonth() + 1,//显示月
        SelectDay:        null,//选择日期
        onSelectDay:    function(){},//在选择日期触发
        onToday:        function(){},//在当天日期触发
        onFinish:        function(){}//日历画完后触发
    };
    Object.extend(
this.options, options || {});
  },
  
//上一个月
  PreMonth: function() {
    
//先取得上一个月的日期对象
    var d = new Date(this.Year, this.Month - 21);
    
//再设置属性
    this.Year = d.getFullYear();
    
this.Month = d.getMonth() + 1;
    
//重新画日历
    this.Draw();
  },  
  
//下一个月
  NextMonth: function() {
    
var d = new Date(this.Year, this.Month, 1);
    
this.Year = d.getFullYear();
    
this.Month = d.getMonth() + 1;
    
this.Draw();
  },
  
//画日历
  Draw: function() {
    
//用来保存日期列表
    var arr = [];
    
//用当月第一天在一周中的日期值作为当月离第一天的天数
    for(var i = 1, firstDay = new Date(this.Year, this.Month - 11).getDay(); i <= firstDay; i++){ arr.push("&nbsp;"); }
    
//用当月最后一天在一个月中的日期值作为当月的天数
    for(var i = 1, monthDay = new Date(this.Year, this.Month, 0).getDate(); i <= monthDay; i++){ arr.push(i); }
    
    
var frag = document.createDocumentFragment();
    
    
this.Days = [];
    
    
while(arr.length > 0){
        
//每个星期插入一个tr
        var row = document.createElement("tr");
        
//每个星期有7天
        for(var i = 1; i <= 7; i++){
            
var cell = document.createElement("td");
            cell.innerHTML 
= "&nbsp;";
            
            
if(arr.length > 0){
                
var d = arr.shift();
                cell.innerHTML 
= d;
                
if(d > 0){
                    
this.Days[d] = cell;
                    
//判断是否今日
                    if(this.IsSame(new Date(this.Year, this.Month - 1, d), new Date())){ this.onToday(cell); }
                    
//判断是否选择日期
                    if(this.SelectDay && this.IsSame(new Date(this.Year, this.Month - 1, d), this.SelectDay)){ this.onSelectDay(cell); }
                }
            }
            row.appendChild(cell);
        }
        frag.appendChild(row);
    }
    
    
//先清空内容再插入(ie的table不能用innerHTML)
    while(this.Container.hasChildNodes()){ this.Container.removeChild(this.Container.firstChild); }
    
this.Container.appendChild(frag);
    
    
this.onFinish();
  },
  
//判断是否同一日
  IsSame: function(d1, d2) {
    
return (d1.getFullYear() == d2.getFullYear() && d1.getMonth() == d2.getMonth() && d1.getDate() == d2.getDate());
  } 
};

</script>
<style type="text/css">

.Calendar 
{
    font-family
:Verdana;
    font-size
:12px;
    background-color
:#e0ecf9;
    text-align
:center;
    width
:200px;
    height
:160px;
    padding
:10px;
    line-height
:1.5em;
}
.Calendar a
{
    color
:#1e5494;
}

.Calendar table
{
width
:100%; 
border
:0;
}

.Calendar table thead
{color:#acacac;}

.Calendar table td 
{
    font-size
: 11px;
    padding
:1px;
}
#idCalendarPre
{
    cursor
:pointer;
    float
:left;
    padding-right
:5px;
}
#idCalendarNext
{
    cursor
:pointer;
    float
:right;
    padding-right
:5px;
}
#idCalendar td.onToday 
{
    font-weight
:bold;
    color
:#C60;
}
#idCalendar td.onSelect 
{
    font-weight
:bold;
}
</style>
<div class="Calendar">
  
<div id="idCalendarPre">&lt;&lt;</div>
  
<div id="idCalendarNext">&gt;&gt;</div>
  
<span id="idCalendarYear">2008</span>年 <span id="idCalendarMonth">8</span>
  
<table cellspacing="0">
    
<thead>
      
<tr>
        
<td></td>
        
<td></td>
        
<td></td>
        
<td></td>
        
<td></td>
        
<td></td>
        
<td></td>
      
</tr>
    
</thead>
    
<tbody id="idCalendar">
    
</tbody>
  
</table>
</div>
<script language="JavaScript">

var cale = new Calendar("idCalendar", {
    SelectDay: 
new Date().setDate(10),
    onSelectDay: 
function(o){ o.className = "onSelect"; },
    onToday: 
function(o){ o.className = "onToday"; },
    onFinish: 
function(){
        $(
"idCalendarYear").innerHTML = this.Year; $("idCalendarMonth").innerHTML = this.Month;
        
var flag = [10,15,20];
        
for(var i = 0, len = flag.length; i < len; i++){
            
this.Days[flag[i]].innerHTML = "<a href='javascript:void(0);' οnclick=/"alert('日期是:"+this.Year+"/"+this.Month+"/"+flag[i]+"');return false;/">" + flag[i] + "</a>";
        }
    }
});

$(
"idCalendarPre").onclick = function(){ cale.PreMonth(); }
$(
"idCalendarNext").onclick = function(){ cale.NextMonth(); }

</script>
</body>
</html>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值