index.vue 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. <!-- 精英管理 -->
  2. <template>
  3. <v-card class="pa-3 card-box">
  4. <div class="d-flex justify-space-between mb-3">
  5. <v-tabs v-model="tab" align-tabs="start" color="primary" bg-color="#f7f8fa" @update:model-value="handleChangeTab">
  6. <v-tab v-for="k in tabList" :value="k.value" :key="k.value">{{ k.label }}</v-tab>
  7. </v-tabs>
  8. <!-- <div class="d-flex align-center">
  9. <TextInput v-model="textItems.value" :item="textItems" @appendInnerClick="handleSearch" @enter="handleSearch"></TextInput>
  10. </div> -->
  11. <v-btn color="primary" prependIcon="mdi-filter-multiple-outline" class="ml-3" @click="showDrawer = true">筛选{{ rawData.length > 0 ? rawData.length : '' }}</v-btn>
  12. </div>
  13. <div class="d-flex justify-space-between align-center mb-3" v-if="tab === 0">
  14. <div class="mr-5 d-flex align-center">
  15. <v-radio-group v-model="bounty" inline style="height: 28px;" @update:modelValue="handleChangeBounty">
  16. <v-radio v-model="bounty" label="普通职位" :value="false" color="primary" hide-details density="compact" class="mr-3"></v-radio>
  17. <v-radio v-model="bounty" label="赏金职位" :value="true" color="primary" hide-details density="compact"></v-radio>
  18. </v-radio-group>
  19. <v-radio-group class="ml-5" v-model="selected" inline style="height: 28px;" @update:modelValue="handleSelect">
  20. <v-radio v-model="selected" label="新投递" value="0" color="primary" hide-details density="compact" class="mr-3"></v-radio>
  21. <v-radio v-model="selected" label="已查看" value="1" color="primary" hide-details density="compact"></v-radio>
  22. </v-radio-group>
  23. </div>
  24. <div class="color-primary font-size-14 cursor-pointer" @click="FilterPageRef.handleReset(false)">重置</div>
  25. </div>
  26. <div v-if="rawData && rawData.length > 0">
  27. <v-chip v-for="item in rawData" :key="item.key" closable class="mr-2 mb-2" label @click:close="handleClose(item)">{{ item.title }}: {{ item.label || item.value }}</v-chip>
  28. </div>
  29. <!-- 筛选抽屉 -->
  30. <v-navigation-drawer v-model="showDrawer" location="right" absolute temporary width="700">
  31. <FilterPage ref="FilterPageRef" :jobId="route.query?.id" :jobFairId="route.query?.jobFairId" @confirm="handleConfirm" @cancel="showDrawer = false" @reset="handleScreenReset" />
  32. </v-navigation-drawer>
  33. <v-window v-model="tab" class="mt-1">
  34. <v-window-item v-for="k in tabList" :value="k.value" :key="k.value">
  35. <TablePage :items="items" :total="total" :pageInfo="query" :tab="k.value" :statusList="statusList" @refresh="getList" @page="handleChangePage"></TablePage>
  36. </v-window-item>
  37. </v-window>
  38. </v-card>
  39. </template>
  40. <script setup>
  41. defineOptions({ name: 'enterprise-elite-management'})
  42. import { ref, onMounted } from 'vue'
  43. import { getPersonCvPage } from '@/api/enterprise'
  44. import { personCvUnfitPage } from '@/api/recruit/enterprise/personnel'
  45. import { dealDictObjData } from '@/utils/position'
  46. import { getDict } from '@/hooks/web/useDictionaries'
  47. import { getInterviewInvitePage } from '@/api/recruit/enterprise/interview'
  48. import TablePage from './components/table.vue'
  49. import { timesTampChange } from '@/utils/date'
  50. import { useRoute } from 'vue-router'
  51. import FilterPage from './components/filterPage.vue'
  52. const route = useRoute()
  53. const total = ref(0)
  54. const query = ref({
  55. pageNo: 1,
  56. pageSize: 10,
  57. status: null,
  58. type: null
  59. })
  60. const selected = ref()
  61. const bounty = ref(null)
  62. const showDrawer = ref(false)
  63. const FilterPageRef = ref()
  64. const tab = ref(0)
  65. const tabList = ref([
  66. { label: '投递简历', value: 0, api: getPersonCvPage, status: null },
  67. { label: '已邀约', value: 1, api: getInterviewInvitePage, status: '0' },
  68. { label: '已入职', value: 2, api: getInterviewInvitePage, status: '1' },
  69. { label: '已结算', value: 3, api: getInterviewInvitePage, status: '2' },
  70. { label: '不合适', value: 4, api: personCvUnfitPage },
  71. ])
  72. const textItems = ref({
  73. type: 'text',
  74. value: '',
  75. width: 250,
  76. label: '搜索姓名',
  77. hideDetails: true,
  78. clearable: true,
  79. appendInnerIcon: 'mdi-magnify'
  80. })
  81. // 状态字典
  82. const statusList = ref([])
  83. getDict('menduner_interview_invite_status').then(({data}) => {
  84. if (data && data.length) statusList.value = data
  85. })
  86. // 获取牛人列表
  87. const items = ref([])
  88. const getList = async () => {
  89. const api = tabList.value[tab.value].api
  90. if (tab.value !== 0) {
  91. query.value.conversationStatus = tabList.value[tab.value].status
  92. delete query.value.status
  93. delete query.value.type
  94. }
  95. const { list, total: number } = await api(query.value)
  96. if (!list.length) {
  97. items.value = []
  98. total.value = 0
  99. return
  100. }
  101. total.value = number
  102. items.value = list.map(e => {
  103. let obj = e
  104. obj.jobClosed = e?.job?.status === '1' || false // 职位已关闭
  105. if (e.person) obj.person = Object.assign(e.person, dealDictObjData({}, e.person))
  106. obj.job = Object.assign(e.job, dealDictObjData({}, e.job))
  107. obj.createTime = timesTampChange(e.createTime, 'Y-M-D h:m')
  108. return obj
  109. })
  110. }
  111. // 没有带id时一进来才刷新,带id由组件传值刷新
  112. if (!route.query?.id) getList()
  113. onMounted(() => {
  114. // 众聘职位
  115. if (route.query?.hire) {
  116. query.value.hire = true
  117. bounty.value = true
  118. }
  119. // 招聘会职位
  120. if (route.query?.jobFairId) {
  121. if (tab.value !== 0) return delete query.value.jobFairId
  122. query.value.jobFairId = route.query.jobFairId
  123. }
  124. })
  125. // 分页
  126. const handleChangePage = (i) => {
  127. query.value.pageNo = i
  128. getList()
  129. }
  130. // 牛人姓名检索
  131. const handleSearch = () => {
  132. if (textItems.value.value) query.value.name = textItems.value.value
  133. else delete query.value.name
  134. query.value.pageNo = 1
  135. getList()
  136. }
  137. // 筛选
  138. const rawData = ref([])
  139. const handleConfirm = (params, data) => {
  140. rawData.value = data
  141. query.value.pageNo = 1
  142. query.value = Object.assign(query.value, params)
  143. showDrawer.value = false
  144. getList()
  145. }
  146. // 筛选条件单个清除
  147. const handleClose = (item) => {
  148. FilterPageRef.value.handleClear(item)
  149. delete query.value[item.key]
  150. rawData.value = rawData.value.filter(e => e.key !== item.key)
  151. getList()
  152. }
  153. // 重置
  154. const handleScreenReset = (close) => {
  155. rawData.value = []
  156. bounty.value = ''
  157. selected.value = ''
  158. query.value = {
  159. pageSize: 10,
  160. pageNo: 1,
  161. status: tab.value
  162. }
  163. if (tab.value === 0) {
  164. query.value.status = null
  165. query.value.type = null
  166. }
  167. if (close) showDrawer.value = false
  168. getList()
  169. }
  170. // 职位类型选择
  171. const handleSelect = (e) => {
  172. query.value.pageNo = 1
  173. query.value.status = e
  174. getList()
  175. }
  176. // 赏金职位
  177. const handleChangeBounty = (e) => {
  178. query.value.type = e ? 1 : 0
  179. query.value.pageNo = 1
  180. getList()
  181. }
  182. const handleChangeTab = (val) => {
  183. query.value.pageNo = 1
  184. if (val === 0) {
  185. query.value.status = null
  186. query.value.type = null
  187. } else query.value.status = val
  188. getList()
  189. }
  190. </script>
  191. <style scoped lang="scss">
  192. </style>