index.vue 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752
  1. <template>
  2. <div class="app-container">
  3. <el-card shadow="never" class="search-card" :body-style="{ padding: '15px 20px 10px' }">
  4. <el-form :model="queryParams" ref="queryRef" label-width="80px">
  5. <el-row :gutter="24">
  6. <el-col :span="6">
  7. <el-form-item label="所属公司" prop="companyNo">
  8. <el-select v-model="queryParams.companyNo" placeholder="请选择" style="width: 100%" clearable filterable>
  9. <el-option v-for="item in companyOptions" :key="item.companyCode || item.id" :label="item.companyName" :value="String(item.companyCode || item.id)" />
  10. </el-select>
  11. </el-form-item>
  12. </el-col>
  13. <el-col :span="6">
  14. <el-form-item label="项目名称" prop="projectName">
  15. <el-input v-model="queryParams.projectName" placeholder="请输入项目名称" clearable />
  16. </el-form-item>
  17. </el-col>
  18. <el-col :span="6">
  19. <el-form-item label="客户名称" prop="customerName">
  20. <el-input v-model="queryParams.customerName" placeholder="请输入客户名称" clearable />
  21. </el-form-item>
  22. </el-col>
  23. <el-col :span="6">
  24. <el-form-item label="负责人" prop="managerId">
  25. <el-select v-model="queryParams.managerId" placeholder="请选择" style="width: 100%" clearable filterable>
  26. <el-option v-for="item in userOptions" :key="item.staffId || item.userId" :label="(item.staffCode || item.userName ? '(' + (item.staffCode || item.userName) + ') ' : '') + (item.staffName || item.nickName)" :value="String(item.staffId || item.userId)" />
  27. </el-select>
  28. </el-form-item>
  29. </el-col>
  30. </el-row>
  31. <el-row :gutter="24">
  32. <el-col :span="6">
  33. <el-form-item label="部门" prop="deptId">
  34. <el-tree-select
  35. v-model="queryParams.deptId"
  36. :data="deptOptions"
  37. :props="{ value: 'id', label: 'label', children: 'children' }"
  38. value-key="id"
  39. placeholder="所属部门"
  40. check-strictly
  41. style="width: 100%"
  42. clearable
  43. filterable
  44. />
  45. </el-form-item>
  46. </el-col>
  47. <el-col :span="6">
  48. <el-form-item label="产品支持" prop="productSupport">
  49. <el-select v-model="queryParams.productSupport" placeholder="请选择" style="width: 100%" clearable filterable>
  50. <el-option v-for="item in userOptions" :key="item.staffId || item.userId" :label="(item.staffCode || item.userName ? '(' + (item.staffCode || item.userName) + ') ' : '') + (item.staffName || item.nickName)" :value="String(item.staffId || item.userId)" />
  51. </el-select>
  52. </el-form-item>
  53. </el-col>
  54. <el-col :span="6">
  55. <el-form-item label="项目级别" prop="projectLevel">
  56. <el-select v-model="queryParams.projectLevel" placeholder="请选择" style="width: 100%" clearable>
  57. <el-option v-for="item in projectLevelOptions" :key="item.value" :label="item.label" :value="item.value" />
  58. </el-select>
  59. </el-form-item>
  60. </el-col>
  61. <el-col :span="6">
  62. <el-form-item label="项目类型" prop="businessType">
  63. <el-select v-model="queryParams.businessType" placeholder="请选择" style="width: 100%" clearable>
  64. <el-option v-for="item in projectTypeOptions" :key="item.value" :label="item.label" :value="item.value" />
  65. </el-select>
  66. </el-form-item>
  67. </el-col>
  68. </el-row>
  69. <el-row :gutter="24">
  70. <el-col :span="6">
  71. <el-form-item label="入围类型" prop="selectionType">
  72. <el-select v-model="queryParams.selectionType" placeholder="请选择" style="width: 100%" clearable>
  73. <el-option v-for="item in shortlistedTypeOptions" :key="item.value" :label="item.label" :value="item.value" />
  74. </el-select>
  75. </el-form-item>
  76. </el-col>
  77. <el-col :span="6">
  78. <el-form-item label="项目状态" prop="projectStatus">
  79. <el-select v-model="queryParams.projectStatus" placeholder="请选择" style="width: 100%" clearable>
  80. <el-option v-for="item in statusOptions" :key="item.value" :label="item.label" :value="item.value" />
  81. </el-select>
  82. </el-form-item>
  83. </el-col>
  84. <el-col :span="6">
  85. <el-form-item label="成交结果" prop="result">
  86. <el-select v-model="queryParams.result" placeholder="请选择" style="width: 100%" clearable>
  87. <el-option v-for="item in resultOptions" :key="item.value" :label="item.label" :value="item.value" />
  88. </el-select>
  89. </el-form-item>
  90. </el-col>
  91. <el-col :span="6">
  92. <el-form-item label="查询类型" prop="timeType">
  93. <el-select v-model="queryParams.timeType" placeholder="请选择" style="width: 100%" clearable>
  94. <el-option v-for="item in timeTypeOptions" :key="item.value" :label="item.label" :value="item.value" />
  95. </el-select>
  96. </el-form-item>
  97. </el-col>
  98. </el-row>
  99. <el-row :gutter="24">
  100. <el-col :span="7">
  101. <el-form-item label="时间范围">
  102. <el-date-picker v-model="dateRange" type="daterange" range-separator="至" start-placeholder="开始时间" end-placeholder="结束时间" style="width: 100%" value-format="YYYY-MM-DD" />
  103. </el-form-item>
  104. </el-col>
  105. <el-col :span="17">
  106. <el-form-item label-width="20px">
  107. <div class="search-btns">
  108. <el-button icon="Search" @click="handleQuery">搜索</el-button>
  109. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  110. <el-button type="primary" icon="Plus" @click="handleAdd" class="btn-new">新增</el-button>
  111. </div>
  112. </el-form-item>
  113. </el-col>
  114. </el-row>
  115. </el-form>
  116. </el-card>
  117. <el-card shadow="never" class="table-card">
  118. <div class="tabs-control">
  119. <div class="tabs-left">
  120. <span class="page-title">年度入围(项目)信息列表</span>
  121. <div class="tab-items">
  122. <div class="tab-item" :class="{ current: activeTab === 'mine' }" @click="handleTabChange('mine')">我负责的年度入围</div>
  123. <div class="tab-item" :class="{ current: activeTab === 'join' }" @click="handleTabChange('join')">我参与的年度入围</div>
  124. <div class="tab-item badge-item" :class="{ current: activeTab === 'all' }" @click="handleTabChange('all')">
  125. 全部年度入围
  126. <span class="badge" v-if="total > 0">{{ total > 99 ? '99+' : total }}</span>
  127. </div>
  128. </div>
  129. <div class="summary-info">
  130. <span class="total-label">金额总计</span>
  131. <span class="total-value">{{ Number(totalAmount).toFixed(2) }}</span>
  132. <span class="total-unit">(万)</span>
  133. </div>
  134. </div>
  135. <div class="tabs-right">
  136. <div class="action-btns">
  137. <el-button type="info" plain size="default" icon="Delete" :disabled="multipleSelection.length === 0" @click="handleBatchDelete">批量删除</el-button>
  138. <el-button type="primary" size="default" icon="Switch" :disabled="multipleSelection.length === 0" @click="handleTransfer">转移给他人</el-button>
  139. </div>
  140. </div>
  141. </div>
  142. <el-table ref="selectionTable" v-loading="loading" :data="dataList" border class="standard-table" @selection-change="handleSelectionChange">
  143. <el-table-column type="selection" width="50" align="center" fixed />
  144. <el-table-column label="项目名称" align="left" prop="projectName" min-width="220" show-overflow-tooltip fixed>
  145. <template #default="{ row }">
  146. <span class="project-name-link" @click="handleDetail(row)">{{ row.projectName }}</span>
  147. </template>
  148. </el-table-column>
  149. <el-table-column label="所属公司" align="center" min-width="180" show-overflow-tooltip>
  150. <template #default="{ row }">
  151. {{ companyOptions.find(c => String(c.companyCode || c.id) === String(row.companyNo))?.companyName || row.companyName || row.companyNo }}
  152. </template>
  153. </el-table-column>
  154. <el-table-column label="客户名称" align="center" min-width="200" show-overflow-tooltip>
  155. <template #default="{ row }">
  156. {{ row.customerName || row.customName || '--' }}
  157. </template>
  158. </el-table-column>
  159. <el-table-column label="项目级别" align="center" prop="projectLevel" width="100">
  160. <template #default="{ row }">
  161. {{ projectLevelOptions.find(o => String(o.value) === String(row.projectLevel))?.label || row.projectLevel }}
  162. </template>
  163. </el-table-column>
  164. <el-table-column label="项目类型" align="center" prop="businessType" width="100">
  165. <template #default="{ row }">
  166. {{ projectTypeOptions.find(o => String(o.value) === String(row.businessType))?.label || row.businessType }}
  167. </template>
  168. </el-table-column>
  169. <el-table-column label="金额(万)" align="center" prop="amount" width="100">
  170. <template #default="{ row }">
  171. <span>{{ row.amount != null ? Number(row.amount).toFixed(2) : '' }}</span>
  172. </template>
  173. </el-table-column>
  174. <el-table-column label="服务期(年)" align="center" prop="standardPeriod" width="100" />
  175. <el-table-column label="服务时间段" align="center" width="180" show-overflow-tooltip>
  176. <template #default="{ row }">
  177. <span v-if="row.serviceStartTime">{{ row.serviceStartTime }} ~ {{ row.serviceEndTime }}</span>
  178. <span v-else>{{ row.serviceTime || '--' }}</span>
  179. </template>
  180. </el-table-column>
  181. <el-table-column label="投标截止时间" align="center" prop="tenderDeadline" width="120">
  182. <template #default="{ row }">
  183. <span>{{ proxy.parseTime(row.tenderDeadline, '{y}-{m}-{d}') }}</span>
  184. </template>
  185. </el-table-column>
  186. <el-table-column label="部门" align="center" prop="deptName" min-width="120" show-overflow-tooltip />
  187. <el-table-column label="业务负责人" align="center" prop="leaderName" width="100">
  188. <template #default="{ row }">
  189. <span>{{ row.leaderName || findUserName(row.managerId || row.leader) }}</span>
  190. </template>
  191. </el-table-column>
  192. <el-table-column label="产品支持" align="center" prop="productSupportName" width="100">
  193. <template #default="{ row }">
  194. <span>{{ row.productSupportName || findUserName(row.productSupport) }}</span>
  195. </template>
  196. </el-table-column>
  197. <el-table-column label="赢单率" align="center" width="80">
  198. <template #default="{ row }">
  199. <span>{{ (row.winRate || row.winningRate) ? (row.winRate || row.winningRate) + '%' : '--' }}</span>
  200. </template>
  201. </el-table-column>
  202. <el-table-column label="行业" align="center" prop="industryName" width="120" show-overflow-tooltip>
  203. <template #default="{ row }">
  204. {{ row.industryName || industryOptions.find(o => String(o.id) === String(row.industryId || row.industry || row.profession))?.industryCategoryName || '' }}
  205. </template>
  206. </el-table-column>
  207. <el-table-column label="创建时间" align="center" prop="createTime" width="120">
  208. <template #default="{ row }">
  209. <span>{{ proxy.parseTime(row.createTime, '{y}-{m}-{d}') }}</span>
  210. </template>
  211. </el-table-column>
  212. <el-table-column label="报名截止时间" align="center" width="120">
  213. <template #default="{ row }">
  214. <span>{{ proxy.parseTime(row.registrationDeadline || row.signUpDeadline, '{y}-{m}-{d}') }}</span>
  215. </template>
  216. </el-table-column>
  217. <el-table-column label="入围类型" align="center" prop="finalizationType" width="100">
  218. <template #default="{ row }">
  219. {{ shortlistedTypeOptions.find(o => String(o.value) === String(row.finalizationType))?.label || row.finalizationType }}
  220. </template>
  221. </el-table-column>
  222. <el-table-column label="项目状态" align="center" prop="projectStatus" width="100">
  223. <template #default="{ row }">
  224. <span class="status-tag" :class="getStatusClass(row.projectStatus)">
  225. {{ statusOptions.find(o => String(o.value) === String(row.projectStatus))?.label || (String(row.projectStatus) === '2' ? '进行中' : (String(row.projectStatus) === '0' || String(row.projectStatus) === '1' ? '跟进中' : row.projectStatus)) }}
  226. </span>
  227. </template>
  228. </el-table-column>
  229. <el-table-column label="成交结果" align="center" width="100">
  230. <template #default="{ row }">
  231. <template v-if="row.result === '1' || row.dealResult === '1' || row.result === 1 || row.dealResult === 1">
  232. <span style="color: #67c23a;">赢单</span>
  233. </template>
  234. <template v-else-if="row.result === '2' || row.dealResult === '2' || row.result === 2 || row.dealResult === 2">
  235. <span style="color: #f56c6c;">丢单</span>
  236. </template>
  237. <template v-else>
  238. <span :style="{ color: (row.result === '赢单' || row.dealResult === '赢单') ? '#67c23a' : (row.result === '丢单' || row.dealResult === '丢单') ? '#f56c6c' : '' }">
  239. {{ row.result || row.dealResult || resultOptions.find(o => String(o.value) === String(row.result || row.dealResult))?.label || '' }}
  240. </span>
  241. </template>
  242. </template>
  243. </el-table-column>
  244. <el-table-column label="操作" align="center" width="180" fixed="right">
  245. <template #default="{ row }">
  246. <el-button link type="primary" @click="handleDetail(row)">详情</el-button>
  247. <el-button link type="primary" @click="handleProgress(row)">进度</el-button>
  248. <el-button link type="danger" @click="handleDelete(row)">删除</el-button>
  249. </template>
  250. </el-table-column>
  251. </el-table>
  252. <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
  253. </el-card>
  254. <!-- 弹窗组件 -->
  255. <SelectionAdd
  256. v-model="addVisible"
  257. :companyOptions="companyOptions"
  258. :userOptions="userOptions"
  259. :projectLevelOptions="projectLevelOptions"
  260. :projectTypeOptions="projectTypeOptions"
  261. :shortlistedTypeOptions="shortlistedTypeOptions"
  262. :professionOptions="professionOptions"
  263. :industryOptions="industryOptions"
  264. @success="getList"
  265. />
  266. <SelectionEdit
  267. v-model="editVisible"
  268. :id="currentId"
  269. :companyOptions="companyOptions"
  270. :userOptions="userOptions"
  271. :projectLevelOptions="projectLevelOptions"
  272. :projectTypeOptions="projectTypeOptions"
  273. :shortlistedTypeOptions="shortlistedTypeOptions"
  274. :professionOptions="professionOptions"
  275. :industryOptions="industryOptions"
  276. @success="getList"
  277. />
  278. <SelectionDetail
  279. ref="detailDrawerRef"
  280. v-model="detailVisible"
  281. :companyOptions="companyOptions"
  282. :userOptions="userOptions"
  283. :statusOptions="statusOptions"
  284. :projectLevelOptions="projectLevelOptions"
  285. :projectTypeOptions="projectTypeOptions"
  286. :shortlistedTypeOptions="shortlistedTypeOptions"
  287. :professionOptions="professionOptions"
  288. :industryOptions="industryOptions"
  289. :deptOptions="deptOptions"
  290. @edit="handleEdit"
  291. @success="getList"
  292. />
  293. <!-- 弹窗:转移他人 -->
  294. <el-dialog title="项目转移" v-model="transferVisible" width="500px">
  295. <el-form label-width="100px">
  296. <el-form-item label="新负责人" required>
  297. <el-select v-model="transferOwner" placeholder="请选择" style="width: 100%" filterable>
  298. <el-option v-for="item in userOptions" :key="item.staffId || item.userId" :label="(item.staffCode || item.userName ? '(' + (item.staffCode || item.userName) + ') ' : '') + (item.staffName || item.nickName)" :value="String(item.staffId || item.userId)" />
  299. </el-select>
  300. </el-form-item>
  301. <el-form-item>
  302. <el-checkbox v-model="keepAsMember">保留原负责人为团队成员</el-checkbox>
  303. </el-form-item>
  304. </el-form>
  305. <template #footer>
  306. <div class="dialog-footer">
  307. <el-button type="primary" @click="confirmTransfer">确认</el-button>
  308. <el-button @click="transferVisible = false">取消</el-button>
  309. </div>
  310. </template>
  311. </el-dialog>
  312. <!-- 抽屉:查看/发布进度 -->
  313. <el-drawer title="查看进度" v-model="progressVisible" size="500px" class="progress-drawer">
  314. <div class="progress-container">
  315. <!-- 发布区域 -->
  316. <div class="progress-publish-box">
  317. <el-input
  318. v-model="progressContent"
  319. type="textarea"
  320. :rows="6"
  321. placeholder="请输入进度描述"
  322. maxlength="200"
  323. show-word-limit
  324. resize="none"
  325. />
  326. <div class="publish-footer">
  327. <el-button type="primary" icon="Plus" @click="submitProgress" :loading="publishLoading">发 布</el-button>
  328. </div>
  329. </div>
  330. <!-- 列表区域 -->
  331. <div class="progress-list" v-loading="progressListLoading">
  332. <div v-for="(item, index) in progressList" :key="index" class="progress-item">
  333. <div class="item-header">
  334. <span class="user-name">{{ item.userName || item.createBy }}</span>
  335. <span class="time">{{ item.createTime || item.followUpTime }}</span>
  336. </div>
  337. <div class="item-content">{{ item.content || item.followUpCondition }}</div>
  338. </div>
  339. <el-empty v-if="progressList.length === 0 && !progressListLoading" description="暂无进度数据" :image-size="60" />
  340. </div>
  341. <!-- 分页 -->
  342. <div class="progress-pagination" v-if="progressTotal > 0">
  343. <el-pagination
  344. v-model:current-page="progressPage"
  345. v-model:page-size="progressPageSize"
  346. layout="prev, pager, next"
  347. :total="progressTotal"
  348. @current-change="loadProgress"
  349. small
  350. />
  351. </div>
  352. </div>
  353. </el-drawer>
  354. </div>
  355. </template>
  356. <script setup name="ProjectSelection">
  357. import { ref, reactive, onMounted, getCurrentInstance, toRefs } from 'vue';
  358. import { listProjectSelection, delProjectSelection, transferProjectSelection, publishProjectProgress } from '@/api/saleManage/projectSelection/index';
  359. import { listSalesResultAnalyze } from '@/api/saleManage/leads/salesResultAnalyze';
  360. import { listRecord, addRecord } from "@/api/visit/record";
  361. import { listIndustryCategory } from "@/api/customer/industryCategory";
  362. import { listComStaff } from "@/api/system/comStaff/index";
  363. import { listCompanyOption, getCustomerInfo } from "@/api/customer/customerInfo/index";
  364. import { deptTreeSelect } from "@/api/system/user";
  365. import SelectionAdd from './add.vue';
  366. import SelectionEdit from './edit.vue';
  367. import SelectionDetail from './detail.vue';
  368. const proxy = getCurrentInstance().proxy;
  369. const {
  370. L0001: projectLevelOptions,
  371. XMJB0001: projectTypeOptions,
  372. R0001: shortlistedTypeOptions,
  373. J0001: statusOptions,
  374. deal_result: resultOptions,
  375. time_query_type: timeTypeOptions,
  376. ZBPL0001: professionOptions
  377. } = toRefs(reactive(proxy.useDict('XMJB0001', 'L0001', 'R0001', 'J0001', 'deal_result', 'time_query_type', 'ZBPL0001')));
  378. const loading = ref(false);
  379. const progressVisible = ref(false);
  380. const progressContent = ref('');
  381. const publishLoading = ref(false);
  382. const total = ref(0);
  383. const dataList = ref([]);
  384. const dateRange = ref([]);
  385. const multipleSelection = ref([]);
  386. const totalAmount = ref(0);
  387. const activeTab = ref('all');
  388. const addVisible = ref(false);
  389. const editVisible = ref(false);
  390. const detailVisible = ref(false);
  391. const currentId = ref(null);
  392. const detailDrawerRef = ref(null);
  393. const transferVisible = ref(false);
  394. const transferOwner = ref('');
  395. const keepAsMember = ref(false);
  396. const progressList = ref([]);
  397. const progressListLoading = ref(false);
  398. const currentProjectNo = ref('');
  399. const progressTotal = ref(0);
  400. const progressPage = ref(1);
  401. const progressPageSize = ref(5);
  402. const userOptions = ref([]);
  403. const companyOptions = ref([]);
  404. const deptOptions = ref([]);
  405. const industryOptions = ref([]);
  406. const queryParams = reactive({
  407. pageNum: 1, pageSize: 10,
  408. companyNo: undefined, projectName: undefined, customerName: undefined, managerId: undefined,
  409. deptId: undefined, projectLevel: undefined, businessType: undefined, projectStatus: undefined,
  410. timeType: undefined,
  411. productSupport: undefined,
  412. result: undefined,
  413. queryType: undefined,
  414. finalizationType: 2
  415. });
  416. const getList = async () => {
  417. loading.value = true;
  418. try {
  419. const params = { ...queryParams, dealResult: queryParams.result };
  420. const res = await listProjectSelection(proxy.addDateRange(params, dateRange.value));
  421. const list = res.rows || [];
  422. // 异步补全成交结果数据
  423. if (list.length > 0) {
  424. const projectNos = list.map(i => i.projectNo).filter(Boolean);
  425. if (projectNos.length > 0) {
  426. try {
  427. const analysisRes = await listSalesResultAnalyze({ objectNos: projectNos.join(',') });
  428. const analysisMap = {};
  429. (analysisRes.rows || analysisRes.data || []).forEach(a => {
  430. analysisMap[a.objectNo] = a.dealResult;
  431. });
  432. list.forEach(item => {
  433. if (item.projectNo && analysisMap[item.projectNo]) {
  434. item.dealResult = analysisMap[item.projectNo];
  435. }
  436. });
  437. } catch (e) {
  438. console.error('补全成交结果失败', e);
  439. }
  440. }
  441. // 异步补全行业信息 (根据客户动态显示)
  442. const missingIndustryList = list.filter(i => (i.customNo || i.customerNo) && !i.industryName);
  443. if (missingIndustryList.length > 0) {
  444. const uniqueNos = [...new Set(missingIndustryList.map(i => i.customNo || i.customerNo))];
  445. uniqueNos.forEach(async (no) => {
  446. try {
  447. const cRes = await getCustomerInfo(no);
  448. const cData = cRes.data || cRes;
  449. const industryId = cData.industryCategoryId || cData.industryId;
  450. // 尝试从返回的行业名称或根据 ID 从选项中查找
  451. const industryName = cData.industryName || industryOptions.value.find(o => String(o.id) === String(industryId))?.industryCategoryName;
  452. if (industryName) {
  453. list.forEach(item => {
  454. if ((item.customNo || item.customerNo) === no) {
  455. item.industryName = industryName;
  456. }
  457. });
  458. // 触发列表强制更新
  459. dataList.value = [...list];
  460. }
  461. } catch (e) {
  462. console.error('动态补全行业失败', e);
  463. }
  464. });
  465. }
  466. }
  467. dataList.value = list;
  468. total.value = res.total || 0;
  469. totalAmount.value = dataList.value.reduce((sum, item) => sum + (Number(item.amount) || 0), 0);
  470. } catch (err) {
  471. } finally {
  472. loading.value = false;
  473. }
  474. };
  475. const handleQuery = () => { queryParams.pageNum = 1; getList(); };
  476. const resetQuery = () => { dateRange.value = []; proxy.resetForm("queryRef"); handleQuery(); };
  477. const handleSelectionChange = (selection) => { multipleSelection.value = selection; };
  478. const handleTabChange = (tab) => {
  479. activeTab.value = tab;
  480. if (tab === 'mine') queryParams.queryType = 1;
  481. else if (tab === 'join') queryParams.queryType = 2;
  482. else delete queryParams.queryType;
  483. handleQuery();
  484. };
  485. const handleAdd = () => { addVisible.value = true; };
  486. const handleEdit = (row) => { currentId.value = row.id; editVisible.value = true; };
  487. const handleDetail = (row) => { detailDrawerRef.value.open(row); };
  488. const handleProgress = (row) => {
  489. currentId.value = row.id;
  490. currentProjectNo.value = row.projectNo;
  491. progressContent.value = '';
  492. progressPage.value = 1;
  493. progressVisible.value = true;
  494. loadProgress();
  495. };
  496. const loadProgress = async () => {
  497. if (!currentProjectNo.value) return;
  498. progressListLoading.value = true;
  499. try {
  500. const res = await listRecord({
  501. objectNo: currentProjectNo.value,
  502. pageNum: progressPage.value,
  503. pageSize: progressPageSize.value
  504. });
  505. progressList.value = res.rows || [];
  506. progressTotal.value = res.total || 0;
  507. } catch (err) {
  508. } finally {
  509. progressListLoading.value = false;
  510. }
  511. };
  512. const submitProgress = async () => {
  513. if (!progressContent.value.trim()) return proxy.$modal.msgWarning("请输入进度内容");
  514. publishLoading.value = true;
  515. try {
  516. await addRecord({
  517. objectNo: currentProjectNo.value,
  518. followUpCondition: progressContent.value,
  519. dataType: '2',
  520. callDate: proxy.parseTime(new Date(), '{y}-{m}-{d}'),
  521. callTypeCode: '1',
  522. callAim: '项目进度跟进'
  523. });
  524. proxy.$modal.msgSuccess("发布成功");
  525. progressContent.value = '';
  526. progressPage.value = 1;
  527. loadProgress();
  528. getList();
  529. } catch (err) {
  530. console.error(err);
  531. } finally {
  532. publishLoading.value = false;
  533. }
  534. };
  535. const handleTransfer = () => { transferVisible.value = true; };
  536. const confirmTransfer = async () => {
  537. if (!transferOwner.value) return proxy.$modal.msgWarning('请选择新负责人');
  538. const ids = multipleSelection.value.map(item => item.id);
  539. const user = userOptions.value.find(u => String(u.staffId || u.userId) === String(transferOwner.value));
  540. try {
  541. await transferProjectSelection({
  542. ids, leader: transferOwner.value, leaderName: user?.staffName || user?.nickName || '',
  543. keepOldOwner: keepAsMember.value ? 1 : 0
  544. });
  545. proxy.$modal.msgSuccess('转移成功');
  546. transferVisible.value = false;
  547. getList();
  548. } catch (err) {}
  549. };
  550. const handleDelete = (row) => {
  551. proxy.$modal.confirm(`确认删除「${row.projectName}」吗?`).then(() => {
  552. return delProjectSelection(row.id);
  553. }).then(() => {
  554. getList();
  555. proxy.$modal.msgSuccess("删除成功");
  556. }).catch(() => {});
  557. };
  558. const getStatusClass = (status) => {
  559. const s = String(status);
  560. if (s === '5') return 'status-success';
  561. return 'status-following';
  562. };
  563. const handleBatchDelete = () => {
  564. const ids = multipleSelection.value.map(item => item.id);
  565. proxy.$modal.confirm(`确认批量删除选中的 ${ids.length} 项记录吗?`).then(() => {
  566. return delProjectSelection(ids);
  567. }).then(() => {
  568. getList();
  569. proxy.$modal.msgSuccess("批量删除成功");
  570. }).catch(() => {});
  571. };
  572. const initOptions = () => {
  573. listIndustryCategory().then(res => {
  574. industryOptions.value = res.data || res.rows || [];
  575. });
  576. listComStaff({ pageSize: 1000 }).then(r => userOptions.value = r.rows || r.data || []);
  577. listCompanyOption().then(r => companyOptions.value = r.data);
  578. deptTreeSelect().then(res => {
  579. deptOptions.value = res.data;
  580. });
  581. };
  582. const findUserName = (id) => {
  583. if (!id) return '';
  584. const user = userOptions.value.find(u => String(u.staffId || u.userId) === String(id));
  585. return user ? (user.staffName || user.nickName) : id;
  586. };
  587. onMounted(() => {
  588. getList();
  589. initOptions();
  590. });
  591. </script>
  592. <style scoped lang="scss">
  593. .app-container { padding: 15px; background-color: #f7f9fb; min-height: calc(100vh - 84px); }
  594. .search-card {
  595. margin-bottom: 10px; border: none; background: #fff; border-radius: 8px;
  596. :deep(.el-form-item) { margin-bottom: 6px; }
  597. :deep(.el-form-item__label) { white-space: nowrap; font-weight: normal !important; color: #4e5969; }
  598. }
  599. .search-btns {
  600. display: flex;
  601. align-items: center;
  602. justify-content: flex-start;
  603. gap: 12px;
  604. .el-button { border-radius: 6px; height: 32px; padding: 8px 16px; }
  605. }
  606. .tabs-control {
  607. display: flex; align-items: center; justify-content: space-between; margin-bottom: 15px;
  608. .tabs-left { display: flex; align-items: center; gap: 20px;
  609. .page-title { font-size: 16px; color: #333; font-weight: normal; }
  610. .tab-items { display: flex; gap: 10px;
  611. .tab-item { padding: 6px 15px; border: 1px solid #dcdfe6; border-radius: 4px; font-size: 14px; color: #606266; cursor: pointer; transition: all 0.2s; position: relative; background: #fff;
  612. &:hover { color: #409eff; border-color: #c6e2ff; background-color: #ecf5ff; }
  613. &.current { color: #409eff; border-color: #409eff; background-color: #ecf5ff; }
  614. &.badge-item { padding-right: 25px; }
  615. .badge { position: absolute; top: -8px; right: -12px; background: #f56c6c; color: #fff; font-size: 12px; padding: 2px 6px; border-radius: 10px; line-height: 1; transform: scale(0.85); }
  616. }
  617. }
  618. .summary-info { font-size: 14px; color: #f56c6c; margin-left: 10px;
  619. .total-value { font-size: 16px; font-weight: normal; margin: 0 4px; }
  620. }
  621. }
  622. .tabs-right {
  623. .action-btns { display: flex; gap: 10px; }
  624. }
  625. }
  626. .table-card { border: none; }
  627. .standard-table {
  628. :deep(th.el-table__cell) { background-color: #f8fafc !important; color: #475569; font-weight: normal; }
  629. }
  630. .project-name-link { color: #409eff; cursor: pointer; &:hover { text-decoration: underline; } }
  631. .dialog-footer { display: flex; justify-content: center; gap: 20px; }
  632. .progress-publish-box {
  633. padding-bottom: 20px;
  634. .publish-footer {
  635. margin-top: 15px;
  636. display: flex;
  637. justify-content: flex-end;
  638. }
  639. }
  640. .progress-container {
  641. padding: 0 10px;
  642. display: flex;
  643. flex-direction: column;
  644. height: 100%;
  645. }
  646. .progress-list {
  647. flex: 1;
  648. overflow-y: auto;
  649. padding-top: 10px;
  650. .progress-item {
  651. padding: 15px 0;
  652. .item-header {
  653. display: flex;
  654. align-items: center;
  655. margin-bottom: 10px;
  656. .user-name { color: #333; font-size: 14px; margin-right: 15px; }
  657. .time { color: #333; font-size: 14px; }
  658. }
  659. .item-content {
  660. font-size: 14px;
  661. color: #333;
  662. line-height: 1.6;
  663. word-break: break-all;
  664. }
  665. }
  666. }
  667. .progress-pagination {
  668. margin-top: 10px;
  669. padding-bottom: 20px;
  670. display: flex;
  671. justify-content: flex-end;
  672. }
  673. /* 状态标签通用样式 */
  674. .status-tag {
  675. display: inline-block;
  676. padding: 1px 8px;
  677. border-radius: 4px;
  678. font-size: 12px;
  679. line-height: 18px;
  680. border: 1px solid transparent;
  681. }
  682. /* 跟进中 - 橙色 */
  683. .status-following {
  684. background-color: #fff7e8;
  685. color: #ff7d00;
  686. border-color: #ffcf8b;
  687. }
  688. /* 结案/成功 - 绿色 */
  689. .status-success {
  690. background-color: #e8ffea;
  691. color: #00b42a;
  692. border-color: #aff0b5;
  693. }
  694. .standard-table {
  695. :deep(.project-name-link) {
  696. color: #409eff !important;
  697. text-decoration: none !important;
  698. border-bottom: none !important;
  699. cursor: pointer;
  700. &:hover {
  701. color: #66b1ff !important;
  702. text-decoration: none !important;
  703. }
  704. }
  705. }
  706. </style>