基于ThinkPHP实现的日历功能实例详解


Posted in PHP onApril 15, 2017

本文实例讲述了基于ThinkPHP实现的日历功能。分享给大家供大家参考,具体如下:

开发环境介绍

最新,闲来没事,便开发了一款简单的日历,来统计工作情况。为了开发便捷,使用ThinkPHP架构。界面如下图

基于ThinkPHP实现的日历功能实例详解

备注:每页包含上一个月,当前月,下一个月的日期,并用不同的颜色区分,如果某天工作了,便圈出来。
主要是以下两个文件

重要文件描述

功能文件

基于ThinkPHP实现的日历功能实例详解

CalenDar.class.php主要负责,获取日历详细信息的,不涉及用户数据操作。

代码如下:

<?php
namespace Util;
class CalenDar{
  //上一个月信息
  private $lastYear=null;
  private $lastMonth=null;
  //当前月信息
  private $curYear=null;
  private $curMonth=null;
  private $curWek=null;
  private $curDay=null;
  private $curDaySum=0;
  //下一个月月信息
  private $nextYear=null; //下一个月是哪一年
  private $nextMonth=null;//下一个月
  private $calendar=null;
  public function __construct($dateTime=null){
    if(isset($_get['yeal']) && is_numeric($_get['yeal'])){
      $this->curYear=$_get['yeal'];
    }else{
      $this->curYear=date('Y');
    }
    if(isset($_get['month']) && is_numeric($_get['month'])){
      $this->curMonth=$_get['month'];
    }else{
      $this->curMonth=date('n');
    }
    if(isset($_get['day']) && is_numeric($_get['day'])){
      $this->curDay=$_get['day'];
    }else{
      $this->curDay=date('j');
    }
    $this->init($dateTime);
    $this->createCalendar();
  }
  /**
  *初始化
  */
  public function init($dateTime=null){
    if(!empty($dateTime)){ //当月
      $this->curYear=date('Y',strtotime($dateTime));
      $this->curMonth=date('n',strtotime($dateTime));
      $this->curDay=date('j',strtotime($dateTime));
    }
    $this->curWek=date('w',strtotime($this->curYear.'-'.$this->curMonth.'-1'));
    //上一个月
    $this->lastMonth=$this->curMonth-1; //上一个月
    $this->lastYear=$this->curYear; //上一个月属于哪一年
    if($this->lastMonth<0){
      $this->lastMonth=12;
      $this->lastYear-=1;
    }
    //下一个月
    $this->nextMonth=$this->curMonth+1;//下一个月
    $this->nextYear=$this->curYear; //下一个月属于哪一年
    if($this->nextMonth > 12){
      $this->nextMonth=1;
      $this->nextYear+=1;
    }
  }
  public function getCalendar(){
    return $this->calendar;
  }
  /**
  *创建日历从周日 周一 周二 周三 周四 周五 周六,7*5方格,前面补上月后几天,后面补下月开始几天
  **/
  public function createCalendar(){
    //判断当月共计多少天
    $nextStr=$this->nextYear.'-'.$this->nextMonth.'-1 -1 days';
    $curDaySum=date('j',strtotime($nextStr));
    //判断上一个月最后一天是多少号
    $lastStr=$this->curYear.'-'.$this->curMonth.'-1 -1 days';
    $lastDaySum=date('j',strtotime($lastStr));
    $prefixLId=$this->lastYear.'-'.$this->lastMonth;
    $prefixCId=$this->curYear.'-'.$this->curMonth;
    $prefixNId=$this->nextYear.'-'.$this->nextMonth;
    if($this->curWek == 0){
      $lastMonthSum=7; //需要添加上个月的$lastMonthSum天
    }else{
      $lastMonthSum=$this->curWek;
    }
    $lastMonthStart=$lastDaySum - $lastMonthSum+1;
    for($i=0,$j=1,$k=1;$i<42;$i++){
      $dateInfo=array();
      if($i<$lastMonthSum){ //上一个月
        $dateInfo['day']=$lastMonthStart + $i;
        $dateInfo['type']=1;
        $id=$prefixLId.'-'.$dateInfo['day'];
        $this->calendar[]=array('id'=>$id,
                    'info'=>$dateInfo);
      }else if($j > $curDaySum){//下一个月
        $id=$prefixNId.'-'.$k;
        $dateInfo['day']=$k;
        $dateInfo['type']=3;
        $this->calendar[]=array('id'=>$id,
                    'info'=>$dateInfo);
        $k++;
      }else{//本月
        $dateInfo['day']=$j;
        $dateInfo['type']=2;
        $this->calendar[]=array('id'=>($prefixCId.'-'.$j),
                    'info'=>$dateInfo);
        $j++;
        $this->curDaySum+=1;
      }
    }
  }
  public function getDayTime(){
    return $this->curYear.'-'.$this->curMonth.'-'.$this->curDay;
  }
  /**
  *获取当前月属于哪个月
  **/
  public function getCurMonth(){
    return $this->curYear.'-'.$this->curMonth;
  }
  /**
  *获取上一个月属于哪个月
  **/
  public function getLastMonth(){
    return $this->lastYear.'-'.$this->lastMonth;
  }
  /**
  *获取下一个月属于哪个月
  **/
  public function getNextMonth(){
    return $this->nextYear.'-'.$this->nextMonth;
  }
  /**
  *判断当前月有多少天
  **/
  public function getCurDaySum(){
    return $this->curDaySum;
  }
}

WorkLog.class.php文件,主要负责将用户工作信息与日历信息结合起来。

<?php
namespace Util;
class WorkLog{
  private $uid;//用户id
  private $calendar;//日历对象
  private $lastMonth;//上一个月
  private $curMonth; //本月
  private $nextMonth;//下一个月
  private $monthWorkedDays;//当月工作的天数
  private $workLog=null;//当前月的工作日历
  public function __construct($uid=0,$daytime=null){
    $this->uid=$uid;
    $this->calendar=new \Util\CalenDar($daytime);
    $this->lastMonth=$this->calendar->getLastMonth();
    $this->curMonth=$this->calendar->getCurMonth();
    $this->nextMonth=$this->calendar->getNextMonth();
    $this->init();
  }
  public function init(){
    $info=$this->calendar->getCalendar();
    $userLog=array();
    if($this->uid !=0){
      $lastMonth=explode('-', $this->lastMonth);
      $curMonth=explode('-', $this->curMonth);
      $nextMonth=explode('-', $this->nextMonth);
      //获取上一个月,当前月,下一个月的工作日志,后期使用缓存
      $where='uid='.$this->uid.' AND ((`year`='.$lastMonth[0].' AND `month`='.$lastMonth[1].' ) or (`year`='.$curMonth[0].' AND `month`='.$curMonth[1].' ) or (`year`='.$nextMonth[0].' AND `month`='.$nextMonth[1].' ))';
      $rs=M('work_log')->field('year,month,day,status')->where($where)->select();
      foreach ($rs as $value) {
        $userLog[$value['year'].'-'.$value['month'].'-'.$value['day']]=$value['status'];
      }
    }
    $flag=1;
    foreach ($info as $key=>$value) {
      if($flag % 7 ==1 || $flag % 7 ==0){
        $cellbgtype=3;//没有工作
      }else{
        $cellbgtype=1;//没有工作
      }
      if(isset($userLog[$value['id']]) && $userLog[$value['id']] ==1){
        //判断是不是当前月份
        $str=date('Y-n',strtotime($value['id']));
        if($this->curMonth == $str){
          $this->monthWorkedDays+=1;
        }
        $cellbgtype+=1;
      }
      $cellbgtype='daytype'.$cellbgtype.'_'.$value['info']['type'];
      $info[$key]['info']['class']=$cellbgtype;
      $flag++;
    }
    $this->workLog=$info;
  }
  public function getLastMonth(){
    return $this->lastMonth;
  }
  public function getCurMonth(){
    return $this->curMonth;
  }
  public function getNextMonth(){
    return $this->nextMonth;
  }
  /**
  *当月已经工作的天数
  */
  public function monthWorkedDays(){
    return $this->monthWorkedDays;
  }
  /**
  *当月的天数
  **/
  public function monthDays(){
    return $this->calendar->getCurDaySum();
  }
  /**
  *获取累计工作的天数
  **/
  public function workedDays(){
  }
  /**
  *当前工作日历的月份
  **/
  public function logMonth(){
  }
  /**
  *当月截止到目前可得薪水
  **/
  public function hadSalary(){
  }
  /**
  *预计可得最高薪水
  **/
  public function maxSalary(){
  }
  /**
  *当前的工作日历
  **/
  public function workInfo(){
    return $this->workLog;
  }
}

调用文件

IndexController.class.php

<?php
namespace Home\Controller;
use Think\Controller;
class IndexController extends Controller {
  public function index(){
   $uid=1;
   $daytime=I('daytime');
   if(!empty($daytime)){
    $daytime=date('Y-m-d',$daytime);
   }
   $WorkLog=new \Util\WorkLog($uid,$daytime);
   $curDay=$WorkLog->getCurMonth();
   $lastMDay=strtotime($curDay.' -1 month');
   $nextMDay=strtotime($curDay.' +1 month');
   $nowTime=date('当前时间:Y年m月d号 H:i:s');
   $curDayStr=date('日历时间:Y年m月d号',strtotime($curDay));
   $info=$WorkLog->workInfo();
   $curDaySum=$WorkLog->monthDays();
   $workDays=$WorkLog->monthWorkedDays();
   $this->assign('lastMDay',$lastMDay);
   $this->assign('nextMDay',$nextMDay);
   $this->assign('nowTime',$nowTime);
   $this->assign('curDay',$curDayStr);
   $this->assign('info',$info);
   $this->assign('curDaySum',$curDaySum);
   $this->assign('workDays',$workDays);
   $this->display();
  }
}

显示文件

index.html

<extend name="Public/base" />
<block name="css">
<style type="text/css">
  *{margin: 0; padding: 0; border: 0; font-size: 16px;}
  body{background-color: #4d56a3;}
  ul{list-style: none;}
  #mainBox{width: 1024px; height:auto; margin:10px auto 0 auto; }
  #leftBox{width:60%; height: 700px; float: left; border-radius:15px; margin-right:3%;  background-color:#ffbd66; }
  #rightBox{width:37%; height: 700px; float: right; border-radius:15px; background-color:#faf7dd}
  #calendartitle{width: 95%; margin: 10px auto 5px auto; height: 110px; }
  #calendartitle ul li{float: left; margin-right: 10px;}
  #logoImg{width: 100px; height: 100px; border-radius: 10px; background-image: url('__IMG__/logo.png');background-repeat: no-repeat;}
  #cellHead{width: 95%; background-color:#ffe786; border-radius:10px 10px 0 0; }
  #cellHead td{width:80px; height:45px; font-size: 22px; }
  #calendarcell{width: 95%; margin: 0 auto;}
  #calendarTable td{width:80px; height:80px;background-repeat: no-repeat; border: 1px; font-size: 18px; }
  .daytype1_1{background-image:url("__IMG__/cellbg1_0_1.png");}
  .daytype2_1{background-image:url("__IMG__/cellbg1_1_1.png");}
  .daytype1_2{background-image:url("__IMG__/cellbg1_0_2.png");}
  .daytype2_2{background-image:url("__IMG__/cellbg1_1_2.png");}
  .daytype1_3{background-image:url("__IMG__/cellbg1_0_3.png");}
  .daytype2_3{background-image:url("__IMG__/cellbg1_1_3.png");}
  .daytype3_1{background-image:url("__IMG__/cellbg2_0_1.png");}
  .daytype4_1{background-image:url("__IMG__/cellbg2_1_1.png");}
  .daytype3_2{background-image:url("__IMG__/cellbg2_0_2.png");}
  .daytype4_2{background-image:url("__IMG__/cellbg2_1_2.png");}
  .daytype3_3{background-image:url("__IMG__/cellbg2_0_3.png");}
  .daytype4_3{background-image:url("__IMG__/cellbg2_1_3.png");}
  .aBlock{display: block; text-decoration: none;}
  .ainblock{display: inline-block; text-decoration: none;}
  .actionBox{width: 80%; margin: 0 auto; margin-top:15px; background-color:#fadfbb; height: auto; padding-top: 20px; padding-bottom: 20px; border-radius: 10px; text-align: center;}
  #action1 a{width:80%; margin: 0 auto 15px auto; height: 45px; background-color: #ffec42; border-radius: 10px; line-height: 45px; text-align: center; font-size: 22px; color: #ad5408; font-weight: 700;}
  #action1 a:hover{font-size: 24px; color:#F12;}
  #action2{text-align: center;}
  #action2 a{width: 80px; background-color: #ffec42; border-radius: 10px; height: 35px; line-height: 35px; text-align: center; margin-right: 10px;}
  #action3{text-align: left;}
  #action3 ul li{margin-left:20px; border-bottom: 1px dashed #999;margin-right:10px; margin-bottom: 10px; font-size: 20px;}
</style>
</block>
<block name="title">日历</block>
<block name="main">
  <div id="mainBox">
    <div id='leftBox'>
      <div id="calendartitle">
        <ul>
          <li id="logoImg">
          </li>
          <li >
            <div style="height:45px; line-height:45px; font-size:20px;">{$nowTime}</div>
            <div style="height:45px; line-height:45px; font-size:20px;">{$curDay}</div>
          </li>
        </ul>
      </div>
      <div id="calendarcell">
        <div id="cellHead">
          <table cellpadding="0" cellspacing="0" border="0" align="center">
            <tr>
              <td align="center">周日</td>
              <td align="center">周一</td>
              <td align="center">周二</td>
              <td align="center">周三</td>
              <td align="center">周四</td>
              <td align="center">周五</td>
              <td align="center">周六</td>
            </tr>
          </table>
        </div>
      <table cellpadding="0" cellspacing="0" border="0" id='calendarTable' align="center">
        <tr>
        <volist name="info" id="dayinfo" key='flag'>
          <td align="center" class="{$dayinfo['info']['class']}">{$dayinfo['info']['day']}</td>
          <if condition="$flag % 7 == 0 && $flag % 7 != 42 ">
          </tr><tr>
          </if>
        </volist>
      </table>
      </div>
    </div>
    <div id='rightBox'>
      <div id="action1" class="actionBox">
        <ul>
          <li><a href="javascript:;" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="aBlock" onclick="todayAction(1)">工作</a></li>
          <li><a href="javascript:;" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="aBlock" onclick="todayAction(2)">休息</a></li>
        </ul>
      </div>
      <div id="action2" class="actionBox">
        <ul>
          <li><a href="?daytime={$lastMDay}" rel="external nofollow" class="ainblock" >上一月</a><a href="?daytime={$nextMDay}" rel="external nofollow" class="ainblock" >下一月</a></li>
        </ul>
      </div>
      <div id="action3" class="actionBox">
        <ul>
          <li>本月共计:{$curDaySum} 天</li>
          <li>本月已工作:{$workDays} 天</li>
          <li>本月剩余工作日:** 天</li>
          <li>预计工资:**天</li>
        </ul>
      </div>
      <div id="action4" class="actionBox">
        <ul>
          <li><a href="javascript:;" rel="external nofollow" rel="external nofollow" rel="external nofollow" class="aBlock">设置</a></li>
        </ul>
      </div>
    </div>
  </div>
</block>
<block name="js">
<script type="text/javascript" src="__JS__/jquery-1.12.4.min.js"></script>
<script type="text/javascript">
  function todayAction(type){
    $.post("{:U('Log/addLog')}",{'type':type},function(data){
      if(data['status']==1){
        alert('数据添加完成');
      }else{
        alert('数据异常');
      }
    });
  }
</script>
</block>

思路分析

1.在CalenDar.class.php中,封装每个月的日期信息。如果读者需要做日历,只需要将该文件作为一个类调用即可。
如下图

$calendar=new \Util\CalenDar();
$info=$calendar->getCalendar();
echo json_encode($info);
//输出结果如下
[{"id":"2016-5-29","info":{"day":29,"type":1}},{"id":"2016-5-30","info":{"day":30,"type":1}},{"id":"2016-5-31","info":{"day":31,"type":1}},{"id":"2016-6-1","info":{"day":1,"type":2}},{"id":"2016-6-2","info":{"day":2,"type":2}},{"id":"2016-6-3","info":{"day":3,"type":2}},{"id":"2016-6-4","info":{"day":4,"type":2}},{"id":"2016-6-5","info":{"day":5,"type":2}},{"id":"2016-6-6","info":{"day":6,"type":2}},{"id":"2016-6-7","info":{"day":7,"type":2}},{"id":"2016-6-8","info":{"day":8,"type":2}},{"id":"2016-6-9","info":{"day":9,"type":2}},{"id":"2016-6-10","info":{"day":10,"type":2}},{"id":"2016-6-11","info":{"day":11,"type":2}},{"id":"2016-6-12","info":{"day":12,"type":2}},{"id":"2016-6-13","info":{"day":13,"type":2}},{"id":"2016-6-14","info":{"day":14,"type":2}},{"id":"2016-6-15","info":{"day":15,"type":2}},{"id":"2016-6-16","info":{"day":16,"type":2}},{"id":"2016-6-17","info":{"day":17,"type":2}},{"id":"2016-6-18","info":{"day":18,"type":2}},{"id":"2016-6-19","info":{"day":19,"type":2}},{"id":"2016-6-20","info":{"day":20,"type":2}},{"id":"2016-6-21","info":{"day":21,"type":2}},{"id":"2016-6-22","info":{"day":22,"type":2}},{"id":"2016-6-23","info":{"day":23,"type":2}},{"id":"2016-6-24","info":{"day":24,"type":2}},{"id":"2016-6-25","info":{"day":25,"type":2}},{"id":"2016-6-26","info":{"day":26,"type":2}},{"id":"2016-6-27","info":{"day":27,"type":2}},{"id":"2016-6-28","info":{"day":28,"type":2}},{"id":"2016-6-29","info":{"day":29,"type":2}},{"id":"2016-6-30","info":{"day":30,"type":2}},{"id":"2016-7-1","info":{"day":1,"type":3}},{"id":"2016-7-2","info":{"day":2,"type":3}},{"id":"2016-7-3","info":{"day":3,"type":3}},{"id":"2016-7-4","info":{"day":4,"type":3}},{"id":"2016-7-5","info":{"day":5,"type":3}},{"id":"2016-7-6","info":{"day":6,"type":3}},{"id":"2016-7-7","info":{"day":7,"type":3}},{"id":"2016-7-8","info":{"day":8,"type":3}},{"id":"2016-7-9","info":{"day":9,"type":3}}]

2.在WorkLog.class.php中,获取该用户上一个月、当前月、下一个月的工作信息,之所以使用一次性获取三个月的工作信息,因为如果每天的去读取,这样数据查询的次数过大,当然最好的还是做一下缓存比较好。读取到工作信息后,然后结合日历,判断每天是否工作,以及是否是周末,来决定日历中每个方格的背景样式。工作信息数据库如下图:

基于ThinkPHP实现的日历功能实例详解

PS:这里再为大家分享几款本站的在线日期工具供大家参考:

在线万年历日历:
http://tools.3water.com/bianmin/wannianli

网页万年历日历:
http://tools.3water.com/bianmin/webwannianli

在线万年历黄历flash版:
http://tools.3water.com/bianmin/flashwnl

希望本文所述对大家基于ThinkPHP框架的PHP程序设计有所帮助。

PHP 相关文章推荐
PHP新手上路(五)
Oct 09 PHP
php一句话cmdshell新型 (非一句话木马)
Apr 18 PHP
php+javascript的日历控件
Nov 19 PHP
php导入导出excel实例
Oct 25 PHP
phpnow php探针环境检测代码
Nov 04 PHP
PHP准确取得服务器IP地址的方法
Jun 02 PHP
php正则表达式验证(邮件地址、Url地址、电话号码、邮政编码)
Mar 14 PHP
人脸识别测颜值、测脸龄、测相似度微信接口
Apr 07 PHP
Python中使用django form表单验证的方法
Jan 16 PHP
php对xml文件的增删改查操作实现方法分析
May 19 PHP
PHP使用正则表达式实现过滤非法字符串功能示例
Jun 04 PHP
laravel框架中控制器的创建和使用方法分析
Nov 23 PHP
php修改数组键名的方法示例
Apr 15 #PHP
php自定义函数实现统计中文字符串长度的方法小结
Apr 15 #PHP
php+mysql+jquery实现简易的检索自动补全提示功能
Apr 15 #PHP
php+mysql+ajax实现单表多字段多关键词查询的方法
Apr 15 #PHP
PHP快速推送微信模板消息
Apr 14 #PHP
PHP 中常量的知识整理
Apr 14 #PHP
php动态读取数据清除最右边距的方法
Apr 12 #PHP
You might like
图片存储与浏览一例(Linux+Apache+PHP+MySQL)
2006/10/09 PHP
PHP 基于文件头的文件类型验证类函数
2012/05/01 PHP
PHP使用json_encode函数时不转义中文的解决方法
2014/11/12 PHP
在Javascript中为String对象添加trim,ltrim,rtrim方法
2006/09/22 Javascript
子窗口、父窗口和Silverlight之间的相互调用
2010/08/16 Javascript
javascript获取网页中指定节点的父节点、子节点的方法小结
2013/04/24 Javascript
jquery 延迟执行实例介绍
2013/08/20 Javascript
DOM基础教程之使用DOM控制表格
2015/01/20 Javascript
javascript表格的渲染组件
2015/07/03 Javascript
JS控制静态页面之间传递参数获取参数并应用的简单实例
2016/08/10 Javascript
微信小程序 开发工具快捷键整理
2016/10/31 Javascript
AngularJS中指令的四种基本形式实例分析
2016/11/22 Javascript
实例解析angularjs的filter过滤器
2016/12/14 Javascript
100行代码实现一个vue分页组功能
2018/11/06 Javascript
vue-cli 目录结构详细讲解总结
2019/01/15 Javascript
JS将时间秒转换成天小时分钟秒的字符串
2019/07/10 Javascript
layui 数据表格复选框实现单选功能的例子
2019/09/19 Javascript
Vue SSR 即时编译技术的实现
2020/05/06 Javascript
JavaScript图像放大镜效果实现方法详解
2020/06/28 Javascript
[47:39]2018DOTA2亚洲邀请赛 3.31 小组赛 A组 LGD vs OPTIC
2018/03/31 DOTA
使用IPython来操作Docker容器的入门指引
2015/04/08 Python
12步入门Python中的decorator装饰器使用方法
2016/06/20 Python
python批量制作雷达图的实现方法
2016/07/26 Python
pandas实现将dataframe满足某一条件的值选出
2019/06/12 Python
python matplotlib画盒图、子图解决坐标轴标签重叠的问题
2020/01/19 Python
Python列表操作方法详解
2020/02/09 Python
python3+openCV 获取图片中文本区域的最小外接矩形实例
2020/06/02 Python
解决python 执行sql语句时所传参数含有单引号的问题
2020/06/06 Python
python导入库的具体方法
2020/06/18 Python
美国瑜伽品牌:Gaiam
2017/10/31 全球购物
英国快时尚女装购物网站:PrettyLittleThing
2018/08/15 全球购物
Farah官方网站:男士服装及配件
2019/11/01 全球购物
绿色环保口号
2014/06/12 职场文书
健康状况证明书
2014/11/26 职场文书
公司员工辞职信范文
2015/05/12 职场文书
浅谈JS的二进制家族
2021/05/09 Javascript