效果图

甘特图jquery 插件 甘特图js组件_甘特图

在线预览

GitHub链接(包含 knockoutJS 版本与 Vue 版本)

推荐组合效果

  • 推荐与双表头固定效果组合,实现如上例中横表头(日期)纵向固定,纵表头(类型)横向固定效果。
  • 参照连接 表头固定,表身滚动实例

Vue.js 引入

<script src="https://cdn.bootcss.com/vue/2.6.10/vue.js"></script>

CSS 代码

<style>
    table {
        border-collapse: collapse;
        border-width: 1;
    }

    p {
        width: 100px;
    }

    th,
    td {
        min-width: 20px;
        text-align: center;
        border-width: 1px;
        border-style: solid;
        border-color: #666;
    }

    th {
        background-color: #34B4A0;
    }

    span {
        border-radius: 50%;
        height: 10px;
        width: 10px;
        display: block;
        display: inline-block;
    }

    .yellow {
        background-color: #F9A100;
        border: #FCAA00 1px solid;
    }

    .green {
        background-color: #34B4A0;
        border: #46D8BC 1px solid;
    }

    .red {
        background-color: #FF0D0D;
        border: #FF2E2E 1px solid
    }

    .purple {
        background-color: #8B64FF;
        border: #8B64FF 1px solid;
    }

    .blue {
        background-color: #299CED;
        border: #299CED 1px solid;
    }
</style>

vue 页面

<div id="demo">
    <table>
        <caption>Vue.JS 制作甘特图</caption>
        <thead>
            <tr>
                <th rowspan="2" colspan="2">项目/时间</th>
                <th :colspan="days(day)" v-for="(day, index) in AllMonths">{{day|longDateToMonth}}</th>
            </tr>
            <tr>
                <th v-for="(day, index) in Dateinterval">{{day|longDateToDay}}</th>
            </tr>
        </thead>
        <tbody>
            <tr v-for="(item, index) in jsonData">
                <td :rowspan="colNum(item, index)" v-if="showFormType(item, index)">
                    <p>
                        {{item.formType}}
                    </p>
                </td>
                <td>
                    <p>{{item.type}}</p>
                </td>
                <td v-for="day in Dateinterval">
                    <span :class="spanShow(item.value, day).color" v-show="spanShow(item.value, day).show"></span>
                </td>
            </tr>
        </tbody>
    </table>
</div>

javascript 代码

<script>
    /**
     * 日期格式化。
     * 小于 10 的数字在前面 + 0。如 01、06
     * */
    var fDate = function (value) {
        return value < 10 ? '0' + value : value;
    }
    /**
     * 日期格式化:yyyy-mm-dd
     * */
    Date.prototype.format = function () {
        return this.getFullYear() + '-' + fDate(this.getMonth() + 1) + '-' + fDate(this.getDate());
    };

    /**
     * 获取指定时间段内所有天。
     * 输入起止日期。格式:yyyy-mm-dd。
     * 返回格式:yyyy-mm-dd
     * */
    var getAllDay = function (startDate, endDate) {
        var allDays = [];
        var a = startDate.split('-');
        var b = endDate.split('-');
        var uDb = new Date().setUTCFullYear(a[0], a[1] - 1, a[2]); //start
        var uDe = new Date().setUTCFullYear(b[0], b[1] - 1, b[2]); //end
        for (var i = uDb; i <= uDe; i = i + 24 * 60 * 60 * 1000) {
            allDays.push(new Date(parseInt(i)).format());
        }
        return allDays;
    };
    /**
     * 获取指定时间段内所有月。
     * 输入起止日期。格式:yyyy-mm-dd。
     * 返回格式:yyyy-mm
     * */
    var getAllMonth = function (startDate, endDate) {
        var allMonth = [];
        var a = startDate.split('-');
        var b = endDate.split('-');
        var uDb = new Date().setUTCFullYear(a[0], a[1] - 1, a[2]); //start
        var uDe = new Date().setUTCFullYear(b[0], b[1] - 1, new Date(b[0], b[1], 0).getDate()); //结束月份设置为指定月最后一天
        while (uDb <= uDe) { // 等于号防止 31 号计算遗漏
            allMonth.push(new Date(uDb).getFullYear() + '-' + fDate(new Date(uDb).getMonth() + 1));
            uDb = new Date(uDb).setMonth(new Date(uDb).getMonth() + 1);
        }
        return allMonth;
    };
</script>
<script>
    /**
     * 创建 Vue 实例
     * */
    var demo = new Vue({
        el: '#demo',
        data: {
            /**
             * 数据源 JSON 对象数组
             * 格式:
             * {
             *  startDate:'',        //开始日期(表头所用)
             *  endDate:''           //结束日期(表头所用)
             *  data:[{
             *      formType:'',        //数据类型1。可自动进行行合并
             *      type:'',            //数据类型2。如需行合并需参照类型1自行优化
             *      value:[{            //甘特图所需数据
             *              fromDate:'',    //开始时间
             *              toDate:'',      //截止时间
             *              spanClass:''    //该段时间内的样式
             *          },{
             *              //允许有多组时间段。
             *          }]
             *      }]
             *  }
             * 
             * */
            jsonData: sourceData.data,
            startDate: sourceData.startDate,
            endDate: sourceData.endDate,
            /*获取指定时间段内所有月,用作表头一*/
            AllMonths: getAllMonth(sourceData.startDate, sourceData.endDate),
            /*获取指定时间段内所有天,用作表头二*/
            Dateinterval: getAllDay(sourceData.startDate, sourceData.endDate),
        },
        methods: {
            /**
             * 甘特图显示状况
             * 传入参数:
             *      1、甘特图所需数据数组。即上述 JSON 对象的 value 数组
             *      2、当前日期,即此刻对应表头中的日期列。
             *  */
            spanShow: function (valArray, currentDate) {
                /*默认不显示*/
                var show = false;
                /*默认样式颜色为空*/
                var color = "";
                /*调用 Vue 过滤器*/
                var filter = this.$options.filters['strDateToTimeStamp'];
                /*循环 value 数据。判断当前对应日期是否位于该组 value 数据的 fromdate 与 todate 之间。*/
                for (var i = 0; i < valArray.length; i++) {
                    var inInterval = false;
                    inInterval = filter(valArray[i].fromDate) <= filter(currentDate) && filter(
                        currentDate) <= filter(valArray[i].toDate)
                    if (inInterval) {
                        color = valArray[i].spanClass;
                    }
                    show = show || inInterval;
                }
                /*返回判断结果与当前日期在该组数据下对应的样式颜色*/
                return {
                    show: show,
                    color: color
                };
            },
            /**
             * 计算指定时间段内指定月份的总天数。
             * 这里需要根据指定月的总天数确定合并表头一时跨列的数量
             * 传入参数:指定月份。格式 yyyy-mm
             * */
            days: function (strDate) {
                debugger;
                var sArr = this.startDate.split('-');
                var eArr = this.endDate.split('-');
                var arr = strDate.split('-');
                //指定月份不在指定时间段内
                if ('' + arr[0] + arr[1] < '' + sArr[0] + sArr[1] || '' + arr[0] + arr[1] > '' + eArr[0] +
                    eArr[1]) {
                    return 0;
                }
                //指定月份等于开始时间段的月份时,返回本月剩余天数 = 本月最后一天 - 开始日期 + 1
                if ('' + arr[0] + arr[1] === '' + sArr[0] + sArr[1]) {
                    return new Date(arr[0], arr[1], 0).getDate() - new Date(sArr[0] + '-' + sArr[1] + '-' +
                            sArr[2])
                        .getDate() + 1;
                }
                //指定月份等于结束时间段的月份时,返回时间段内本月有效天数 = 结束日期
                if ('' + arr[0] + arr[1] === '' + eArr[0] + eArr[1]) {
                    return new Date(eArr[0] + '-' + eArr[1] + '-' + eArr[2]).getDate()
                }
                return new Date(arr[0], arr[1], 0).getDate(); //取当前月的最后一天,即本月天数
            },
            /**
             * 表单类型显示状态。
             * 即第一列。
             * */
            showFormType: function (item, index) {
                /*与下一组数据的类型相同时,不显示*/
                var show = true;
                if (index > 0 && item.formType == this.jsonData[index - 1]
                    .formType) {
                    show = false;
                }
                return show;
            },
            /**
             * 表单类型需要跨行的数量。
             * 用于自动合并第一列的类型
             * */
            colNum: function (item, index) {
                var iCount = 0;
                if (!(index > 0 && item.formType == this.jsonData[index - 1]
                        .formType)) {
                    this.jsonData.forEach(ele => {
                        if (item.formType == ele.formType) {
                            iCount++;
                        }
                    });
                }
                return iCount;
            }
        },
        /*Vue 过滤器用于格式化日期*/
        filters: {
            /*字符串格式的日期转为时间戳*/
            strDateToTimeStamp: function (strDate) {
                return Date.parse(strDate);
            },
            /*字符串格式的日期转为天。yyyy-mm-dd  -->  dd*/
            longDateToDay: function (longDate) {
                return fDate(new Date(Date.parse(longDate)).getDate());
            },
            /*字符串格式的日期转为月。yyyy-mm-dd  -->  mm*/
            longDateToMonth: function (longDate) {
                return fDate(new Date(Date.parse(longDate)).getMonth() + 1);
            }
        }
    })
</script>

JSON 实例数据(为了方便,直接写在JS中)

<script>
        var sourceData = {
        'startDate': '2018-12-01',
        'endDate': '2019-01-31',
        'data': [{
                "formType": "DFM",
                "type": "计划",
                "value": [{
                    "fromDate": "2018-12-01",
                    "toDate": "2018-12-06",
                    "spanClass": "green"
                }, {
                    "fromDate": "2018-12-10",
                    "toDate": "2018-12-11",
                    "spanClass": "green"
                }, {
                    "fromDate": "2018-12-15",
                    "toDate": "2018-12-16",
                    "spanClass": "green"
                }]
            },
            {
                "formType": "DFM",
                "type": "实际",
                "value": [{
                    "fromDate": "2018-12-01",
                    "toDate": "2018-12-07",
                    "spanClass": "red"
                }, {
                    "fromDate": "2018-12-10",
                    "toDate": "2018-12-11",
                    "spanClass": "blue"
                }, {
                    "fromDate": "2018-12-15",
                    "toDate": "2018-12-16",
                    "spanClass": "blue"
                }]
            }, {
                "formType": "DFM",
                "type": "客户回复",
                "value": [{
                    "fromDate": "2018-12-09",
                    "toDate": "2018-12-09",
                    "spanClass": "purple"
                }, {
                    "fromDate": "2018-12-14",
                    "toDate": "2018-12-14",
                    "spanClass": "purple"
                }, {
                    "fromDate": "2018-12-18",
                    "toDate": "2018-12-18",
                    "spanClass": "purple"
                }]
            },

            {
                "formType": "开模检讨",
                "type": "实际",
                "value": [{
                    "fromDate": "2018-12-20",
                    "toDate": "2018-12-20",
                    "spanClass": "yellow"
                }]
            },

            {
                "formType": "模具开发",
                "type": "计划",
                "value": [{
                    "fromDate": "2018-12-19",
                    "toDate": "2019-01-10",
                    "spanClass": "green"
                }]
            },
            {
                "formType": "模具开发",
                "type": "实际",
                "value": [{
                    "fromDate": "2018-12-19",
                    "toDate": "2019-01-13",
                    "spanClass": "red"
                }]
            }, {
                "formType": "设计",
                "type": "计划",
                "value": [{
                    "fromDate": "2018-12-19",
                    "toDate": "2018-12-21",
                    "spanClass": "green"
                }]
            }, {
                "formType": "设计",
                "type": "实际",
                "value": [{
                    "fromDate": "2018-12-19",
                    "toDate": "2018-12-21",
                    "spanClass": "yellow"
                }]
            }, {
                "formType": "加工",
                "type": "计划",
                "value": [{
                    "fromDate": "2018-12-22",
                    "toDate": "2019-01-06",
                    "spanClass": "green"
                }]
            },
            {
                "formType": "加工",
                "type": "客户回复",
                "value": [{
                    "fromDate": "2018-12-22",
                    "toDate": "2019-01-08",
                    "spanClass": "purple"
                }]
            },
            {
                "formType": "加工",
                "type": "实际",
                "value": [{
                    "fromDate": "2018-12-22",
                    "toDate": "2019-01-09",
                    "spanClass": "red"
                }]
            },
        ],
    };
</script>