vitae.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470
  1. <template>
  2. <div class="card-search">
  3. <div class="jl-content">
  4. <!-- 头像等信息 -->
  5. <div class="avt-content">
  6. <!--头像 -->
  7. <div class="avt-image">
  8. <img src="../../../assets/images/njl-avt.png" v-if="jobUserInfo.genderName=='女'">
  9. <img src="../../../assets/images/jl-avt.png" v-if="jobUserInfo.genderName=='男'">
  10. <!-- <a-avatar :size="72" :src="jobUserInfo.genderName === '男'?avtImg:avtNImg"></a-avatar>-->
  11. </div>
  12. <!-- 姓名 -->
  13. <p class="nameTitle">{{ jobUserInfo.name }}</p>
  14. <!-- 性别,年龄,民族 -->
  15. <div class="age-info">
  16. <span>
  17. <man-outlined v-if="jobUserInfo.genderName === '男'"/>
  18. <woman-outlined v-if="jobUserInfo.genderName === '女'"/>
  19. {{ jobUserInfo.genderName }}
  20. </span>
  21. <span>
  22. {{ jobUserInfo.age }}岁
  23. </span>
  24. <span>
  25. {{ jobUserInfo.nationName }}
  26. </span>
  27. </div>
  28. <!-- 政治面貌,学历,手机 -->
  29. <div class="politics-info">
  30. <span class="label-span">
  31. 政治面貌:
  32. </span>
  33. <span>
  34. {{ jobUserInfo.politicsStatusName }}
  35. </span>
  36. <span class="label-span">
  37. 学历:
  38. </span>
  39. <span>
  40. {{ jobUserInfo.cultureName }}
  41. </span>
  42. <span class="label-span">
  43. 手机:
  44. </span>
  45. <span>
  46. {{ jobUserInfo.userMobile }}
  47. </span>
  48. </div>
  49. <!-- 标签 -->
  50. <div class="label-info" v-if="jobUserInfo.listLabel.length >= 1">
  51. <p class="title">标签</p>
  52. <div class="flex-box">
  53. <a-tag v-for="(item, index) in jobUserInfo.listLabel" :key="index">
  54. {{ item.labelName }}
  55. </a-tag>
  56. </div>
  57. </div>
  58. </div>
  59. <!-- 基本信息与学历,工作经历 -->
  60. <div class="info-content">
  61. <a-divider orientation="left">基本信息</a-divider>
  62. <a-descriptions :column="2" bordered style="margin-left: 25px;">
  63. <a-descriptions-item label="重点人员类别">{{ jobUserInfo.keyTypeName }}</a-descriptions-item>
  64. <a-descriptions-item label="健康状况">{{ jobUserInfo.healthName }}</a-descriptions-item>
  65. <a-descriptions-item label="就业状态">{{ jobUserInfo.jobStatusName }}</a-descriptions-item>
  66. <a-descriptions-item label="常住地址">{{ jobUserInfo.address }}</a-descriptions-item>
  67. <a-descriptions-item :span="2" label="个人评价">{{ jobUserInfo.hobby }}</a-descriptions-item>
  68. <a-descriptions-item :span="2" label="专业技术特长">{{ jobUserInfo.personalSkills }}</a-descriptions-item>
  69. </a-descriptions>
  70. <a-divider orientation="left">教育经历</a-divider>
  71. <a-table :columns="educationColumns" :data-source="educationData" :pagination="false" bordered
  72. style="margin-left: 11px;">
  73. <template #bodyCell="{ column, index, record}">
  74. <template v-if="column.key === 'cultureRank'">
  75. <div>
  76. {{
  77. getCultureName(record.cultureRank)
  78. }}
  79. </div>
  80. </template>
  81. <template v-if="column.key === 'schoolTime'">
  82. <div>
  83. {{
  84. dayjs(record.schoolTime).format('YYYY-MM-DD')
  85. }}
  86. </div>
  87. </template>
  88. <template v-if="column.key === 'overTime'">
  89. <div>
  90. {{
  91. dayjs(record.overTime).format('YYYY-MM-DD')
  92. }}
  93. </div>
  94. </template>
  95. </template>
  96. </a-table>
  97. <a-divider orientation="left">求职意向</a-divider>
  98. <a-table :columns="jobHuntColumns" :data-source="jobHuntList" bordered :pagination="false"
  99. style="margin-left: 11px">
  100. </a-table>
  101. <a-divider orientation="left">工作经历</a-divider>
  102. <a-timeline style="margin-left: 25px;">
  103. <a-timeline-item v-for="(item, key) in experienceData" :key="key" position="left">
  104. <p>{{ dayjs(item.startTime).format('YYYY-MM-DD') }}至{{ dayjs(item.endTime).format('YYYY-MM-DD') }}</p>
  105. <p>工作单位:{{ item.workAddress }}</p>
  106. <p>职务:{{ item.duties }}</p>
  107. </a-timeline-item>
  108. </a-timeline>
  109. <a-empty :image="simpleImage" v-if="experienceData.length==0" ></a-empty>
  110. <a-divider orientation="left">个人荣誉</a-divider>
  111. <a-table :columns="HonorColumns" :data-source="honorData" bordered :pagination="false"
  112. style="margin-left: 11px">
  113. <template #bodyCell="{ column, index, record}">
  114. <template v-if="column.key === 'getTime'">
  115. <div>
  116. {{
  117. dayjs(record.getTime).format('YYYY-MM-DD')
  118. }}
  119. </div>
  120. </template>
  121. </template>
  122. </a-table>
  123. <a-divider orientation="left">附件</a-divider>
  124. <b-upload-file :fileRefId="jobUserInfo.jobUserID" :readonly="true" :multiple="true" style="width: 100%;"
  125. :setFileList="setFileList" :accept="'.pdf,.png,.jpg'"></b-upload-file>
  126. </div>
  127. </div>
  128. </div>
  129. </template>
  130. <script setup lang="ts">
  131. import {onMounted, reactive, ref} from "vue";
  132. import {getDataById, getEducationList, getExperienceList, getHonorList} from "@/api/jobUserManager/jobuser";
  133. import {ManOutlined, WomanOutlined} from '@ant-design/icons-vue';
  134. import type {SelectProps, TableColumnsType} from "ant-design-vue";
  135. import dayjs from "dayjs";
  136. import {getSysDictionaryList} from "@/api/system/dictionary";
  137. import {getJobHuntList} from "@/api/jobUserManager/jobhunt";
  138. import {useUserStore} from "@/store/modules/user";
  139. import crtyptoHelp from "@/utils/crypto";
  140. import BUploadFile from "@/components/file/uploadFile.vue";
  141. import {Empty} from "ant-design-vue";
  142. const userStore = useUserStore();
  143. const userInfo = ref(userStore.getUserInfo)
  144. // 求职人员信息
  145. const jobUserInfo = reactive({
  146. jobUserID: "",
  147. name: "",
  148. genderName: "",
  149. age: null,
  150. nationName: "",
  151. politicsStatusName: "",
  152. cultureName: "",
  153. userMobile: "",
  154. birthPlace: "",
  155. identityNumber: "",
  156. keyTypeName: "",
  157. healthName: "",
  158. jobStatusName: "",
  159. familyNatureName: "",
  160. address: "",
  161. familyAddress: "",
  162. hobby: "",
  163. personalSkills: "",
  164. listLabel: new Array<any>()
  165. })
  166. // 受教育经历
  167. const educationData = ref<Array<any>>([]);
  168. const simpleImage = Empty.PRESENTED_IMAGE_SIMPLE;
  169. // 受教育经历表格定义
  170. const educationColumns: TableColumnsType = [
  171. {
  172. title: '序号',
  173. align: "center",
  174. key: 'educationID',
  175. width: 120,
  176. customRender: item => `${searchParams.pageSize * (searchParams.pageIndex - 1) + item.index + 1}`
  177. },
  178. {
  179. title: '学校名',
  180. dataIndex: 'schoolName',
  181. key: 'schoolName',
  182. align: "center",
  183. width: 120
  184. },
  185. {
  186. title: '文化程度',
  187. dataIndex: 'cultureRank',
  188. key: 'cultureRank',
  189. align: "center",
  190. width: 120
  191. },
  192. {
  193. title: '就读时间',
  194. dataIndex: 'schoolTime',
  195. key: 'schoolTime',
  196. align: "center",
  197. width: 120
  198. },
  199. {
  200. title: '毕业时间',
  201. dataIndex: 'overTime',
  202. key: 'overTime',
  203. align: "center",
  204. width: 120
  205. },
  206. {
  207. title: '专业',
  208. dataIndex: 'major',
  209. key: 'major',
  210. align: "center",
  211. width: 120
  212. },
  213. ];
  214. const searchParams = reactive({
  215. pageIndex: 1,
  216. pageSize: 99
  217. });
  218. // 工作经验数据
  219. const experienceData = ref<Array<any>>([]);
  220. // 求职意向数据
  221. const jobHuntList = ref<Array<any>>([]);
  222. // 求职意向表格定义
  223. const jobHuntColumns: TableColumnsType = [
  224. {
  225. title: '序号',
  226. align: 'center',
  227. width: 80,
  228. key: 'jobHuntID',
  229. customRender: (item) =>
  230. `${searchParams.pageSize * (searchParams.pageIndex - 1) + item.index + 1}`,
  231. },
  232. {title: '希望工作地区', dataIndex: 'areaWork', key: 'areaWork', width: 150, align: "center",},
  233. {
  234. title: '可到职日期', dataIndex: 'inDate', key: 'inDate', width: 100, align: "center",
  235. customRender: ({record}) => record.inDate == null ? "" : dayjs(record.inDate).format('YYYY-MM-DD'),
  236. },
  237. {title: '工作年限', dataIndex: 'workYear', key: 'workYear', align: "center",},
  238. {title: '求职类型', dataIndex: 'jobHuntTypeStr', key: 'jobHuntTypeStr', align: "center",},
  239. {title: '求职岗位', dataIndex: 'professionName', key: 'professionName', align: "center",},
  240. {title: '人才类型', dataIndex: 'jobUserTypeStr', key: 'jobUserTypeStr', align: "center",},
  241. {
  242. title: '月薪要求', dataIndex: 'salary', key: 'salary', align: "center",
  243. customRender: (item) => {
  244. const salary = showSalary(item.record.minSalary, item.record.maxSalary);
  245. return salary;
  246. }
  247. },
  248. ];
  249. const HonorColumns: TableColumnsType = [
  250. {
  251. title: '序号',
  252. align: "center",
  253. key: 'honorID',
  254. width: 100,
  255. customRender: item => `${searchParams.pageSize * (searchParams.pageIndex - 1) + item.index + 1}`
  256. },
  257. {
  258. title: '荣誉名称',
  259. dataIndex: 'honorName',
  260. key: 'honorName',
  261. align: "center",
  262. width: 300
  263. },
  264. {
  265. title: '获得时间',
  266. dataIndex: 'getTime',
  267. key: 'getTime',
  268. align: "center",
  269. width: 200
  270. },
  271. ]
  272. const honorData = ref([]);
  273. // 期望月薪显示优化方法
  274. const showSalary = (minSalary: any, maxSalary: any) => {
  275. if (minSalary != null) {
  276. if (maxSalary != null) {
  277. return minSalary.toString() + "-" + maxSalary.toString();
  278. } else {
  279. return "≥" + minSalary.toString();
  280. }
  281. } else {
  282. if (maxSalary != null) {
  283. return "≤" + maxSalary.toString();
  284. } else {
  285. return "";
  286. }
  287. }
  288. }
  289. // 教育经历数据
  290. const cultureList = ref<SelectProps['options']>();
  291. // 加载求职人员数据
  292. const loadData = (id: any) => {
  293. getDataById(id, userInfo.value.userID, false).then(data => {
  294. Object.keys(jobUserInfo).forEach(key => {
  295. jobUserInfo[key] = data[key];
  296. })
  297. jobUserInfo.userMobile = crtyptoHelp.decryptDesText(jobUserInfo.userMobile);
  298. jobUserInfo.identityNumber = crtyptoHelp.decryptDesText(jobUserInfo.identityNumber);
  299. });
  300. };
  301. // 加载教育经历
  302. const loadEducation = (id: any) => {
  303. getEducationList(id, true).then(data => {
  304. if (data) {
  305. educationData.value = data;
  306. }
  307. });
  308. }
  309. // 加载工作经历
  310. const loadExperienceData = (id: any) => {
  311. getExperienceList(id, true).then(data => {
  312. if (data) {
  313. experienceData.value = data;
  314. }
  315. })
  316. }
  317. // 加载个人荣誉
  318. const loadHonorData = (id: any) => {
  319. getHonorList(id).then(data => {
  320. if (data) {
  321. honorData.value = data;
  322. }
  323. })
  324. }
  325. // 加载求职意向数据
  326. function loadJobHuntData(jobUserID: any) {
  327. let params = {
  328. pageIndex: 1,
  329. pageSize: 999,
  330. jobUserID
  331. }
  332. getJobHuntList(params).then(result => {
  333. jobHuntList.value = result.list
  334. })
  335. }
  336. // 获取教育经历数据
  337. const getCultureList = () => {
  338. getSysDictionaryList('HighestDegree').then((data) => {
  339. cultureList.value = data;
  340. });
  341. };
  342. function getCultureName(value: any) {
  343. let filter = cultureList.value?.filter(item => item.value == value);
  344. if (filter && filter.length > 0) {
  345. return filter[0].name
  346. }
  347. return "";
  348. }
  349. const fileList = ref();
  350. const setFileList = (files) => {
  351. fileList.value = files;
  352. };
  353. // 页面初始化
  354. onMounted(() => {
  355. const id = history.state.params?.id;
  356. if (id) {
  357. loadData(id);
  358. loadEducation(id);
  359. loadExperienceData(id);
  360. loadJobHuntData(id);
  361. loadHonorData(id);
  362. }
  363. getCultureList()
  364. })
  365. </script>
  366. <script lang="ts">
  367. // 设置页面名称进行组件缓存
  368. export default {
  369. name: "JobUserVitae"
  370. }
  371. </script>
  372. <style lang="less" scoped>
  373. .card-search {
  374. .jl-content {
  375. min-height: calc(100vh - 240px);
  376. width: 100%;
  377. display: flex;
  378. .avt-content {
  379. width: 20%;
  380. background-color: #e7e7e7;
  381. padding: 10px;
  382. .avt-image {
  383. display: flex;
  384. flex-wrap: wrap;
  385. justify-content: center;
  386. img{
  387. width:120px;
  388. height:120px;
  389. }
  390. }
  391. .age-info {
  392. width: 100%;
  393. display: flex;
  394. justify-content: space-between;
  395. margin-top: 15px;
  396. padding: 0 20%;
  397. }
  398. .politics-info {
  399. margin-top: 25px;
  400. display: grid;
  401. row-gap: 22px;
  402. grid-template-columns: repeat(2, minmax(0, 1fr));
  403. .label-span {
  404. text-align: right;
  405. }
  406. }
  407. .nameTitle {
  408. margin-top: 1rem;
  409. text-align: center;
  410. font-size: 20px;
  411. font-weight: 700;
  412. }
  413. .label-info {
  414. margin-top: 15px;
  415. padding: 0 5%;
  416. .title {
  417. margin-bottom: 10px;
  418. }
  419. .flex-box {
  420. display: flex;
  421. flex-wrap: wrap;
  422. .ant-tag {
  423. text-align: center;
  424. margin-bottom: 8px;
  425. overflow: hidden;
  426. text-overflow: ellipsis;
  427. white-space: nowrap;
  428. }
  429. }
  430. }
  431. }
  432. .info-content {
  433. width: 80%;
  434. height: 100%;
  435. margin-left: 10px;
  436. }
  437. }
  438. }
  439. </style>