Browse Source

折线图和员工信息

master
deartibers 2 years ago
parent
commit
fc8e1d518e
  1. BIN
      web/client/assets/images/hrImg/!.png
  2. BIN
      web/client/assets/images/hrImg/V.png
  3. BIN
      web/client/assets/images/hrImg/setUp.png
  4. 2
      web/client/index.ejs
  5. 2
      web/client/index.html
  6. 9
      web/client/src/components/setup.jsx
  7. 344
      web/client/src/sections/humanAffairs/containers/employeeInformation.jsx
  8. 3
      web/client/src/sections/humanAffairs/containers/index.js
  9. 2
      web/client/src/sections/humanAffairs/containers/personnelFiles.jsx
  10. 156
      web/client/src/sections/humanAffairs/containers/personnelFilesDetail.jsx
  11. 10
      web/client/src/sections/humanAffairs/nav-item.jsx
  12. 14
      web/client/src/sections/humanAffairs/routes.js

BIN
web/client/assets/images/hrImg/!.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 554 B

BIN
web/client/assets/images/hrImg/V.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 838 B

BIN
web/client/assets/images/hrImg/setUp.png

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.0 KiB

2
web/client/index.ejs

@ -9,7 +9,7 @@
<link rel="shortcut icon" href="/assets/images/favicon.ico">
<script src="https://lf1-cdn-tos.bytegoofy.com/obj/iconpark/icons_19077_10.1efd80a22a5e53e48737fd5ab150ffd2.es5.js"></script>
<script src="https://lf1-cdn-tos.bytegoofy.com/obj/iconpark/icons_20231_3.d613c57d3bf10acf9916c8162dae204a.es5.js"></script>
</head>
<body>

2
web/client/index.html

@ -8,7 +8,7 @@
<link rel="shortcut icon" href="/assets/images/favicon.ico">
<script
src="https://lf1-cdn-tos.bytegoofy.com/obj/iconpark/icons_19077_11.27aacefc59cea1cbc86236576463a6d2.es5.js"></script>
src="https://lf1-cdn-tos.bytegoofy.com/obj/iconpark/icons_20231_3.d613c57d3bf10acf9916c8162dae204a.es5.js"></script>
</head>
<body>

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

@ -9,7 +9,8 @@ function Setup(props) {
const {
close,
tableType,
tableList
tableList,
length
} = props;
console.log(tableType,
@ -25,7 +26,7 @@ function Setup(props) {
ischeck();
}, []);
function ischeck(value) {
if (check.length >= 8) {
if (check.length >= length) {
if (check.includes(value)) {
return false;
} else {
@ -49,12 +50,12 @@ function Setup(props) {
textAlign: "center",
marginLeft: 6,
background:
check.length == 8
check.length == length
? "rgba(40, 123, 255, 1)"
: "rgba(176, 176, 176, 1)",
}}
>
{check.length}/8
{check.length}/length
</span>
</div>
}

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

@ -0,0 +1,344 @@
import React, { useEffect, useState, useRef } from 'react';
import { connect } from 'react-redux';
import { Select, SkeletonScreen, Button, Pagination, Skeleton, Form } from '@douyinfe/semi-ui';
import { IconSearch } from '@douyinfe/semi-icons';
import '../style.less'
import { Setup } from "$components";
import moment from 'moment'
import { set } from 'nprogress';
const employeeInformation = (props) => {
const { dispatch, actions, history, user, loading, socket, xqMembers } = props
const { humanAffairs } = actions;
const form = useRef();//
let [archivesList, setArchivesList] = useState([]);
const [personnelModal, setPersonnelModal] = useState(false);//
const [exportModalVs, setExportModalVs] = useState(false);
const [keyword, setKeyword] = useState('');//
const [keywordTarget, setKeywordTarget] = useState('');//
const [downloadUrl, setDownloadUrl] = useState('')
let [typeChoose, setTypeChoose] = useState('');
const [setup, setSetup] = useState(false);//
const [setupp, setSetupp] = useState([]);//
const [query, setQuery] = useState({ limit: 10, page: 0 }); //
const [limits, setLimits] = useState()//
const EMPLOYEEINFORMATION = "employeeInformation";
const tableList = [//
{
title: '推送信息',
list: [
{ name: "策略类型", value: "pushWay" },
{ name: "推送机制", value: "noticeWay" },
{ name: "监听设备数量", value: "monitorCount" },
{ name: "累计推送次数", value: "logCount" },
]
}
];
useEffect(() => {
getMemberSearchList()
// attribute();
// localStorage.getItem(EMPLOYEEINFORMATION) == null
// ? localStorage.setItem(
// EMPLOYEEINFORMATION,
// JSON.stringify(['pushWay','noticeWay','logCount','monitorCount'])
// )
// : "";
}, [typeChoose])
function getMemberSearchList () {//
dispatch(humanAffairs.getMemberList({ keywordTarget, keyword, state: typeChoose })).then((res) => {//
if (res.success) {
setArchivesList(res.payload?.data?.rows)
}
})
}
function typeOnChange (e) {//
setTypeChoose(e.target.value);
}
function seachValueChange (value) {
setKeyword(value)
}
//
function attribute () {
const arr = localStorage.getItem(EMPLOYEEINFORMATION)
? JSON.parse(localStorage.getItem(EMPLOYEEINFORMATION))
: [];
const column = [
{
title: "关联项目",
dataIndex: "noticeWay",
key: "noticeWay",
render: (_, r, index) => {
return r.noticeWay;
},
},
{
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++) {
let colum = column.filter((item) => {
return item.key === arr[i];
});
columns.splice(i + 2, 0, colum[0]);
}
setSetupp(columns);
}
return (
<>
<div style={{ padding: '0px 12px' }}>
<div style={{ display: 'flex' }}>
<div style={{ color: 'rgba(0,0,0,0.45)', fontSize: 14 }}>人事管理</div>
<div style={{ color: 'rgba(0,0,0,0.45)', fontSize: 14, margin: '0px 8px' }}>/</div>
<div style={{ color: 'rgba(0,0,0,0.45)', fontSize: 14 }}>档案中心</div>
<div style={{ color: '#033C9A', fontSize: 14, margin: '0px 8px' }}>/</div>
<div style={{ color: '#033C9A', fontSize: 14 }}>人员档案</div>
</div>
<div style={{ background: '#FFFFFF', boxShadow: '0px 0px 12px 2px rgba(220,222,224,0.2)', borderRadius: 2, padding: '20px 0px 20px 19px ', marginTop: 12 }}>
<div style={{ display: 'flex', alignItems: 'center', justifyContent: 'space-between' }}>
<div style={{ display: 'flex', alignItems: 'baseline' }}>
<div style={{ width: 0, height: 20, borderLeft: '3px solid #0F7EFB', borderTop: '3px solid transparent', borderBottom: '3px solid transparent' }}></div>
<div style={{ fontFamily: "YouSheBiaoTiHei", fontSize: 24, color: '#033C9A', marginLeft: 8 }}>员工信息</div>
<div style={{ marginLeft: 6, fontSize: 12, color: '#969799', fontFamily: "DINExp", }}>EMPLOYEE INFORMATION</div>
</div>
</div>
<div style={{ marginRight: 20, marginTop: 18 }}>
<Form
labelPosition="left"
labelAlign="right"
labelWidth="80px"
onValueChange={(values, field) => {
console.log('values', values);
}}
getFormApi={(formApi) => (form.current = formApi)}
>
<div>
<div style={{ display: 'flex', alignItems: 'center' }}>
<Form.Select value={keywordTarget}
pure onChange={setKeywordTarget} placeholder='请选择搜索类型' style={{ width: 200 }} showClear>
<Select.Option value='role'>职位</Select.Option>
<Select.Option value='dep'>部门</Select.Option>
<Select.Option value='number'>编号</Select.Option>
<Select.Option value='name'>姓名</Select.Option>
</Form.Select>
<Form.Input suffix={<IconSearch />}
pure
showClear
placeholder='请输入或选择关键词'
value={keyword}
style={{ width: 346, marginLeft: 12, marginRight: 12 }}
onChange={seachValueChange}>
</Form.Input>
<Form.DatePicker
label='入职时间:'
field='enableType' type="dateRange" density="compact" showClear style={{ width: 370, color: "#F9F9F9" }} />
</div>
<div style={{ display: 'flex', alignItems: 'center', justifyContent: 'space-between' }}>
<div style={{ display: 'flex', alignItems: 'center' }}>
<Form.Select
label='婚姻状况:'
labelPosition="left"
field='enableType'
style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
placeholder="全部"
filter
showClear
>
{/* {.map((item) => {
return (
<Form.Select.Option key={item.value} value={item.value}>
{item.name}
</Form.Select.Option>
);
})} */}
</Form.Select>
<Form.Select
label='户籍地:'
labelPosition="left"
field='enableType'
style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
placeholder="全部"
filter
showClear
>
{/* {.map((item) => {
return (
<Form.Select.Option key={item.value} value={item.value}>
{item.name}
</Form.Select.Option>
);
})} */}
</Form.Select>
<Form.Select
label='工作地点:'
labelPosition="left"
field='enableType'
style={{ width: 246, marginRight: 20, color: "#F9F9F9", }}
placeholder="全部"
filter
showClear
>
{/* {.map((item) => {
return (
<Form.Select.Option key={item.value} value={item.value}>
{item.name}
</Form.Select.Option>
);
})} */}
</Form.Select>
</div>
<div style={{ display: 'flex', alignItems: 'center' }}>
<img src="/assets/images/hrImg/export.png" alt="" style={{ width: 20, height: 20, cursor: "pointer", marginRight: 15 }}
// onClick={() => setSetup(true)}
/>
<img src="/assets/images/hrImg/setup.png" alt="" style={{ width: 20, height: 20, cursor: "pointer", marginRight: 15 }}
onClick={() => setSetup(true)}
/>
<Button theme='solid' type='primary' style={{ width: 80, borderRadius: 2, height: 32, background: '#DBECFF', color: '#005ABD' }}
onClick={() => {
// dispatch(humanAffairs.getMemberList({ keywordTarget, keyword, state: typeChoose })).then((res) => {//
// if (res.success) {
// setArchivesList(res.payload.data.rows)
// }
// })
}}>查询</Button>
</div>
</div>
</div>
</Form>
<div style={{ border: '1px solid #C7E1FF', background: '#F4F8FF', borderRadius: 2, height: 32, width: 669, padding: '8px 0px 7px 12px', display: 'flex', alignItems: 'center', color: '#0F7EFB', fontSize: 12 }}>
<img src="/assets/images/hrImg/!.png" alt="" style={{ width: 14, height: 14, marginRight: 8 }} />
表格中带有认证标识"
<img src="/assets/images/hrImg/V.png" alt="" style={{ width: 14, height: 14 }} />
"信息的为系统基础数据来源于项企PEP钉钉等系统其他数据均为导入或自定义数据
</div>
{/* <div style={{ marginTop: 20 }}>
<Skeleton
// loading={loading}
loading={false}
active={true}
placeholder={SkeletonScreen()}
>
<Table
columns={setupp.filter((s) => s)}
dataSource={tableData}
bordered={false}
empty="暂无数据"
pagination={false}
onRow={handleRow}
rowSelection={rowSelection}
/>
</Skeleton>
<div
style={{
display: "flex",
justifyContent: "space-between",
padding: "20px 20px",
}}
>
<div>
</div>
<div style={{ display: 'flex', }}>
<span style={{ lineHeight: "30px", fontSize: 13, color: 'rgba(0,90,189,0.8)' }}>
{limits}条信息
</span>
<Pagination
className="22"
total={limits}
showSizeChanger
currentPage={query.page + 1}
pageSizeOpts={[10, 20, 30, 40]}
onChange={(currentPage, pageSize) => {
setQuery({ limit: pageSize, page: currentPage - 1 });
page.current = currentPage - 1
}}
/>
</div>
</div>
</div> */}
</div>
</div>
</div>
{setup ? (
<Setup
tableType={EMPLOYEEINFORMATION}
tableList={tableList}
length={25}
close={() => {
setSetup(false);
attribute();
// setcameraSetup(false);
}}
/>
) : (
""
)}
</>
)
}
function mapStateToProps (state) {
const { auth, global, MemberSearch, webSocket } = state;
return {
// loading: members.isRequesting,
user: auth.user,
actions: global.actions,
xqMembers: MemberSearch.data,
// socket: webSocket.socket
};
}
export default connect(mapStateToProps)(employeeInformation);

3
web/client/src/sections/humanAffairs/containers/index.js

@ -2,5 +2,6 @@
import PersonnelFiles from './personnelFiles';
import PersonnelFilesDetail from './personnelFilesDetail';
import EmployeeInformation from './employeeInformation';
export { PersonnelFiles, PersonnelFilesDetail };
export { PersonnelFiles, PersonnelFilesDetail, EmployeeInformation };

2
web/client/src/sections/humanAffairs/containers/personnelFiles.jsx

@ -161,7 +161,7 @@ const Rest = (props) => {
<div style={{ marginLeft: 4 }}>
<div style={{ display: 'flex' }}>
<div style={{ color: '#005ABD', fontSize: 14, marginRight: 6 }}>
{item.pepUserId}
{item.userCode}
</div>
{
item.departmrnt.map((ite, idx) => {

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

@ -151,23 +151,65 @@ const Rest = (props) => {
}
})
}
function getWorkOption () {//线
function getWorkOption () {//线
let date = [];
let showdate = []
let showdate1 = []
dispatch(humanAffairs.getMemberOvertime({ pepUserId: history?.location?.search.slice(1), startDate: startDate, endDate: endDate })).then((res) => {//
if (res.success) {
setTableData(res.payload?.data?.data)
setTableStatistic(res.payload?.data)
let textdate = [];
for (let i = 0; i < res.payload?.data?.dayStatisticData?.length; i++) {
if (textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')
}) !== -1) {
textdate[textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')
})].num
=
textdate[textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')
})].num + res.payload.data.dayStatisticData[i].duration / 3600
}
else {
textdate.push({
time: moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM'),
num: res.payload.data.dayStatisticData[i].duration / 3600,
compensate: res.payload.data.dayStatisticData[i].compensate
})
var date = [];
date.push([2017, 6, 1].join("/"));
date.push([2017, 6, 2].join("/"));
date.push([2017, 6, 3].join("/"));
date.push([2017, 6, 4].join("/"));
date.push([2017, 6, 5].join("/"));
date.push([2017, 6, 6].join("/"));
date.push([2017, 6, 7].join("/"));
date.push([2017, 6, 8].join("/"));
date.push([2017, 6, 9].join("/"));
}
}
console.log('textdate',textdate);
for (let l = 0; l < textdate.length; l++) {
if (textdate[l].compensate == '调休') {
showdate.push([textdate[l].time, textdate[l].num])
}
else {
showdate1.push([textdate[l].time, textdate[l].num])
}
}
if (!startDate && !endDate) {
for (let i = 0; i < 12; i++) {
date.unshift(moment(new Date()).subtract(i, 'months').format('YYYY/MM'));
}
}
else {
let nowdate = moment(startDate).format('YYYY/MM')
date.push(nowdate)
for (let o = 0; o < 10000; o++) {
if (nowdate !== moment(endDate).format('YYYY/MM')) {
nowdate = moment(nowdate).add(1, 'months').format('YYYY/MM')
date.push(nowdate)
}
else {
break
}
}
}
let data = {
legend: {
data: ["调休", "折算"],
@ -217,13 +259,7 @@ const Rest = (props) => {
opacity: 0.1
},
smooth: true,
data: [
["2017/6/1", 1],
["2017/6/3", 3],
["2017/6/5", 5],
["2017/6/7", 4],
["2017/6/9", 4],
],
data: showdate,
},
{
name: "折算",
@ -232,18 +268,17 @@ const Rest = (props) => {
color: 'rgba(254,152,18,0.2)',
},
smooth: true,
data: [
["2017/6/2", 3],
["2017/6/4", 1],
["2017/6/6", 6],
["2017/6/8", 3],
],
data: showdate1,
},
],
}
setOption(data)
}
})
}
function getLeaveOption () {//线
let date = [];
let showdate = []
dispatch(humanAffairs.getMemberVacate({ pepUserId: history?.location?.search.slice(1), startDate: startDate, endDate: endDate })).then((res) => {//
if (res.success) {
setLeaveData(res.payload?.data?.data)
@ -270,18 +305,49 @@ const Rest = (props) => {
setLeaveStatistic({
compassionate, sick, year, other, all
})
let textdate = [];
for (let i = 0; i < res.payload?.data?.dayStatisticData?.length; i++) {
if (textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')
}) !== -1) {
textdate[textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')
})].num
=
textdate[textdate.findIndex((ev) => {
return ev.time == moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM')
})].num + res.payload.data.dayStatisticData[i].duration / 3600
}
else {
textdate.push({
time: moment(res.payload.data.dayStatisticData[i].day).format('YYYY/MM'),
num: res.payload.data.dayStatisticData[i].duration / 3600
})
var date = [];
date.push([2017, 6, 1].join("/"));
date.push([2017, 6, 2].join("/"));
date.push([2017, 6, 3].join("/"));
date.push([2017, 6, 4].join("/"));
date.push([2017, 6, 5].join("/"));
date.push([2017, 6, 6].join("/"));
date.push([2017, 6, 7].join("/"));
date.push([2017, 6, 8].join("/"));
date.push([2017, 6, 9].join("/"));
}
}
for (let l = 0; l < textdate.length; l++) {
showdate.push([textdate[l].time, textdate[l].num])
}
if (!startDate && !endDate) {
for (let i = 0; i < 12; i++) {
date.unshift(moment(new Date()).subtract(i, 'months').format('YYYY/MM'));
}
}
else {
let nowdate = moment(startDate).format('YYYY/MM')
date.push(nowdate)
for (let o = 0; o < 10000; o++) {
if (nowdate !== moment(endDate).format('YYYY/MM')) {
nowdate = moment(nowdate).add(1, 'months').format('YYYY/MM')
date.push(nowdate)
}
else {
break
}
}
}
let data = {
color: ['#0F7EFB', '#FE9812'], //线
xAxis: [
@ -325,23 +391,23 @@ const Rest = (props) => {
opacity: 0.1
},
smooth: true,
data: [
["2017/6/1", 1],
["2017/6/3", 3],
["2017/6/5", 5],
["2017/6/7", 4],
["2017/6/9", 4],
],
data: showdate,
},
],
}
setLeaveOption(data)
}
})
}
function handleChange (date) {
console.log('datedatedatedate', date);
if (date.length > 1) {
setStartDate(moment(date[0]).format('YYYY-MM-DD'))
setEndDate(moment(date[1]).format('YYYY-MM-DD'))
}
else {
setStartDate('')
setEndDate('')
}
}
return (
<>
@ -646,8 +712,8 @@ const Rest = (props) => {
</div>
<div style={{ borderBottom: '1px solid #DCDEE0', margin: '24px 0px 24px -20px' }}></div>
<div style={{ paddingLeft: 17 }}>
<div style={{ color: '#4A4A4A', fontSize: 14, marginBottom: 10 }}>
动作
<div style={{ color: '#4A4A4A', fontSize: 14, marginBottom: 10, fontWeight: 'bold' }}>
动作
</div>
<div style={{ color: '#4A4A4A', fontSize: 14, display: 'flex', alignItems: 'center' }}>
时间筛选<DatePicker onChange={handleChange} type="dateRange" density="compact" style={{ width: 242 }} />

10
web/client/src/sections/humanAffairs/nav-item.jsx

@ -12,11 +12,19 @@ export function getNavItem (user, dispatch) {
{
itemKey: 'archivesCenter',
text: '档案中心',
icon: <iconpark-icon style={{ width: 20, height: 20 }} name="iconjianshezhong"></iconpark-icon>,
icon: <iconpark-icon style={{ width: 20, height: 20 }} name="iconcbrengongdangan"></iconpark-icon>,
to: '/humanAffairs/archivesCenter/personnelFiles',
items: [{
itemKey: 'personnelFiles', to: '/humanAffairs/archivesCenter/personnelFiles', text: '人员档案'
}]
},{
itemKey: 'employeeRelations',
text: '员工关系',
icon: <iconpark-icon style={{ width: 20, height: 20 }} name="iconcbyuangongguanxi"></iconpark-icon>,
to: '/humanAffairs/employeeRelations/employeeInformation',
items: [{
itemKey: 'employeeInformation', to: '/humanAffairs/employeeRelations/employeeInformation', text: '员工信息'
}]
},
]
},

14
web/client/src/sections/humanAffairs/routes.js

@ -1,4 +1,4 @@
import { PersonnelFiles,PersonnelFilesDetail } from './containers';
import { PersonnelFiles, PersonnelFilesDetail, EmployeeInformation } from './containers';
export default [{
type: 'inner',
@ -17,7 +17,17 @@ export default [{
component: PersonnelFiles,
breadcrumb: '人员档案',
}]
},]
}, {
path: '/employeeRelations',
key: 'employeeRelations',
breadcrumb: '员工关系',
childRoutes: [{
path: '/employeeInformation',
key: 'employeeInformation',
component: EmployeeInformation,
breadcrumb: '员工信息',
}]
}]
}
}, {
type: 'inner',

Loading…
Cancel
Save