2024-10-15 09:53:49 +00:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="work-log-container">
|
|
|
|
|
|
<!-- 左侧项目列表 -->
|
|
|
|
|
|
<div class="project-list" :class="{ collapsed: isCollapsed }">
|
|
|
|
|
|
<div class="list-header">
|
|
|
|
|
|
<span v-if="!isCollapsed">项目列表</span>
|
|
|
|
|
|
<div class="collapse-button-wrapper">
|
|
|
|
|
|
<el-button
|
|
|
|
|
|
type="text"
|
|
|
|
|
|
@click="toggleCollapse"
|
|
|
|
|
|
class="collapse-button"
|
|
|
|
|
|
>
|
2024-10-16 09:32:16 +00:00
|
|
|
|
{{ !isCollapsed ? "收起" : "展开" }}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
</el-button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="f1">
|
|
|
|
|
|
<CustomTable
|
|
|
|
|
|
v-show="!isCollapsed"
|
|
|
|
|
|
:columns="[
|
|
|
|
|
|
{ prop: 'name', label: '项目', align: 'center' },
|
|
|
|
|
|
{ prop: 'workTime', label: '工时(天)', align: 'center' },
|
|
|
|
|
|
]"
|
|
|
|
|
|
:tableData="projectList"
|
2024-10-16 09:32:16 +00:00
|
|
|
|
:rowClick="
|
|
|
|
|
|
(row) => {
|
|
|
|
|
|
handleProjectClick(row, 1);
|
|
|
|
|
|
}
|
|
|
|
|
|
"
|
2024-10-15 09:53:49 +00:00
|
|
|
|
:show-summary="true"
|
|
|
|
|
|
:summary-method="getSummaries"
|
2024-10-16 09:32:16 +00:00
|
|
|
|
sum-text="合计工时(天)"
|
2024-10-15 09:53:49 +00:00
|
|
|
|
:showPagination="false"
|
|
|
|
|
|
:highligt="true"
|
|
|
|
|
|
ref="projectRef"
|
2024-10-16 09:32:16 +00:00
|
|
|
|
style="height: 100%"
|
2024-11-06 02:04:43 +00:00
|
|
|
|
rowKey="projectId"
|
2024-10-15 09:53:49 +00:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 右侧项目信息和日历 -->
|
|
|
|
|
|
<div class="project-info-calendar">
|
|
|
|
|
|
<h2 class="mb20 textC">工作日志</h2>
|
|
|
|
|
|
<!-- 项目信息表单 -->
|
|
|
|
|
|
<el-form
|
|
|
|
|
|
:model="projectInfo"
|
|
|
|
|
|
label-width="100px"
|
|
|
|
|
|
disabled
|
|
|
|
|
|
class="project-info-form log-form"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-form-item label="项目名称">
|
|
|
|
|
|
<el-input v-model="projectInfo.projectName"></el-input>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
|
|
|
<el-col :span="12">
|
|
|
|
|
|
<el-form-item label="项目编号">
|
|
|
|
|
|
<el-input v-model="projectInfo.projectCode"></el-input>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="12">
|
|
|
|
|
|
<el-form-item label="工时填报人">
|
|
|
|
|
|
<el-input v-model="projectInfo.nickName"></el-input>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
</el-row>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
|
|
|
<el-col :span="12">
|
|
|
|
|
|
<el-form-item label="项目开始时间">
|
|
|
|
|
|
<el-date-picker
|
|
|
|
|
|
v-model="projectInfo.startDate"
|
|
|
|
|
|
type="date"
|
|
|
|
|
|
></el-date-picker>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="12">
|
|
|
|
|
|
<el-form-item label="项目结束时间">
|
|
|
|
|
|
<el-date-picker
|
|
|
|
|
|
v-model="projectInfo.endDate"
|
|
|
|
|
|
type="date"
|
|
|
|
|
|
></el-date-picker>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-col>
|
|
|
|
|
|
</el-row>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 日历视图 -->
|
|
|
|
|
|
<div class="calendar-header">
|
|
|
|
|
|
<el-date-picker
|
|
|
|
|
|
v-model="selectedDate"
|
|
|
|
|
|
type="month"
|
|
|
|
|
|
format="yyyy年MM月"
|
|
|
|
|
|
:clearable="false"
|
2024-10-17 10:00:03 +00:00
|
|
|
|
@change="changeMonth"
|
2024-10-15 09:53:49 +00:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="calendar-view" v-if="projectInfo">
|
|
|
|
|
|
<el-calendar v-model="selectedDate">
|
|
|
|
|
|
<template #dateCell="{ data }">
|
|
|
|
|
|
<div
|
|
|
|
|
|
:key="data.day"
|
|
|
|
|
|
:id="data.day"
|
|
|
|
|
|
@click="openLogDialog(data)"
|
|
|
|
|
|
:class="{
|
|
|
|
|
|
'date-cell': true,
|
|
|
|
|
|
'in-range': isInProjectRange(data),
|
|
|
|
|
|
'out-range': !isInProjectRange(data),
|
|
|
|
|
|
disabled: isFutureDate(data.date) && isInProjectRange(data),
|
|
|
|
|
|
}"
|
|
|
|
|
|
>
|
2024-11-06 02:04:43 +00:00
|
|
|
|
{{ data.day.split("-")[1] + "/" + data.day.split("-")[2] }}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-calendar>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<!-- 工作日志对话框 -->
|
|
|
|
|
|
<el-dialog
|
|
|
|
|
|
:visible="logDialogVisible"
|
|
|
|
|
|
:title="
|
|
|
|
|
|
logForm.loggerId && projectInfo.userId == $store.state.user.id
|
|
|
|
|
|
? '编辑日志'
|
|
|
|
|
|
: projectInfo.userId == $store.state.user.id
|
|
|
|
|
|
? '新增日志'
|
|
|
|
|
|
: '查看日志'
|
|
|
|
|
|
"
|
|
|
|
|
|
width="30%"
|
|
|
|
|
|
center
|
|
|
|
|
|
append-to-body
|
|
|
|
|
|
:before-close="
|
|
|
|
|
|
() => {
|
|
|
|
|
|
logDialogVisible = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-form :model="logForm" label-width="120px" class="log-form">
|
|
|
|
|
|
<el-form-item label="工时分配(天)">
|
|
|
|
|
|
<el-input
|
|
|
|
|
|
type="number"
|
|
|
|
|
|
v-model="logForm.workTime"
|
|
|
|
|
|
:max="logForm.max"
|
2024-10-16 09:32:16 +00:00
|
|
|
|
:placeholder="
|
|
|
|
|
|
projectInfo.userId == $store.state.user.id
|
|
|
|
|
|
? `可填报工时:${logForm.max}人天`
|
|
|
|
|
|
: ''
|
|
|
|
|
|
"
|
2024-10-15 09:53:49 +00:00
|
|
|
|
:disabled="!(projectInfo.userId == $store.state.user.id)"
|
2024-10-16 09:32:16 +00:00
|
|
|
|
:step="0.1"
|
|
|
|
|
|
min="0"
|
2024-10-15 09:53:49 +00:00
|
|
|
|
></el-input>
|
|
|
|
|
|
<el-button
|
|
|
|
|
|
type="text"
|
|
|
|
|
|
v-if="projectInfo.userId == $store.state.user.id"
|
2025-01-06 09:52:52 +00:00
|
|
|
|
@click="logForm.workTime = logForm.max"
|
2024-10-16 09:32:16 +00:00
|
|
|
|
>{{ `当天剩余可分配工时:${logForm.max}人天` }}</el-button
|
2024-10-15 09:53:49 +00:00
|
|
|
|
>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item label="工作内容">
|
|
|
|
|
|
<el-input
|
|
|
|
|
|
v-model="logForm.workContent"
|
|
|
|
|
|
:disabled="!(projectInfo.userId == $store.state.user.id)"
|
|
|
|
|
|
type="textarea"
|
|
|
|
|
|
:rows="4"
|
|
|
|
|
|
></el-input>
|
|
|
|
|
|
</el-form-item>
|
|
|
|
|
|
</el-form>
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
|
<span
|
|
|
|
|
|
class="dialog-footer"
|
|
|
|
|
|
v-if="projectInfo.userId == $store.state.user.id"
|
|
|
|
|
|
>
|
|
|
|
|
|
<el-button @click="logDialogVisible = false">取消</el-button>
|
2024-11-06 09:49:10 +00:00
|
|
|
|
<el-button v-if="logForm.loggerId" type="danger" @click="delWorkLog"
|
|
|
|
|
|
>删除</el-button
|
|
|
|
|
|
>
|
2024-10-15 09:53:49 +00:00
|
|
|
|
<el-button type="primary" @click="saveWorkLog">确定</el-button>
|
|
|
|
|
|
</span>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
</el-dialog>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script>
|
|
|
|
|
|
import { workLogApi, projectApi } from "@/utils/api";
|
|
|
|
|
|
import CustomTable from "@/components/CustomTable.vue";
|
|
|
|
|
|
|
|
|
|
|
|
export default {
|
|
|
|
|
|
components: {
|
|
|
|
|
|
CustomTable,
|
|
|
|
|
|
// ArrowLeft,
|
|
|
|
|
|
// ArrowRight,
|
|
|
|
|
|
},
|
|
|
|
|
|
data() {
|
|
|
|
|
|
return {
|
|
|
|
|
|
isCollapsed: false,
|
|
|
|
|
|
selectedDate: new Date(),
|
|
|
|
|
|
projectList: [],
|
|
|
|
|
|
logData: [],
|
|
|
|
|
|
logDialogVisible: false,
|
|
|
|
|
|
projectInfo: {},
|
|
|
|
|
|
logForm: {
|
|
|
|
|
|
loggerDate: "",
|
|
|
|
|
|
workContent: "",
|
|
|
|
|
|
workTime: "",
|
|
|
|
|
|
status: -1,
|
|
|
|
|
|
max: 1,
|
|
|
|
|
|
loggerId: "",
|
|
|
|
|
|
},
|
2024-10-16 09:32:16 +00:00
|
|
|
|
routeQuery: this.$route.query,
|
2024-10-15 09:53:49 +00:00
|
|
|
|
};
|
|
|
|
|
|
},
|
|
|
|
|
|
methods: {
|
|
|
|
|
|
toggleCollapse() {
|
|
|
|
|
|
this.isCollapsed = !this.isCollapsed;
|
|
|
|
|
|
},
|
2024-10-16 09:32:16 +00:00
|
|
|
|
async handleProjectClick(row, first) {
|
2024-10-15 09:53:49 +00:00
|
|
|
|
if (!row.projectId) {
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const res = await projectApi.getProjectDetail(row.projectId);
|
|
|
|
|
|
this.projectInfo.projectName = res.data.projectName;
|
|
|
|
|
|
this.projectInfo.projectId = res.data.projectId;
|
|
|
|
|
|
this.projectInfo.projectCode = res.data.projectCode;
|
|
|
|
|
|
this.projectInfo.startDate =
|
|
|
|
|
|
res.data.startDate.split(" ")[0] + " 00:00:00";
|
|
|
|
|
|
this.projectInfo.endDate = res.data.endDate.split(" ")[0] + " 23:59:59";
|
|
|
|
|
|
|
2024-10-17 10:00:03 +00:00
|
|
|
|
if (this.projectInfo.userId == this.$store.state.user.id) {
|
|
|
|
|
|
this.selectedDate = new Date();
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.selectedDate = new Date(this.projectInfo.startDate); // 设置为项目开始时间
|
|
|
|
|
|
}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
|
|
|
|
|
|
const res2 = await workLogApi.getLogData({
|
|
|
|
|
|
projectId: this.projectInfo.projectId,
|
|
|
|
|
|
startDate: this.projectInfo.startDate,
|
|
|
|
|
|
endDate: this.projectInfo.endDate,
|
|
|
|
|
|
userId: this.projectInfo.userId,
|
|
|
|
|
|
});
|
|
|
|
|
|
this.logData = res2.data;
|
2024-10-16 09:32:16 +00:00
|
|
|
|
this.initDateColor(first);
|
2024-10-15 09:53:49 +00:00
|
|
|
|
},
|
2024-10-16 09:32:16 +00:00
|
|
|
|
initDateColor(first) {
|
2025-01-06 09:52:52 +00:00
|
|
|
|
let boxs = document.getElementsByClassName("date-cell");
|
|
|
|
|
|
for (var i = 0; i < boxs.length; i++) {
|
|
|
|
|
|
boxs[i].style = "";
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-10-15 09:53:49 +00:00
|
|
|
|
this.logData.map((item) => {
|
|
|
|
|
|
var ele = document.getElementById(item.date.split(" ")[0]);
|
|
|
|
|
|
if (ele) {
|
|
|
|
|
|
if (item.state == -1) {
|
|
|
|
|
|
ele.style = "background:#ecf5ff";
|
2025-01-06 09:52:52 +00:00
|
|
|
|
} else if (item.state == 0) {
|
|
|
|
|
|
ele.style = `background:linear-gradient(to right, #409eff ${
|
|
|
|
|
|
(item.workTime || 1) * 100
|
|
|
|
|
|
}% , #ecf5ff ${(item.workTime || 1) * 100}%);color:${
|
|
|
|
|
|
item.workTime > 0.65 ? "#fff" : "#333"
|
|
|
|
|
|
};`;
|
2024-10-15 09:53:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2024-10-16 09:32:16 +00:00
|
|
|
|
|
|
|
|
|
|
if (document.getElementsByClassName("is-selected")[0] && first == 1) {
|
|
|
|
|
|
document.getElementsByClassName("is-selected")[0].className = document
|
|
|
|
|
|
.getElementsByClassName("is-selected")[0]
|
|
|
|
|
|
.className.replace("is-selected", "");
|
|
|
|
|
|
}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
},
|
|
|
|
|
|
async openLogDialog(data) {
|
|
|
|
|
|
if (!this.projectInfo.projectId) {
|
|
|
|
|
|
this.$modal.msgWarning("请先选择一个项目");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const clickedDate = new Date(data.day);
|
|
|
|
|
|
const currentDate = new Date();
|
|
|
|
|
|
if (
|
|
|
|
|
|
clickedDate.getTime() > currentDate.getTime() &&
|
|
|
|
|
|
clickedDate.getTime() < new Date(this.projectInfo.endDate).getTime()
|
|
|
|
|
|
) {
|
2024-10-16 09:32:16 +00:00
|
|
|
|
if (this.projectInfo.userId == this.$store.state.user.id)
|
|
|
|
|
|
this.$modal.msgWarning("不可编辑未来日期");
|
|
|
|
|
|
else this.$modal.msgWarning("不可查看未来日期");
|
|
|
|
|
|
|
2024-10-15 09:53:49 +00:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const date = new Date(data.day);
|
|
|
|
|
|
const start = new Date(this.projectInfo.startDate);
|
|
|
|
|
|
const end = new Date(this.projectInfo.endDate);
|
|
|
|
|
|
let flag =
|
|
|
|
|
|
date.getTime() >= start.getTime() && date.getTime() <= end.getTime();
|
|
|
|
|
|
|
|
|
|
|
|
if (!flag) {
|
|
|
|
|
|
this.$modal.msgWarning("该日期不在项目范围内");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
this.logForm.loggerDate = data.day + " 00:00:00";
|
|
|
|
|
|
let logTime =
|
|
|
|
|
|
this.logData.find((ele) => ele.date == this.logForm.loggerDate) || {};
|
|
|
|
|
|
this.logForm.status = logTime.state;
|
|
|
|
|
|
if (this.logForm.status != -1) {
|
|
|
|
|
|
const res = await workLogApi.getLogDataDetail({
|
|
|
|
|
|
userId: this.projectInfo.userId,
|
|
|
|
|
|
projectId: this.projectInfo.projectId,
|
|
|
|
|
|
loggerDate: this.logForm.loggerDate,
|
|
|
|
|
|
});
|
2024-10-17 10:00:03 +00:00
|
|
|
|
if (res.data) {
|
|
|
|
|
|
this.logForm.workTime = res.data.workTime;
|
|
|
|
|
|
this.logForm.workContent = res.data.workContent;
|
|
|
|
|
|
this.logForm.loggerId = res.data.loggerId;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
this.logForm.workTime = "";
|
|
|
|
|
|
this.logForm.workContent = "";
|
|
|
|
|
|
this.logForm.loggerId = "";
|
|
|
|
|
|
}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
} else {
|
|
|
|
|
|
this.logForm.workTime = "";
|
|
|
|
|
|
this.logForm.workContent = "";
|
|
|
|
|
|
this.logForm.loggerId = "";
|
|
|
|
|
|
}
|
|
|
|
|
|
const res = await workLogApi.getDayTime({
|
|
|
|
|
|
loggerDate: this.logForm.loggerDate,
|
|
|
|
|
|
});
|
2024-10-17 10:00:03 +00:00
|
|
|
|
|
2024-11-06 02:04:43 +00:00
|
|
|
|
this.logForm.max = (
|
|
|
|
|
|
Number(res.data) + (Number(this.logForm.workTime) || 0)
|
|
|
|
|
|
).toFixed(1);
|
2024-10-15 09:53:49 +00:00
|
|
|
|
this.logDialogVisible = true;
|
2024-10-17 10:00:03 +00:00
|
|
|
|
this.$nextTick(() => {
|
|
|
|
|
|
this.changeMonth();
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
changeMonth() {
|
|
|
|
|
|
let that = this;
|
|
|
|
|
|
this.$nextTick(async () => {
|
|
|
|
|
|
const res2 = await workLogApi.getLogData({
|
|
|
|
|
|
projectId: that.projectInfo.projectId,
|
|
|
|
|
|
startDate:
|
|
|
|
|
|
that
|
|
|
|
|
|
.moment(that.selectedDate)
|
|
|
|
|
|
.startOf("month")
|
|
|
|
|
|
.format("YYYY-MM-DD") + " 00:00:00",
|
|
|
|
|
|
endDate:
|
|
|
|
|
|
that.moment(that.selectedDate).endOf("month").format("YYYY-MM-DD") +
|
|
|
|
|
|
" 23:59:59",
|
|
|
|
|
|
userId: that.projectInfo.userId,
|
|
|
|
|
|
});
|
|
|
|
|
|
that.logData = res2.data;
|
|
|
|
|
|
that.initDateColor(1);
|
|
|
|
|
|
});
|
2024-10-15 09:53:49 +00:00
|
|
|
|
},
|
|
|
|
|
|
isInProjectRange(data) {
|
|
|
|
|
|
if (!this.projectInfo) return false;
|
|
|
|
|
|
const date = new Date(data.day);
|
|
|
|
|
|
const start = new Date(this.projectInfo.startDate);
|
|
|
|
|
|
const end = new Date(this.projectInfo.endDate);
|
|
|
|
|
|
return (
|
|
|
|
|
|
date.getTime() >= start.getTime() && date.getTime() <= end.getTime()
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
isFutureDate(date) {
|
|
|
|
|
|
const clickedDate = new Date(date);
|
|
|
|
|
|
const currentDate = new Date();
|
|
|
|
|
|
return clickedDate.getTime() > currentDate.getTime();
|
|
|
|
|
|
},
|
|
|
|
|
|
getSummaries(param) {
|
|
|
|
|
|
const { columns, data } = param;
|
|
|
|
|
|
const sums = [];
|
|
|
|
|
|
columns.forEach((column, index) => {
|
|
|
|
|
|
if (index === 0) {
|
2024-10-16 09:32:16 +00:00
|
|
|
|
sums[index] = "合计工时(天)";
|
2024-10-15 09:53:49 +00:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
const values = data.map((item) => Number(item[column.property]));
|
|
|
|
|
|
if (!values.every((value) => isNaN(value))) {
|
|
|
|
|
|
sums[index] = values.reduce((prev, curr) => {
|
|
|
|
|
|
const value = Number(curr);
|
|
|
|
|
|
if (!isNaN(value)) {
|
|
|
|
|
|
return prev + curr;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return prev;
|
|
|
|
|
|
}
|
|
|
|
|
|
}, 0);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
sums[index] = "N/A";
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
if (sums[1] != "N/A") sums[1] = Number(sums[1]).toFixed(2);
|
|
|
|
|
|
return sums;
|
|
|
|
|
|
},
|
|
|
|
|
|
async saveWorkLog() {
|
|
|
|
|
|
if (this.logForm.workTime > this.logForm.max) {
|
|
|
|
|
|
this.$modal.msgWarning("工时超过最大值");
|
|
|
|
|
|
return;
|
|
|
|
|
|
} else if (!this.logForm.workContent) {
|
|
|
|
|
|
this.$modal.msgWarning("工作日志不能为空");
|
|
|
|
|
|
return;
|
|
|
|
|
|
} else if (
|
|
|
|
|
|
(String(this.logForm.workTime).split(".")[1] || "").length > 1
|
|
|
|
|
|
) {
|
|
|
|
|
|
this.$modal.msgWarning("工时只允许一位小数");
|
|
|
|
|
|
return;
|
2024-10-16 09:32:16 +00:00
|
|
|
|
} else if (this.logForm.workTime <= 0) {
|
|
|
|
|
|
this.$modal.msgWarning("工时不能小于等于0");
|
|
|
|
|
|
return;
|
2024-10-15 09:53:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
let state =
|
|
|
|
|
|
new Date(this.logForm.loggerDate).getTime() ==
|
|
|
|
|
|
new Date().setHours(0, 0, 0, 0)
|
|
|
|
|
|
? 0
|
|
|
|
|
|
: 1;
|
|
|
|
|
|
let param = {
|
|
|
|
|
|
...this.logForm,
|
|
|
|
|
|
projectId: this.projectInfo.projectId,
|
|
|
|
|
|
state,
|
|
|
|
|
|
userId: this.projectInfo.userId,
|
|
|
|
|
|
};
|
|
|
|
|
|
if (this.logForm.loggerId) {
|
|
|
|
|
|
await workLogApi.editLog(param);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
await workLogApi.addLog(param);
|
|
|
|
|
|
}
|
|
|
|
|
|
this.logDialogVisible = false;
|
|
|
|
|
|
this.$modal.msgSuccess("操作成功");
|
|
|
|
|
|
const response = await workLogApi.userProject(this.projectInfo.userId);
|
|
|
|
|
|
this.projectList = response.data;
|
2024-11-06 02:04:43 +00:00
|
|
|
|
this.handleProjectClick(this.projectInfo);
|
2024-10-15 09:53:49 +00:00
|
|
|
|
},
|
|
|
|
|
|
async fetchUserProjects() {
|
|
|
|
|
|
try {
|
|
|
|
|
|
const response = await workLogApi.userProject(this.projectInfo.userId);
|
|
|
|
|
|
this.projectList = response.data;
|
|
|
|
|
|
if (this.projectList.length) {
|
|
|
|
|
|
let arr = [];
|
|
|
|
|
|
if (this.$route.query.projectId)
|
|
|
|
|
|
arr = this.projectList.filter(
|
|
|
|
|
|
(ele) => ele.projectId == this.$route.query.projectId
|
|
|
|
|
|
);
|
|
|
|
|
|
if (this.$route.query.projectId && arr.length) {
|
2024-10-16 09:32:16 +00:00
|
|
|
|
this.handleProjectClick(arr[0], 1);
|
2024-10-15 09:53:49 +00:00
|
|
|
|
this.$refs.projectRef.setCurrentRow(arr[0]);
|
|
|
|
|
|
this.projectList = arr;
|
|
|
|
|
|
} else {
|
2024-10-16 09:32:16 +00:00
|
|
|
|
this.handleProjectClick(this.projectList[0], 1);
|
2024-10-15 09:53:49 +00:00
|
|
|
|
this.$refs.projectRef.setCurrentRow(this.projectList[0]);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
} catch (error) {}
|
|
|
|
|
|
},
|
2024-10-16 09:32:16 +00:00
|
|
|
|
init() {
|
|
|
|
|
|
if (this.$route.query.userId) {
|
|
|
|
|
|
this.projectInfo.userId = this.$route.query.userId;
|
|
|
|
|
|
this.projectInfo.nickName = this.$route.query.nickName;
|
|
|
|
|
|
} else {
|
|
|
|
|
|
let userInfo = this.$store.state.user;
|
|
|
|
|
|
this.projectInfo.userId = userInfo.id;
|
|
|
|
|
|
this.projectInfo.nickName = userInfo.nickName;
|
|
|
|
|
|
}
|
|
|
|
|
|
//获取项目列表
|
|
|
|
|
|
this.fetchUserProjects();
|
|
|
|
|
|
},
|
2024-11-06 02:04:43 +00:00
|
|
|
|
// 删除日志
|
|
|
|
|
|
delWorkLog() {
|
|
|
|
|
|
this.$modal.confirm(`是否确认删日志`).then(async () => {
|
|
|
|
|
|
await workLogApi.delLog(this.logForm.loggerId);
|
|
|
|
|
|
this.$modal.msgSuccess("操作成功");
|
2024-11-06 09:49:10 +00:00
|
|
|
|
this.logDialogVisible = false;
|
2024-11-06 02:04:43 +00:00
|
|
|
|
this.changeMonth();
|
|
|
|
|
|
const response = await workLogApi.userProject(this.projectInfo.userId);
|
|
|
|
|
|
this.projectList = response.data;
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
2024-10-16 09:32:16 +00:00
|
|
|
|
},
|
|
|
|
|
|
watch: {
|
|
|
|
|
|
$route(to, from) {
|
|
|
|
|
|
if (!this.$route.query.userId) {
|
|
|
|
|
|
this.init();
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2024-10-15 09:53:49 +00:00
|
|
|
|
},
|
|
|
|
|
|
mounted() {
|
2024-10-16 09:32:16 +00:00
|
|
|
|
this.init();
|
2024-10-15 09:53:49 +00:00
|
|
|
|
},
|
|
|
|
|
|
};
|
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
|
|
.work-log-container {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
height: 88vh;
|
|
|
|
|
|
background-color: white;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-list {
|
2024-10-16 09:32:16 +00:00
|
|
|
|
height: 88vh;
|
2024-10-15 09:53:49 +00:00
|
|
|
|
width: 300px;
|
|
|
|
|
|
border-right: 1px solid #dcdfe6;
|
|
|
|
|
|
transition: all 0.3s;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-list.collapsed {
|
|
|
|
|
|
width: 50px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-list .list-header {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: space-between;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
padding: 10px;
|
|
|
|
|
|
font-size: 16px;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
border-bottom: 1px solid #dcdfe6;
|
|
|
|
|
|
position: relative;
|
|
|
|
|
|
height: 40px;
|
|
|
|
|
|
// flex: 1;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-list .collapse-button-wrapper {
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
right: 0;
|
|
|
|
|
|
top: 0;
|
|
|
|
|
|
bottom: 0;
|
|
|
|
|
|
width: 40px; /* 固定宽度 */
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
transition: all 0.3s;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
padding: 20px;
|
|
|
|
|
|
overflow-y: auto;
|
|
|
|
|
|
padding-bottom: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-picker {
|
|
|
|
|
|
margin-bottom: 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
padding: 0 20px;
|
|
|
|
|
|
font-size: 14px; /* 缩小字体大小 */
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .el-calendar {
|
|
|
|
|
|
--el-calendar-cell-width: 40px; /* 缩小日历单元格宽度 */
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .el-calendar__header {
|
|
|
|
|
|
display: none;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .el-calendar__body {
|
|
|
|
|
|
padding: 10px 0; /* 减小主体内边距 */
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .el-calendar__week {
|
|
|
|
|
|
background-color: #4a4a4a; /* 深灰色背景 */
|
|
|
|
|
|
color: white; /* 白色文字 */
|
|
|
|
|
|
padding: 5px 0; /* 增加一些内边距 */
|
|
|
|
|
|
}
|
2024-11-06 02:04:43 +00:00
|
|
|
|
.project-info-calendar ::v-deep .el-calendar__body th {
|
|
|
|
|
|
height: 60px !important;
|
2024-10-17 10:00:03 +00:00
|
|
|
|
font-weight: bold !important;
|
|
|
|
|
|
font-size: 20px;
|
|
|
|
|
|
}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .el-calendar-day {
|
2024-10-22 01:46:54 +00:00
|
|
|
|
height: 70px; /* 增加日期单元格高度(原高度 + 5px) */
|
2024-10-16 09:32:16 +00:00
|
|
|
|
// padding-top: 5px;
|
2024-10-17 10:00:03 +00:00
|
|
|
|
padding: 10px;
|
|
|
|
|
|
border-radius: 10px;
|
2024-10-15 09:53:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .el-calendar-day.disabled {
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
}
|
2024-11-06 02:04:43 +00:00
|
|
|
|
.project-info-calendar .calendar-view ::v-deep table * {
|
|
|
|
|
|
border: none;
|
2024-10-17 10:00:03 +00:00
|
|
|
|
}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .date-cell {
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
text-align: center;
|
2024-10-22 01:46:54 +00:00
|
|
|
|
line-height: 50px;
|
2024-10-17 10:00:03 +00:00
|
|
|
|
border-radius: 10px;
|
2024-10-15 09:53:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .in-range {
|
|
|
|
|
|
background-color: #ecf5ff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .out-range {
|
2024-10-17 10:00:03 +00:00
|
|
|
|
background-color: rgba(0, 0, 0, 0.1) !important;
|
2024-10-15 09:53:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.project-info-calendar .calendar-view ::v-deep .disabled {
|
|
|
|
|
|
background-color: #fff !important;
|
|
|
|
|
|
color: #c0c4cc;
|
|
|
|
|
|
cursor: not-allowed;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.dialog-footer {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
justify-content: center;
|
|
|
|
|
|
margin-top: 20px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep .el-table {
|
|
|
|
|
|
width: 100% !important;
|
|
|
|
|
|
}
|
2024-10-16 09:32:16 +00:00
|
|
|
|
::v-deep .log-form .el-date-editor {
|
|
|
|
|
|
width: 100% !important;
|
|
|
|
|
|
}
|
2024-10-15 09:53:49 +00:00
|
|
|
|
::v-deep .el-table__body-wrapper {
|
|
|
|
|
|
overflow-x: hidden;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep .el-table .cell {
|
|
|
|
|
|
white-space: nowrap;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
text-overflow: ellipsis;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep .el-table__footer-wrapper {
|
|
|
|
|
|
background-color: #f5f7fa;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep .el-table__footer td {
|
|
|
|
|
|
background-color: #f5f7fa !important;
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
color: #606266;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep .el-table__row {
|
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep tr.el-table__row:hover .el-table__cell {
|
|
|
|
|
|
background: #409eff !important;
|
|
|
|
|
|
color: #fff;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
::v-deep tr.el-table__row.current-row .el-table__cell {
|
|
|
|
|
|
background-color: #409eff !important;
|
|
|
|
|
|
color: #fff;
|
|
|
|
|
|
}
|
|
|
|
|
|
.calendar-header {
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
.custom-table {
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
}
|
|
|
|
|
|
/* 调整合计行的样式 */
|
|
|
|
|
|
::v-deep .el-table__footer-wrapper {
|
|
|
|
|
|
bottom: 0;
|
|
|
|
|
|
position: absolute;
|
|
|
|
|
|
}
|
|
|
|
|
|
::v-deep .el-table__footer-wrapper td {
|
|
|
|
|
|
background-color: #c0c4cc !important;
|
|
|
|
|
|
}
|
|
|
|
|
|
</style>
|