vue实现一个简单的日程表:

写在home.vue文件中了

<template>
    <div class="homeBox">
        <table>
            <!-- 时间轴 -->
            <tr>
                <td ><div :style="{width:2*tableWidth+'px',height:tableHeight+'px'}"></div></td>
                <td v-for="(item,index) in arrLength" :key="index">
                    <div class="timeWrap" :style="{width:tableWidth+'px',height:tableHeight+'px'}">
                        <span>{{minTOhm(index,step,startTime)}}</span>
                    </div>
                </td>
            </tr>
            <!-- 表格 -->
            <tr v-for="(tableItem,index) in sessionDataArr" :key="'table'+index">
                <td ><div class="roomName" :style="{width:2*tableWidth+'px',height:tableHeight+'px',lineHeight:tableHeight+'px'}">{{tableItem.roomName}}</div></td>
                <td v-for="(item,index) in tableItem.sessionArr"
                    class="itemcell"
                    :key="index+'tableItem'" 
                    :colspan="item.colspan?item.colspan:''">
                    <div class="itemTd" 
                        :style="{width:item.colspan?(item.colspan*tableWidth)+'px':tableWidth+'px',height:tableHeight+'px'}"
                        :title="item.text">
                        <a class="linkEle" href="javascript:">
                            <span class="session_text">{{item.text}}</span>
                        </a>
                    </div>
                </td>
            </tr>
        </table>
    </div>
</template>

<script>
// @ is an alias to /src
import tools from "@/util/tools.js";
import global from "@/util/global.js";
import api from "@/util/api.js";
export default {
    name: "home",
    components: {
    },
    data(){
        return{
            tableWidth:50,
            tableHeight:50,
            startTime:"08:00",//日程开始时间
            endTime:"22:15",//日程结束时间
            step:10,//时间间隔
            sessionData:[//模拟的session数据
                {
                    roomName:"多功能厅A",
                    sessionArr:[
                        {startTime:"08:00",endTime:"08:30",text:"心血管研究心血管研究",colspan:""},
                        {startTime:"09:00",endTime:"09:10",text:"心血管研究",colspan:""},
                        {startTime:"09:20",endTime:"09:30",text:"是打发",colspan:""},
                        {startTime:"09:40",endTime:"10:00",text:"分发大时分发时分发送到",colspan:""},
                    ]
                },
                {
                    roomName:"多功能厅B",
                    sessionArr:[
                        {startTime:"08:00",endTime:"08:50",text:"sessionB心血管研究",colspan:""},
                        {startTime:"09:00",endTime:"09:10",text:"大脑研究",colspan:""},
                        {startTime:"15:20",endTime:"15:50",text:"哈哈哈",colspan:""},
                        {startTime:"21:00",endTime:"21:20",text:"分发大时",colspan:""},
                    ]
                },
                {
                    roomName:"多功能厅C",
                    sessionArr:[
                        {startTime:"08:00",endTime:"08:50",text:"sessionB心血管研究",colspan:""},
                        {startTime:"09:00",endTime:"09:10",text:"大脑研究",colspan:""},
                        {startTime:"15:20",endTime:"15:50",text:"哈哈哈",colspan:""},
                        {startTime:"21:00",endTime:"21:20",text:"分发大时",colspan:""},
                    ]
                },
            ],

            arrLength:0,//总共与多上格子(不用后台传,是计算出来的))
            sessionDataArr:[],//根据后台数据 生成的最终表格数据(不用后台传)
        }
    },
    created(){
        this.generaterAllTable(this.sessionData);//生成整个表格数据  
    },
    mounted(){
       
    },
    methods:{
        hmTOmin(hm){//hh:mm=>分钟或则秒
            var arr=hm.split(":"),H,M,sum={M:'',S:''};
            if(/^0[0-9]$/.test(arr[0])){
                H=Number(arr[0].replace("0",''));
            }else{
                H=Number(arr[0]);
            }
            if(/^0[0-9]$/.test(arr[1])){
                M=Number(arr[1].replace("0",''));
            }else{
                M=Number(arr[1]);
            }
            sum.M=Number(H*60+M);
            sum.S=Number((H*60+M)*60);
            return sum;
        },
        minTOhm(index,step,startTime){//计算时间轴
            var str="",h,m,self=this;
            if(index==0){
                str=startTime;
            }else{
                var sumMIn=self.hmTOmin(startTime).M+(step*index);
                h=parseInt(sumMIn/60);
                m=sumMIn%60;
                if(h<10){
                    h='0'+h;
                }
                if(m<10){
                    m='0'+m;
                }
                str=h+":"+m;
            }
            return str;
        },
        generaterCol(sessionArr){//生成某一列的session数组
            var self=this,colArr=[];
            self.generaterArrLength();//计算时间轴的长度 下面能用到
            var sessionTd=0;//session所占的总格数
            sessionArr.forEach((item,index,arr)=>{
                item.colspan=Math.ceil((self.hmTOmin(item.endTime).M-self.hmTOmin(item.startTime).M)/self.step);
                sessionTd+=item.colspan;
            })
            
            var tableColArrLength=self.arrLength-sessionTd+sessionArr.length;//日程某一列有多少个单元格,也就是某一列数组的长度
            
            for(let i=0,len=tableColArrLength;i<len;i++){//可以考虑抽离出去
                var obj={startTime:"",endTime:"",text:"",colspan:""};
                colArr[i]=obj;
            }

            function addData(){//向某一列数组中添加数据
                var count=0;//计算每个session前面有
                sessionArr.forEach((item,index,arr)=>{
                    var itemIndex=Math.floor((self.hmTOmin(item.startTime).M-self.hmTOmin(self.startTime).M)/self.step);
                    colArr[itemIndex-count].startTime=item.startTime;
                    colArr[itemIndex-count].endTime=item.endTime;
                    colArr[itemIndex-count].text=item.text;
                    colArr[itemIndex-count].colspan=item.colspan;
                    count+=(item.colspan-1);
                })
            }
            addData();
            return colArr;//将生成的某一列数据返回出去
        },
        generaterArrLength(){//计算出事件轴的长度
            var self=this;
            var sumMIN=self.hmTOmin(self.endTime).M-self.hmTOmin(self.startTime).M;
            self.arrLength=Math.ceil(sumMIN/self.step);//总共有多少格子
        },
        generaterAllTable(sessionData){//生成整个session表
            var self=this,arr=[];
            sessionData.forEach((item,index,arr)=>{
                var obj={
                    roomName:"",
                    sessionArr:[],
                };
                obj.roomName=item.roomName;
                obj.sessionArr=self.generaterCol(item.sessionArr);
                self.sessionDataArr.push(obj);
            });
            console.log(self.sessionDataArr);
        }
    }
};
</script>

<style scoped>
    .homeBox{
    }
    @media screen and (max-width:768px){
        .min768{
            display:block;
        }
        .max768{
            display: none;
        }
    }
    @media screen and (min-width:768px){
        .min768{
            display:none;
        }
        .max768{
            display: block;
        }
    }
    .timeWrap{
        position: relative;
    }
    .timeWrap span{
        position: absolute;
        bottom:0;
        left:0;
        transform-origin: left bottom;
        transform:rotateZ(-45deg)
    }
    .itemTd{
        border:2px solid #ffffff;
        border-radius:5px;
        background:brown;
        box-sizing: border-box;
        text-align: center;
        position: relative;
    }
    .session_text{
        position: absolute;
        top:50%;
        width:100%;
        transform: translateY(-50%);
        display: -webkit-box;
        overflow: hidden;
        white-space: normal;
        text-overflow: ellipsis;
        word-wrap: break-word;
        -webkit-line-clamp: 2;
        -webkit-box-orient: vertical;
    }
    .itemcell{
        vertical-align: middle;
    }
    .roomName{
        border:2px solid #ffffff;
        border-radius:5px;
        background:yellow;
        text-align: center;
        box-sizing: border-box;
    }
    .linkEle{
        text-decoration: none;
        color:black;
    }
</style>

。。。

 另一种实现:

<template>
    <div class="scheduleBox">
        <div class="contentBox padding0">
            <div class="topImgBox">
                <img :src="bannerImgUrl" alt="banner图">
            </div>
            <!-- 检索框 -->
            <div class="row paddingTB bottomBorder margin0">
                <div class="col col-xs-1 title lineHeight text-right">模糊搜索:</div>
                <div class="col col-xs-6">
                    <div>
                        <el-input v-show="searchPattern=='姓名'" placeholder="请输入专家姓名/讲题进行搜索" v-model="realName" class="input-with-select">
                            <el-select v-model="searchPattern" slot="prepend" style="width:130px;" placeholder="请选择">
                                <el-option label="姓名" value="姓名"></el-option>
                                <el-option label="讲题名称" value="讲题名称"></el-option>
                            </el-select>
                            <el-button @click="clickSearch()" slot="append" icon="el-icon-search"></el-button>
                        </el-input>
                        <el-input v-show="searchPattern=='讲题名称'" placeholder="请输入专家姓名/讲题进行搜索" v-model="jiangti" class="input-with-select">
                            <el-select v-model="searchPattern" slot="prepend" style="width:130px;" placeholder="请选择">
                                <el-option label="姓名" value="姓名"></el-option>
                                <el-option label="讲题名称" value="讲题名称"></el-option>
                            </el-select>
                            <el-button @click="clickSearch()" slot="append" icon="el-icon-search"></el-button>
                        </el-input>
                    </div>
                </div>
                <div class="col col-xs-5">

                </div>
            </div>
            <!-- 选择日期     -->
            <div class="row paddingTB bottomBorder margin0">
                <div class="col col-xs-1 title lineHeight28 text-right">选择日期:</div>
                <div class="col col-xs-11">
                    <ul class="timeBox">
                        <li class="time_item" @click="clickDate('')" :class="{'dateActive':date==''}">不限</li>
                        <li class="time_item" @click="clickDate(item.value)" :class="{'dateActive':date==item.value}" v-for="(item,index) in dateList" :key="'date'+index">
                            <a href="javascript:">{{item.value}}</a>
                        </li>
                    </ul>
                </div>
            </div>
            <!-- 日程一览表 -->
            <div class="schedule_wraper" v-if="schedule_show">
                <table>
                    <tr class="topBox">
                        <td>
                            <div class="table_header_box" style="border-radius:0;" :style="{height:tableHeight+'px'}"></div>
                        </td>
                        <td>
                            <div class="timeRow" :style="{width:(sumMin*control)+'px',height:tableHeight+'px'}">
                                <div class="timeCol" 
                                v-for="(item,index) in arrLength" :key="index"
                                :style="{left:(index*step*control)+'px'}">
                                    <span class="timeText">{{minTOhm(index,step,startTime)}}</span>
                                </div>
                            </div>
                        </td>
                    </tr>
                    <tr class="scheduleRows" v-for="(parentItem,index) in sessionDataArr" :key="'parent'+index">
                        <td>
                            <div :style="{height:tableHeight+'px'}"class="table_header_box" :title="parentItem.cn_name">
                                <span class="room_name">
                                    {{parentItem.cn_name}}
                                </span>
                            </div>
                        </td>
                        <td >
                            <div class="eleRow" :style="{width:(sumMin*control)+'px',height:tableHeight+'px'}">
                                <div class="eleCol" 
                                :title="childItem.cn_name"
                                v-for="(childItem,index) in parentItem.sessionArr" :key="'child'+index"
                                :style="{left:(childItem.startMin*control)+'px',width:(childItem.sumMin*control)+'px',height:(tableHeight-2)+'px'}"
                                >
                                    <span class="eleCol_text">
                                        {{childItem.cn_name}}
                                    </span>
                                </div>
                            </div>
                        </td>
                    </tr>
                </table>
            </div>
        </div>
    </div>
</template>

<script>
import api from "@/util/api.js";
import global from "@/util/global.js";
import tools from "@/util/tools.js";
export default {
    watch:{
        searchPattern:{
            handler(newVal,oldVal){
                if(newVal=='姓名'){
                    this.jiangti="";
                }else{
                    this.realName="";
                }
            },
            immediate:true,
        }
    },
    data(){
        return {
            searchPattern:"姓名",//用户是按'姓名'精确检索的还是按'讲题'精确搜索的
            jiangti:"",//讲题名
            realName:"",//姓名(模糊搜索时)
            date:"",//会场日期
            dateList:[],//日期列表
            mid:this.$route.params.mid,//会议id
            schedule_show:false,//控制一览表的显示隐藏
            //以下是日程一览相关
            control:1,//精度值 1分钟代表多少像素、有总宽度和总分钟计算得出
            fixedWidth:0,//后端后去的表的总宽度(px)
            tableHeight:50,
            startTime:"08:00",//日程开始时间
            endTime:"18:00",//日程结束时间
            step:30,//时间间隔
            arrLength:0,//总共与多上格子(不用后台传,是计算出来的))
            sumMin:0,//session总共有多少分钟
            sessionDataArr:[],//根据后台数据 生成的最终表格数据(不用后台传)


            sessionData:[//模拟的session数据
                {
                    cn_name:"多功能厅A",
                    sessionArr:[
                        {startTime:"08:00",endTime:"08:30",text:"心血管研究心血管研究",colspan:""},
                        {startTime:"09:00",endTime:"09:10",text:"心血管研究",colspan:""},
                        {startTime:"09:20",endTime:"09:30",text:"是打发",colspan:""},
                        {startTime:"09:40",endTime:"10:00",text:"分发大时分发时分发送到",colspan:""},
                    ]
                },
                {
                    cn_name:"多功能厅B",
                    sessionArr:[
                        {startTime:"08:00",endTime:"08:50",text:"sessionB心血管研究",colspan:""},
                        {startTime:"09:00",endTime:"09:10",text:"大脑研究",colspan:""},
                        {startTime:"15:20",endTime:"15:50",text:"哈哈哈",colspan:""},
                        {startTime:"21:00",endTime:"21:20",text:"分发大时",colspan:""},
                    ]
                },
                {
                    cn_name:"多功能厅C",
                    sessionArr:[
                        {startTime:"08:00",endTime:"08:50",text:"sessionB心血管研究",colspan:""},
                        {startTime:"09:00",endTime:"09:10",text:"大脑研究",colspan:""},
                        {startTime:"15:20",endTime:"15:50",text:"哈哈哈",colspan:""},
                        {startTime:"21:00",endTime:"21:20",text:"分发大时",colspan:""},
                    ]
                },
            ],
        }
    },
    created(){
        var self=this;
        self.scheduleInit();
    },
    mounted(){

    },
    methods:{
        scheduleInit(){//初始化
            var self=this;
            self.getMeetingInfo();//获取会议信息
            self.getMeetingDateList().then(()=>{//获取会议日期列表,
                // console.log(self.dateList);
                if(self.dateList.length>0){//默认显示 第一个时间的一览表
                    self.date=self.dateList[0].value;
                }
                self.oncegeneraterSchedule(self.date);//根据日期 生成一览表
            });

            
        },
        oncegeneraterSchedule(date){//根据日期 一次性生成一览表
            var self=this;
            self.getScheduleData(date).then((data)=>{//获取日程一览表数据
                //生成日程表
                self.generaterArrLength();//计算出格子总数
                self.sessionDataArr=self.generaterScheduleData(data);//生成最终数据
                self.control=self.fixedWidth/self.sumMin;//后端给一个指定的宽度,计算出1分钟=多少像素
                self.schedule_show=true;//显示一览表
            });
        },
        getMeetingInfo(){//获取会议信息
            var self=this;
            var obj={
                mid:self.mid
            }
            api.get(global.xuehui+"/pcRcInterface/getMeet",obj,res=>{
                console.log(res)
                if(res.data.success){
                    self.bannerImgUrl=res.data.data.cn_banner_img;
                }else{
                    tools.msgErr(res.data.message);
                }
            },err=>{
                console.log(err);
            })
        },
        getMeetingDateList(){//获取会议日期列表
            var self=this;
            return new Promise((resolve,reject)=>{
                api.get(global.xuehui+"/pcRcInterface/listDate",{mid:self.mid},res=>{
                    if(res.data.success){
                        self.dateList=res.data.data;
                        resolve();
                    }else{
                        self.dateList=[];
                        tools.msgErr(res.data.message||'error');
                    }
                },err=>{
                    console.log(err);
                })
            })  
        },
        clickDate(date){//点击日期项
            var self=this;
            self.schedule_show=false;//隐藏一览表
            self.date=date;
            if(!date){
                if(self.dateList.length>0){//默认显示 第一个时间的一览表
                    self.date=self.dateList[0].value;
                }
            }
            self.oncegeneraterSchedule(self.date);
        },
        //以下是日程一览相关
        getScheduleData(date){//获取日程一览表数据
            var self=this;
            console.log(date);
            return new Promise((resolve,reject)=>{
                var obj={
                    mid:self.mid,//会议唯一标识
                    date:date,//所需一栏表的日期
                    lang:1,//中英文 1:中文  2:英文
                }
                api.get(global.xuehui+"/pcRcInterface/getScheduleAll",obj,res=>{
                    self.startTime=res.data.data.startTime;
                    self.endTime=res.data.data.endTime;
                    self.fixedWidth=res.data.data.fixedWidth?res.data.data.fixedWidth:1200;//拿到最大表的宽度
                    self.step=res.data.data.step;
                    self.tableHeight=res.data.data.tableHeight
                    resolve(res.data.data.RoomData);
                },err=>{
                    console.log(err);
                })
            })
                
        },
        hmTOmin(hm){//hh:mm=>分钟或则秒
            var arr=hm.split(":"),H,M,sum={M:'',S:''};
            if(/^0[0-9]$/.test(arr[0])){
                H=Number(arr[0].replace("0",''));
            }else{
                H=Number(arr[0]);
            }
            if(/^0[0-9]$/.test(arr[1])){
                M=Number(arr[1].replace("0",''));
            }else{
                M=Number(arr[1]);
            }
            sum.M=Number(H*60+M);
            sum.S=Number((H*60+M)*60);
            return sum;
        },
        minTOhm(index,step,startTime){//计算时间轴
            var str="",h,m,self=this;
            if(index==0){
                str=startTime.slice(0,5);
            }else{
                var sumMIn=self.hmTOmin(startTime).M+(step*index);
                h=parseInt(sumMIn/60);
                m=sumMIn%60;
                if(h<10){
                    h='0'+h;
                }
                if(m<10){
                    m='0'+m;
                }
                str=h+":"+m;
            }
            return str;
        },
        generaterArrLength(){//计算出事件轴的长度
            var self=this;
            self.sumMin=self.hmTOmin(self.endTime).M-self.hmTOmin(self.startTime).M;
            self.arrLength=Math.ceil(self.sumMin/self.step);//总共有多少格子
        },
        generaterColArr(sessionArr){//生成某一列的session数组
            var self=this;
            var array=[];
            sessionArr.forEach((item,index,arr)=>{
                item.startMin=self.hmTOmin(item.startTime).M-self.hmTOmin(self.startTime).M;
                item.sumMin=self.hmTOmin(item.endTime).M-self.hmTOmin(item.startTime).M;
            });
            array=sessionArr;
            return array;
        },
        generaterScheduleData(sessionData){//生成整个session表数据
            var self=this;
            var array=[];
            sessionData.forEach((item,index,arr)=>{
                var obj={};
                obj.cn_name=item.cn_name;
                obj.sessionArr=self.generaterColArr(item.sessionArr);
                array.push(obj);
            });
            return array;
        },
        
    }
}
</script>

<style scoped>
    .scheduleBox{

    }
    .contentBox{
        width:100%;
        margin:0 auto;
        box-shadow:0px 0px 0px 0px #e0e0e0,   /*上边阴影*/
                -5px 0px 10px 0px #e0e0e0,   /*左边阴影*/
                5px 0px 10px 0px #e0e0e0,    /*右边阴影*/
                0px 0px 0px 0px #e0e0e0;    /*下边阴影*/
        background:#ffffff;
    }
    /* 检索块 */
    .padding0{
        padding:0;
    }
    .paddingTB{
        padding:10px 0;
    }
    .lineHeight{
        line-height:40px;
    }
    .lineHeight28{
        line-height:28px;
    }
    .bottomBorder{
        border-bottom: 1px solid #dddddd;
    }
    .margin0{
        margin:0;
    }
    .title{
        color:#777;
        font-weight: bold;
        font-size:14px;
    }
    
    /* 顶部图 */
    .topImgBox img{
        width:100%;
        height:auto;
    }
    /* 选择日期 */
    .timeBox{
        overflow: hidden;
    }
    .timeBox .dateActive{
        background:#ff7b00;
        color:#ffffff;
        padding: 0 10px;
        border-radius:5px;
        cursor: pointer;
    }
    .timeBox .dateActive a{
        color:#ffffff;
    }
    .time_item{
        float: left;
        line-height:28px;
        color:#333333;
        margin-right:20px;
        padding: 0 10px;
        border-radius:5px;
        cursor: pointer;
    }
    .time_item a{
          text-decoration: none;
        color:#333333;
    }
    .time_item:hover{
        color:#ffffff;
        background:#ff7b00;
    }
    .time_item:hover a{
        color:#ffffff;
    }
    /* 日程一览表 */
    .schedule_wraper{
        width:100%;
        overflow-x: auto;
        text-align:center;
        padding:20px 20px 50px;
    }
    .schedule_wraper>table{
        margin:0 auto;
    }
    .topBox{
        background:#0d8cdd;
    }
    .eleRow{
        background:#f1f5f9;
        margin:0 auto;
        position: relative;
        border-bottom:2px solid #ffffff;
        border-radius:6px;
    }
    .eleCol{
        position: absolute;
        /* background:red; */
        border-left:1px solid #ffffff;
        border-right:1px solid #ffffff;
        border-radius:6px;
        top:0;
        background:rgba(1,54,126,.8);
        padding:2px;
        box-sizing: border-box;
        color:#ffffff;
        cursor: pointer;
    }
    .eleCol:hover{
        background:rgba(1,54,126,1);
    }
    .eleCol .eleCol_text{
        position: absolute;
        width:100%;
        top:50%;
        left:0;
        transform: translateY(-50%);
        display: -webkit-box;
        overflow: hidden;
        white-space: normal;
        text-overflow: ellipsis;
        word-wrap: break-word;
        -webkit-line-clamp: 2;
        -webkit-box-orient: vertical;
    }
    .timeRow{
        height:50px;
        margin:0 auto;
        position: relative;
        border-bottom:2px solid #ffffff;
    }
    .timeCol{
        width:2px;
        height:4px;
        position: absolute;
        bottom:0;
        background:#ffffff;

    }
    .timeText{
        position: absolute;
        bottom:0;
        left:0;
        transform-origin: left bottom;
        transform:rotateZ(-45deg);
        color:#ffffff;
    }
    .table_header_box{
        width:120px;
        border-bottom:2px solid #ffffff;
        border-radius:6px;
        background:#0d8cdd;
        text-align: center;
        position: relative;
        color:#ffffff;
    }
    .room_name{
        position: absolute;
        width:100%;
        top:50%;
        left:0;
        transform: translateY(-50%);
        display: -webkit-box;
        overflow: hidden;
        white-space: normal;
        text-overflow: ellipsis;
        word-wrap: break-word;
        -webkit-line-clamp: 2;
        -webkit-box-orient: vertical;
    }
    
</style>