1、VUE页面甘特图实现效果


2、甘特图实现过程
复制
复制

复制
| |
| import 'dhtmlx-gantt/codebase/dhtmlxgantt.css'; |
| import gantt from 'dhtmlx-gantt'; |
复制
3、甘特图实现效果
复制
| |
| init() { |
| |
| gantt.init("gantt-chart"); |
| gantt.config.grid_width = 600; |
| gantt.config.min_grid_column_width = 120; |
| gantt.config.scale_height = 80; |
| gantt.config.row_height = 35; |
| |
| |
| |
| gantt.config.readonly = true; |
| gantt.config.fit_tasks = true; |
| gantt.config.autofit = true; |
| gantt.config.show_quick_info = true; |
| |
| gantt.config.work_time = true; |
| |
| gantt.templates.timeline_cell_class = function (task, date) { |
| if (!gantt.isWorkTime({ task: task, date: date })) return "weekend"; |
| }; |
| |
| gantt.templates.scale_cell_class = function (date) { |
| if (!gantt.isWorkTime(date)) return "weekend"; |
| }; |
| |
| |
| gantt.config.start_date = new Date(2020, 1, 1); |
| |
| |
| gantt.config.date_format = "%Y-%m-%d"; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.date_scale = "%Y年 %m月"; |
| |
| |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, date: "%d" } |
| ]; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, template: "%d日" } |
| ]; |
| |
| |
| gantt.config.columns = [ |
| { |
| name: "text", label: "生产计划单号", width: 80, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${task.text}</div>`; |
| } |
| }, |
| { |
| name: "start_date", label: "开始时间", width: 50, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${gantt.date.date_to_str("%Y-%m-%d")(task.start_date)}</div>`; |
| } |
| }, |
| { |
| name: "end_date", label: "结束时间", width: 50, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${gantt.date.date_to_str("%Y-%m-%d")(task.end_date)}</div>`; |
| } |
| }, |
| { |
| name: "duration", label: "任务天数", width: 50, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${task.duration}</div>`; |
| } |
| } |
| |
| ]; |
| |
| gantt.plugins({ |
| tooltip: true, |
| }); |
| |
| gantt.templates.tooltip_text = function (start, end, task) { |
| |
| return ` |
| <div style="display: flex; flex-wrap: wrap; align-items: center; width: 200px;"> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">工单编号:${task.text}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品编码:${task.productCode}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品名称:${task.productName}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品数量:${task.productQty}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">完工数量:${task.finishQty}</div> |
| <div style="width: 100%; line-height: 18px;">持续时间:${task.duration} 天</div> |
| <div style="width: 100%; line-height: 18px;">开始时间:${gantt.date.date_to_str('%Y-%m-%d')(task.start_date)}</div> |
| <div style="width: 100%; line-height: 18px;">结束时间:${gantt.date.date_to_str('%Y-%m-%d')(task.end_date)}</div> |
| </div> |
| `; |
| }; |
| |
| gantt.init("gantt-chart"); |
| }, |
复制
复制
| |
| getProductData() { |
| this.http.post('/api/Production_ProductPlan/接口方法名', { |
| }).then((response) => { |
| console.log("甘特图数据源:", response); |
| |
| if (response.status == true) { |
| const data = response.data; |
| console.log("数据源:", response.data); |
| |
| const formattedData = data.result.map(item => ({ |
| id: item.productPlan_Id, |
| text: item.productPlanCode, |
| start_date: new Date(item.planStartDate), |
| end_date: new Date(item.planEndDate), |
| duration: 0, |
| documentStatus: item.documentStatus, |
| productCode: item.productCode, |
| productName: item.productName, |
| productQty: item.qty, |
| finishQty: item.finishQty |
| })); |
| |
| this.ganttBarColor(formattedData); |
| |
| |
| gantt.templates.task_text = function (start, end, task) { |
| return "" + task.text; |
| }; |
| |
| |
| gantt.parse({ data: formattedData }); |
| } else { |
| this.$message.error('生产计划列表查询失败,请联系管理员!'); |
| } |
| }); |
| }, |
复制
复制
| |
| filterTasks() { |
| const searchOrder = this.searchOrder; |
| const searchStartDate = this.searchStartDate; |
| const searchEndDate = this.searchEndDate; |
| |
| gantt.clearAll(); |
| |
| |
| this.http.post('/api/Production_ProductPlan/selectProductData', { |
| searchOrder: searchOrder, |
| searchStartDate: searchStartDate, |
| searchEndDate: searchEndDate |
| }).then((response) => { |
| console.log("搜索甘特图数据源:", response); |
| if (response.status == true) { |
| const data = response.data; |
| |
| const formattedData = data.result.map(item => ({ |
| id: item.productPlan_Id, |
| text: item.productPlanCode, |
| start_date: new Date(item.planStartDate), |
| end_date: new Date(item.planEndDate), |
| duration: 0, |
| documentStatus: item.documentStatus, |
| productCode: item.productCode, |
| productName: item.productName, |
| productQty: item.qty, |
| finishQty: item.finishQty |
| })); |
| |
| this.ganttBarColor(formattedData); |
| |
| |
| gantt.parse({ data: formattedData }); |
| |
| } else { |
| this.$message.error('生产计划列表查询失败,请联系管理员!'); |
| } |
| }); |
| |
| }, |
复制
复制
| |
| ganttBarColor(formattedData) { |
| |
| formattedData.forEach(task => { |
| if (task.documentStatus === 0) { |
| task.color = "#0033ff"; |
| } else if (task.documentStatus === 1) { |
| task.color = "#ff6633"; |
| } else if (task.documentStatus === 2) { |
| task.color = "#006600"; |
| } else if (task.documentStatus === 3) { |
| task.color = "#ff3399"; |
| } else if (task.documentStatus === 4) { |
| task.color = "#ff66ff"; |
| } else { |
| task.color = "#3366ff"; |
| } |
| }); |
| }, |
复制
复制
| |
| changeScale(unit) { |
| |
| if (unit === 'day') { |
| |
| gantt.config.date_format = "%Y-%m-%d"; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.date_scale = "%Y年 %m月"; |
| |
| |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, date: "%d" } |
| ]; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, template: "%d日" } |
| ]; |
| } else if (unit === 'week') { |
| |
| gantt.config.date_format = "%Y-%m-%d"; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.date_scale = "%Y年 %m月"; |
| |
| gantt.config.subscales = [{ |
| unit: 'week', |
| step: 1, |
| template: this.weekScaleTemplate |
| }]; |
| } |
| this.scale = unit; |
| gantt.render(); |
| }, |
| |
| |
| weekScaleTemplate(date) { |
| var dateToStr = gantt.date.date_to_str("%m/%d"); |
| var endDate = gantt.date.add(gantt.date.add(date, 1, "week"), -1, "day"); |
| |
| return `<div style="text-align: center; font-size: 11px; color: black; white-space: nowrap;"> |
| ${dateToStr(date) + "~" + dateToStr(endDate)} |
| </div>`; |
| }, |
| getDay(day) { |
| var today = new Date(); |
| var targetday_milliseconds = today.getTime() + 1000 * 60 * 60 * 24 * day; |
| today.setTime(targetday_milliseconds); |
| var tYear = today.getFullYear(); |
| var tMonth = today.getMonth(); |
| var tDate = today.getDate(); |
| tMonth = this.doHandleMonth(tMonth + 1); |
| tDate = this.doHandleMonth(tDate); |
| return tYear + "-" + tMonth + "-" + tDate; |
| }, |
| doHandleMonth(month) { |
| var m = month; |
| if (month.toString().length == 1) { |
| m = "0" + month; |
| } |
| return m; |
| } |
复制
复制
| gantt.plugins({ |
| tooltip: true, |
| }); |
| |
| gantt.templates.tooltip_text = function (start, end, task) { |
| |
| return ` |
| <div style="display: flex; flex-wrap: wrap; align-items: center; width: 200px;"> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">工单编号:${task.text}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品编码:${task.productCode}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品名称:${task.productName}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品数量:${task.productQty}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">完工数量:${task.finishQty}</div> |
| <div style="width: 100%; line-height: 18px;">持续时间:${task.duration} 天</div> |
| <div style="width: 100%; line-height: 18px;">开始时间:${gantt.date.date_to_str('%Y-%m-%d')(task.start_date)}</div> |
| <div style="width: 100%; line-height: 18px;">结束时间:${gantt.date.date_to_str('%Y-%m-%d')(task.end_date)}</div> |
| </div> |
| `; |
| }; |
复制
4、甘特图功能扩展:实现拖拉拽等效果
| 更多扩展功能及甘特图属性详细介绍参考CSDN博主:https: |
复制
5、甘特图源码(数据源进行替换,复制即可使用)
| <template> |
| <div> |
| |
| <div class="search-bar"> |
| |
| 生产计划单号: |
| <el-input v-model="searchOrder" placeholder="输入单号搜索" style="width: 200px; margin-right: 10px;" clearable /> |
| |
| |
| 开始时间: |
| <el-date-picker v-model="searchStartDate" type="date" placeholder="选择开始日期" |
| style="width: 180px; margin-right: 10px;" /> |
| |
| |
| 结束时间: |
| <el-date-picker v-model="searchEndDate" type="date" placeholder="选择结束日期" |
| style="width: 180px; margin-right: 10px;" /> |
| |
| |
| <el-button type="primary" @click="filterTasks" style="width: 100px;">搜索</el-button> |
| |
| <div style="float: right; margin-left: 35%;"> |
| <el-button :class="{ active: scale === 'day' }" @click="changeScale('day')" |
| style="margin-right: 3px;">日视图</el-button> |
| <el-button :class="{ active: scale === 'week' }" @click="changeScale('week')">周视图</el-button> |
| </div> |
| </div> |
| |
| |
| <div id="gantt-chart" style="width: 100%; height: 800px;"></div> |
| </div> |
| </template> |
| |
| <script> |
| |
| import 'dhtmlx-gantt/codebase/dhtmlxgantt.css'; |
| import gantt from 'dhtmlx-gantt'; |
| import { color } from 'echarts'; |
| |
| export default { |
| name: 'GanttChart', |
| data() { |
| return { |
| searchOrder: '', |
| searchStartDate: '', |
| searchEndDate: '', |
| scale: "day", |
| tasks: [] |
| }; |
| }, |
| watch() { }, |
| mounted() { |
| |
| this.init(); |
| |
| this.getProductData(); |
| |
| }, |
| methods: { |
| |
| init() { |
| |
| gantt.init("gantt-chart"); |
| gantt.config.grid_width = 600; |
| gantt.config.min_grid_column_width = 120; |
| gantt.config.scale_height = 80; |
| gantt.config.row_height = 35; |
| |
| |
| |
| gantt.config.readonly = true; |
| gantt.config.fit_tasks = true; |
| gantt.config.autofit = true; |
| gantt.config.show_quick_info = true; |
| |
| gantt.config.work_time = true; |
| |
| gantt.templates.timeline_cell_class = function (task, date) { |
| if (!gantt.isWorkTime({ task: task, date: date })) return "weekend"; |
| }; |
| |
| gantt.templates.scale_cell_class = function (date) { |
| if (!gantt.isWorkTime(date)) return "weekend"; |
| }; |
| |
| |
| gantt.config.start_date = new Date(2020, 1, 1); |
| |
| |
| gantt.config.date_format = "%Y-%m-%d"; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.date_scale = "%Y年 %m月"; |
| |
| |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, date: "%d" } |
| ]; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, template: "%d日" } |
| ]; |
| |
| |
| gantt.config.columns = [ |
| { |
| name: "text", label: "生产计划单号", width: 80, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${task.text}</div>`; |
| } |
| }, |
| { |
| name: "start_date", label: "开始时间", width: 50, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${gantt.date.date_to_str("%Y-%m-%d")(task.start_date)}</div>`; |
| } |
| }, |
| { |
| name: "end_date", label: "结束时间", width: 50, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${gantt.date.date_to_str("%Y-%m-%d")(task.end_date)}</div>`; |
| } |
| }, |
| { |
| name: "duration", label: "任务天数", width: 50, template: function (task) { |
| return `<div style="text-align: center; font-size: 14px; color: black;">${task.duration}</div>`; |
| } |
| } |
| |
| ]; |
| |
| gantt.plugins({ |
| tooltip: true, |
| }); |
| |
| gantt.templates.tooltip_text = function (start, end, task) { |
| |
| return ` |
| <div style="display: flex; flex-wrap: wrap; align-items: center; width: 200px;"> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">工单编号:${task.text}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品编码:${task.productCode}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品名称:${task.productName}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">产品数量:${task.productQty}</div> |
| <div style="width: 100%; line-height: 18px; font-weight: bold;">完工数量:${task.finishQty}</div> |
| <div style="width: 100%; line-height: 18px;">持续时间:${task.duration} 天</div> |
| <div style="width: 100%; line-height: 18px;">开始时间:${gantt.date.date_to_str('%Y-%m-%d')(task.start_date)}</div> |
| <div style="width: 100%; line-height: 18px;">结束时间:${gantt.date.date_to_str('%Y-%m-%d')(task.end_date)}</div> |
| </div> |
| `; |
| }; |
| |
| gantt.init("gantt-chart"); |
| }, |
| |
| getProductData() { |
| this.http.post('/api/Production_ProductPlan/后端接口数据源', { |
| }).then((response) => { |
| console.log("甘特图数据源:", response); |
| |
| if (response.status == true) { |
| const data = response.data; |
| console.log("数据源:", response.data); |
| |
| const formattedData = data.result.map(item => ({ |
| id: item.productPlan_Id, |
| text: item.productPlanCode, |
| start_date: new Date(item.planStartDate), |
| end_date: new Date(item.planEndDate), |
| duration: 0, |
| documentStatus: item.documentStatus, |
| productCode: item.productCode, |
| productName: item.productName, |
| productQty: item.qty, |
| finishQty: item.finishQty |
| })); |
| |
| this.ganttBarColor(formattedData); |
| |
| |
| gantt.templates.task_text = function (start, end, task) { |
| return "" + task.text; |
| }; |
| |
| |
| gantt.parse({ data: formattedData }); |
| } else { |
| this.$message.error('生产计划列表查询失败,请联系管理员!'); |
| } |
| }); |
| }, |
| |
| |
| filterTasks() { |
| const searchOrder = this.searchOrder; |
| const searchStartDate = this.searchStartDate; |
| const searchEndDate = this.searchEndDate; |
| |
| gantt.clearAll(); |
| |
| |
| this.http.post('/api/Production_ProductPlan/后端接口数据源', { |
| searchOrder: searchOrder, |
| searchStartDate: searchStartDate, |
| searchEndDate: searchEndDate |
| }).then((response) => { |
| console.log("搜索甘特图数据源:", response); |
| if (response.status == true) { |
| const data = response.data; |
| |
| const formattedData = data.result.map(item => ({ |
| id: item.productPlan_Id, |
| text: item.productPlanCode, |
| start_date: new Date(item.planStartDate), |
| end_date: new Date(item.planEndDate), |
| duration: 0, |
| documentStatus: item.documentStatus, |
| productCode: item.productCode, |
| productName: item.productName, |
| productQty: item.qty, |
| finishQty: item.finishQty |
| })); |
| |
| this.ganttBarColor(formattedData); |
| |
| |
| gantt.parse({ data: formattedData }); |
| |
| } else { |
| this.$message.error('生产计划列表查询失败,请联系管理员!'); |
| } |
| }); |
| |
| }, |
| |
| ganttBarColor(formattedData) { |
| |
| formattedData.forEach(task => { |
| if (task.documentStatus === 0) { |
| task.color = "#0033ff"; |
| } else if (task.documentStatus === 1) { |
| task.color = "#ff6633"; |
| } else if (task.documentStatus === 2) { |
| task.color = "#006600"; |
| } else if (task.documentStatus === 3) { |
| task.color = "#ff3399"; |
| } else if (task.documentStatus === 4) { |
| task.color = "#ff66ff"; |
| } else { |
| task.color = "#3366ff"; |
| } |
| }); |
| }, |
| |
| changeScale(unit) { |
| |
| if (unit === 'day') { |
| |
| gantt.config.date_format = "%Y-%m-%d"; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.date_scale = "%Y年 %m月"; |
| |
| |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, date: "%d" } |
| ]; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.subscales = [ |
| { unit: "day", step: 1, template: "%d日" } |
| ]; |
| } else if (unit === 'week') { |
| |
| gantt.config.date_format = "%Y-%m-%d"; |
| |
| gantt.config.scale_unit = "month"; |
| gantt.config.date_scale = "%Y年 %m月"; |
| |
| gantt.config.subscales = [{ |
| unit: 'week', |
| step: 1, |
| template: this.weekScaleTemplate |
| }]; |
| } |
| this.scale = unit; |
| gantt.render(); |
| }, |
| |
| |
| weekScaleTemplate(date) { |
| var dateToStr = gantt.date.date_to_str("%m/%d"); |
| var endDate = gantt.date.add(gantt.date.add(date, 1, "week"), -1, "day"); |
| |
| return `<div style="text-align: center; font-size: 11px; color: black; white-space: nowrap;"> |
| ${dateToStr(date) + "~" + dateToStr(endDate)} |
| </div>`; |
| }, |
| getDay(day) { |
| var today = new Date(); |
| var targetday_milliseconds = today.getTime() + 1000 * 60 * 60 * 24 * day; |
| today.setTime(targetday_milliseconds); |
| var tYear = today.getFullYear(); |
| var tMonth = today.getMonth(); |
| var tDate = today.getDate(); |
| tMonth = this.doHandleMonth(tMonth + 1); |
| tDate = this.doHandleMonth(tDate); |
| return tYear + "-" + tMonth + "-" + tDate; |
| }, |
| doHandleMonth(month) { |
| var m = month; |
| if (month.toString().length == 1) { |
| m = "0" + month; |
| } |
| return m; |
| } |
| |
| } |
| }; |
| </script> |
| |
| |
| <style scoped> |
| #app { |
| font-family: Arial, Helvetica, sans-serif; |
| text-align: center; |
| } |
| |
| #gantt-chart { |
| width: 100%; |
| height: 600px; |
| } |
| |
| |
| .search-bar { |
| display: flex; |
| align-items: center; |
| margin-left: 20px; |
| margin-top: 10px; |
| |
| margin-bottom: 10px; |
| } |
| |
| .search-bar input, |
| .search-bar button { |
| padding: 5px; |
| margin: 0 5px; |
| font-size: 14px; |
| border: 1px solid #ccc; |
| border-radius: 4px; |
| } |
| |
| .search-bar button { |
| cursor: pointer; |
| background-color: #007bff; |
| color: white; |
| border: none; |
| } |
| |
| .search-bar button:hover { |
| background-color: #0056b3; |
| } |
| |
| |
| .gantt_scale .gantt_scale_line { |
| line-height: 200px; |
| |
| } |
| |
| |
| .gantt_grid_head { |
| height: 50px; |
| |
| } |
| |
| |
| .gantt_grid_head_item { |
| height: 50px; |
| |
| line-height: 50px; |
| |
| } |
| |
| |
| .gantt_task_scale { |
| color: #000000 !important; |
| } |
| |
| .weekend { |
| background: #f4f7f4 !important; |
| } |
| |
| |
| .el-button { |
| background-color: rgb(57, 216, 110); |
| color: black; |
| border: 1px solid #dcdfe6; |
| } |
| |
| |
| .el-button.active { |
| background-color: #409EFF; |
| |
| color: rgb(255, 255, 255); |
| border-color: #409EFF; |
| } |
| |
| |
| .el-button:not(.active):hover { |
| background-color: #00ff22; |
| |
| border-color: #91d5ff; |
| |
| } |
| </style> |
| |
复制