list.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415
  1. <!--
  2. * @Author: wanglj 471442253@qq.com
  3. * @Date: 2022-12-26 16:34:37
  4. * @LastEditors: wanglj
  5. * @LastEditTime: 2023-02-13 17:46:36
  6. * @Description: file content
  7. * @FilePath: \opms_frontend\src\views\customer\list.vue
  8. -->
  9. <template>
  10. <div class="list-container">
  11. <el-tabs v-model="activeName" @tab-click="handleClick">
  12. <el-tab-pane label="全部客户" name="all" />
  13. <el-tab-pane label="我的客户" name="self" />
  14. <el-tab-pane label="下属的客户" name="other" />
  15. </el-tabs>
  16. <el-row :gutter="10" style="margin-bottom: 10px">
  17. <el-col :span="4">
  18. <el-input v-model="queryForm.custCode" placeholder="客户编码" />
  19. </el-col>
  20. <el-col :span="4">
  21. <el-input v-model="queryForm.custName" placeholder="客户名称" />
  22. </el-col>
  23. <el-col :span="4">
  24. <el-select v-model="queryForm.custIndustry" placeholder="客户类型" style="width: 100%">
  25. <el-option v-for="item in industryOptions" :key="item.value" :label="item.value" :value="item.value" />
  26. </el-select>
  27. </el-col>
  28. <!-- <el-col :span="4">
  29. <el-select v-model="queryForm.custLevel" placeholder="客户级别" style="width: 100%">
  30. <el-option v-for="item in levelOptions" :key="item.value" :label="item.value" :value="item.value" />
  31. </el-select>
  32. </el-col> -->
  33. <el-col :span="4">
  34. <el-date-picker
  35. v-model="queryForm.followUpDate"
  36. placeholder="最后跟进时间"
  37. style="width: 100%"
  38. type="date"
  39. value-format="yyyy-MM-dd" />
  40. </el-col>
  41. <el-col :span="4">
  42. <el-button icon="el-icon-plus" type="primary" @click="handleSearch">查询</el-button>
  43. <el-button icon="el-icon-refresh-right" @click="reset">重置</el-button>
  44. </el-col>
  45. </el-row>
  46. <vab-query-form>
  47. <vab-query-form-left-panel :span="12">
  48. <el-button
  49. v-permissions="['cust:list:add']"
  50. icon="el-icon-plus"
  51. size="mini"
  52. type="primary"
  53. @click="$refs.edit.init()">
  54. 新建
  55. </el-button>
  56. <el-button
  57. v-permissions="['cust:list:shift']"
  58. icon="el-icon-plus"
  59. size="mini"
  60. type="primary"
  61. @click="handleShift">
  62. 转移客户
  63. </el-button>
  64. <el-button
  65. v-permissions="['cust:list:open']"
  66. icon="el-icon-plus"
  67. size="mini"
  68. type="primary"
  69. @click="handleToOpen">
  70. 移入公海
  71. </el-button>
  72. <el-button
  73. v-permissions="['cust:list:merge']"
  74. icon="el-icon-plus"
  75. size="mini"
  76. type="primary"
  77. @click="handleMerge">
  78. 合并客户
  79. </el-button>
  80. </vab-query-form-left-panel>
  81. <vab-query-form-right-panel :span="12">
  82. <el-button icon="el-icon-download" @click="exportData" />
  83. <table-tool :check-list.sync="checkList" :columns="columns" />
  84. </vab-query-form-right-panel>
  85. </vab-query-form>
  86. <el-table
  87. v-loading="listLoading"
  88. border
  89. :data="list"
  90. :height="$baseTableHeight(3)"
  91. @selection-change="setSelectRows">
  92. <el-table-column align="center" show-overflow-tooltip type="selection" />
  93. <el-table-column
  94. v-for="(item, index) in finallyColumns"
  95. :key="index"
  96. align="center"
  97. :label="item.label"
  98. :min-width="item.width"
  99. :prop="item.prop"
  100. show-overflow-tooltip
  101. :sortable="item.sortable">
  102. <template #default="{ row }">
  103. <el-button v-if="item.prop === 'custName'" class="link-button" type="text" @click="handleDetail(row)">
  104. {{ row.custName }}
  105. </el-button>
  106. <span v-else-if="item.prop === 'custStatus'">
  107. {{ row.custStatus == 10 ? '正常' : '异常' }}
  108. </span>
  109. <span v-else-if="item.prop === 'custIndustry'">
  110. {{ selectDictLabel(industryOptions, row.custIndustry) }}
  111. </span>
  112. <span v-else-if="item.prop === 'followUpDate'">
  113. {{ parseTime(row.followUpDate, '{y}-{m}-{d} {h}:{i}') }}
  114. </span>
  115. <span v-else-if="item.prop === 'createdTime'">
  116. {{ parseTime(row.createdTime, '{y}-{m}-{d} {h}:{i}') }}
  117. </span>
  118. <span v-else>{{ row[item.prop] }}</span>
  119. </template>
  120. </el-table-column>
  121. <el-table-column align="center" fixed="right" label="操作" width="90px">
  122. <template slot-scope="scope">
  123. <el-button v-permissions="['cust:list:edit']" type="text" @click="handleEdit(scope.row)">编辑</el-button>
  124. <el-button v-permissions="['cust:list:delete']" type="text" @click="handleDelete(scope.row)">删除</el-button>
  125. </template>
  126. </el-table-column>
  127. </el-table>
  128. <el-pagination
  129. background
  130. :current-page="queryForm.pageNum"
  131. :layout="layout"
  132. :page-size="queryForm.pageSize"
  133. :total="total"
  134. @current-change="handleCurrentChange"
  135. @size-change="handleSizeChange" />
  136. <!-- 新增编辑客户弹窗 -->
  137. <Edit ref="edit" @createContact="createContact" @customerSave="fetchData" />
  138. <!-- 新建联系人弹窗 -->
  139. <Contact ref="contact" />
  140. <!-- 转移客户 -->
  141. <Shift ref="shift" @refresh="fetchData" />
  142. <!-- 移入公海 -->
  143. <ToOpen ref="toOpen" @refresh="fetchData" />
  144. <!-- 合并客户 -->
  145. <Merge ref="merge" @refresh="fetchData" />
  146. </div>
  147. </template>
  148. <script>
  149. import to from 'await-to-js'
  150. import api from '@/api/customer'
  151. import Edit from './components/Edit'
  152. import Contact from './components/Contact'
  153. import Shift from './components/Shift'
  154. import ToOpen from './components/ToOpen'
  155. import Merge from './components/Merge'
  156. import TableTool from '@/components/table/TableTool'
  157. import downloadFileByByte from '@/utils/base64ToFile'
  158. export default {
  159. components: {
  160. Edit,
  161. Contact,
  162. Shift,
  163. ToOpen,
  164. Merge,
  165. TableTool,
  166. },
  167. data() {
  168. return {
  169. activeName: 'all',
  170. layout: 'total, sizes, prev, pager, next, jumper',
  171. queryForm: {
  172. custCode: '',
  173. custName: '',
  174. indusTry: '',
  175. targetType: '',
  176. pageNum: 1,
  177. pageSize: 10,
  178. isPublic: false,
  179. },
  180. total: 0,
  181. listLoading: false,
  182. list: [],
  183. selectRows: [],
  184. industryOptions: [], //客户类型
  185. levelOptions: [], //客户级别
  186. // 自定义列表
  187. checkList: [],
  188. columns: [
  189. {
  190. label: '客户编码',
  191. width: '120px',
  192. prop: 'custCode',
  193. sortable: false,
  194. disableCheck: false,
  195. },
  196. {
  197. label: '客户名称',
  198. width: '160px',
  199. prop: 'custName',
  200. sortable: false,
  201. disableCheck: true,
  202. },
  203. {
  204. label: '助记名',
  205. width: '100px',
  206. prop: 'abbrName',
  207. sortable: false,
  208. disableCheck: false,
  209. },
  210. {
  211. label: '所在省',
  212. width: '120px',
  213. prop: 'custProvince',
  214. sortable: false,
  215. disableCheck: false,
  216. },
  217. {
  218. label: '所在市',
  219. width: '100px',
  220. prop: 'custCity',
  221. sortable: false,
  222. disableCheck: false,
  223. },
  224. {
  225. label: '客户类型',
  226. width: '100px',
  227. prop: 'custIndustry',
  228. sortable: false,
  229. disableCheck: false,
  230. },
  231. // {
  232. // label: '客户级别',
  233. // width: 'auto',
  234. // prop: 'custLevel',
  235. // sortable: false,
  236. // disableCheck: false,
  237. // },
  238. {
  239. label: '客户状态',
  240. width: '100px',
  241. prop: 'custStatus',
  242. sortable: false,
  243. disableCheck: false,
  244. },
  245. {
  246. label: '最后跟进时间',
  247. width: '150px',
  248. prop: 'followUpDate',
  249. sortable: false,
  250. disableCheck: false,
  251. },
  252. {
  253. label: '最后跟进人',
  254. width: '100px',
  255. prop: 'followUpMan',
  256. sortable: false,
  257. disableCheck: false,
  258. },
  259. {
  260. label: '创建时间',
  261. width: '150px',
  262. prop: 'createdTime',
  263. sortable: false,
  264. disableCheck: false,
  265. },
  266. ],
  267. }
  268. },
  269. computed: {
  270. finallyColumns() {
  271. return this.columns.filter((item) => this.checkList.includes(item.label))
  272. },
  273. },
  274. mounted() {
  275. this.getOptions()
  276. this.fetchData()
  277. },
  278. methods: {
  279. getOptions() {
  280. Promise.all([this.getDicts('cust_level'), this.getDicts('cust_idy')])
  281. .then(([level, industry]) => {
  282. this.levelOptions = level.data.values || []
  283. this.industryOptions = industry.data.values || []
  284. })
  285. .catch((err) => console.log(err))
  286. },
  287. handleClick(tab) {
  288. console.log(tab, 'tab')
  289. this.queryForm.pageNum = 1
  290. this.queryForm.targetType = this.activeName
  291. this.fetchData()
  292. },
  293. async fetchData() {
  294. this.listLoading = true
  295. const params = { ...this.queryForm }
  296. const [err, res] = await to(api.getList(params))
  297. if (err) return (this.listLoading = false)
  298. this.list = res.data.list || []
  299. this.total = res.data.total
  300. this.listLoading = false
  301. },
  302. handleSearch() {
  303. this.queryForm.pageNum = 1
  304. this.fetchData()
  305. },
  306. exportData() {
  307. this.queryForm.isPublic = false
  308. let exportFrom = JSON.parse(JSON.stringify(this.queryForm))
  309. exportFrom.columns = this.finallyColumns.map((item) => item.label)
  310. api
  311. .deriveList(exportFrom)
  312. .then((res) => {
  313. if (res.data.list.content) {
  314. downloadFileByByte(res.data.list.content, '客户数据.xlsx')
  315. }
  316. })
  317. .catch((err) => {
  318. console.error(err)
  319. })
  320. },
  321. reset() {
  322. this.queryForm = {
  323. pageNum: 1,
  324. pageSize: 10,
  325. custCode: '', // 客户编码
  326. custName: '', //客户名称
  327. indusTry: '', // 客户类型 ()
  328. }
  329. this.fetchData()
  330. },
  331. // 客户编辑
  332. handleEdit(row) {
  333. this.$refs.edit.init([row.id])
  334. },
  335. // 联系人弹窗
  336. async createContact(res) {
  337. this.$refs.contact.contactForm.custId = res.id
  338. this.$refs.contact.contactForm.custName = res.name
  339. this.$refs.contact.contactVisible = true
  340. },
  341. // 客户详情
  342. handleDetail(row) {
  343. this.$router.push({
  344. path: '/customer/detail',
  345. query: {
  346. id: row.id,
  347. },
  348. })
  349. },
  350. // 客户删除
  351. handleDelete(row) {
  352. this.$confirm('确认删除?', '提示', {
  353. confirmButtonText: '确定',
  354. cancelButtonText: '取消',
  355. type: 'warning',
  356. })
  357. .then(async () => {
  358. const [err, res] = await to(api.deleteCustomer({ Ids: [row.id] }))
  359. if (err) return
  360. if (res.code == 200) {
  361. this.$message({
  362. type: 'success',
  363. message: '删除成功!',
  364. })
  365. }
  366. })
  367. .catch(() => {})
  368. },
  369. handleSizeChange(val) {
  370. this.queryForm.pageSize = val
  371. this.fetchData()
  372. },
  373. handleCurrentChange(val) {
  374. this.queryForm.pageNum = val
  375. this.fetchData()
  376. },
  377. setSelectRows(val) {
  378. this.selectRows = val
  379. },
  380. // 转移客户
  381. handleShift() {
  382. if (!this.selectRows.length) return this.$message.warning('请选择需要转移的客户')
  383. this.$refs.shift.form.Ids = this.selectRows.map((item) => item.id)
  384. this.$refs.shift.visible = true
  385. },
  386. // 移入公海
  387. handleToOpen() {
  388. if (!this.selectRows.length) return this.$message.warning('请选择要移入公海的客户')
  389. this.$refs.toOpen.form.ids = this.selectRows.map((item) => item.id)
  390. this.$refs.toOpen.visible = true
  391. },
  392. async handleMerge() {
  393. if (this.selectRows.length < 2) return this.$message.warning('请选择两个以上客户进行合并')
  394. const ids = this.selectRows.map((item) => item.id)
  395. const [err, res] = await to(api.getDetail({ ids }))
  396. if (err) return
  397. this.$refs.merge.init(res, ids)
  398. },
  399. },
  400. }
  401. </script>
  402. <style lang="scss" scoped>
  403. $base: '.list';
  404. .link-button {
  405. font-size: 14px;
  406. width: 100%;
  407. overflow: hidden;
  408. text-overflow: ellipsis;
  409. white-space: nowrap;
  410. }
  411. </style>