index.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746
  1. <script setup lang="ts">
  2. import { useTableComponents } from '@/components/ZmTable/useTableComponents'
  3. import { IotProjectTaskScheduleApi } from '@/api/pms/iotprojecttaskschedule'
  4. import { IotProjectTaskApi, IotProjectTaskVO } from '@/api/pms/iotprojecttask'
  5. import * as DeptApi from '@/api/system/dept'
  6. import { DICT_TYPE, getStrDictOptions } from '@/utils/dict'
  7. import { dateFormatter } from '@/utils/formatTime'
  8. import download from '@/utils/download'
  9. import dayjs from 'dayjs'
  10. defineOptions({ name: 'IotProjectTask' })
  11. type ProjectTaskRow = IotProjectTaskVO & {
  12. manufactureName?: string
  13. contractName?: string
  14. contractCode?: string
  15. deptNames?: string
  16. statusLabel?: string
  17. createTime?: string
  18. }
  19. interface QueryParams extends PageParam {
  20. companyId?: number
  21. projectId?: number
  22. manufactureName?: string
  23. contractName?: string
  24. contractCode?: string
  25. wellName?: string
  26. wellType?: string
  27. location?: string
  28. technique?: string
  29. workloadDesign?: string
  30. createTime?: string[]
  31. userName?: string
  32. userId?: number
  33. platformFlag?: string
  34. remark?: string
  35. deptName?: string
  36. }
  37. interface PlanRow {
  38. id?: number
  39. status: string
  40. startTime: string
  41. endTime: string
  42. showEndTime: boolean
  43. }
  44. const message = useMessage()
  45. const { t } = useI18n()
  46. const { push } = useRouter()
  47. const { ZmTable, ZmTableColumn } = useTableComponents<ProjectTaskRow>()
  48. const COMPLETED_STATUS = 'wg'
  49. const REPORT_DEPT_ID = 163
  50. const initQuery: QueryParams = {
  51. pageNo: 1,
  52. pageSize: 10,
  53. companyId: undefined,
  54. projectId: undefined,
  55. manufactureName: '',
  56. contractName: undefined,
  57. contractCode: undefined,
  58. wellName: undefined,
  59. wellType: undefined,
  60. location: undefined,
  61. technique: undefined,
  62. workloadDesign: undefined,
  63. createTime: [],
  64. userName: undefined,
  65. userId: undefined,
  66. platformFlag: '',
  67. remark: undefined,
  68. deptName: undefined
  69. }
  70. const queryParams = reactive<QueryParams>({ ...initQuery })
  71. const queryFormRef = ref()
  72. const loading = ref(false)
  73. const exportLoading = ref(false)
  74. const list = ref<ProjectTaskRow[]>([])
  75. const total = ref(0)
  76. const companyDeptList = ref<any[]>([])
  77. const planDialogVisible = ref(false)
  78. const planList = ref<PlanRow[]>([])
  79. const saveLoading = ref(false)
  80. const currentRow = ref<ProjectTaskRow | null>(null)
  81. const workProgressDictOptions = ref<any[]>([])
  82. const generateReportDialogVisible = ref(false)
  83. const generateReportLoading = ref(false)
  84. const generateReportRow = ref<ProjectTaskRow | null>(null)
  85. const reportDate = ref('')
  86. const getWorkProgressDictOptions = () => {
  87. workProgressDictOptions.value = getStrDictOptions(DICT_TYPE.PMS_PROJECT_TASK_SCHEDULE)
  88. }
  89. const timestampToDateTime = (timestamp: number | string | null | undefined): string => {
  90. if (timestamp === null || timestamp === undefined || timestamp === '') return ''
  91. let value = typeof timestamp === 'string' ? Number.parseInt(timestamp, 10) : timestamp
  92. if (Number.isNaN(value)) return ''
  93. if (value < 1000000000000) value *= 1000
  94. return dayjs(value).format('YYYY-MM-DD HH:mm')
  95. }
  96. const openPlanDialog = async (row: ProjectTaskRow) => {
  97. currentRow.value = row
  98. planList.value = []
  99. try {
  100. getWorkProgressDictOptions()
  101. const taskSchedules = await IotProjectTaskScheduleApi.getIotProjectTaskSchedules({
  102. taskId: row.id
  103. })
  104. if (taskSchedules?.length) {
  105. planList.value = taskSchedules.map((plan: any) => {
  106. const status = plan.status
  107. return {
  108. id: plan.id,
  109. status,
  110. startTime: timestampToDateTime(plan.startTime),
  111. endTime: timestampToDateTime(plan.endTime),
  112. showEndTime: status !== COMPLETED_STATUS
  113. }
  114. })
  115. }
  116. planDialogVisible.value = true
  117. } catch {
  118. message.error('获取计划数据失败')
  119. }
  120. }
  121. const rowShowEndTime = (row: PlanRow) => row.showEndTime
  122. const onStatusChange = (row: PlanRow) => {
  123. if (row.status === COMPLETED_STATUS) {
  124. row.showEndTime = false
  125. row.endTime = ''
  126. return
  127. }
  128. row.showEndTime = true
  129. }
  130. const addNewRow = () => {
  131. planList.value.push({
  132. status: '',
  133. startTime: '',
  134. endTime: '',
  135. showEndTime: true
  136. })
  137. }
  138. const removeRow = (index: number) => {
  139. planList.value.splice(index, 1)
  140. }
  141. const savePlan = async () => {
  142. try {
  143. saveLoading.value = true
  144. for (let index = 0; index < planList.value.length; index++) {
  145. if (!planList.value[index].status) {
  146. message.error(`第${index + 1}行请选择施工状态`)
  147. return
  148. }
  149. }
  150. const submitData = planList.value.map((item) => ({
  151. id: item.id,
  152. taskId: currentRow.value?.id,
  153. status: item.status,
  154. startTime: item.startTime ? new Date(item.startTime).getTime() : null,
  155. endTime:
  156. item.status !== COMPLETED_STATUS && item.endTime ? new Date(item.endTime).getTime() : null
  157. }))
  158. await IotProjectTaskScheduleApi.saveTaskSchedule(submitData)
  159. message.success('保存成功')
  160. planDialogVisible.value = false
  161. getList()
  162. } catch {
  163. message.error('保存失败')
  164. } finally {
  165. saveLoading.value = false
  166. }
  167. }
  168. const openGenerateReportDialog = (row: ProjectTaskRow) => {
  169. generateReportRow.value = row
  170. reportDate.value = ''
  171. generateReportDialogVisible.value = true
  172. }
  173. const submitGenerateReport = async () => {
  174. if (!generateReportRow.value?.id) {
  175. message.error('未找到任务信息')
  176. return
  177. }
  178. if (!reportDate.value) {
  179. message.error('请选择日报日期')
  180. return
  181. }
  182. try {
  183. generateReportLoading.value = true
  184. await IotProjectTaskApi.generateReport({
  185. id: generateReportRow.value.id,
  186. reportDate: dayjs(reportDate.value).valueOf()
  187. })
  188. message.success('生成日报成功')
  189. generateReportDialogVisible.value = false
  190. } catch {
  191. message.error('生成日报失败')
  192. } finally {
  193. generateReportLoading.value = false
  194. }
  195. }
  196. const getList = async () => {
  197. loading.value = true
  198. try {
  199. const data = await IotProjectTaskApi.getIotProjectTaskList(queryParams)
  200. list.value = data.list
  201. total.value = data.total
  202. } finally {
  203. loading.value = false
  204. }
  205. }
  206. const handleQuery = () => {
  207. queryParams.pageNo = 1
  208. getList()
  209. }
  210. const resetQuery = () => {
  211. Object.assign(queryParams, { ...initQuery })
  212. queryFormRef.value?.resetFields()
  213. handleQuery()
  214. }
  215. const handleSizeChange = (val: number) => {
  216. queryParams.pageSize = val
  217. handleQuery()
  218. }
  219. const handleCurrentChange = (val: number) => {
  220. queryParams.pageNo = val
  221. getList()
  222. }
  223. const openForm = (type: string, id?: number, projectId?: number) => {
  224. if (id === undefined) {
  225. push({ name: 'IotProjectTaskInfo', params: { type } })
  226. return
  227. }
  228. push({ name: 'IotProjectTaskInfo', params: { type, id, projectId } })
  229. }
  230. const handleDelete = async (id: number) => {
  231. try {
  232. await message.delConfirm()
  233. await IotProjectTaskApi.deleteIotProjectTask(id)
  234. message.success(t('common.delSuccess'))
  235. await getList()
  236. } catch {}
  237. }
  238. const handleExport = async () => {
  239. try {
  240. exportLoading.value = true
  241. const data = await IotProjectTaskApi.exportIotProjectTask(queryParams)
  242. download.excel(data, '项目信息任务拆分.xls')
  243. } finally {
  244. exportLoading.value = false
  245. }
  246. }
  247. onMounted(async () => {
  248. companyDeptList.value = await DeptApi.companyLevelDepts()
  249. getWorkProgressDictOptions()
  250. getList()
  251. })
  252. </script>
  253. <template>
  254. <div
  255. class="iot-project-task-page grid grid-rows-[auto_1fr] gap-4 h-[calc(100vh-20px-var(--top-tool-height)-var(--tags-view-height)-var(--app-footer-height))]"
  256. >
  257. <el-form
  258. ref="queryFormRef"
  259. :model="queryParams"
  260. size="default"
  261. label-width="72px"
  262. class="iot-project-task-query bg-white dark:bg-[#1d1e1f] rounded-lg shadow px-6 py-3 min-w-0"
  263. >
  264. <div class="query-row">
  265. <el-form-item label="公司" prop="companyId">
  266. <el-select
  267. v-model="queryParams.companyId"
  268. placeholder="请选择公司"
  269. clearable
  270. filterable
  271. class="w-full"
  272. >
  273. <el-option
  274. v-for="item in companyDeptList"
  275. :key="item.id"
  276. :label="item.name"
  277. :value="item.id"
  278. />
  279. </el-select>
  280. </el-form-item>
  281. <el-form-item label="客户名称" prop="manufactureName">
  282. <el-input
  283. v-model="queryParams.manufactureName"
  284. placeholder="请输入客户名称"
  285. clearable
  286. class="w-full"
  287. @keyup.enter="handleQuery"
  288. />
  289. </el-form-item>
  290. <el-form-item label="合同名称" prop="contractName">
  291. <el-input
  292. v-model="queryParams.contractName"
  293. placeholder="请输入合同名称"
  294. clearable
  295. class="w-full"
  296. @keyup.enter="handleQuery"
  297. />
  298. </el-form-item>
  299. <el-form-item label="合同编号" prop="contractCode">
  300. <el-input
  301. v-model="queryParams.contractCode"
  302. placeholder="请输入合同编号"
  303. clearable
  304. class="w-full"
  305. @keyup.enter="handleQuery"
  306. />
  307. </el-form-item>
  308. <el-form-item label="施工队伍" prop="deptName">
  309. <el-input
  310. v-model="queryParams.deptName"
  311. placeholder="请输入施工队伍"
  312. clearable
  313. class="w-full"
  314. @keyup.enter="handleQuery"
  315. />
  316. </el-form-item>
  317. <el-form-item label="井号" prop="wellName">
  318. <el-input
  319. v-model="queryParams.wellName"
  320. placeholder="请输入井号"
  321. clearable
  322. class="w-full"
  323. @keyup.enter="handleQuery"
  324. />
  325. </el-form-item>
  326. <el-form-item label="平台井" prop="platformFlag">
  327. <el-select
  328. v-model="queryParams.platformFlag"
  329. placeholder="请选择平台井"
  330. clearable
  331. class="w-full"
  332. @change="handleQuery"
  333. >
  334. <el-option label="全部" value="A" />
  335. <el-option label="是" value="Y" />
  336. <el-option label="否" value="N" />
  337. </el-select>
  338. </el-form-item>
  339. <el-form-item label="创建时间" prop="createTime">
  340. <el-date-picker
  341. v-model="queryParams.createTime"
  342. value-format="YYYY-MM-DD HH:mm:ss"
  343. type="daterange"
  344. start-placeholder="开始日期"
  345. end-placeholder="结束日期"
  346. :default-time="[new Date('1 00:00:00'), new Date('1 23:59:59')]"
  347. class="w-full"
  348. />
  349. </el-form-item>
  350. <el-form-item class="query-actions">
  351. <el-button type="primary" @click="handleQuery">
  352. <Icon icon="ep:search" class="mr-5px" />搜索
  353. </el-button>
  354. <el-button @click="resetQuery"> <Icon icon="ep:refresh" class="mr-5px" />重置 </el-button>
  355. <el-button type="success" plain :loading="exportLoading" @click="handleExport">
  356. <Icon icon="ep:download" class="mr-5px" />导出
  357. </el-button>
  358. </el-form-item>
  359. </div>
  360. </el-form>
  361. <div class="bg-white dark:bg-[#1d1e1f] shadow rounded-lg flex flex-col p-4 min-w-0 min-h-0">
  362. <div class="flex-1 relative min-h-0">
  363. <el-auto-resizer class="absolute">
  364. <template #default="{ width, height }">
  365. <ZmTable
  366. :data="list"
  367. :loading="loading"
  368. :width="width"
  369. :height="height"
  370. :max-height="height"
  371. show-border
  372. >
  373. <ZmTableColumn
  374. type="index"
  375. :label="t('monitor.serial')"
  376. :width="70"
  377. fixed="left"
  378. hide-in-column-settings
  379. />
  380. <ZmTableColumn prop="manufactureName" label="客户名称" min-width="180" fixed="left" />
  381. <ZmTableColumn prop="contractName" label="合同名称" min-width="220" />
  382. <ZmTableColumn prop="contractCode" label="合同编号" min-width="150" />
  383. <ZmTableColumn prop="wellName" label="井号" min-width="120" />
  384. <ZmTableColumn prop="location" label="施工地点" min-width="140" />
  385. <ZmTableColumn prop="deptNames" label="施工队伍" min-width="150" />
  386. <ZmTableColumn prop="statusLabel" label="施工状态" min-width="120" />
  387. <ZmTableColumn prop="createTime" label="创建时间" min-width="180">
  388. <template #default="{ row }">
  389. {{ dateFormatter(row, null, row.createTime) }}
  390. </template>
  391. </ZmTableColumn>
  392. <ZmTableColumn label="操作" width="220" fixed="right" action>
  393. <template #default="{ row }">
  394. <el-button
  395. link
  396. type="primary"
  397. @click="openPlanDialog(row)"
  398. v-hasPermi="['rq:iot-project-task:update']"
  399. >
  400. 计划
  401. </el-button>
  402. <el-button
  403. v-if="Number(row.deptId) === REPORT_DEPT_ID"
  404. link
  405. type="primary"
  406. @click="openGenerateReportDialog(row)"
  407. >
  408. 生成日报
  409. </el-button>
  410. <el-button
  411. link
  412. type="primary"
  413. @click="openForm('update', row.id, row.projectId)"
  414. v-hasPermi="['rq:iot-project-task:update']"
  415. >
  416. 编辑
  417. </el-button>
  418. <el-button
  419. link
  420. type="danger"
  421. @click="handleDelete(row.id)"
  422. v-hasPermi="['rq:iot-project-task:delete']"
  423. >
  424. 删除
  425. </el-button>
  426. </template>
  427. </ZmTableColumn>
  428. </ZmTable>
  429. </template>
  430. </el-auto-resizer>
  431. </div>
  432. <div class="h-8 mt-2 flex items-center justify-end">
  433. <el-pagination
  434. v-show="total > 0"
  435. size="default"
  436. :current-page="queryParams.pageNo"
  437. :page-size="queryParams.pageSize"
  438. :background="true"
  439. :page-sizes="[10, 20, 30, 50, 100]"
  440. :total="total"
  441. layout="total, sizes, prev, pager, next, jumper"
  442. @size-change="handleSizeChange"
  443. @current-change="handleCurrentChange"
  444. />
  445. </div>
  446. </div>
  447. </div>
  448. <el-dialog
  449. v-model="planDialogVisible"
  450. :title="`${currentRow?.contractName || ''} - ${currentRow?.wellName || ''} - 任务计划`"
  451. width="80%"
  452. >
  453. <div class="mb-15px">
  454. <el-button type="primary" @click="addNewRow">
  455. <Icon icon="ep:plus" class="mr-5px" />新增行
  456. </el-button>
  457. </div>
  458. <el-table :data="planList" border stripe>
  459. <el-table-column label="序号" width="60" align="center">
  460. <template #default="scope">
  461. {{ scope.$index + 1 }}
  462. </template>
  463. </el-table-column>
  464. <el-table-column label="施工状态" min-width="200">
  465. <template #default="scope">
  466. <el-select
  467. v-model="scope.row.status"
  468. placeholder="请选择施工状态"
  469. clearable
  470. class="w-full"
  471. @change="onStatusChange(scope.row)"
  472. >
  473. <el-option
  474. v-for="dict in workProgressDictOptions"
  475. :key="dict.value"
  476. :label="dict.label"
  477. :value="dict.value"
  478. />
  479. </el-select>
  480. </template>
  481. </el-table-column>
  482. <el-table-column label="开始时间" min-width="200">
  483. <template #default="scope">
  484. <el-date-picker
  485. v-model="scope.row.startTime"
  486. type="datetime"
  487. placeholder="选择开始时间"
  488. value-format="YYYY-MM-DD HH:mm"
  489. format="YYYY-MM-DD HH:mm"
  490. class="w-full"
  491. />
  492. </template>
  493. </el-table-column>
  494. <el-table-column label="结束时间" min-width="200">
  495. <template #default="scope">
  496. <el-date-picker
  497. v-if="rowShowEndTime(scope.row)"
  498. v-model="scope.row.endTime"
  499. type="datetime"
  500. placeholder="选择结束时间"
  501. value-format="YYYY-MM-DD HH:mm"
  502. format="YYYY-MM-DD HH:mm"
  503. class="w-full"
  504. />
  505. </template>
  506. </el-table-column>
  507. <el-table-column label="操作" width="100" align="center">
  508. <template #default="scope">
  509. <el-button link type="danger" @click="removeRow(scope.$index)">删除</el-button>
  510. </template>
  511. </el-table-column>
  512. </el-table>
  513. <template #footer>
  514. <span class="dialog-footer">
  515. <el-button @click="planDialogVisible = false">取消</el-button>
  516. <el-button type="primary" :loading="saveLoading" @click="savePlan">保存</el-button>
  517. </span>
  518. </template>
  519. </el-dialog>
  520. <el-dialog v-model="generateReportDialogVisible" title="生成日报" width="420px">
  521. <el-form size="default" label-width="80px">
  522. <el-form-item label="日报日期" required>
  523. <el-date-picker
  524. v-model="reportDate"
  525. type="date"
  526. placeholder="请选择日期"
  527. value-format="YYYY-MM-DD"
  528. format="YYYY-MM-DD"
  529. class="w-full!"
  530. />
  531. </el-form-item>
  532. </el-form>
  533. <template #footer>
  534. <span class="dialog-footer">
  535. <el-button size="default" @click="generateReportDialogVisible = false">取消</el-button>
  536. <el-button
  537. size="default"
  538. type="primary"
  539. :loading="generateReportLoading"
  540. @click="submitGenerateReport"
  541. >
  542. 确定
  543. </el-button>
  544. </span>
  545. </template>
  546. </el-dialog>
  547. <!-- 生成日报 Dialog -->
  548. <el-dialog v-model="generateReportDialogVisible" title="生成日报" width="420px">
  549. <el-form size="default" label-width="80px">
  550. <el-form-item label="日报日期" required>
  551. <el-date-picker
  552. v-model="reportDate"
  553. type="date"
  554. placeholder="请选择日期"
  555. value-format="YYYY-MM-DD"
  556. format="YYYY-MM-DD"
  557. class="w-full!"
  558. />
  559. </el-form-item>
  560. </el-form>
  561. <template #footer>
  562. <span class="dialog-footer">
  563. <el-button size="default" @click="generateReportDialogVisible = false">取消</el-button>
  564. <el-button
  565. size="default"
  566. type="primary"
  567. @click="submitGenerateReport"
  568. :loading="generateReportLoading"
  569. >
  570. 确定
  571. </el-button>
  572. </span>
  573. </template>
  574. </el-dialog>
  575. </template>
  576. <style scoped>
  577. .iot-project-task-query {
  578. display: block;
  579. }
  580. .query-row {
  581. display: grid;
  582. grid-template-columns: repeat(4, minmax(270px, 1fr));
  583. align-items: center;
  584. gap: 14px 28px;
  585. min-width: 0;
  586. }
  587. .query-actions {
  588. justify-self: end;
  589. }
  590. .query-actions :deep(.el-form-item__content) {
  591. display: flex;
  592. flex-wrap: wrap;
  593. justify-content: flex-end;
  594. gap: 8px 10px;
  595. }
  596. .query-actions :deep(.el-button) {
  597. margin-left: 0;
  598. }
  599. :deep(.el-form-item) {
  600. margin-bottom: 0;
  601. }
  602. @media (width >= 2200px) {
  603. .query-actions {
  604. grid-column: 4;
  605. }
  606. }
  607. @media (width <= 1800px) {
  608. .query-row {
  609. grid-template-columns: repeat(3, minmax(270px, 1fr));
  610. }
  611. .query-actions {
  612. justify-self: start;
  613. }
  614. }
  615. @media (width <= 1500px) {
  616. .query-row {
  617. grid-template-columns: repeat(2, minmax(260px, 1fr));
  618. gap: 12px 18px;
  619. }
  620. .query-control {
  621. width: 210px;
  622. }
  623. .query-control--small {
  624. width: 160px;
  625. }
  626. .query-control--date {
  627. width: 280px;
  628. }
  629. }
  630. @media (width <= 1200px) {
  631. .iot-project-task-page {
  632. grid-template-rows: auto minmax(480px, 1fr);
  633. height: auto;
  634. min-height: calc(
  635. 100vh - 20px - var(--top-tool-height) - var(--tags-view-height) - var(--app-footer-height)
  636. );
  637. }
  638. .query-actions {
  639. width: 100%;
  640. justify-self: start;
  641. }
  642. .query-actions :deep(.el-form-item__content) {
  643. justify-content: flex-start;
  644. }
  645. }
  646. @media (width <= 768px) {
  647. .iot-project-task-query {
  648. padding: 12px;
  649. }
  650. .query-row,
  651. .query-row :deep(.el-form-item),
  652. .query-actions {
  653. width: 100%;
  654. }
  655. .query-row {
  656. grid-template-columns: minmax(0, 1fr);
  657. }
  658. .query-control,
  659. .query-control--small,
  660. .query-control--date {
  661. width: 100%;
  662. }
  663. .query-actions :deep(.el-form-item__content) {
  664. display: grid;
  665. grid-template-columns: repeat(2, minmax(0, 1fr));
  666. gap: 8px;
  667. width: 100%;
  668. }
  669. .query-actions :deep(.el-button) {
  670. width: 100%;
  671. margin-left: 0;
  672. }
  673. }
  674. </style>