|
@ -1,17 +1,16 @@ |
|
|
'use strict'; |
|
|
'use strict'; |
|
|
import React, { useState } from 'react'; |
|
|
import React, { useState } from 'react'; |
|
|
import { connect } from 'react-redux'; |
|
|
import { connect } from 'react-redux'; |
|
|
//import { Button, Input, Card, Modal, Upload, message } from 'antd';
|
|
|
import { Modal, Form, Button, Notification } from '@douyinfe/semi-ui'; |
|
|
import { Modal, Form, Button, Upload, Notification } from '@douyinfe/semi-ui'; |
|
|
|
|
|
import { IconUpload } from '@douyinfe/semi-icons'; |
|
|
import { IconUpload } from '@douyinfe/semi-icons'; |
|
|
//import { Request } from '@peace/utils'
|
|
|
import cityData from '../components/city.json'; |
|
|
//import request from 'superagent'
|
|
|
|
|
|
import XLSX from 'xlsx' |
|
|
import XLSX from 'xlsx' |
|
|
//import { userBulkAdd } from '../../actions'
|
|
|
import { membersBulkAdd } from '../actions/personnelFiles' |
|
|
|
|
|
import ExportJsonExcel from 'js-export-excel'; |
|
|
//TODO 下载模板和上传文件读取
|
|
|
import { rule } from './table-input-rule' |
|
|
const ImportUser = props => { |
|
|
//下载模板和上传文件读取
|
|
|
const { user, dispatch, onCancel } = props |
|
|
const ImportMembersModal = props => { |
|
|
|
|
|
const { dispatch, onCancel } = props |
|
|
const [msg, setMsg] = useState('') |
|
|
const [msg, setMsg] = useState('') |
|
|
const [loading, setLoading] = useState('') |
|
|
const [loading, setLoading] = useState('') |
|
|
const [postData, setPostData] = useState([]) |
|
|
const [postData, setPostData] = useState([]) |
|
@ -19,12 +18,13 @@ const ImportUser = props => { |
|
|
const confirm = () => { |
|
|
const confirm = () => { |
|
|
if (postData.length) { |
|
|
if (postData.length) { |
|
|
setLoading(true) |
|
|
setLoading(true) |
|
|
// dispatch(userBulkAdd(postData, params.departmentId ? params.departmentId : null)).then(res => {
|
|
|
dispatch(membersBulkAdd(postData)).then(res => { |
|
|
// if (res.success) {
|
|
|
if (res.success) { |
|
|
|
|
|
Notification.success('导入员工信息成功') |
|
|
// }
|
|
|
onCancel() |
|
|
// setLoading(false)
|
|
|
} |
|
|
// })
|
|
|
setLoading(false) |
|
|
|
|
|
}) |
|
|
} else { |
|
|
} else { |
|
|
Notification.warn('没有数据可以提交,请上传数据文件') |
|
|
Notification.warn('没有数据可以提交,请上传数据文件') |
|
|
} |
|
|
} |
|
@ -32,7 +32,11 @@ const ImportUser = props => { |
|
|
|
|
|
|
|
|
const download = () => { |
|
|
const download = () => { |
|
|
dldTemplate(); |
|
|
dldTemplate(); |
|
|
dldText("填写说明.txt", "12121212121212"); |
|
|
let str = ""; |
|
|
|
|
|
rule.forEach((v, i) => { |
|
|
|
|
|
str += `${v}\r\n` |
|
|
|
|
|
}) |
|
|
|
|
|
dldText("填写说明.txt", str); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const dldTemplate = () => { |
|
|
const dldTemplate = () => { |
|
@ -100,9 +104,38 @@ const ImportUser = props => { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
request.send(); |
|
|
}) |
|
|
}) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const judgePlace = (place) => { |
|
|
|
|
|
let valid = true; |
|
|
|
|
|
if (place.split('-').length == 1) {//判断籍贯
|
|
|
|
|
|
if (['北京市', '上海市', '天津市', '重庆市'].indexOf(place) == -1) { |
|
|
|
|
|
valid = false; |
|
|
|
|
|
} |
|
|
|
|
|
} else if (place.split('-').length == 2) { |
|
|
|
|
|
let province = place.split('-')[0]; |
|
|
|
|
|
let city = place.split('-')[1]; |
|
|
|
|
|
let existProvince = cityData.find(cd => cd.name == province); |
|
|
|
|
|
if (!existProvince) { |
|
|
|
|
|
valid = false; |
|
|
|
|
|
} else { |
|
|
|
|
|
let existCity = existProvince.children.find(cd => cd.name == city); |
|
|
|
|
|
if (!existCity) { |
|
|
|
|
|
valid = false; |
|
|
|
|
|
} |
|
|
|
|
|
} |
|
|
|
|
|
} else { |
|
|
|
|
|
valid = false; |
|
|
|
|
|
} |
|
|
|
|
|
return valid; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
const judgeNull = (value) => { |
|
|
|
|
|
return value ? String(value).trim().replace(/\s*/g, "") : null; |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
return ( |
|
|
return ( |
|
|
<Modal |
|
|
<Modal |
|
|
title="导入信息" visible={true} |
|
|
title="导入信息" visible={true} |
|
@ -136,57 +169,122 @@ const ImportUser = props => { |
|
|
error('请填写至少一行数据') |
|
|
error('请填写至少一行数据') |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
let postData = [] |
|
|
let postData = []; |
|
|
|
|
|
const zmsz = /^[A-Za-z0-9]+$/;//字母数字组合
|
|
|
const pattern = /^1[3|4|5|6|7|8|9]\d{9}$/; |
|
|
const pattern = /^1[3|4|5|6|7|8|9]\d{9}$/; |
|
|
const sfz = /^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/;//身份证
|
|
|
const sfz = /^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/;//身份证
|
|
|
|
|
|
const ymd = /^((19|20)[0-9]{2})[\/\-]((0[1-9])|(1[0-2]))[\/\-]((0[1-9])|((1|2)[0-9])|(3[0-1]))$/;//年月日
|
|
|
|
|
|
const num0 = /^\d+$/;//正整数+0
|
|
|
for (let i = 0; i < res.length; i++) { |
|
|
for (let i = 0; i < res.length; i++) { |
|
|
let d = res[i] |
|
|
let d = res[i]; |
|
|
let number = String(d['人员编号']).trim(); |
|
|
|
|
|
let name = String(d['姓名']).trim(); |
|
|
let number = judgeNull(d['人员编号']); |
|
|
let idNumber = String(d['证件照']).trim(); |
|
|
let name = judgeNull(d['姓名']); |
|
|
let gender = String(d['性别(男/女)']).trim(); |
|
|
let idNumber = judgeNull(d['证件号']); |
|
|
let birthday = String(d['出生年月日(例2022/02/01)']).trim(); |
|
|
let gender = judgeNull(d['性别(男/女)']); |
|
|
let nativePlace = String(d['籍贯']).trim(); |
|
|
let birthday = judgeNull(d['出生年月日(例2022/02/01)']); |
|
|
let marital = String(d['婚育状态(已婚/未婚/已婚已育)']).trim(); |
|
|
let nativePlace = judgeNull(d['籍贯']); |
|
|
let politicsStatus = String(d['政治面貌']).trim(); |
|
|
let marital = judgeNull(d['婚育状态(已婚/未婚/已婚已育)']); |
|
|
let phoneNumber = String(d['联系方式']).trim(); |
|
|
|
|
|
|
|
|
let politicsStatus = judgeNull(d['政治面貌']); |
|
|
let workPlace = String(d['工作地点']).trim(); |
|
|
let phoneNumber = judgeNull(d['联系方式']); |
|
|
let graduatedFrom = String(d['毕业院校']).trim(); |
|
|
let workPlace = judgeNull(d['工作地点']); |
|
|
let educationBackground = String(d['学历']).trim(); |
|
|
let graduatedFrom = judgeNull(d['毕业院校']); |
|
|
let specialty = String(d['专业']).trim(); |
|
|
let educationBackground = judgeNull(d['学历']); |
|
|
let graduationDate = String(d['毕业时间']).trim(); |
|
|
let specialty = judgeNull(d['专业']); |
|
|
let hiredate = String(d['入职时间']).trim(); |
|
|
let graduationDate = judgeNull(d['毕业时间']); |
|
|
|
|
|
|
|
|
let turnProbationPeriod = String(d['转试用期时间']).trim(); |
|
|
let hiredate = judgeNull(d['入职时间']); |
|
|
let regularDate = String(d['转正时间']).trim(); |
|
|
let turnProbationPeriod = judgeNull(d['转试用期时间']); |
|
|
let dimissionDate = String(d['离职日期']).trim(); |
|
|
let regularDate = judgeNull(d['转正时间']); |
|
|
let experienceYear = String(d['工作经验(年)']).trim(); |
|
|
let dimissionDate = judgeNull(d['离职日期']); |
|
|
let occupationalHistory = String(d['历史工作经历与职务']).trim(); |
|
|
let experienceYear = judgeNull(d['工作经验(年)']); |
|
|
|
|
|
let occupationalHistory = judgeNull(d['历史工作经历与职务']); |
|
|
//let account = String(d['账号']).trim()
|
|
|
|
|
|
if (!number || !name) { |
|
|
if (!number) {//人员编号不为空,唯一,字母和数字
|
|
|
error(`第${i + 1} 行有空值,请填写后重新上传`) |
|
|
error(`第${i + 1}行人员编号为空,请填写`) |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
if (!sfz.test(idcard)) { |
|
|
if (postData.some(p => p.number == number)) {//人员编号 唯一
|
|
|
error(`第${i + 1} 行证件号错误`) |
|
|
error(`第${i + 1}行人员编号重复,请更改后重新上传`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
if (!zmsz.test(number)) { |
|
|
|
|
|
error(`第${i + 1}行人员编号错误,请填写字母和数字的组合`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
if (!name) {//姓名必填
|
|
|
|
|
|
error(`第${i + 1}行姓名为空,请填写`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
if (!sfz.test(idNumber)) { |
|
|
|
|
|
error(`第${i + 1}行证件号错误`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
if (['男', '女'].indexOf(gender) == -1) { |
|
|
|
|
|
error(`第${i + 1}行性别错误`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
// if (!ymd.test(birthday)) {
|
|
|
|
|
|
// error(`第${i + 1}行出生年月日错误,请填写yyyy/mm/dd格式`)
|
|
|
|
|
|
// return
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
let jgValid = judgePlace(nativePlace); |
|
|
|
|
|
if (!jgValid) { |
|
|
|
|
|
error(`第${i + 1}行籍贯错误`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if (['已婚', '未婚', '已婚已育'].indexOf(marital) == -1) { |
|
|
|
|
|
error(`第${i + 1}行婚育状态错误`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
if (['群众', '党员'].indexOf(politicsStatus) == -1) { |
|
|
|
|
|
error(`第${i + 1}行政治面貌错误`) |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
if (!pattern.test(phoneNumber)) { |
|
|
if (!pattern.test(phoneNumber)) { |
|
|
error(`第${i + 1} 行手机号码错误`) |
|
|
error(`第${i + 1}行联系方式错误`) |
|
|
|
|
|
return |
|
|
|
|
|
} |
|
|
|
|
|
let wpValid = judgePlace(workPlace); |
|
|
|
|
|
if (!wpValid) { |
|
|
|
|
|
error(`第${i + 1}行工作地点错误`) |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
if (name.length > 128 || account.length > 128) { |
|
|
if (['小学', '初中', '高中', '大专', '本科', '研究生', '博士'].indexOf(educationBackground) == -1) { |
|
|
error(`第${i + 1} 行数据字符长度大于 128,请更改后重新上传`) |
|
|
error(`第${i + 1}行学历错误`) |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
if (postData.some(p => p.number == number)) { |
|
|
// if (!ymd.test(graduationDate)) {
|
|
|
error(`第${i + 1} 行人员编号重复,请更改后重新上传`) |
|
|
// error(`第${i + 1}行毕业时间错误,请填写yyyy/mm/dd格式`)
|
|
|
|
|
|
// return
|
|
|
|
|
|
// }
|
|
|
|
|
|
// if (!ymd.test(hiredate)) {
|
|
|
|
|
|
// error(`第${i + 1}行入职时间错误,请填写yyyy/mm/dd格式`)
|
|
|
|
|
|
// return
|
|
|
|
|
|
// }
|
|
|
|
|
|
// if (!ymd.test(turnProbationPeriod)) {
|
|
|
|
|
|
// error(`第${i + 1}行转试用期时间错误,请填写yyyy/mm/dd格式`)
|
|
|
|
|
|
// return
|
|
|
|
|
|
// }
|
|
|
|
|
|
// if (!ymd.test(regularDate)) {
|
|
|
|
|
|
// error(`第${i + 1}行转正时间错误,请填写yyyy/mm/dd格式`)
|
|
|
|
|
|
// return
|
|
|
|
|
|
// }
|
|
|
|
|
|
// if (!ymd.test(dimissionDate)) {
|
|
|
|
|
|
// error(`第${i + 1}行离职日期错误,请填写yyyy/mm/dd格式`)
|
|
|
|
|
|
// return
|
|
|
|
|
|
// }
|
|
|
|
|
|
if (!num0.test(experienceYear)) { |
|
|
|
|
|
error(`第${i + 1}行工作经验(年)错误,请填写非负整数`) |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
postData.push({ |
|
|
postData.push({//人员编号 待办todotodo
|
|
|
name, phoneNumber, account |
|
|
pepUserId: 555, name, idNumber, gender, birthday, nativePlace, marital, |
|
|
|
|
|
politicsStatus, phoneNumber, workPlace, graduatedFrom, educationBackground, specialty, graduationDate, |
|
|
|
|
|
hiredate, turnProbationPeriod, regularDate, dimissionDate, experienceYear, occupationalHistory |
|
|
}) |
|
|
}) |
|
|
} |
|
|
} |
|
|
if (postData.length) { |
|
|
if (postData.length) { |
|
@ -196,12 +294,12 @@ const ImportUser = props => { |
|
|
setMsg(msg) |
|
|
setMsg(msg) |
|
|
onSuccess({ message: msg }) |
|
|
onSuccess({ message: msg }) |
|
|
}) |
|
|
}) |
|
|
}} |
|
|
}}> |
|
|
> |
|
|
|
|
|
<Button icon={<IconUpload />} theme="light"> |
|
|
<Button icon={<IconUpload />} theme="light"> |
|
|
请选择文件 |
|
|
请选择文件 |
|
|
</Button> |
|
|
</Button> |
|
|
</Form.Upload> |
|
|
</Form.Upload> |
|
|
|
|
|
<span>{msg}</span> |
|
|
<div style={{ color: '#ccc' }}>最大不超过200M,导入文件需与 |
|
|
<div style={{ color: '#ccc' }}>最大不超过200M,导入文件需与 |
|
|
<span onClick={() => download()} style={{ cursor: 'pointer', color: '#0066FF' }}>导入模板</span> |
|
|
<span onClick={() => download()} style={{ cursor: 'pointer', color: '#0066FF' }}>导入模板</span> |
|
|
一致</div> |
|
|
一致</div> |
|
@ -211,10 +309,10 @@ const ImportUser = props => { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
function mapStateToProps(state) { |
|
|
function mapStateToProps(state) { |
|
|
const { auth, customizeList } = state; |
|
|
const { auth } = state; |
|
|
return { |
|
|
return { |
|
|
user: auth.user, |
|
|
user: auth.user, |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export default connect(mapStateToProps)(ImportUser); |
|
|
export default connect(mapStateToProps)(ImportMembersModal); |