Files
td_official/src/views/project/attendance/index.vue

568 lines
20 KiB
Vue
Raw Normal View History

2025-04-07 18:07:52 +08:00
<template>
<div class="p-2">
<transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
<div v-show="showSearch" class="mb-[10px]">
<el-card shadow="hover">
<el-form ref="queryFormRef" :model="queryParams" :inline="true">
<el-form-item label="人员姓名" prop="userName">
<el-input v-model="queryParams.userName" placeholder="请输入人员姓名" clearable @keyup.enter="handleQuery" />
</el-form-item>
<el-form-item label="班组" prop="teamId">
<el-select v-model="queryParams.teamId" placeholder="请选择班组" clearable>
<el-option v-for="dict in projectTeamOpt" :key="dict.value" :label="dict.label" :value="dict.value" />
</el-select>
</el-form-item>
<el-form-item label="工种" prop="typeOfWork">
<el-select v-model="queryParams.typeOfWork" placeholder="请选择工种" clearable>
<el-option v-for="dict in type_of_work" :key="dict.value" :label="dict.label" :value="dict.value" />
</el-select>
</el-form-item>
<el-form-item label="打卡日期" prop="clockMonth">
<el-date-picker clearable v-model="queryParams.clockMonth" type="month" value-format="YYYY-MM" placeholder="请选择打卡日期" />
</el-form-item>
<el-form-item>
<el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
<el-button icon="Refresh" @click="resetQuery">重置</el-button>
</el-form-item>
</el-form>
</el-card>
</div>
</transition>
<el-row :gutter="20">
<el-col :span="24" :offset="0">
<el-card shadow="hover">
<!-- <template #header>
<PieChart style="width: 1em; height: 1em; vertical-align: middle" />
<span style="vertical-align: middle">命令统计</span>
</template> -->
<div class="el-table el-table--enable-row-hover el-table--medium">
<div ref="commandstats" style="height: 200px" />
</div>
</el-card>
</el-col>
</el-row>
<el-card shadow="never">
<!-- <template #header>
<el-row :gutter="10" class="mb8">
<el-col :span="1.5">
<el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['project:attendance:add']">新增</el-button>
</el-col>
<el-col :span="1.5">
<el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['project:attendance:edit']"
>修改</el-button
>
</el-col>
<el-col :span="1.5">
<el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['project:attendance:remove']"
>删除</el-button
>
</el-col>
<el-col :span="1.5">
<el-button type="warning" plain icon="Download" @click="handleExport" v-hasPermi="['project:attendance:export']">导出</el-button>
</el-col>
<right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
</el-row>
</template> -->
<el-table v-loading="loading" :data="attendanceList">
<el-table-column type="selection" width="55" align="center" />
<el-table-column label="主键id" align="center" prop="id" v-if="false" />
<el-table-column label="人员姓名" align="center" prop="userName" />
<el-table-column label="班组" align="center" prop="teamName" />
<el-table-column label="工种" align="center" prop="typeOfWork">
<template #default="scope">
<dict-tag :options="type_of_work" :value="scope.row.typeOfWork" />
</template>
</el-table-column>
<el-table-column label="出勤(天)" align="center" prop="attendanceDays" />
<el-table-column label="迟到(次)" align="center" prop="lateDays" />
<el-table-column label="早退(次)" align="center" prop="leaveEarlyDays" />
<el-table-column label="缺卡(次)" align="center" prop="unClockDays" />
<el-table-column label="操作" align="center" class-name="small-padding fixed-width">
<template #default="scope">
<el-button link type="primary" icon="View" @click="handleDetails(scope.row)" v-hasPermi="['project:attendance:edit']">详情</el-button>
</template>
</el-table-column>
</el-table>
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
</el-card>
<!-- 添加或修改考勤对话框 -->
2025-04-08 18:02:55 +08:00
<!-- <el-dialog :title="dialog.title" v-model="dialog.visible" width="500px" append-to-body>
2025-04-07 18:07:52 +08:00
<el-form ref="attendanceFormRef" :model="form" :rules="rules" label-width="80px">
<el-form-item label="人员id" prop="userId">
<el-input v-model="form.userId" placeholder="请输入人员id" />
</el-form-item>
<el-form-item label="人脸照" prop="facePic">
<image-upload v-model="form.facePic" />
</el-form-item>
<el-form-item label="项目id" prop="projectId">
<el-input v-model="form.projectId" placeholder="请输入项目id" />
</el-form-item>
<el-form-item label="上班打卡时间" prop="onClockTime">
<el-date-picker clearable v-model="form.onClockTime" type="datetime" value-format="YYYY-MM-DD HH:mm:ss" placeholder="请选择上班打卡时间">
</el-date-picker>
</el-form-item>
<el-form-item label="下班打卡时间" prop="offClockTime">
<el-date-picker clearable v-model="form.offClockTime" type="datetime" value-format="YYYY-MM-DD HH:mm:ss" placeholder="请选择下班打卡时间">
</el-date-picker>
</el-form-item>
<el-form-item label="打卡日期" prop="clockDate">
<el-date-picker clearable v-model="form.clockDate" type="datetime" value-format="YYYY-MM-DD HH:mm:ss" placeholder="请选择打卡日期">
</el-date-picker>
</el-form-item>
<el-form-item label="1正常,2迟到,3早退,4缺勤,5补卡" prop="clockStatus">
<el-radio-group v-model="form.clockStatus">
<el-radio v-for="dict in clock_status_type" :key="dict.value" :value="dict.value">{{ dict.label }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item label="代打人员id" prop="pinchUserId">
<el-input v-model="form.pinchUserId" placeholder="请输入代打人员id" />
</el-form-item>
<el-form-item label="多次打卡时间记录" prop="clockRecord">
<el-input v-model="form.clockRecord" type="textarea" placeholder="请输入内容" />
</el-form-item>
<el-form-item label="上下班" prop="commuter">
<el-input v-model="form.commuter" placeholder="请输入上下班" />
</el-form-item>
<el-form-item label="日薪" prop="dailyWage">
<el-input v-model="form.dailyWage" placeholder="请输入日薪" />
</el-form-item>
<el-form-item label="经度" prop="lng">
<el-input v-model="form.lng" placeholder="请输入经度" />
</el-form-item>
<el-form-item label="纬度" prop="lat">
<el-input v-model="form.lat" placeholder="请输入纬度" />
</el-form-item>
<el-form-item label="备注" prop="remark">
<el-input v-model="form.remark" type="textarea" placeholder="请输入内容" />
</el-form-item>
</el-form>
<template #footer>
<div class="dialog-footer">
<el-button :loading="buttonLoading" type="primary" @click="submitForm"> </el-button>
<el-button @click="cancel"> </el-button>
</div>
</template>
2025-04-08 18:02:55 +08:00
</el-dialog> -->
2025-04-07 18:07:52 +08:00
<!-- 考勤详情对话框 -->
<el-dialog v-model="dialog.details" width="1300px">
<el-calendar ref="calendar" class="h170 pos-relative">
<template #header="{ date }">
<span>
<el-button-group>
<el-button type="primary" size="small" icon="ArrowLeft" @click="selectDate('prev-month', date)">上一月</el-button>
<el-button type="primary" size="small" @click="selectDate('next-month', date)">
下一月<el-icon class="el-icon--right"><ArrowRight /></el-icon>
</el-button>
</el-button-group>
</span>
<span class="label">{{ date }} {{ dialog.title }}出勤</span>
<div class="status-detail flex items-center justify-between">
<div class="dot1">全天考勤正常</div>
<div class="dot2">当天存在异常迟到早退缺卡</div>
<div class="dot3">当天提交过补卡申请</div>
</div>
</template>
<template #date-cell="{ data }">
<div class="flex-c">
<p class="time">{{ day(data) }}</p>
<img v-if="!isplayCard(data)" src="http://zmkg.cqet.top:8899/assets/empty-CZvxqguX.png" /><span v-if="!isplayCard(data)"
>暂无打卡记录</span
>
<div v-if="isplayCard(data)" class="flex-r"><div class="circle" :class="'status' + attendanceStatus(data)"></div></div>
<div v-if="isplayCard(data)" class="flex justify-center flex-col w100% items-center">
<el-button type="primary" plain size="small" class="w70% my-2" v-if="workTime(data)">{{ workTime(data) }} 上班打卡</el-button>
<el-button type="danger" plain size="small" class="w50% my-2" v-else>上班缺卡</el-button>
<span></span>
<el-button type="warning" plain size="small" class="w70%" v-if="workFromTime(data)">{{ workFromTime(data) }} 下班打卡</el-button>
<el-button type="danger" plain size="small" class="w50%" v-else>下班缺卡</el-button>
</div>
</div>
</template>
</el-calendar>
</el-dialog>
</div>
</template>
<script setup name="Attendance" lang="ts">
import {
listAttendance,
getAttendance,
delAttendance,
addAttendance,
updateAttendance,
listAttendanceTwoWeek,
listAttendanceMonth
} from '@/api/project/attendance';
import { echartsConfig } from '@/api/project/attendance/echarts';
import { AttendanceVO, AttendanceQuery, AttendanceForm, AttendanceTwoWeekVO, AttendanceMonthVO } from '@/api/project/attendance/types';
import { listProjectTeam } from '@/api/project/projectTeam';
import { ProjectTeamVO } from '@/api/project/projectTeam/types';
import { useUserStoreHook } from '@/store/modules/user';
const commandstats = ref();
const { proxy } = getCurrentInstance() as ComponentInternalInstance;
const { clock_status_type, type_of_work } = toRefs<any>(proxy?.useDict('clock_status_type', 'type_of_work'));
import type { CalendarDateType, CalendarInstance } from 'element-plus';
// 获取用户 store
const userStore = useUserStoreHook();
// 从 store 中获取项目列表和当前选中的项目
const currentProject = computed(() => userStore.selectedProject);
const attendanceList = ref<AttendanceVO[]>([]);
const attendanceTwoWeekList = ref<AttendanceTwoWeekVO[]>([]);
const buttonLoading = ref(false);
const loading = ref(true);
const showSearch = ref(true);
const ids = ref<Array<string | number>>([]);
const single = ref(true);
const multiple = ref(true);
const total = ref(0);
const calendarList = ref<AttendanceMonthVO[]>();
const queryFormRef = ref<ElFormInstance>();
const attendanceFormRef = ref<ElFormInstance>();
//班组列表
const projectTeamOpt = ref([]);
const dialog = reactive<DialogOption>({
visible: false,
details: false,
title: ''
});
const initFormData: AttendanceForm = {
id: undefined,
userId: undefined,
facePic: undefined,
onClockTime: undefined,
offClockTime: undefined,
clockDate: undefined,
clockStatus: undefined,
pinchUserId: undefined,
clockRecord: undefined,
commuter: undefined,
dailyWage: undefined,
projectId: currentProject.value.id,
lng: undefined,
lat: undefined,
remark: undefined,
typeOfWork: undefined,
teamId: undefined,
clockMonth: undefined
};
const data = reactive<PageData<AttendanceForm, AttendanceQuery>>({
form: { ...initFormData },
queryParams: {
pageNum: 1,
pageSize: 10,
userName: undefined,
clockDate: undefined,
clockStatus: undefined,
commuter: undefined,
projectId: currentProject.value.id,
typeOfWork: undefined,
teamId: undefined,
clockMonth: undefined,
params: {}
},
rules: {
id: [{ required: true, message: '主键id不能为空', trigger: 'blur' }],
userId: [{ required: true, message: '人员id不能为空', trigger: 'blur' }],
facePic: [{ required: true, message: '人脸照不能为空', trigger: 'blur' }],
projectId: [{ required: true, message: '项目id不能为空', trigger: 'blur' }],
clockDate: [{ required: true, message: '打卡日期不能为空', trigger: 'blur' }],
clockStatus: [{ required: true, message: '1正常,2迟到,3早退,4缺勤,5补卡不能为空', trigger: 'change' }]
}
});
const day = computed(() => (date) => {
return date.day.split('-').slice(1).join('-');
});
//是否打卡
const isplayCard = computed(() => (date) => {
return calendarList.value.some((item) => item.clockDate == date.day);
});
//打卡时间下标
const playCardIdx = computed(() => (date) => {
return calendarList.value.findIndex((item) => item.clockDate == date.day);
});
//上班时间
const workTime = computed(() => (date) => {
return calendarList.value[playCardIdx.value(date)].attendanceList[0].clockTime?.slice(10);
});
//下班时间
const workFromTime = computed(() => (date) => {
2025-04-08 18:02:55 +08:00
return calendarList.value[playCardIdx.value(date)].attendanceList[1]?.clockTime?.slice(10);
2025-04-07 18:07:52 +08:00
});
//考勤状态
const attendanceStatus = computed(() => (date) => {
return calendarList.value[playCardIdx.value(date)].status;
});
const { queryParams, form, rules } = toRefs(data);
const calendar = ref<CalendarInstance>();
const selectDate = async (val: CalendarDateType, date) => {
if (!calendar.value) return;
calendar.value.selectDate(val);
const clockMonth = incrementMonth(date, val == 'prev-month' ? -1 : 1);
const res = await listAttendanceMonth({ id: dialog.id, clockMonth });
calendarList.value = res.data;
};
/** 查询考勤列表 */
const getList = async () => {
loading.value = true;
const res = await listAttendance(queryParams.value);
attendanceList.value = res.rows;
total.value = res.total;
loading.value = false;
};
const getProjectTeamList = async () => {
loading.value = true;
const res = await listProjectTeam({
pageNum: 1,
pageSize: 20,
orderByColumn: 'createTime',
isAsc: 'desc',
projectId: currentProject.value.id
});
projectTeamOpt.value = res.rows.map((projectTeam: ProjectTeamVO) => ({
value: projectTeam.id,
label: projectTeam.teamName
}));
loading.value = false;
};
/** 查询近两周考勤列表 */
const getListTwoWeek = async () => {
loading.value = true;
const res = await listAttendanceTwoWeek({ projectId: currentProject.value.id });
attendanceTwoWeekList.value = res.data;
echartsConfig(commandstats.value, attendanceTwoWeekList.value);
};
/** 取消按钮 */
const cancel = () => {
reset();
dialog.visible = false;
};
/** 表单重置 */
const reset = () => {
form.value = { ...initFormData };
attendanceFormRef.value?.resetFields();
};
/** 搜索按钮操作 */
const handleQuery = () => {
queryParams.value.pageNum = 1;
getList();
};
/** 重置按钮操作 */
const resetQuery = () => {
queryFormRef.value?.resetFields();
handleQuery();
};
/** 多选框选中数据 */
// const handleSelectionChange = (selection: AttendanceVO[]) => {
// ids.value = selection.map((item) => item.id);
// single.value = selection.length != 1;
// multiple.value = !selection.length;
// };
/** 新增按钮操作 */
const handleAdd = () => {
reset();
dialog.visible = true;
dialog.title = '添加考勤';
};
//处理获取到的月份
const incrementMonth = (dateStr: string, monthsToAdd: number) => {
const [yearPart, monthPart] = dateStr.replace(/\s/g, '').split('年');
const year = parseInt(yearPart, 10);
const month = parseInt(monthPart.replace('月', ''), 10);
// 创建一个新的 Date 对象,设置为输入日期的第一天
const date = new Date(year, month - 1, 1);
// 增加一个月
date.setMonth(date.getMonth() + monthsToAdd);
// 提取增加一个月后的年份和月份
const newYear = date.getFullYear();
const newMonth = String(date.getMonth() + 1).padStart(2, '0');
// 返回格式化后的日期字符串
return `${newYear}-${newMonth}`;
};
/** 详情按钮操作 */
const handleDetails = async (row?: AttendanceVO) => {
2025-04-08 18:02:55 +08:00
const res = await listAttendanceMonth({ userId: row?.id });
2025-04-07 18:07:52 +08:00
calendarList.value = res.data;
dialog.details = true;
dialog.id = row?.id;
dialog.title = row?.userName || '';
};
/** 提交按钮 */
const submitForm = () => {
attendanceFormRef.value?.validate(async (valid: boolean) => {
if (valid) {
buttonLoading.value = true;
if (form.value.id) {
await updateAttendance(form.value).finally(() => (buttonLoading.value = false));
} else {
await addAttendance(form.value).finally(() => (buttonLoading.value = false));
}
proxy?.$modal.msgSuccess('操作成功');
dialog.visible = false;
await getList();
}
});
};
/** 删除按钮操作 */
// const handleDelete = async (row?: AttendanceVO) => {
// const _ids = row?.id || ids.value;
// await proxy?.$modal.confirm('是否确认删除考勤编号为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
// await delAttendance(_ids);
// proxy?.$modal.msgSuccess('删除成功');
// await getList();
// };
/** 导出按钮操作 */
// const handleExport = () => {
// proxy?.download(
// 'project/attendance/export',
// {
// ...queryParams.value
// },
// `attendance_${new Date().getTime()}.xlsx`
// );
// };
onMounted(() => {
getList();
getListTwoWeek();
getProjectTeamList();
});
</script>
<style lang="scss" scoped>
.label {
font-size: 24px;
position: absolute;
left: 50%;
top: 0;
transform: translate(-50%);
color: #000;
font-weight: 500;
}
.status-detail {
margin: 0 15px;
position: relative;
font-size: 12px;
> div {
margin: 0 15px;
position: relative;
font-size: 12px;
&::before {
position: absolute;
content: '';
display: inline-block;
left: -15px;
top: 30%;
width: 8px;
height: 8px;
border-radius: 50%;
}
}
.dot1 {
&::before {
background-color: #1d6fe9;
}
}
.dot2 {
&::before {
background-color: #f55f4e;
}
}
.dot3 {
&::before {
background-color: #ff8d1a;
}
}
}
.flex-c {
height: 110px;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
position: relative;
.time {
position: absolute;
z-index: 10;
right: 0;
top: 0;
}
img {
width: 50%;
height: 50%;
}
> span {
font-size: 12px;
color: #ccc;
padding-top: 5px;
}
}
.el-calendar-table__row {
height: 100px;
}
.flex-r {
width: 100%;
display: flex;
flex-direction: row;
justify-content: center;
padding-top: 5px;
height: 10px;
.circle {
width: 7px;
height: 7px;
border-radius: 50%;
margin: 0 2px;
position: absolute;
z-index: 10;
right: 12px;
top: 35px;
}
.status2 {
background: #f55f4e;
}
.status1 {
background: #1d6fe9;
}
.status3 {
background: #ff8d1a;
}
}
::v-deep(.el-calendar) {
.el-calendar__body {
height: 600px;
overflow: auto;
}
td {
height: 110px;
.el-calendar-day {
height: 100%;
}
}
}
</style>