巴林闲侠 2 years ago
parent
commit
88cdfb8089
  1. 8
      web/client/src/components/setup.jsx
  2. 11
      web/client/src/sections/humanAffairs/actions/employeeInformation.js
  3. 325
      web/client/src/sections/humanAffairs/containers/employeeInformation.jsx
  4. 14
      web/client/src/sections/humanAffairs/containers/personnelFilesDetail.jsx
  5. 9
      web/client/src/sections/humanAffairs/style.less
  6. 2
      web/client/src/utils/webapi.js

8
web/client/src/components/setup.jsx

@ -12,9 +12,6 @@ function Setup(props) {
tableList, tableList,
length length
} = props; } = props;
console.log(tableType,
tableList);
const [check, setCheck] = useState([]); const [check, setCheck] = useState([]);
const checkboxcss = { width: "25%", height: 16, margin: "0 0 20px 0" }; const checkboxcss = { width: "25%", height: 16, margin: "0 0 20px 0" };
@ -55,7 +52,7 @@ function Setup(props) {
: "rgba(176, 176, 176, 1)", : "rgba(176, 176, 176, 1)",
}} }}
> >
{check.length}/length {check.length}/{length}
</span> </span>
</div> </div>
} }
@ -116,7 +113,8 @@ function Setup(props) {
})} })}
</div> </div>
</div> </div>
)})} )
})}
</CheckboxGroup> </CheckboxGroup>
</Modal> </Modal>
); );

11
web/client/src/sections/humanAffairs/actions/employeeInformation.js

@ -13,3 +13,14 @@ export function getMemberNativePlace(query) {//查询籍贯列表
reducer: { name: "MemberNativePlace", params: { noClear: true } }, reducer: { name: "MemberNativePlace", params: { noClear: true } },
}); });
} }
export function getMemberWorkPlace(query) {//查询工作地列表
return (dispatch) => basicAction({
type: "get",
dispatch: dispatch,
actionType: "GET_MemberWorkPlace",
query: query,
url: `${ApiTable.getMemberWorkPlace}`,
msg: { option: "查询工作地列表" },
reducer: { name: "MemberWorkPlace", params: { noClear: true } },
});
}

325
web/client/src/sections/humanAffairs/containers/employeeInformation.jsx

@ -1,7 +1,8 @@
import React, { useEffect, useState, useRef } from 'react'; import React, { useEffect, useState, useRef } from 'react';
import { connect } from 'react-redux'; import { connect } from 'react-redux';
import { Select, SkeletonScreen, Button, Pagination, Skeleton, Form } from '@douyinfe/semi-ui'; import { Table, Button, Pagination, Skeleton, Form, Tooltip } from '@douyinfe/semi-ui';
import { IconSearch } from '@douyinfe/semi-icons'; import { IconSearch } from '@douyinfe/semi-icons';
import { SkeletonScreen } from "$components";
import '../style.less' import '../style.less'
import { Setup } from "$components"; import { Setup } from "$components";
import moment from 'moment' import moment from 'moment'
@ -13,13 +14,9 @@ const employeeInformation = (props) => {
const { humanAffairs } = actions; const { humanAffairs } = actions;
const form = useRef();// const form = useRef();//
let [archivesList, setArchivesList] = useState([]); let [archivesList, setArchivesList] = useState([]);//
const [personnelModal, setPersonnelModal] = useState(false);// let [workPlaceList, setWorkPlaceList] = useState([]);//
const [exportModalVs, setExportModalVs] = useState(false); let [nativeList, setNativeList] = useState([]);//
const [keyword, setKeyword] = useState('');//
const [keywordTarget, setKeywordTarget] = useState('');//
const [downloadUrl, setDownloadUrl] = useState('')
let [typeChoose, setTypeChoose] = useState('');
const [setup, setSetup] = useState(false);// const [setup, setSetup] = useState(false);//
const [setupp, setSetupp] = useState([]);// const [setupp, setSetupp] = useState([]);//
@ -29,49 +26,62 @@ const employeeInformation = (props) => {
const EMPLOYEEINFORMATION = "employeeInformation"; const EMPLOYEEINFORMATION = "employeeInformation";
const tableList = [// const tableList = [//
{ {
title: '推送信息', title: '基础信息',
list: [ list: [
{ name: "策略类型", value: "pushWay" }, { name: "员工编号", value: "userCode" },
{ name: "推送机制", value: "noticeWay" }, { name: "姓名", value: "userName" },
{ name: "监听设备数量", value: "monitorCount" }, { name: "所属部门", value: "departmrnt" },
{ name: "累计推送次数", value: "logCount" }, // { name: "", value: "monitorCount" },
// { name: "", value: "logCount" },
] ]
} }
]; ];
useEffect(() => { useEffect(() => {
getMemberSearchList() getMemberNativePlaceList()//
getMemberNativePlaceList() getMemberWorkPlaceList()//
getMemberSearchList()//
// attribute(); attribute();
// localStorage.getItem(EMPLOYEEINFORMATION) == null localStorage.getItem(EMPLOYEEINFORMATION) == null
// ? localStorage.setItem( ? localStorage.setItem(
// EMPLOYEEINFORMATION, EMPLOYEEINFORMATION,
// JSON.stringify(['pushWay','noticeWay','logCount','monitorCount']) JSON.stringify(['userCode', 'userName', 'departmrnt'])
// ) )
// : ""; : "";
}, [typeChoose]) }, [])
function getMemberSearchList () {// function getMemberSearchList () {//
dispatch(humanAffairs.getMemberList({ keywordTarget, keyword, state: typeChoose })).then((res) => {// let obj = form.current.getValues()
if (form.current.getValues().entryTime?.length > 1) {
obj.hiredateStart = moment(form.current.getValues().entryTime[0]).format('YYYY-MM-DD')
obj.hiredateEnd = moment(form.current.getValues().entryTime[1]).format('YYYY-MM-DD')
}
else {
obj.hiredateStart = ''
obj.hiredateEnd = ''
}
setQuery({ limit: 10, page: 0 })
dispatch(humanAffairs.getMemberList({ ...obj, ...query })).then((res) => {//
if (res.success) { if (res.success) {
setArchivesList(res.payload?.data?.rows) setArchivesList(res.payload?.data?.rows)
setLimits(res.payload?.data?.count)
} }
}) })
} }
function getMemberNativePlaceList () { function getMemberNativePlaceList () {
dispatch(humanAffairs.getMemberNativePlace()).then((res) => {// dispatch(humanAffairs.getMemberNativePlace()).then((res) => {//
if (res.success) { if (res.success) {
// setArchivesList(res.payload?.data?.rows) setNativeList(res.payload?.data)
console.log('res.payload?.data?.rows',res.payload?.data);
} }
}) })
} }
function typeOnChange (e) {// function getMemberWorkPlaceList () {
setTypeChoose(e.target.value); dispatch(humanAffairs.getMemberWorkPlace()).then((res) => {//
if (res.success) {
setWorkPlaceList(res.payload?.data)
} }
function seachValueChange (value) { })
setKeyword(value)
} }
const columns = [];
// //
function attribute () { function attribute () {
const arr = localStorage.getItem(EMPLOYEEINFORMATION) const arr = localStorage.getItem(EMPLOYEEINFORMATION)
@ -80,61 +90,117 @@ const employeeInformation = (props) => {
const column = [ const column = [
{ {
title: "关联项目", title: (
dataIndex: "noticeWay", <div>
key: "noticeWay", <img src="/assets/images/hrImg/V.png" alt="" style={{ width: 14, height: 14 }} /> 员工编号
render: (_, r, index) => { </div>
return r.noticeWay; ),
}, dataIndex: "userCode",
}, key: "userCode",
{
title: "创建时间",
dataIndex: "logCount",
key: "logCount",
render: (_, r, index) => {
return (r.logCount + '次')
},
},
{
title: "接收人",
dataIndex: "monitorCount",
key: "monitorCount",
render: (_, r, index) => { render: (_, r, index) => {
return r.monitorCount return r.userCode;
},
}, },
{ }, {
title: "监听问题", title: (
dataIndex: "pushWay", <div>
key: "pushWay", <img src="/assets/images/hrImg/V.png" alt="" style={{ width: 14, height: 14 }} /> 姓名
</div>
),
dataIndex: "userName",
key: "userName",
render: (_, r, index) => { render: (_, r, index) => {
return r.pushWay == 'email' ? '邮件通知' : '短信通知'; return r.userName;
},
}, },
{ }, {
title: "通知时效", title: (
dataIndex: "text1", <div>
key: "text1", <img src="/assets/images/hrImg/V.png" alt="" style={{ width: 14, height: 14 }} /> 所属部门
</div>
),
dataIndex: "departmrnt",
key: "departmrnt",
render: (_, r, index) => { render: (_, r, index) => {
return r.text1 return (
}, <div>
},
{ {
title: "启用状态", r.departmrnt.map((ite, idx) => {
dataIndex: "text2", let departmentsArr = []
key: "text2", for (let i = 0; i < r.departmrnt.length; i++) {
render: (_, r, index) => { departmentsArr.push(r.departmrnt[i].name)
return r.text2 }
}, return (
}, <div key={idx} style={{ display: 'flex' }}>
{idx == 0 ?
(<div style={{ padding: '0px 4px 1px 4px ', color: '#FFFFFF', fontSize: 12, background: 'rgba(0,90,189,0.8)', borderRadius: 2, marginRight: 4 }}>
{ite.name}
</div>) : ('')
}
{ {
title: "推送次数", r.departmrnt.length > 1 && idx == 1 ? (
dataIndex: "time", <Tooltip content={departmentsArr.join(',')} trigger="click" style={{ lineHeight: 2 }}>
key: "time", <div style={{ padding: '0px 4px 1px 4px ', color: '#FFFFFF', fontSize: 12, background: 'rgba(0,90,189,0.8)', borderRadius: 2, marginRight: 4, cursor: "pointer", }}>
render: (_, r, index) => { ...
return r.time </div>
</Tooltip>
) : ('')
}
</div>
)
})
}
</div>
)
}, },
}, },
// {
// title: "",
// dataIndex: "logCount",
// key: "logCount",
// render: (_, r, index) => {
// return (r.logCount + '')
// },
// },
// {
// title: "",
// dataIndex: "monitorCount",
// key: "monitorCount",
// render: (_, r, index) => {
// return r.monitorCount
// },
// },
// {
// title: "",
// dataIndex: "pushWay",
// key: "pushWay",
// render: (_, r, index) => {
// return r.pushWay == 'email' ? '' : '';
// },
// },
// {
// title: "",
// dataIndex: "text1",
// key: "text1",
// render: (_, r, index) => {
// return r.text1
// },
// },
// {
// title: "",
// dataIndex: "text2",
// key: "text2",
// render: (_, r, index) => {
// return r.text2
// },
// },
// {
// title: "",
// dataIndex: "time",
// key: "time",
// render: (_, r, index) => {
// return r.time
// },
// },
]; ];
for (let i = 0; i < arr.length; i++) { for (let i = 0; i < arr.length; i++) {
let colum = column.filter((item) => { let colum = column.filter((item) => {
@ -144,6 +210,18 @@ const employeeInformation = (props) => {
} }
setSetupp(columns); setSetupp(columns);
} }
function handleRow (record, index) {//
//
if (index % 2 === 0) {
return {
style: {
background: '#FAFCFF',
}
};
} else {
return {};
}
}
return ( return (
<> <>
<div style={{ padding: '0px 12px' }}> <div style={{ padding: '0px 12px' }}>
@ -174,77 +252,85 @@ const employeeInformation = (props) => {
> >
<div> <div>
<div style={{ display: 'flex', alignItems: 'center' }}> <div style={{ display: 'flex', alignItems: 'center' }}>
<Form.Select value={keywordTarget} <Form.Select
pure onChange={setKeywordTarget} placeholder='请选择搜索类型' style={{ width: 200 }} showClear> pure
<Select.Option value='role'>职位</Select.Option> field="keywordTarget"
<Select.Option value='dep'>部门</Select.Option> placeholder="请选择搜索类型"
<Select.Option value='number'>编号</Select.Option> style={{ width: 200 }}
<Select.Option value='name'>姓名</Select.Option> showClear
>
<Form.Select.Option value='role'>职位</Form.Select.Option>
<Form.Select.Option value='dep'>部门</Form.Select.Option>
<Form.Select.Option value='number'>编号</Form.Select.Option>
<Form.Select.Option value='name'>姓名</Form.Select.Option>
</Form.Select> </Form.Select>
<Form.Input suffix={<IconSearch />} <Form.Input
suffix={<IconSearch />}
field="keyword"
pure pure
showClear showClear
placeholder='请输入或选择关键词'
value={keyword}
style={{ width: 346, marginLeft: 12, marginRight: 12 }} style={{ width: 346, marginLeft: 12, marginRight: 12 }}
onChange={seachValueChange}> placeholder="请输入或选择关键词"
</Form.Input> />
<Form.DatePicker <Form.DatePicker
label='入职时间:' label='入职时间:'
field='enableType' type="dateRange" density="compact" showClear style={{ width: 370, color: "#F9F9F9" }} /> field='entryTime' type="dateRange" density="compact" showClear style={{ width: 370, color: "#F9F9F9" }} />
</div> </div>
<div style={{ display: 'flex', alignItems: 'center', justifyContent: 'space-between' }}> <div style={{ display: 'flex', alignItems: 'center', justifyContent: 'space-between' }}>
<div style={{ display: 'flex', alignItems: 'center' }}> <div style={{ display: 'flex', alignItems: 'center' }}>
<Form.Select <Form.Select
label='婚姻状况:' label="婚育状态:"
field="marital"
labelPosition="left" labelPosition="left"
field='enableType'
style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
placeholder="全部" placeholder="全部"
filter style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
showClear showClear
> >
{/* {.map((item) => { <Form.Select.Option value=''>
return ( 全部
<Form.Select.Option key={item.value} value={item.value}> </Form.Select.Option>
{item.name} <Form.Select.Option value='已婚已育'>
已婚已育
</Form.Select.Option>
<Form.Select.Option value='已婚'>
已婚
</Form.Select.Option>
<Form.Select.Option value='未婚'>
未婚
</Form.Select.Option> </Form.Select.Option>
);
})} */}
</Form.Select> </Form.Select>
<Form.Select <Form.Select
label='户籍地:' label='户籍地:'
labelPosition="left" labelPosition="left"
field='enableType' field='native'
style={{ width: 246, marginRight: 20, color: "#F9F9F9", }} style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
placeholder="全部" placeholder="全部"
filter
showClear showClear
> >
{/* {.map((item) => { {nativeList.map((item) => {
return ( return (
<Form.Select.Option key={item.value} value={item.value}> <Form.Select.Option key={item.nativePlace} value={item.nativePlace}>
{item.name} {item.nativePlace}
</Form.Select.Option> </Form.Select.Option>
); );
})} */} })}
</Form.Select> </Form.Select>
<Form.Select <Form.Select
label='工作地点:' label='工作地点:'
labelPosition="left" labelPosition="left"
field='enableType' field='workPlace'
style={{ width: 246, marginRight: 20, color: "#F9F9F9", }} style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
placeholder="全部" placeholder="全部"
filter
showClear showClear
> >
{/* {.map((item) => { {workPlaceList.map((item) => {
return ( return (
<Form.Select.Option key={item.value} value={item.value}> <Form.Select.Option key={item.workPlace} value={item.workPlace}>
{item.name} {item.workPlace}
</Form.Select.Option> </Form.Select.Option>
); );
})} */} })}
</Form.Select> </Form.Select>
</div> </div>
<div style={{ display: 'flex', alignItems: 'center' }}> <div style={{ display: 'flex', alignItems: 'center' }}>
@ -256,11 +342,7 @@ const employeeInformation = (props) => {
/> />
<Button theme='solid' type='primary' style={{ width: 80, borderRadius: 2, height: 32, background: '#DBECFF', color: '#005ABD' }} <Button theme='solid' type='primary' style={{ width: 80, borderRadius: 2, height: 32, background: '#DBECFF', color: '#005ABD' }}
onClick={() => { onClick={() => {
// dispatch(humanAffairs.getMemberList({ keywordTarget, keyword, state: typeChoose })).then((res) => {// getMemberSearchList()
// if (res.success) {
// setArchivesList(res.payload.data.rows)
// }
// })
}}>查询</Button> }}>查询</Button>
</div> </div>
</div> </div>
@ -272,7 +354,7 @@ const employeeInformation = (props) => {
<img src="/assets/images/hrImg/V.png" alt="" style={{ width: 14, height: 14 }} /> <img src="/assets/images/hrImg/V.png" alt="" style={{ width: 14, height: 14 }} />
"信息的为系统基础数据来源于项企PEP钉钉等系统其他数据均为导入或自定义数据 "信息的为系统基础数据来源于项企PEP钉钉等系统其他数据均为导入或自定义数据
</div> </div>
{/* <div style={{ marginTop: 20 }}> <div style={{ marginTop: 20 }}>
<Skeleton <Skeleton
// loading={loading} // loading={loading}
loading={false} loading={false}
@ -281,12 +363,11 @@ const employeeInformation = (props) => {
> >
<Table <Table
columns={setupp.filter((s) => s)} columns={setupp.filter((s) => s)}
dataSource={tableData} dataSource={archivesList}
bordered={false} bordered={false}
empty="暂无数据" empty="暂无数据"
pagination={false} pagination={false}
onRow={handleRow} onRow={handleRow}
rowSelection={rowSelection}
/> />
</Skeleton> </Skeleton>
<div <div
@ -315,7 +396,7 @@ const employeeInformation = (props) => {
/> />
</div> </div>
</div> </div>
</div> */} </div>
</div> </div>
</div> </div>
</div> </div>

14
web/client/src/sections/humanAffairs/containers/personnelFilesDetail.jsx

@ -164,14 +164,14 @@ const Rest = (props) => {
let textdate = []; let textdate = [];
for (let i = 0; i < res.payload?.data?.dayStatisticData?.length; i++) { for (let i = 0; i < res.payload?.data?.dayStatisticData?.length; i++) {
if (textdate.findIndex((ev) => { if (textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM') return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')&&ev.compensate == res.payload.data.dayStatisticData[i].compensate
}) !== -1) { }) !== -1) {
textdate[textdate.findIndex((ev) => { textdate[textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM') return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')&&ev.compensate == res.payload.data.dayStatisticData[i].compensate
})].num })].num
= =
textdate[textdate.findIndex((ev) => { textdate[textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM') return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')&&ev.compensate == res.payload.data.dayStatisticData[i].compensate
})].num + res.payload.data.dayStatisticData[i].duration / 3600 })].num + res.payload.data.dayStatisticData[i].duration / 3600
} }
else { else {
@ -753,7 +753,7 @@ const Rest = (props) => {
工作日 工作日
</div> </div>
<div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}> <div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}>
{tableStatistic.sumPayWorkday / 60 / 60 + tableStatistic.sumTakeRestWorkday / 60 / 60 || 0}小时 {tableStatistic.sumPayWorkday / 3600 + tableStatistic.sumTakeRestWorkday /3600 || 0}小时
</div> </div>
</div> </div>
<div style={{ display: 'flex', alignItems: 'center', width: '19.87%' }}> <div style={{ display: 'flex', alignItems: 'center', width: '19.87%' }}>
@ -761,7 +761,7 @@ const Rest = (props) => {
普通假日 普通假日
</div> </div>
<div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}> <div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}>
{tableStatistic.sumTakeRestDayoff / 60 / 60 + tableStatistic.sumTakeRestDayoff / 60 / 60 || 0}小时 {tableStatistic.sumTakeRestDayoff / 3600 + tableStatistic.sumPayDayoff / 3600 || 0}小时
</div> </div>
</div> </div>
<div style={{ display: 'flex', alignItems: 'center', width: '19.03%' }}> <div style={{ display: 'flex', alignItems: 'center', width: '19.03%' }}>
@ -769,7 +769,7 @@ const Rest = (props) => {
法定假日 法定假日
</div> </div>
<div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}> <div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}>
{tableStatistic.sumPayFestivals / 60 / 60 + tableStatistic.sumTakeRestFestivals / 60 / 60 || 0}小时 {tableStatistic.sumPayFestivals / 3600 + tableStatistic.sumTakeRestFestivals / 3600 || 0}小时
</div> </div>
</div> </div>
<div style={{ display: 'flex', alignItems: 'center', width: '19.217%' }}> <div style={{ display: 'flex', alignItems: 'center', width: '19.217%' }}>
@ -777,7 +777,7 @@ const Rest = (props) => {
累计加班时长 累计加班时长
</div> </div>
<div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}> <div style={{ fontSize: 14, color: '#005ABD', marginLeft: 5 }}>
{tableStatistic.sumPayWorkday / 60 / 60 + tableStatistic.sumTakeRestWorkday / 60 / 60 + tableStatistic.sumTakeRestDayoff / 60 / 60 + tableStatistic.sumTakeRestDayoff / 60 / 60 + tableStatistic.sumPayFestivals / 60 / 60 + tableStatistic.sumTakeRestFestivals / 60 / 60 || 0}小时 {(tableStatistic.sumPayWorkday / 3600 + tableStatistic.sumTakeRestWorkday / 3600 + tableStatistic.sumTakeRestDayoff / 3600 + tableStatistic.sumPayDayoff /3600 + tableStatistic.sumPayFestivals / 3600 + tableStatistic.sumTakeRestFestivals / 3600) || 0}小时
</div> </div>
</div> </div>
</div> </div>

9
web/client/src/sections/humanAffairs/style.less

@ -0,0 +1,9 @@
.semi-table{
.semi-table-row:first-child{
.semi-table-row-head{
background: #F2F3F5;
color: #4A4A4A;
font-size: 14px;
}
}
}

2
web/client/src/utils/webapi.js

@ -27,7 +27,7 @@ export const ApiTable = {
getMemberVacate: 'member/vacate',//查询单个人员请假统计数据 getMemberVacate: 'member/vacate',//查询单个人员请假统计数据
getMemberExport: 'member/export',//导出员工信息 getMemberExport: 'member/export',//导出员工信息
getMemberNativePlace: 'member/native_place',//查询籍贯列表 getMemberNativePlace: 'member/native_place',//查询籍贯列表
getMemberWorkPlace: 'member/work_place',//查询工作地列表
}; };
export const RouteTable = { export const RouteTable = {
apiRoot: "/api/root", apiRoot: "/api/root",

Loading…
Cancel
Save