project_gantt.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584
  1. {{ src/App.vue }}
  2. <template>
  3. <div class="container">
  4. <div class="gantt_head">
  5. <div class="head_RorX">
  6. <el-radio-group v-model="radio1" @change="selChange()" size="small" style="margin-right:9px">
  7. <el-radio-button label="按人员查看" value="renyuan"></el-radio-button>
  8. <el-radio-button label="按项目查看" value="xiangmu"></el-radio-button>
  9. <el-radio-button label="资源需求" value="demand"></el-radio-button>
  10. </el-radio-group>
  11. </div>
  12. <!-- 时间段筛选 -->
  13. <div class="head_date">
  14. <span>时间段</span>
  15. <el-date-picker
  16. style="margin-left:9px;width:17vw"
  17. size="small"
  18. v-model="valueDate"
  19. type="daterange"
  20. range-separator="至"
  21. start-placeholder="开始日期"
  22. end-placeholder="结束日期"
  23. value-format="yyyy-MM-dd"
  24. @change="dateupdata()">
  25. </el-date-picker>
  26. </div>
  27. <!-- 任务分组筛选 -->
  28. <div v-if="reqpar1" class="head_taskgroup">
  29. <span>任务分组</span>
  30. <el-select clearable filterable v-model="valuex2" placeholder="请选择" size="small" style="margin-left:9px;width:10vw" @change="taskgroupSel()">
  31. <el-option
  32. v-for="item in taskgroupList"
  33. :key="item.name"
  34. :label="item.name"
  35. :value="item.name"
  36. >
  37. </el-option>
  38. </el-select>
  39. </div>
  40. <!-- 人员/项目筛选 -->
  41. <div class="head_select">
  42. <span>{{(this.radio1 == "按人员查看" ? "人员" : "项目")}}</span>
  43. <el-select clearable filterable v-model="valuex" placeholder="请选择" size="small" style="margin-left:9px;width:10vw" @change="optupdata()">
  44. <el-option
  45. v-for="item in screenList"
  46. :key="item.id"
  47. :label="reqpar1 ? item.projectName : item.name"
  48. :value="item.id"
  49. >
  50. <span v-if="reqpar1" style="float: left;color: #8492a6;">{{ item.projectCode }}</span>
  51. <span v-if="reqpar1" style="float: right;font-size: 13px;margin-left: 20px">{{ item.projectName }}</span>
  52. </el-option>
  53. </el-select>
  54. </div>
  55. <!-- 资源需求导入/导出 -->
  56. <div class="head_files" v-if="!isDataLoaded">
  57. <el-link type="primary" style="margin-left:10px;float:left;" :underline="false" href="./upload/资源需求导入模板.xlsx" download="资源需求导入模板.xlsx">模板下载</el-link>
  58. <el-upload ref="upload" style="margin-left:10px;float:left;" action="#" :limit="1" :http-request="importProject" :show-file-list="false">
  59. <el-link type="primary" :underline="false" >导入需求</el-link>
  60. </el-upload>
  61. <el-link type="primary" style="margin-left:10px;float:left;" :underline="false" @click="exportProjectData" download="资源需求导出.xlsx">导出需求</el-link>
  62. </div>
  63. </div>
  64. <gantt v-if="isDataLoaded" ref="ganttTable1" class="left-container" :tasks="tasks"
  65. :stafforpro="radio1"
  66. :valueDate="valueDate"
  67. :key="updatakey1"></gantt>
  68. <div class="demand-container" v-if="!isDataLoaded">
  69. <el-table height="90%" :loading="demandListLoading" :data="demandList">
  70. <el-table-column label="项目编号" prop="projectCode" width="160"></el-table-column>
  71. <el-table-column label="项目名称" prop="projectName" min-width="240"></el-table-column>
  72. <el-table-column label="近七日活跃人员" prop="activeUsers" min-width="240"></el-table-column>
  73. <el-table-column label="开始时间 - 结束时间" min-width="240">
  74. <template slot-scope="scope">
  75. <span>{{scope.row.startDate ? scope.row.startDate + ' ~ ' + scope.row.endDate : ''}}</span>
  76. </template>
  77. </el-table-column>
  78. <el-table-column label="人员需求" prop="membReq" min-width="280"></el-table-column>
  79. <el-table-column label="任务需求" prop="taskReq" min-width="280"></el-table-column>
  80. <el-table-column label="操作" width="120" align="center">
  81. <template slot-scope="scope">
  82. <el-button @click="demandEdit(scope.row)" size="small">修改</el-button>
  83. </template>
  84. </el-table-column>
  85. </el-table>
  86. <div class="poss">
  87. <el-pagination
  88. @size-change="handleSizeChange"
  89. @current-change="handleCurrentChange"
  90. :current-page="pageIndex"
  91. :page-sizes="[20, 50, 100, 200]"
  92. :page-size="20"
  93. layout="total, sizes, prev, pager, next"
  94. :total="total">
  95. </el-pagination>
  96. </div>
  97. <!-- 资源需求修改 -->
  98. <el-dialog v-if="demandEditDialog" width="500px" append-to-body :visible.sync="demandEditDialog" :title="'修改 - ' + editParameter.projectName">
  99. <el-form label-width="150">
  100. <el-form-item label="开始时间">
  101. <el-date-picker
  102. style="width:350px"
  103. v-model="editDate1"
  104. type="date"
  105. placeholder="请选择开始时间"
  106. value-format="yyyy-MM-dd"></el-date-picker>
  107. </el-form-item>
  108. <el-form-item label="结束时间">
  109. <el-date-picker
  110. style="width:350px"
  111. v-model="editDate2"
  112. type="date"
  113. placeholder="请选择结束时间"
  114. value-format="yyyy-MM-dd"></el-date-picker>
  115. </el-form-item>
  116. <el-form-item label="人员需求">
  117. <el-input style="width:350px" v-model="editParameter.membReq" clearable></el-input>
  118. </el-form-item>
  119. <el-form-item label="任务需求">
  120. <el-input style="width:350px" v-model="editParameter.taskReq" clearable></el-input>
  121. </el-form-item>
  122. </el-form>
  123. <div slot="footer" class="dialog-footer">
  124. <el-button type="default" @click="demandEditDialog = false">取消</el-button>
  125. <el-button type="primary" @click="demandEditSure" >确定</el-button>
  126. </div>
  127. </el-dialog>
  128. </div>
  129. </div>
  130. </template>
  131. <script>
  132. import { error } from 'dingtalk-jsapi';
  133. import Gantt from './gantt.vue';
  134. export default {
  135. name: 'project_gantt',
  136. components: {Gantt},
  137. data () {
  138. return {
  139. isDataLoaded:false,
  140. tasks: {
  141. data : [],
  142. links: []
  143. },
  144. tasks1: {links:[]},
  145. updatakey1: 1,
  146. updatakey2: -1,
  147. radio1:"按人员查看",
  148. valueDate:[],
  149. options:[{value:"选项1",label:"全部"},{value:"选项2",label:"人员1"}],
  150. valuex:'',
  151. screenList:[],
  152. // 请求参数
  153. reqpar1:0,
  154. reqpar2:[],
  155. demandListLoading: false,
  156. demandList: [],
  157. pageIndex: 1,
  158. pageSize: 20,
  159. demandEditDialog: false,
  160. editParameter: {},
  161. editDate1: '',
  162. editDate2: '',
  163. taskgroupList: [],
  164. valuex2: ''
  165. }
  166. },
  167. methods: {
  168. setGroup() {
  169. this.$refs.ganttTable1.setGroup();
  170. // this.$refs.ganttTable2.setGroup();
  171. },
  172. // 资源需求导出
  173. exportProjectData() {
  174. this.http.post('/project-requirement/exportData',{},
  175. res => {
  176. if (res.code == "ok") {
  177. let filePath = res.data;
  178. const a = document.createElement('a'); // 创建a标签
  179. a.setAttribute('download', '资源需求导出.xls');// download属性
  180. a.setAttribute('href', filePath);// href链接
  181. a.click(); //自执行点击事件
  182. a.remove();
  183. }
  184. },
  185. error => {
  186. this.$message({
  187. message: error,
  188. type: "error"
  189. });
  190. }
  191. );
  192. },
  193. // 资源需求导入
  194. importProject(item) {
  195. //首先判断文件类型
  196. let str = item.file.name.split(".");
  197. let format = str[str.length - 1];
  198. if (format != "xls" && format != "xlsx") {
  199. this.$message({
  200. message: "请选择.xls或.xlsx文件",
  201. type: "error"
  202. });
  203. } else {
  204. let formData = new FormData();
  205. formData.append("multipartFile", item.file);
  206. this.http.uploadFile('/project-requirement/importData', formData,
  207. res => {
  208. this.$refs.upload.clearFiles();
  209. if (res.code == "ok") {
  210. this.$message({
  211. message: "导入成功",
  212. type: "success"
  213. });
  214. this.getDemandList();
  215. } else {
  216. this.$message({
  217. message: res.msg,
  218. type: "error"
  219. });
  220. }
  221. },
  222. error => {
  223. this.$refs.upload.clearFiles();
  224. this.$message({
  225. message: error,
  226. type: "error"
  227. });
  228. });
  229. }
  230. },
  231. // 人员/项目切换
  232. selChange(){
  233. this.valuex = null
  234. // console.log("切换按钮",this.radio1);
  235. if (this.radio1 == "按人员查看") {
  236. this.isDataLoaded = true
  237. this.reqpar1 = 0
  238. this.getList()
  239. this.getScreen()
  240. }else if(this.radio1 == "按项目查看"){
  241. this.isDataLoaded = true
  242. this.reqpar1 = 1
  243. this.getList()
  244. this.getXmScreen()
  245. this.getTaskgroupList()
  246. }else {
  247. this.reqpar1 = 1
  248. this.isDataLoaded = false
  249. this.pageIndex = 1
  250. this.pageSize = 20
  251. this.valuex = ''
  252. this.valuex2 = ''
  253. this.getDemandList()
  254. this.getXmScreen()
  255. this.getTaskgroupList()
  256. }
  257. },
  258. // 时间段改变
  259. dateupdata(){
  260. this.reqpar2 = this.valueDate
  261. if(this.isDataLoaded){
  262. this.getList()
  263. }else {
  264. this.getDemandList()
  265. }
  266. },
  267. // 人员/项目筛选改变
  268. optupdata(){
  269. // console.log(this.valuex);
  270. if(this.isDataLoaded){
  271. this.getList()
  272. }else{
  273. this.getDemandList()
  274. }
  275. },
  276. // 任务分组筛选改变
  277. taskgroupSel(){
  278. // console.log(this.valuex2);
  279. if(this.isDataLoaded){
  280. this.getList()
  281. }else{
  282. this.getDemandList()
  283. }
  284. },
  285. handleSizeChange(val){
  286. this.pageSize = val
  287. this.getDemandList()
  288. },
  289. handleCurrentChange(val){
  290. this.pageIndex = val
  291. this.getDemandList()
  292. },
  293. // 资源需求修改
  294. demandEdit(row){
  295. console.log(row);
  296. this.demandEditDialog = true
  297. this.editParameter = JSON.parse(JSON.stringify(row))
  298. this.editDate1 = this.editParameter.reStartDate
  299. this.editDate2 = this.editParameter.reEndDate
  300. // console.log('edit',this.demandEditDialog);
  301. },
  302. demandEditSure(){
  303. this.editParameter.startDate = this.editDate1
  304. this.editParameter.endDate = this.editDate2
  305. delete this.editParameter.reStartDate
  306. delete this.editParameter.reEndDate
  307. this.http.post('/project-requirement/addOrMod',this.editParameter,
  308. res => {
  309. if(res.code == 'ok'){
  310. this.$message({
  311. message: '修改成功',
  312. type: 'success'
  313. })
  314. this.demandEditDialog = false
  315. this.getDemandList()
  316. }else {
  317. this.$message({
  318. message: res.msg,
  319. type: 'error'
  320. })
  321. }
  322. },err => {
  323. this.$message({
  324. message: err,
  325. type: 'error'
  326. })
  327. })
  328. },
  329. // 获取人员项目筛选列表
  330. getScreen(){
  331. this.http.get('/project/getMyUsers',
  332. res => {
  333. if (res.code == "ok") {
  334. this.screenList = res.data
  335. }else{
  336. this.$message({
  337. message: res.msg,
  338. type: "error"
  339. });
  340. }
  341. },
  342. error => {
  343. this.$message({
  344. message : error,
  345. type : "error"
  346. })
  347. }
  348. )
  349. },
  350. getXmScreen(){
  351. this.http.get('/project/getProjectList',
  352. res => {
  353. if (res.code == "ok") {
  354. this.screenList = res.data
  355. // console.log("screen",this.screenList);
  356. }else{
  357. this.$message({
  358. message: res.msg,
  359. type: "error"
  360. });
  361. }
  362. },
  363. error => {
  364. this.$message({
  365. message : error,
  366. type : "error"
  367. })
  368. }
  369. )
  370. },
  371. // 获取任务分组筛选列表
  372. getTaskgroupList(){
  373. this.http.post('/task-group/getGroupNames',{},
  374. res => {
  375. if(res.code == 'ok'){
  376. this.taskgroupList = res.data
  377. }else {
  378. this.$message({
  379. message: res.msg,
  380. type: 'error'
  381. })
  382. }
  383. },err => {
  384. this.$message({
  385. message: err,
  386. type: 'error'
  387. })
  388. })
  389. },
  390. // 获取甘特图数据
  391. getList() {
  392. let getlistcs = {type : this.reqpar1 , startDate : this.reqpar2[0] , endDate : this.reqpar2[1]}
  393. if(this.reqpar1) {
  394. if(this.valuex != ''){
  395. getlistcs.projectId = this.valuex
  396. }
  397. if(this.valuex2 != ''){
  398. getlistcs.groupName = this.valuex2
  399. }
  400. }else {
  401. if(this.valuex != ''){
  402. getlistcs.userId = this.valuex
  403. }
  404. }
  405. this.http.post('/project/getGanttData', getlistcs ,
  406. res => {
  407. if (res.code == "ok") {
  408. for(var i in res.data) {
  409. if(res.data[i].id.indexOf('出差') != '-1') {
  410. res.data[i].color = '#E6A23C'
  411. }
  412. if(res.data[i].id.indexOf('请假') != '-1') {
  413. res.data[i].color = '#F56C6C'
  414. }
  415. }
  416. this.tasks = {data:res.data};
  417. for(let i in this.tasks.data){
  418. if(this.tasks.data[i].time == 0){
  419. delete this.tasks.data[i].start_date
  420. delete this.tasks.data[i].end_date
  421. this.tasks.data[i].type = 'milestone'
  422. }
  423. }
  424. this.$nextTick(()=>{
  425. this.updatakey1 += 1
  426. })
  427. } else {
  428. this.$message({
  429. message: res.msg,
  430. type: "error"
  431. });
  432. }
  433. },
  434. error => {
  435. this.$message({
  436. message: error,
  437. type: "error"
  438. });
  439. }
  440. );
  441. },
  442. // 获取资源需求列表
  443. getDemandList(){
  444. let parameter = {
  445. pageIndex: this.pageIndex,
  446. pageSize: this.pageSize,
  447. }
  448. if(this.reqpar2 != [] && this.reqpar2 != null){
  449. parameter.reStartDate = this.reqpar2[0],
  450. parameter.reEndDate = this.reqpar2[1]
  451. }
  452. if(this.valuex != ''){
  453. parameter.projectId = this.valuex
  454. }
  455. if(this.valuex2 != ''){
  456. parameter.groupName = this.valuex2
  457. }
  458. this.demandListLoading = true
  459. this.http.post('/project-requirement/listByPage',parameter,
  460. res => {
  461. if(res.code == 'ok'){
  462. this.demandListLoading = false
  463. this.total = res.data.total
  464. this.demandList = res.data.records
  465. }else {
  466. this.demandListLoading = false
  467. this.$message({
  468. message: res.msg,
  469. type: 'error'
  470. })
  471. }
  472. },err => {
  473. this.demandListLoading = false
  474. this.$message({
  475. message: err,
  476. type: 'error'
  477. })
  478. })
  479. },
  480. // tasksEdit(){
  481. // let etasks = JSON.parse(JSON.stringify(this.tasks.data))
  482. // for(let i=0;i<etasks.length;i++){
  483. // if(etasks[i].parent != null){
  484. // let edate = new Date(etasks[i].end_date)
  485. // edate = new Date(edate.setDate(edate.getDate() + 1))
  486. // let edatemonth = edate.getMonth() + 1
  487. // let edateday = edate.getDate()
  488. // edate = edate.getFullYear() + '-' + (edatemonth < 10 ? '0' + edatemonth : edatemonth) + '-' + (edateday < 10 ? '0' + edateday : edateday)
  489. // etasks[i].end_date = edate
  490. // }
  491. // }
  492. // this.tasks = {data:etasks}
  493. // }
  494. },
  495. mounted: function () {
  496. let nowdate = new Date()
  497. let nowmonth = nowdate.getMonth() + 1
  498. let startdate = nowdate.getFullYear() + "-" + (nowmonth < 10 ? "0" + nowmonth : nowmonth) + "-" + (nowdate.getDate() < 10 ? "0" + nowdate.getDate() : nowdate.getDate())
  499. let udate = new Date(nowdate.getFullYear(),nowdate.getMonth(),nowdate.getDate() + 31)
  500. let endmonth = udate.getMonth() + 1
  501. let enddate = udate.getFullYear() + "-" + (endmonth < 10 ? "0" + endmonth : endmonth) + "-" + (udate.getDate() < 10 ? "0" + udate.getDate() : udate.getDate())
  502. this.valueDate = [startdate,enddate]
  503. this.reqpar2 = this.valueDate
  504. // console.log("date",this.valueDate);
  505. this.getList();
  506. this.getScreen()
  507. // this.tasks1 = this.tasks
  508. this.isDataLoaded = true
  509. }
  510. }
  511. </script>
  512. <style>
  513. /* html, body {
  514. height: 100%;
  515. margin: 0;
  516. padding: 0;
  517. } */
  518. .container {
  519. height: 100%;
  520. width: 100%;
  521. }
  522. .left-container {
  523. overflow: hidden;
  524. position: relative;
  525. height: 90%;
  526. }
  527. .demand-container{
  528. overflow: hidden;
  529. position: relative;
  530. height: 92%;
  531. }
  532. .gantt_head{
  533. width: 100%;
  534. height: 80px;
  535. display: flex;
  536. justify-content: space-between;
  537. align-items: center;
  538. }
  539. .gantt_head .head_RorX{
  540. width: 22vw;
  541. }
  542. .gantt_head .head_date{
  543. width: 24vw;
  544. }
  545. .gantt_head .head_taskgroup{
  546. width: 16vw;
  547. }
  548. .gantt_head .head_select{
  549. width: 15vw;
  550. }
  551. .gantt_head .head_files{
  552. width: 17vw;
  553. }
  554. .poss {
  555. height: 8%;
  556. float: right;
  557. padding-top: 15px;
  558. }
  559. </style>