weekly.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469
  1. <template>
  2. <div class="app-container">
  3. <el-form size="mini" :inline="true">
  4. <el-form-item label="关联会议">
  5. <el-select size="small" style="width: 350px" placeholder="关联会议" clearable filterable v-model="meetingWeek" @change="changeWeekData()">
  6. <el-option v-for="(item,index) in meetingData" :label="item.meetingName" :key="item.id" :value="item.weeks"></el-option>
  7. </el-select>
  8. </el-form-item>
  9. <el-form-item>
  10. <el-button type="primary" icon="el-icon-download" @click="generateWeekly()" v-show="false">周报生成</el-button>
  11. </el-form-item>
  12. </el-form>
  13. <div ref="exportPdf" id="exportPdf">
  14. <div>
  15. <h3>本周工作内容</h3>
  16. <el-table
  17. border
  18. stripe
  19. size="mini"
  20. style="width: 100%" v-model="workContents" :data="workContents"
  21. @row-click="rowClick" :span-method="objectSpanMethod" :highlight-current-row="true">
  22. <el-table-column label="分类" prop="classifyName" width="120">
  23. </el-table-column>
  24. <el-table-column label="项目" prop="projectName" width="160">
  25. </el-table-column>
  26. <el-table-column label="模块" prop="moduleName" width="120">
  27. </el-table-column>
  28. <el-table-column label="任务名称" prop="taskName" width="250">
  29. </el-table-column>
  30. <el-table-column label="工作反馈" prop="description" :show-overflow-tooltip="true">
  31. <template slot-scope="scope">
  32. <span v-if="scope.row.description!==''" v-for="(item,index) in scope.row.description.split('$') ">
  33. {{index+1}}.{{item}}<br/>
  34. </span>
  35. </template>
  36. </el-table-column>
  37. <el-table-column label="最新进度" prop="scheduleValue" width="100">
  38. <template slot-scope="scope">
  39. {{scope.row.scheduleValue}}%
  40. </template>
  41. </el-table-column>
  42. </el-table>
  43. </div>
  44. <div>
  45. <h3>下周工作计划</h3>
  46. <el-table
  47. border
  48. stripe
  49. size="mini"
  50. style="width: 100%;margin-top: 10px;" :data="workPlans">
  51. <el-table-column type="index" label="序号" width="60">
  52. </el-table-column>
  53. <el-table-column label="工作计划" prop="workPlan" min-width="250">
  54. <template slot-scope="scope">
  55. <el-input size="mini" v-if="scope.row.taskId!==undefined && scope.row.taskId!==''" :disabled="true" v-model="scope.row.planContent" placeholder="请填写工作计划信息" ></el-input>
  56. <el-input size="mini" v-else v-model="scope.row.planContent" @input="updateWorkPlans(scope.row)" placeholder="请填写工作计划信息" ></el-input>
  57. </template>
  58. </el-table-column>
  59. <el-table-column label="预计开始时间" prop="beginDate" width="180">
  60. <template slot-scope="scope">
  61. <el-date-picker
  62. v-model="scope.row.beginDate" v-if="scope.row.taskId!==undefined && scope.row.taskId!==''"
  63. type="date"
  64. size="mini"
  65. value-format="yyyy-MM-dd"
  66. format="yyyy-MM-dd"
  67. :disabled="true"
  68. placeholder="选择日期" style="width: 150px">
  69. </el-date-picker>
  70. <el-date-picker v-else
  71. v-model="scope.row.beginDate"
  72. type="date"
  73. size="mini"
  74. value-format="yyyy-MM-dd"
  75. format="yyyy-MM-dd"
  76. placeholder="选择日期" style="width: 150px" @change="updateWorkPlans(scope.row)">
  77. </el-date-picker>
  78. </template>
  79. </el-table-column>
  80. <el-table-column label="预计完成时间" prop="completionTimes" width="180">
  81. <template slot-scope="scope">
  82. <el-date-picker
  83. v-model="scope.row.completionTimes" style="width: 150px" v-if="scope.row.taskId!==undefined && scope.row.taskId!==''"
  84. type="date"
  85. size="mini"
  86. value-format="yyyy-MM-dd"
  87. format="yyyy-MM-dd"
  88. :disabled="true"
  89. placeholder="选择日期">
  90. </el-date-picker>
  91. <el-date-picker
  92. v-model="scope.row.completionTimes" style="width: 150px" v-else
  93. type="date"
  94. size="mini"
  95. value-format="yyyy-MM-dd"
  96. format="yyyy-MM-dd"
  97. placeholder="选择日期" @change="updateWorkPlans(scope.row)">
  98. </el-date-picker>
  99. </template>
  100. </el-table-column>
  101. <el-table-column label="所属项目" prop="projectId" width="220">
  102. <template slot-scope="scope">
  103. <el-select :disabled="true" v-model="scope.row.projectId" placeholder="请选择" size="mini" v-if="scope.row.taskId!==undefined && scope.row.taskId!==''">
  104. <el-option
  105. v-for="item in projects"
  106. :key="item.id"
  107. :label="item.projectName"
  108. :value="item.id">
  109. </el-option>
  110. </el-select>
  111. <el-select v-model="scope.row.projectId" placeholder="请选择" size="mini" v-else @change="updateWorkPlans(scope.row)">
  112. <el-option
  113. v-for="item in projects"
  114. :key="item.id"
  115. :label="item.projectName"
  116. :value="item.id">
  117. </el-option>
  118. </el-select>
  119. </template>
  120. </el-table-column>
  121. <el-table-column label="操作" prop="op" width="100" v-if="currentWeek===weeks" center>
  122. <template slot-scope="scope">
  123. <el-button size="mini" type="danger" v-if="scope.row.taskId===undefined || scope.row.taskId===null" @click="deleteRow(scope.$index, scope.row)">删除</el-button>
  124. </template>
  125. </el-table-column>
  126. </el-table>
  127. <div style="margin-top: 15px;width: 100%;text-align: right" >
  128. <el-button size="mini" type="success" @click="addRow()" v-if="currentWeek===weeks">添加</el-button>
  129. </div>
  130. </div>
  131. </div>
  132. <div style="margin-top: 50px;margin-right: 250px; width: 100%;text-align: center">
  133. <el-button size="mini" type="primary" @click="saveWorkPlan()" v-if="currentWeek===weeks">保存工作计划</el-button>
  134. </div>
  135. <!-- 任务详情对话框 -->
  136. <el-dialog title="任务详情" :visible.sync="taskOpen" width="760px" append-to-body
  137. :close-on-click-modal="false">
  138. <task-detail :detail-form="taskForm"></task-detail>
  139. </el-dialog>
  140. </div>
  141. </template>
  142. <script>
  143. import {list} from '@/api/meeting/meeting'
  144. import {
  145. deleteWorkPlanById,
  146. getMeetingsAndWorkPlansByWeeks,
  147. getProjects,
  148. getWorkPlanConfirmFlags,
  149. saveWorkPlans, updateWorkPlan
  150. } from '@/api/meeting/work'
  151. import {getTask} from "@/api/task/task";
  152. import TaskDetail from "../task/components/taskDetail"
  153. export default {
  154. name: 'weekly',
  155. components: {TaskDetail},
  156. data() {
  157. return {
  158. meetingWeek: null,
  159. meetingData: [],
  160. workContents: [],
  161. projects:[],
  162. workPlans: [{
  163. planContent: '请填写工作计划信息',
  164. id: null,
  165. meetingId: null,
  166. completionTimes: null,
  167. completionTime: null,
  168. projectId: null
  169. }],
  170. meetingStatus: null,
  171. weeks: null,
  172. currentWeek: null,
  173. meetingId: null,
  174. taskOpen: false,
  175. taskForm:{},
  176. mergeObj: {}, // 用来记录需要合并行的下标
  177. mergeArr: ['classifyName','projectName','moduleName'], // 表格中的列名
  178. }
  179. },
  180. mounted() {
  181. this.getData()
  182. },
  183. methods: {
  184. stripTags(html) {
  185. return html.replace(/<\/?[^>]+(>|$)/g, '');
  186. },
  187. getData() {
  188. list().then(response => {
  189. this.meetingData = response.data
  190. this.meetingId = this.meetingData.id
  191. //js 获取当前周
  192. const date = new Date();
  193. // 获取当前第几周
  194. let week = this.getYearWeek(date.getFullYear(), Number(date.getMonth()) + 1, date.getDate());
  195. let currentWeek = week ;
  196. this.meetingWeek = currentWeek
  197. this.getMeetingsAndWorkPlansByWeeksData(currentWeek)
  198. })
  199. getProjects().then(res=>{
  200. this.projects=res.data
  201. });
  202. },
  203. getMeetingsAndWorkPlansByWeeksData(currentWeek) {
  204. let weekss = currentWeek
  205. getMeetingsAndWorkPlansByWeeks(currentWeek).then(response => {
  206. this.workContents = response.data.workContent
  207. this.getSpanArr(this.workContents);
  208. if (response.data.workPlan != null && response.data.workPlan.length > 0) {
  209. this.workPlans = response.data.workPlan
  210. }else{
  211. this.workPlans=[{
  212. planContent: '',
  213. id: null,
  214. meetingId: null,
  215. completionTimes: null,
  216. completionTime: null,
  217. projectId: null,
  218. beginDate:null
  219. }]
  220. }
  221. this.weeks = response.data.weeks
  222. const date = new Date();
  223. // 获取当前第几周
  224. let week = this.getYearWeek(date.getFullYear(), Number(date.getMonth()) + 1, date.getDate());
  225. let currentWeek = week ;
  226. this.currentWeek = currentWeek;
  227. if (weekss !== currentWeek) {
  228. this.workPlans = response.data.workPlan
  229. }
  230. let obj = {};
  231. obj = this.meetingData.find(item => {
  232. return item.weeks === currentWeek;
  233. });
  234. this.meetingId =obj.id
  235. })
  236. },
  237. getYearWeek(a, b, c) {
  238. let date1 = new Date(a, parseInt(b) - 1, c),
  239. date2 = new Date(a, 0, 1),
  240. d = Math.round((date1.valueOf() - date2.valueOf()) / 86400000);
  241. return Math.ceil((d + (date2.getDay() + 1 - 1)) / 7);
  242. },
  243. getWorkData() {
  244. },
  245. generateWeekly() {
  246. if (this.meetingWeek == null || this.meetingWeek == undefined || this.meetingWeek === '') {
  247. this.$message({
  248. message: '请选择会议信息!',
  249. type: 'warning'
  250. });
  251. return
  252. }
  253. this.$PDFSave(this.$refs.exportPdf, "我的工作周报");
  254. },
  255. //添加下周周计划权限认证
  256. validateWorkPlanFlag(){
  257. //已经确认的工作不能继续编辑工作计划,临时增加就去任务系统手动添加
  258. let meetingId=this.meetingId
  259. let userId=this.$store.getters.userId
  260. let data={
  261. weeks:this.weeks,
  262. userId:userId,
  263. meetingId:meetingId
  264. }
  265. getWorkPlanConfirmFlags(data).then(res=> {
  266. if (res.success !== true) {
  267. this.$message.warning(res.data)
  268. return false
  269. }else{
  270. return true
  271. }
  272. })
  273. },
  274. saveWorkPlan() {
  275. let meetingId=this.meetingId
  276. let userId=this.$store.getters.userId
  277. let datas={
  278. weeks:this.weeks,
  279. userId:userId,
  280. meetingId:meetingId
  281. }
  282. getWorkPlanConfirmFlags(datas).then(res=> {
  283. if (res.success !== true) {
  284. this.$message.warning(res.data)
  285. }else{
  286. if (this.meetingWeek === null || this.meetingWeek === undefined || this.meetingWeek === '') {
  287. this.$message({
  288. message: '请选择会议信息!',
  289. type: 'warning'
  290. });
  291. return
  292. }
  293. //判断是否存在周会信息,无周会信息,就不保存工作计划
  294. if(meetingId===undefined || meetingId===null){
  295. this.$message.warning("请选择周会会议,再填写工作计划!")
  296. return
  297. }
  298. let workPlanss = this.workPlans;
  299. let newWorkPlans = []
  300. for(let i = 0; i < workPlanss.length; i++){
  301. if(workPlanss[i].taskId===undefined || workPlanss[i].taskId==null){
  302. newWorkPlans.push(workPlanss[i])
  303. }
  304. }
  305. if(newWorkPlans.length===0){
  306. this.$message.warning("请填写新的工作计划在做保存!")
  307. return
  308. }
  309. for (let i = 0; i < newWorkPlans.length; i++) {
  310. if(newWorkPlans[i].planContent==='' || newWorkPlans[i].beginDate===undefined ||
  311. newWorkPlans[i].completionTimes===undefined || newWorkPlans[i].projectId===undefined){
  312. this.$message({
  313. message: '请将工作计划填写完整!',
  314. type: 'warning'
  315. });
  316. return
  317. }
  318. newWorkPlans[i].completionTime = null;
  319. newWorkPlans[i].beginDate = new Date(newWorkPlans[i].beginDate).getTime()
  320. }
  321. let data = {
  322. weeks: this.meetingWeek,
  323. workPlans: newWorkPlans,
  324. id: this.workPlans.id,
  325. meetingId: this.meetingId
  326. }
  327. saveWorkPlans(data).then(res => {
  328. this.$message({
  329. message: '操作成功',
  330. type: 'success'
  331. });
  332. this.getData()
  333. });
  334. }
  335. })
  336. },
  337. addRow() {
  338. let meetingId=this.meetingId
  339. let userId=this.$store.getters.userId
  340. let data={
  341. weeks:this.weeks,
  342. userId:userId,
  343. meetingId:meetingId
  344. }
  345. getWorkPlanConfirmFlags(data).then(res=> {
  346. if (res.success !== true) {
  347. this.$message.warning(res.data)
  348. }else{
  349. let row = {planContent: '', 'op': '', id: null, completionTimes: null,beginDate:null}
  350. this.workPlans.push(row)
  351. }
  352. })
  353. },
  354. deleteRow(index, row) {
  355. let meetingId=this.meetingId
  356. let userId=this.$store.getters.userId
  357. let data={
  358. weeks:this.weeks,
  359. userId:userId,
  360. meetingId:meetingId
  361. }
  362. getWorkPlanConfirmFlags(data).then(res=> {
  363. if (res.success !== true) {
  364. this.$message.warning(res.data)
  365. }else{
  366. if (index == 0) {
  367. /* this.$message({
  368. message: '不能删除此条数据',
  369. type: 'warning'
  370. });
  371. return; */
  372. }
  373. this.workPlans.splice(index, 1)
  374. let id = row.id;
  375. if (id != null) {
  376. deleteWorkPlanById(id).then(res => {
  377. this.getMeetingsAndWorkPlansByWeeksData(this.meetingWeek)
  378. });
  379. }
  380. }
  381. })
  382. },
  383. changeWeekData() {
  384. this.getMeetingsAndWorkPlansByWeeksData(this.meetingWeek)
  385. },
  386. /** 查看任务详情*/
  387. rowClick(row, column, event) {
  388. getTask(row.taskId).then(res => {
  389. this.taskForm = res.data;
  390. this.taskOpen = true;
  391. })
  392. },
  393. objectSpanMethod({row, column, rowIndex, columnIndex}) {
  394. if (this.mergeArr.indexOf(column.property) !== -1) {
  395. // 判断其值是不是为0
  396. if (this.mergeObj[column.property][rowIndex]) {
  397. return [this.mergeObj[column.property][rowIndex], 1]
  398. } else {
  399. // 如果为0则为需要合并的行
  400. return [0, 0];
  401. }
  402. }
  403. },
  404. getSpanArr(data) {
  405. this.mergeArr.forEach((key, index1) => {
  406. let count = 0; // 用来记录需要合并行的起始位置
  407. this.mergeObj[key] = []; // 记录每一列的合并信息
  408. data.forEach((item, index) => {
  409. // index == 0表示数据为第一行,直接 push 一个 1
  410. if (index === 0) {
  411. this.mergeObj[key].push(1);
  412. //item.group=index;
  413. } else {
  414. // 判断当前行是否与上一行其值相等 如果相等 在 count 记录的位置其值 +1 表示当前行需要合并 并push 一个 0 作为占位
  415. if (item[key] === data[index - 1][key]) {
  416. this.mergeObj[key][count] += 1;
  417. this.mergeObj[key].push(0);
  418. //item.group = this.tableData[index - 1].group;
  419. } else {
  420. // 如果当前行和上一行其值不相等
  421. count = index; // 记录当前位置
  422. this.mergeObj[key].push(1); // 重新push 一个 1
  423. // item.group = this.tableData[index - 1].group + 1; //如果不一样 将组号设置为上一个数据的组号加1
  424. }
  425. }
  426. })
  427. })
  428. },
  429. updateWorkPlans(row){
  430. let data=row;
  431. if(row.id!==undefined && row.id!==null){
  432. let datas={
  433. id:row.id,
  434. planContent:data.planContent,
  435. beginDate:new Date(data.beginDate).getTime(),
  436. projectId:row.projectId,
  437. completionTime:new Date(row.completionTimes).getTime()
  438. }
  439. updateWorkPlan(datas).then(res=>{
  440. })
  441. }
  442. }
  443. }
  444. }
  445. </script>
  446. <style scoped lang="scss">
  447. .el-form-item--mini.el-form-item, .el-form-item--small.el-form-item {
  448. margin-bottom: 10px;
  449. }
  450. </style>