safetySystemConstructionWorkPlanManagementViewSender.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354
  1. <template>
  2. <div class="safety-platform-container">
  3. <div class="back-header">
  4. <BreadcrumbBack />
  5. <span class="breadcrumb-title">查看发送对象</span>
  6. </div>
  7. <header class="safety-platform-container__header">
  8. <div class="breadcrumb-title"> {{ detail.workContent }}</div>
  9. <div class="breadcrumb-desc flex" style="margin-top: 20px">
  10. <span style="margin-right: 30px; font-size: 13px">创建人: {{ detail.createdByName }}</span>
  11. <span style="font-size: 13px">创建时间:{{ detail.createdAt }}</span>
  12. </div>
  13. </header>
  14. <main class="safety-platform-container__main">
  15. <div class="search-table-container">
  16. <header class="header-wrapper">
  17. <div class="act-search">
  18. <section class="select-box">
  19. <div class="select-box--item">
  20. <span>状态:</span>
  21. <el-select v-model="tableQuery.queryParam.status" placeholder="请选择状态" clearable>
  22. <el-option
  23. v-for="item in VIEW_SENDER_WORK_PLAN_STATUS_OPTIONS"
  24. :key="item.value"
  25. :label="item.label"
  26. :value="item.value"
  27. />
  28. </el-select>
  29. </div>
  30. <div class="select-box--item">
  31. <span>部门名称:</span>
  32. <el-cascader
  33. v-model="responsibleDept"
  34. :options="deptTree"
  35. :props="cascaderProp"
  36. clearable
  37. collapse-tags
  38. :show-all-levels="false"
  39. :max-collapse-tags="3"
  40. placeholder="请选择责任部门"
  41. style="width: 100%"
  42. />
  43. </div>
  44. <div class="select-box--item">
  45. <span>日期范围:</span>
  46. <el-date-picker
  47. v-model="dateRange"
  48. type="daterange"
  49. range-separator="至"
  50. start-placeholder="开始日期"
  51. end-placeholder="结束日期"
  52. value-format="YYYY-MM-DD"
  53. format="YYYY-MM-DD"
  54. @change="onChangeDateRange"
  55. />
  56. </div>
  57. <div class="select-box--item">
  58. <span>关键词:</span>
  59. <el-input v-model="tableQuery.queryParam.keyword" placeholder="搜索关键词"></el-input>
  60. </div>
  61. </section>
  62. <section class="search-btn">
  63. <el-button type="primary" @click="handleSearch">查询</el-button>
  64. <el-button @click="handleReset">重置</el-button>
  65. </section>
  66. </div>
  67. </header>
  68. <div class="batch-table">
  69. <BasicTable
  70. ref="basicTableRef"
  71. :tableData="tableData"
  72. :tableConfig="tableConfig"
  73. @update:pageSize="handleSizeChange"
  74. @update:pageNumber="handleCurrentChange"
  75. >
  76. <template #action="scope">
  77. <div class="action-container--div" style="justify-content: left">
  78. <!-- 1-未下发、2-待反馈、3-已完成 4- 已作废 5-待审核 -->
  79. <!-- (2)待反馈-->
  80. <template v-if="Number(scope.row.status) === 2">
  81. <ActionButton text="查看" @click="handleView(scope.row)" />
  82. <ActionButton text="作废" @click="handleViewInvalid(scope.row.id)" />
  83. </template>
  84. <!-- 已完成(3)/已作废(4)-->
  85. <template v-else>
  86. <ActionButton
  87. text="删除"
  88. :popconfirm="{
  89. title: '确定要删除?',
  90. }"
  91. @confirm="handleDelete(scope.row.id)"
  92. />
  93. <ActionButton text="查看" @click="handleView(scope.row)" />
  94. </template>
  95. <!-- 待审核(5)-->
  96. <template v-if="Number(scope.row.status) === 5">
  97. <ActionButton text="审核" @click="handleReview(scope.row)" />
  98. </template>
  99. </div>
  100. </template>
  101. </BasicTable>
  102. </div>
  103. </div>
  104. </main>
  105. </div>
  106. </template>
  107. <script setup lang="ts">
  108. import { onMounted, reactive, ref, computed } from 'vue';
  109. import { ElMessage } from 'element-plus';
  110. import { Discount } from '@element-plus/icons-vue';
  111. import BasicTable from '@/components/BasicTable.vue';
  112. import useTableConfig from '@/hooks/useTableConfigHook';
  113. import ActionButton from '@/components/ActionButton.vue';
  114. import BreadcrumbBack from '@/components/BreadcrumbBack.vue';
  115. import {
  116. TABLE_OPTIONS,
  117. VIEW_SENDS_TABLE_COLUMNS,
  118. VIEW_SENDER_WORK_PLAN_STATUS_OPTIONS,
  119. WORK_PLAN_STATUS_LABEL,
  120. } from './configs/tables';
  121. import { useRouter, useRoute } from 'vue-router';
  122. import type { QueryPageRequest } from '@/types/basic-query';
  123. import {
  124. queryViewSender,
  125. deleteWorkPlanDept,
  126. newCancelWorkPlan,
  127. queryWorkPlanDetail,
  128. type SaveWorkPlanRequest,
  129. } from '@/api/safety-system-construction-work-plan';
  130. import { DeptTree } from '@/types/dept/type';
  131. import { getAllDepartments } from '@/api/auth/dept';
  132. const router = useRouter();
  133. const route = useRoute();
  134. const currentId = computed(() => Number(route.query.id));
  135. const detail = reactive({
  136. workContent: '',
  137. createdByName: '',
  138. createdAt: '',
  139. });
  140. // 表格
  141. const basicTableRef = ref<InstanceType<typeof BasicTable>>();
  142. const { tableConfig, pagination } = useTableConfig(VIEW_SENDS_TABLE_COLUMNS, TABLE_OPTIONS);
  143. const tableData = ref<any[]>([]);
  144. // 日期范围
  145. const dateRange = ref<[string, string] | null>(null);
  146. // 分类名称选项
  147. const classifyNameOptions = ref<Array<{ label: string; value: string }>>([
  148. { label: '全部', value: '全部' },
  149. { label: '安全综合工作', value: '安全综合工作' },
  150. { label: '生产安全工作', value: '生产安全工作' },
  151. ]);
  152. const responsibleDept = ref([]);
  153. const tableQuery = reactive<QueryPageRequest<any>>({
  154. pageNumber: pagination.pageNumber,
  155. pageSize: pagination.pageSize,
  156. queryParam: {
  157. planId: currentId.value,
  158. keyword: '',
  159. status: '',
  160. responsibleDeptIds: '',
  161. categoryName: '',
  162. startDate: '',
  163. endDate: '',
  164. /** 排序 */
  165. // sortField: "created_at",
  166. // sortOrder: false,
  167. },
  168. });
  169. /**
  170. * @description: 部门接口
  171. * @return {*}
  172. */
  173. const cascaderProp = {
  174. multiple: true,
  175. expandTrigger: 'hover',
  176. checkStrictly: true,
  177. emitPath: false,
  178. value: 'id',
  179. label: 'deptName',
  180. };
  181. // 获取级联部门数据
  182. const deptTree = ref<DeptTree[]>();
  183. const loadDeptTreeData = async () => {
  184. const result = await getAllDepartments();
  185. deptTree.value = result[0].children;
  186. };
  187. const handleSizeChange = (value: number) => {
  188. pagination.pageSize = value;
  189. tableQuery.pageSize = value;
  190. getTableData();
  191. };
  192. const handleCurrentChange = (value: number) => {
  193. pagination.pageNumber = value;
  194. tableQuery.pageNumber = value;
  195. getTableData();
  196. };
  197. const onChangeDateRange = (value) => {
  198. tableQuery.queryParam.startDate = value[0];
  199. tableQuery.queryParam.endDate = value[1];
  200. getTableData();
  201. };
  202. async function getTableData() {
  203. tableConfig.loading = true;
  204. try {
  205. let option = responsibleDept.value;
  206. if (tableQuery.queryParam.responsibleDeptIds) {
  207. tableQuery.queryParam.responsibleDeptIds = option.toString();
  208. }
  209. const res = await queryViewSender(tableQuery);
  210. if (res) {
  211. res.records.forEach((item) => {
  212. item.responsibleDeptNames = item.responsibleDeptName;
  213. });
  214. tableData.value = res.records;
  215. pagination.total = res.totalRow;
  216. }
  217. } catch (e) {
  218. console.error('获取工作计划列表失败:', e);
  219. ElMessage.error('获取工作计划列表失败');
  220. tableData.value = [];
  221. pagination.total = 0;
  222. } finally {
  223. tableConfig.loading = false;
  224. }
  225. }
  226. const fetchWorkPlanDetail = async () => {
  227. const res = await queryWorkPlanDetail(currentId.value);
  228. Object.assign(detail, res);
  229. };
  230. const handleSearch = () => {
  231. pagination.pageNumber = 1;
  232. tableQuery.pageNumber = 1;
  233. getTableData();
  234. };
  235. const handleReset = () => {
  236. tableQuery.pageNumber = 1;
  237. tableQuery.queryParam = {
  238. planId: currentId.value,
  239. keyword: '',
  240. status: '',
  241. categoryName: '',
  242. startDate: '',
  243. endDate: '',
  244. /** 排序 */
  245. // sortField: "created_at",
  246. // sortOrder: false,
  247. };
  248. dateRange.value = null;
  249. handleSearch();
  250. };
  251. const handleDelete = async (id: number) => {
  252. try {
  253. await deleteWorkPlanDept(id);
  254. ElMessage.success('删除成功');
  255. getTableData();
  256. } catch (e) {
  257. console.error('删除工作计划失败:', e);
  258. ElMessage.error('删除失败,请重试');
  259. }
  260. };
  261. // 查看
  262. const handleView = (row) => {
  263. router.push({
  264. name: 'SafetySystemConstructionWorkPlanManagementDeptItem',
  265. query: {
  266. id: row.id,
  267. planId: row.planId,
  268. operate: 'work-plan-dept-view',
  269. },
  270. });
  271. };
  272. // 审核
  273. const handleReview = (row) => {
  274. router.push({
  275. name: 'SafetySystemConstructionWorkPlanManagementItem',
  276. query: {
  277. id: row.id,
  278. planId: row.planId,
  279. operate: 'work-plan-review',
  280. },
  281. });
  282. };
  283. // 作废
  284. const handleViewInvalid = async (id: number) => {
  285. try {
  286. await newCancelWorkPlan(id);
  287. ElMessage.success('作废成功');
  288. getTableData();
  289. } catch (e) {
  290. console.error('作废工作计划失败:', e);
  291. ElMessage.error('作废失败,请重试');
  292. }
  293. };
  294. onMounted(() => {
  295. fetchWorkPlanDetail();
  296. getTableData();
  297. loadDeptTreeData();
  298. });
  299. </script>
  300. <style scoped lang="scss">
  301. @use '@/styles/page-details-layout.scss' as *;
  302. @use '@/styles/page-main-layout.scss' as *;
  303. @use '@/styles/basic-table-action.scss' as *;
  304. @use '@/views/traffic/violation/style/act-search-table.scss' as *;
  305. .header-wrapper {
  306. display: flex;
  307. justify-content: space-between;
  308. align-items: center;
  309. .act-search {
  310. flex: 1;
  311. display: flex;
  312. justify-content: flex-end;
  313. .search-btn {
  314. margin-left: 40px;
  315. }
  316. }
  317. }
  318. .back-header {
  319. display: flex;
  320. align-items: center;
  321. gap: 16px;
  322. padding: 16px 22px;
  323. flex-shrink: 0;
  324. background-color: white;
  325. }
  326. .select-box{
  327. justify-content: flex-start;
  328. }
  329. </style>