日历完成

This commit is contained in:
kuaifan 2021-06-17 11:59:42 +08:00
parent b417020b9d
commit 5c68caa18a
7 changed files with 217 additions and 132 deletions

View File

@ -25,7 +25,8 @@
:theme="calendarTheme" :theme="calendarTheme"
:template="calendarTemplate" :template="calendarTemplate"
:calendars="calendarList" :calendars="calendarList"
:schedules="scheduleLists" :schedules="calendarTasks"
@beforeCreateSchedule="onBeforeCreateSchedule"
@beforeClickSchedule="onBeforeClickSchedule" @beforeClickSchedule="onBeforeClickSchedule"
@beforeUpdateSchedule="onBeforeUpdateSchedule"/> @beforeUpdateSchedule="onBeforeUpdateSchedule"/>
</div> </div>
@ -71,7 +72,6 @@ export default {
calendarList: [], calendarList: [],
scheduleLoad: 0, scheduleLoad: 0,
scheduleList: [],
} }
}, },
@ -84,10 +84,10 @@ export default {
}, },
computed: { computed: {
...mapState(['projectList']), ...mapState(['projectList', 'calendarTask']),
scheduleLists() { calendarTasks() {
return this.scheduleList.filter(({complete_at}) => !complete_at) return this.calendarTask.filter(({complete_at}) => !complete_at)
} }
}, },
@ -99,7 +99,7 @@ export default {
projectList(data) { projectList(data) {
const list = data.map((project) => { const list = data.map((project) => {
return { return {
id: project.id, id: String(project.id),
name: project.name, name: project.name,
} }
}); });
@ -121,6 +121,9 @@ export default {
titlePlaceholder: () => { titlePlaceholder: () => {
return this.$L("任务描述") return this.$L("任务描述")
}, },
popupSave: () => {
return this.$L("保存");
},
popupEdit: () => { popupEdit: () => {
return this.$L("详情"); return this.$L("详情");
}, },
@ -143,34 +146,7 @@ export default {
}).then(({data}) => { }).then(({data}) => {
this.scheduleLoad--; this.scheduleLoad--;
data.data.some((task) => { data.data.some((task) => {
let schedule = { this.$store.dispatch("saveCalendarTask", task)
id: task.id,
calendarId: task.project_id,
title: task.name,
category: 'allday',
start: new Date(task.start_at).toISOString(),
end: new Date(task.end_at).toISOString(),
color: "#515a6e",
bgColor: task.color || '#E3EAFD',
borderColor: task.p_color,
complete_at: task.complete_at,
preventClick: true,
isChecked: false,
};
if (task.p_name) {
schedule.priority = '<span class="priority" style="background-color:' + task.p_color + '">' + task.p_name + '</span>';
}
if (task.overdue) {
schedule.color = "#f56c6c"
schedule.bgColor = "#fef0f0"
schedule.priority+= '<span class="overdue">' + this.$L('超期未完成') + '</span>';
}
let index = this.scheduleList.findIndex(({id}) => id === task.id);
if (index > -1) {
this.scheduleList.splice(index, 1, schedule)
} else {
this.scheduleList.push(schedule)
}
}); });
}).catch(() => { }).catch(() => {
this.scheduleLoad--; this.scheduleLoad--;
@ -224,8 +200,20 @@ export default {
return currentDate.format(format); return currentDate.format(format);
}, },
onBeforeCreateSchedule(res) {
this.$store.dispatch("taskAdd", {
project_id: res.calendarId,
times: [res.start.toDate(), res.end.toDate()],
name: res.title
}).then(({msg}) => {
$A.messageSuccess(msg);
}).catch(({msg}) => {
$A.modalError(msg);
});
},
onBeforeClickSchedule({type, schedule}) { onBeforeClickSchedule({type, schedule}) {
let data = this.scheduleList.find(({id}) => id === schedule.id); let data = this.calendarTask.find(({id}) => id === schedule.id);
if (!data) { if (!data) {
return; return;
} }
@ -253,7 +241,6 @@ export default {
content: '你确定要删除任务【' + data.title + '】吗?', content: '你确定要删除任务【' + data.title + '】吗?',
loading: true, loading: true,
onOk: () => { onOk: () => {
this.scheduleList = this.scheduleList.filter(({id}) => id !== data.id);
this.$store.dispatch("taskArchivedOrRemove", { this.$store.dispatch("taskArchivedOrRemove", {
task_id: data.id, task_id: data.id,
type: 'delete', type: 'delete',
@ -272,9 +259,27 @@ export default {
}, },
onBeforeUpdateSchedule(res) { onBeforeUpdateSchedule(res) {
console.group('onBeforeUpdateSchedule'); const {changes, schedule} = res;
console.log('BeforeUpdate : ', res); let data = this.calendarTask.find(({id}) => id === schedule.id);
console.groupEnd(); if (!data) {
return;
}
if (changes.start || changes.end) {
const cal = this.$refs.cal.getInstance();
cal.updateSchedule(schedule.id, schedule.calendarId, changes);
//
this.$store.dispatch("taskUpdate", {
task_id: data.id,
times: [
(changes.start || schedule.start).toDate(),
(changes.end || schedule.end).toDate(),
],
}).then(({msg}) => {
$A.messageSuccess(msg);
}).catch(({msg}) => {
$A.modalError(msg);
});
}
} }
} }
} }

View File

@ -108,10 +108,10 @@ export default {
watch: { watch: {
calendars(newValue) { calendars(newValue) {
this.calendarInstance.setCalendars(newValue); this.calendarInstance.setCalendars(newValue);
this.$nextTick(this.resetRender)
}, },
schedules() { schedules() {
this.calendarInstance.clear(); this.resetRender();
this.reflectSchedules();
}, },
view(newValue) { view(newValue) {
this.calendarInstance.changeView(newValue, true); this.calendarInstance.changeView(newValue, true);

View File

@ -6,9 +6,6 @@
<h1>{{projectDetail.name}}</h1> <h1>{{projectDetail.name}}</h1>
<div v-if="projectLoad > 0" class="project-load"><Loading/></div> <div v-if="projectLoad > 0" class="project-load"><Loading/></div>
</div> </div>
<div v-if="projectDetail.desc" class="project-subtitle">{{projectDetail.desc}}</div>
</div>
<div class="project-icobox">
<ul class="project-icons"> <ul class="project-icons">
<li> <li>
<UserAvatar :userid="projectDetail.owner_userid" :size="36"> <UserAvatar :userid="projectDetail.owner_userid" :size="36">
@ -45,14 +42,15 @@
</EDropdown> </EDropdown>
</li> </li>
</ul> </ul>
<div class="project-switch"> </div>
<div v-if="completedCount > 0" class="project-checkbox"> <div v-if="projectDetail.desc" class="project-subtitle">{{projectDetail.desc}}</div>
<Checkbox :value="projectCompleteHide" @on-change="toggleBoolean('projectCompleteHide', $event)">{{$L('隐藏已完成')}}</Checkbox> <div class="project-switch">
</div> <div v-if="completedCount > 0" class="project-checkbox">
<div :class="['project-switch-button', !projectTablePanel ? 'menu' : '']" @click="toggleBoolean('projectTablePanel')"> <Checkbox :value="projectCompleteShow" @on-change="toggleBoolean('projectCompleteShow', $event)">{{$L('显示已完成')}}</Checkbox>
<div><i class="iconfont">&#xe60c;</i></div> </div>
<div><i class="iconfont">&#xe66a;</i></div> <div :class="['project-switch-button', !projectTablePanel ? 'menu' : '']" @click="toggleBoolean('projectTablePanel')">
</div> <div><i class="iconfont">&#xe60c;</i></div>
<div><i class="iconfont">&#xe66a;</i></div>
</div> </div>
</div> </div>
</div> </div>
@ -450,7 +448,7 @@ export default {
'projectChatShow', 'projectChatShow',
'projectTablePanel', 'projectTablePanel',
'projectCompleteHide', 'projectCompleteShow',
'taskMyShow', 'taskMyShow',
'taskUndoneShow', 'taskUndoneShow',
'taskCompletedShow' 'taskCompletedShow'
@ -463,9 +461,9 @@ export default {
}, },
panelTask() { panelTask() {
const {searchText, projectCompleteHide} = this; const {searchText, projectCompleteShow} = this;
return function (project_task) { return function (project_task) {
if (projectCompleteHide) { if (!projectCompleteShow) {
project_task = project_task.filter(({complete_at}) => { project_task = project_task.filter(({complete_at}) => {
return !complete_at; return !complete_at;
}); });
@ -480,11 +478,11 @@ export default {
}, },
myList() { myList() {
const {searchText, projectCompleteHide, userId, projectDetail} = this; const {searchText, projectCompleteShow, userId, projectDetail} = this;
const array = []; const array = [];
projectDetail.project_column.forEach(({project_task, name}) => { projectDetail.project_column.forEach(({project_task, name}) => {
project_task.some((task) => { project_task.some((task) => {
if (projectCompleteHide) { if (!projectCompleteShow) {
if (task.complete_at) { if (task.complete_at) {
return false; return false;
} }
@ -512,11 +510,11 @@ export default {
}, },
undoneList() { undoneList() {
const {searchText, projectCompleteHide, projectDetail} = this; const {searchText, projectCompleteShow, projectDetail} = this;
const array = []; const array = [];
projectDetail.project_column.forEach(({project_task, name}) => { projectDetail.project_column.forEach(({project_task, name}) => {
project_task.some((task) => { project_task.some((task) => {
if (projectCompleteHide) { if (!projectCompleteShow) {
if (task.complete_at) { if (task.complete_at) {
return false; return false;
} }
@ -553,11 +551,11 @@ export default {
}, },
completedList() { completedList() {
const {searchText, projectCompleteHide, projectDetail} = this; const {searchText, projectCompleteShow, projectDetail} = this;
const array = []; const array = [];
projectDetail.project_column.forEach(({project_task, name}) => { projectDetail.project_column.forEach(({project_task, name}) => {
project_task.some((task) => { project_task.some((task) => {
if (projectCompleteHide) { if (!projectCompleteShow) {
if (task.complete_at) { if (task.complete_at) {
return false; return false;
} }
@ -1032,8 +1030,8 @@ export default {
taskHidden(task) { taskHidden(task) {
const {name, desc, complete_at} = task; const {name, desc, complete_at} = task;
const {searchText, projectCompleteHide} = this; const {searchText, projectCompleteShow} = this;
if (projectCompleteHide) { if (!projectCompleteShow) {
if (complete_at) { if (complete_at) {
return true; return true;
} }

View File

@ -378,9 +378,10 @@ export default {
/** /**
* 保存任务信息 * 保存任务信息
* @param state * @param state
* @param dispatch
* @param data * @param data
*/ */
saveTask({state}, data) { saveTask({state, dispatch}, data) {
state.projectDetail.project_column.some(({project_task}) => { state.projectDetail.project_column.some(({project_task}) => {
let index = project_task.findIndex(({id}) => id === data.id); let index = project_task.findIndex(({id}) => id === data.id);
if (index > -1) { if (index > -1) {
@ -396,6 +397,46 @@ export default {
state.projectOpenTask.sub_task.splice(index, 1, Object.assign(state.projectOpenTask.sub_task[index], data)) state.projectOpenTask.sub_task.splice(index, 1, Object.assign(state.projectOpenTask.sub_task[index], data))
} }
} }
dispatch("saveCalendarTask", data)
},
/**
* 保存任务信息日历任务
* @param state
* @param data
*/
saveCalendarTask({state}, data) {
let task = {
id: data.id,
calendarId: String(data.project_id),
title: data.name,
body: data.desc,
category: 'allday',
start: new Date(data.start_at).toISOString(),
end: new Date(data.end_at).toISOString(),
color: "#515a6e",
bgColor: data.color || '#E3EAFD',
borderColor: data.p_color,
complete_at: data.complete_at,
priority: '',
preventClick: true,
isChecked: false,
};
if (data.p_name) {
task.priority = '<span class="priority" style="background-color:' + data.p_color + '">' + data.p_name + '</span>';
}
if (data.overdue) {
task.title = '[' + $A.L('超期') + '] ' + task.title
task.color = "#f56c6c"
task.bgColor = "#fef0f0"
task.priority+= '<span class="overdue">' + $A.L('超期未完成') + '</span>';
}
let index = state.calendarTask.findIndex(({id}) => id === data.id);
if (index > -1) {
state.calendarTask.splice(index, 1, Object.assign(state.calendarTask[index], task))
} else {
state.calendarTask.push(task)
}
}, },
/** /**
@ -560,7 +601,7 @@ export default {
*/ */
taskAdd({state, dispatch}, data) { taskAdd({state, dispatch}, data) {
return new Promise(function (resolve, reject) { return new Promise(function (resolve, reject) {
const post = state.method.cloneJSON(data); const post = state.method.cloneJSON(state.method.date2string(data));
if (state.method.isArray(post.column_id)) { if (state.method.isArray(post.column_id)) {
post.column_id = post.column_id.find((val) => val) post.column_id = post.column_id.find((val) => val)
} }
@ -587,6 +628,7 @@ export default {
} }
} }
} }
dispatch("saveCalendarTask", task);
dispatch("getProjectOne", task.project_id); dispatch("getProjectOne", task.project_id);
resolve(result) resolve(result)
}).catch(result => { }).catch(result => {
@ -628,7 +670,7 @@ export default {
*/ */
taskUpdate({state, dispatch}, data) { taskUpdate({state, dispatch}, data) {
return new Promise(function (resolve, reject) { return new Promise(function (resolve, reject) {
const post = state.method.cloneJSON(data); const post = state.method.cloneJSON(state.method.date2string(data));
if (state.method.isArray(post.owner)) { if (state.method.isArray(post.owner)) {
post.owner = post.owner.find((id) => id) post.owner = post.owner.find((id) => id)
} }
@ -684,6 +726,11 @@ export default {
state.projectOpenTask.sub_task.splice(index, 1) state.projectOpenTask.sub_task.splice(index, 1)
} }
} }
let index = state.calendarTask.findIndex(({id}) => id === data.id);
if (index > -1) {
state.calendarTask.splice(index, 1)
}
dispatch("getProjectOne", data.project_id);
resolve(result); resolve(result);
}).catch(result => { }).catch(result => {
reject(result) reject(result)

View File

@ -236,7 +236,7 @@ const state = { method };
[ [
'projectChatShow', // 项目聊天显示 'projectChatShow', // 项目聊天显示
'projectCompleteHide' // 项目面板显示已完成列表 'projectCompleteShow' // 项目面板显示已完成列表
].forEach((key) => { ].forEach((key) => {
state[key] = state.method.getStorageBoolean("boolean:" + key, false) state[key] = state.method.getStorageBoolean("boolean:" + key, false)
}); });
@ -275,6 +275,7 @@ state.projectOpenTask = {_show: false, id: 0, task_user: [], task_tag: []};
state.projectTaskContent = {}; state.projectTaskContent = {};
state.projectTaskFiles = {}; state.projectTaskFiles = {};
state.projectSubTask = {}; state.projectSubTask = {};
state.calendarTask = [];
// 会话聊天 // 会话聊天
state.dialogId = 0; state.dialogId = 0;

View File

@ -3,11 +3,15 @@
flex-direction: column; flex-direction: column;
.project-head { .project-head {
display: flex; display: flex;
flex-direction: column;
align-items: flex-start; align-items: flex-start;
margin: 32px 32px 0; margin: 32px 32px 0;
.project-titbox { .project-titbox {
flex: 1; width: 100%;
margin-bottom: 16px; display: flex;
align-items: flex-start;
justify-content: space-between;
margin-bottom: 36px;
.project-title { .project-title {
display: flex; display: flex;
align-items: center; align-items: center;
@ -26,23 +30,11 @@
} }
} }
} }
.project-subtitle {
color: #999999;
margin-top: 18px;
line-height: 24px;
}
}
.project-icobox {
display: flex;
flex-direction: column;
justify-content: space-between;
height: 100%;
margin-top: 2px;
margin-left: 80px;
.project-icons { .project-icons {
display: flex; display: flex;
align-items: center; align-items: center;
flex-shrink: 0; flex-shrink: 0;
margin-top: 3px;
> li { > li {
list-style: none; list-style: none;
display: flex; display: flex;
@ -84,63 +76,78 @@
} }
} }
} }
.project-switch { }
.project-subtitle {
width: 100%;
color: #999999;
line-height: 24px;
margin-top: -18px;
margin-bottom: 6px;
padding-right: 260px;
}
.project-switch {
width: 100%;
display: flex;
justify-content: flex-end;
.project-checkbox {
display: flex; display: flex;
justify-content: flex-end; align-items: center;
margin-top: 24px; margin-right: 14px;
.project-checkbox { opacity: 0.9;
display: flex; .ivu-checkbox-focus {
align-items: center; box-shadow: none;
margin-right: 14px;
opacity: 0.9;
} }
.project-switch-button { }
.project-switch-button {
display: flex;
align-items: center;
background-color: #ffffff;
border-radius: 6px;
position: relative;
transition: box-shadow 0.2s;
&:hover {
box-shadow: 0 0 10px #e6ecfa;
}
&:before {
content: "";
position: absolute;
top: 0;
left: 0;
width: 50%;
height: 100%;
z-index: 0;
color: #2d8cf0;
border-radius: 6px;
border: 1px solid #2d8cf0;
background-color: #e6f7ff;
transition: left 0.2s;
}
> div {
z-index: 1;
width: 32px;
height: 30px;
display: flex; display: flex;
align-items: center; align-items: center;
background-color: #ffffff; justify-content: center;
border-radius: 6px; border-radius: 6px;
position: relative; cursor: pointer;
&:before { color: #515a6e;
content: ""; > i {
position: absolute; font-size: 17px;
top: 0; }
left: 0; &:first-child {
width: 50%;
height: 100%;
z-index: 0;
color: #2d8cf0; color: #2d8cf0;
border-radius: 6px;
border: 1px solid #2d8cf0;
background-color: #e6f7ff;
transition: left 0.2s;
} }
> div { }
z-index: 1; &.menu {
width: 32px; &:before {
height: 30px; left: 50%;
display: flex; }
align-items: center; > div:first-child {
justify-content: center;
border-radius: 6px;
cursor: pointer;
color: #515a6e; color: #515a6e;
> i {
font-size: 17px;
}
&:first-child {
color: #2d8cf0;
}
} }
&.menu { > div:last-child {
&:before { color: #2d8cf0;
left: 50%;
}
> div:first-child {
color: #515a6e;
}
> div:last-child {
color: #2d8cf0;
}
} }
} }
} }

View File

@ -56,6 +56,18 @@
background-color: #ffffff; background-color: #ffffff;
z-index: 1; z-index: 1;
} }
.tui-full-calendar-popup {
box-shadow: none;
.tui-full-calendar-popup-container {
border: 0;
box-shadow: 0 1px 6px rgba(0, 0, 0, 0.2);
border-radius: 6px;
}
.tui-full-calendar-arrow-top .tui-full-calendar-popup-arrow-border {
top: -8px;
border-bottom-color: rgba(217, 217, 217, .5);
}
}
.tui-full-calendar-popup-creation { .tui-full-calendar-popup-creation {
.tui-full-calendar-icon { .tui-full-calendar-icon {
&.tui-full-calendar-ic-title, &.tui-full-calendar-ic-title,
@ -73,10 +85,21 @@
.tui-full-calendar-popup-section { .tui-full-calendar-popup-section {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
margin-bottom: 6px; margin-bottom: 10px;
.tui-full-calendar-popup-section-item {
height: 36px;
border-color: #e8e8e8;
border-radius: 4px;
}
.tui-full-calendar-popup-section-item input {
height: 34px;
}
} }
.tui-full-calendar-section-title { .tui-full-calendar-section-title {
width: 100%; width: 100%;
input {
width: 100%;
}
} }
.tui-full-calendar-section-start-date, .tui-full-calendar-section-start-date,
.tui-full-calendar-section-end-date { .tui-full-calendar-section-end-date {
@ -119,12 +142,16 @@
top: -2px; top: -2px;
background-image: url(""); background-image: url("");
} }
.tui-full-calendar-popup-detail-item-separate {
padding-left: 22px;
}
} }
.tui-datepicker { .tui-datepicker {
border-color: #e8e8e8;
.tui-calendar { .tui-calendar {
th, th,
td { td {
height: 35px; height: 32px;
} }
.tui-calendar-prev-month.tui-calendar-date, .tui-calendar-prev-month.tui-calendar-date,
.tui-calendar-next-month.tui-calendar-date { .tui-calendar-next-month.tui-calendar-date {