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

相关文章

Vue和React受控组件的区别小结

《Vue和React受控组件的区别小结》本文主要介绍了Vue和React受控组件的区别小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录背景React 的实现vue3 的实现写法一:直接修改事件参数写法二:通过ref引用 DOMVu

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

C#使用Spire.Doc for .NET实现HTML转Word的高效方案

《C#使用Spire.Docfor.NET实现HTML转Word的高效方案》在Web开发中,HTML内容的生成与处理是高频需求,然而,当用户需要将HTML页面或动态生成的HTML字符串转换为Wor... 目录引言一、html转Word的典型场景与挑战二、用 Spire.Doc 实现 HTML 转 Word1

Vue3绑定props默认值问题

《Vue3绑定props默认值问题》使用Vue3的defineProps配合TypeScript的interface定义props类型,并通过withDefaults设置默认值,使组件能安全访问传入的... 目录前言步骤步骤1:使用 defineProps 定义 Props步骤2:设置默认值总结前言使用T

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

MyBatis ParameterHandler的具体使用

《MyBatisParameterHandler的具体使用》本文主要介绍了MyBatisParameterHandler的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参... 目录一、概述二、源码1 关键属性2.setParameters3.TypeHandler1.TypeHa

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完