Vue 甘特图 gantt 安装使用

2024-01-31 17:36

本文主要是介绍Vue 甘特图 gantt 安装使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Vue 甘特图 gantt 安装使用

(gantt-elastic)参考文章@shenjuncaci

(dhtmlx)参考文章@秃头的铲屎官

dhtmlx@官方文档

安装

npm i dhtmlx-gantt

使用

创建一个容器

  <div ref="gantt" class="gantt-container"></div>

引入依赖

import { gantt } from "dhtmlx-gantt";
import "dhtmlx-gantt/codebase/dhtmlxgantt.css";

初始化及数据解析

        //  初始化gantt.init(this.$refs.gantt)//  数据解析gantt.parse(this.tasks)

其他配置可根据自己需求参考官网添加。

dhtmlx 实操案例 (简易 demo)

<template><div class="container"><div class="select-wrap"><el-select v-model="value" placeholder="请选择" @change="selectChange"><el-optionv-for="item in options":key="item.value":label="item.label":value="item.value"></el-option></el-select></div><div ref="gantt" class="gantt-container"></div></div>
</template>
<script>import { gantt } from "dhtmlx-gantt";import "dhtmlx-gantt/codebase/dhtmlxgantt.css";export default {name: "gantt",data() {return {tasks: {data: [],},options: [{value: "1",label: "全部",},{value: "2",label: "完成",},{value: "3",label: "正常",},{value: "4",label: "异常",},{value: "5",label: "未启动",},],value: "1",};},methods: {//开始时间-结束时间参数DateDifference: function (strDateStart, strDateEnd) {var begintime_ms = Date.parse(new Date(strDateStart.replace(/-/g, "/"))); //begintime 为开始时间var endtime_ms = Date.parse(new Date(strDateEnd.replace(/-/g, "/"))); // endtime 为结束时间var date3 = endtime_ms - begintime_ms; //时间差的毫秒数var days = Math.floor(date3 / (24 * 3600 * 1000));return days;},initData: function () {this.tasks.data = [{id: 1,text: "概念设计",start_date: "2020-04-08",duration: 10,open: true, //默认打开,toolTipsTxt: "xxx项目概念设计",progress: 0.6,status: "parent",},{toolTipsTxt: "xxx项目-项目启动会",text: "项目启动会-外部", // 任务名start_date: "2020-04-08", // 开始时间id: 11, // 任务idduration: 3, // 任务时长,从start_date开始计算parent: 1, // 父任务IDtype: 1,progress: 0.5,status: "yellow",},{toolTipsTxt: "xxx项目-项目启动会议",text: "项目启动会-内部",start_date: "2020-04-11",id: 12,duration: 2,parent: 1,type: 2,progress: 0.6,status: "pink",},{toolTipsTxt: "xxx项目开工会",text: "项目开工会",start_date: "2020-04-13",id: 13,duration: 4,parent: 1,type: 3,progress: 1,status: "green",},{toolTipsTxt: "xxx项目-项目分析",text: "项目分析",start_date: "2020-04-13",id: 14,duration: 4,parent: 1,type: 4,progress: 0.6,status: "popular",},{id: 2,text: "方案设计",start_date: "2020-04-08",duration: 8,open: true,toolTipsTxt: "xxx方案设计",state: "default",// color:"#409EFF", //设置颜色progress: 0.6,status: "parent",},{toolTipsTxt: "xxx新项目原型图设计",text: "原型图设计",start_date: "2020-04-08",id: 21,duration: 2,parent: 2,type: 1,progress: 0.6,status: "yellow",},{toolTipsTxt: "xxx项目-项目设计图",text: "设计图设计",start_date: "2020-04-09",id: 22,duration: 2,parent: 2,type: 2,progress: 0.6,status: "pink",},{toolTipsTxt: "xxx项目-项目确认",text: "项目确认",start_date: "2020-04-11",id: 23,duration: 2,parent: 2,type: 3,progress: 1,status: "green",},].map(function (current, ind, arry) {var newObj = {};if (current.type) {//存在type字段 说明非一级菜单,判断阶段的具体类型 设置不同颜色if (current.type == 1) {//冒烟newObj = Object.assign({}, current, {color: "#fcca02",});} else if (current.type == 2) {//单元newObj = Object.assign({}, current, {color: "#fec0dc",});} else if (current.type == 3) {//回归newObj = Object.assign({}, current, {color: "#62ddd4",});} else if (current.type == 4) {newObj = Object.assign({}, current, {color: "#d1a6ff",});}} else {//一级菜单是蓝色的newObj = Object.assign({}, current, {color: "#5692f0",});}return newObj;});},selectChange(val) {console.log(val);//测试用例var obj = {toolTipsTxt: "新增任务",text: "新增任务", // 任务名start_date: "2020-04-15", // 开始时间id: 24, // 任务idduration: 2, // 任务时长,从start_date开始计算parent: 2, // 父任务IDtype: 4,progress: 0,status: "popular",};this.tasks.data.push(obj);// 数据解析gantt.parse(this.tasks);// 刷新数据gantt.refreshData();},},mounted() {this.initData();//自适应甘特图的尺寸大小, 使得在不出现滚动条的情况下, 显示全部任务gantt.config.autosize = true;//只读模式gantt.config.readonly = true;//是否显示左侧树表格gantt.config.show_grid = true;//表格列设置gantt.config.columns = [{name: "text",label: "阶段名字",tree: true,width: "280",onrender: function (task, node) {node.setAttribute("class","gantt_cell gantt_last_cell gantt_cell_tree " + task.status);},},{name: "duration",label: "时长",align: "center",template: function (obj) {return obj.duration + "天";},hide: true,},];var weekScaleTemplate = function (date) {var dateToStr = gantt.date.date_to_str("%m %d");var endDate = gantt.date.add(gantt.date.add(date, 1, "week"),-1,"day");var weekNum = gantt.date.date_to_str("第 %W 周");return weekNum(date);};var daysStyle = function (date) {var dateToStr = gantt.date.date_to_str("%D");if (dateToStr(date) == "六" || dateToStr(date) == "日")return "weekend";return "";};gantt.config.subscales = [{unit: "week",step: 1,template: weekScaleTemplate,},{unit: "day",step: 1,format: "%d",},];gantt.plugins({tooltip: true,});gantt.attachEvent("onGanttReady", function () {var tooltips = gantt.ext.tooltips;gantt.templates.tooltip_text = function (start, end, task) {return (task.toolTipsTxt +"<br/>" +"阶段:" +task.text +"<br/>" +gantt.templates.tooltip_date_format(start));};});//设置任务条进度内容gantt.templates.progress_text = function (start, end, task) {return ("<div style='text-align:left;color:#fff;padding-left:20px'>" +Math.round(task.progress * 100) +"% </div>");};//任务条显示内容gantt.templates.task_text = function (start, end, task) {// return task.text + '(' + task.duration + '天)';return ("<div style='text-align:center;color:#fff'>" +task.text +"(" +task.duration +"天)" +"</div>");};// gantt.templates.scale_cell_class = function(date) {//     /*if(date.getDay()== 0 || date.getDay()== 6){//       return "weekend";//     }*///     return 'weekend'// }//任务栏周末亮色/*gantt.templates.task_cell_class = function(item,date){if(date.getDay()== 0 || date.getDay()== 6){return "weekend";}};*///任务条上的文字大小 以及取消border自带样式gantt.templates.task_class = function (start, end, item) {return item.$level == 0 ? "firstLevelTask" : "secondLevelTask";};gantt.config.layout = {css: "gantt_container",cols: [{width: 280,min_width: 280,rows: [{view: "grid",scrollX: "gridScroll",scrollable: true,scrollY: "scrollVer",},{view: "scrollbar",id: "gridScroll",group: "horizontal",},],},{resizer: true,width: 1,},{rows: [{view: "timeline",scrollX: "scrollHor",scrollY: "scrollVer",},{view: "scrollbar",id: "scrollHor",group: "horizontal",},],},{view: "scrollbar",id: "scrollVer",},],};//时间轴图表中,任务条形图的高度// gantt.config.task_height = 28//时间轴图表中,甘特图的高度// gantt.config.row_height = 36//时间轴图表中,如果不设置,只有行边框,区分上下的任务,设置之后带有列的边框,整个时间轴变成格子状。gantt.config.show_task_cells = true;//当task的长度改变时,自动调整图表坐标轴区间用于适配task的长度gantt.config.fit_tasks = true;gantt.config.min_column_width = 50;gantt.config.auto_types = true;gantt.config.xml_date = "%Y-%m-%d";gantt.config.scale_unit = "month";gantt.config.step = 1;gantt.config.date_scale = "%Y年%M";gantt.config.start_on_monday = true;gantt.config.scale_height = 90;gantt.config.autoscroll = true;gantt.config.calendar_property = "start_date";gantt.config.calendar_property = "end_date";gantt.config.readonly = true;gantt.i18n.setLocale("cn");// 初始化gantt.init(this.$refs.gantt);// 数据解析gantt.parse(this.tasks);},};
</script>
<style lang="scss">.firstLevelTask {border: none;.gantt_task_content {font-size: 13px;}}.secondLevelTask {border: none;}.thirdLevelTask {border: 2px solid #da645d;color: #da645d;background: #da645d;}.milestone-default {border: none;background: rgba(0, 0, 0, 0.45);}.milestone-unfinished {border: none;background: #5692f0;}.milestone-finished {border: none;background: #84bd54;}.milestone-canceled {border: none;background: #da645d;}html,body {margin: 0px;padding: 0px;height: 100%;overflow: hidden;}.container {height: 100%;width: 100%;position: relative;.gantt_grid_head_cell {padding-left: 20px;text-align: left !important;font-size: 14px;color: #333;}.select-wrap {position: absolute;top: 25px;z-index: 99;width: 90px;left: 180px;.el-input__inner {border: none;}}.left-container {height: 100%;}//   .parent {//     .gantt_tree_icon {//       &.gantt_folder_open {//         background-image: url(assets/gantt-icon.svg) !important;//       }//       &.gantt_folder_closed {//         background-image: url(assets/gantt-icon-up.svg) !important;//       }//     }//   }.green,.yellow,.pink,.popular {.gantt_tree_icon.gantt_file {background: none;position: relative;&::before {content: "";width: 10px;height: 10px;border-radius: 50%;position: absolute;left: 50%;top: 50%;transform: translate(-50%, -50%);}}}.green {.gantt_tree_icon.gantt_file {&::before {background: #84bd54;}}}.yellow {.gantt_tree_icon.gantt_file {&::before {background: #fcca02;}}}.pink {.gantt_tree_icon.gantt_file {&::before {background: #da645d;}}}.popular {.gantt_tree_icon.gantt_file {&::before {background: #d1a6ff;}}}}.left-container {height: 100%;}.gantt_task_content {text-align: left;padding-left: 10px;}
</style>

踩坑记录

样式不生效问题

再给 style 标签添加了 scoped 属性后 可能会出现样式不生效的问题

解决办法

  • scss 环境使用 ::v-deep进行样式穿透
  • less 环境使用 /deep/ 进行样式穿透
  • 删掉 scoped 属性。

甘特图数据未渲染

再调用接口的数据赋值后,甘特图的数据为渲染。

可能原因:

gant 在实例的时候 还没有获取到 data 数据

解决办法:

  • 使用 async await 在获取的数据之后,再让 gant 进行实例
  • 使用 this.$nextTick(()=>{}) 再拿到最新的数据后 进行更新。
//  数据初始化处理的方法initData: function () {this.tasks.data = this.jsondata.map((item) => {return {id: item.id,text: item.workItem,start_date: transitionYmdTime(item.planStartDate),end_date: transitionYmdTime(item.planCompletedDate),open: true, // 默认打开,toolTipsTxt: item.workItem}})},
//  甘特图数据接口gantDataSearch() {let bodyData = { apqpKey: this.apqpKey }API.apqpPowerMeeting(bodyData).then((res) => {this.jsondata = res.workItemsthis.$nextTick(() => {// 获取到数据后 进行数据处理this.initData()// 使用gantt实例 进行 初始化gantt.init(this.$refs.gantt)// 使用gantt实例  数据解析gantt.parse(this.tasks)})console.log('gant数据', res)})}

这篇关于Vue 甘特图 gantt 安装使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/664542

相关文章

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Pandas使用SQLite3实战

《Pandas使用SQLite3实战》本文主要介绍了Pandas使用SQLite3实战,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学... 目录1 环境准备2 从 SQLite3VlfrWQzgt 读取数据到 DataFrame基础用法:读

JSON Web Token在登陆中的使用过程

《JSONWebToken在登陆中的使用过程》:本文主要介绍JSONWebToken在登陆中的使用过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录JWT 介绍微服务架构中的 JWT 使用结合微服务网关的 JWT 验证1. 用户登录,生成 JWT2. 自定义过滤

Linux卸载自带jdk并安装新jdk版本的图文教程

《Linux卸载自带jdk并安装新jdk版本的图文教程》在Linux系统中,有时需要卸载预装的OpenJDK并安装特定版本的JDK,例如JDK1.8,所以本文给大家详细介绍了Linux卸载自带jdk并... 目录Ⅰ、卸载自带jdkⅡ、安装新版jdkⅠ、卸载自带jdk1、输入命令查看旧jdkrpm -qa

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java使用Curator进行ZooKeeper操作的详细教程

《Java使用Curator进行ZooKeeper操作的详细教程》ApacheCurator是一个基于ZooKeeper的Java客户端库,它极大地简化了使用ZooKeeper的开发工作,在分布式系统... 目录1、简述2、核心功能2.1 CuratorFramework2.2 Recipes3、示例实践3

springboot security使用jwt认证方式

《springbootsecurity使用jwt认证方式》:本文主要介绍springbootsecurity使用jwt认证方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录前言代码示例依赖定义mapper定义用户信息的实体beansecurity相关的类提供登录接口测试提供一

go中空接口的具体使用

《go中空接口的具体使用》空接口是一种特殊的接口类型,它不包含任何方法,本文主要介绍了go中空接口的具体使用,具有一定的参考价值,感兴趣的可以了解一下... 目录接口-空接口1. 什么是空接口?2. 如何使用空接口?第一,第二,第三,3. 空接口几个要注意的坑坑1:坑2:坑3:接口-空接口1. 什么是空接

springboot security快速使用示例详解

《springbootsecurity快速使用示例详解》:本文主要介绍springbootsecurity快速使用示例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝... 目录创www.chinasem.cn建spring boot项目生成脚手架配置依赖接口示例代码项目结构启用s

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.