|
|
@ -18,7 +18,8 @@ |
|
|
|
|
|
|
|
<script> |
|
|
|
import { mapState, mapGetters, mapMutations, mapActions } from 'vuex'; |
|
|
|
import { setPlaceholderTasks } from '@/utils/task'; |
|
|
|
import { setPlaceholderTasks, computeFillPlaceholderTaskCount } from '@/utils/task'; |
|
|
|
import { flatten } from 'lodash'; |
|
|
|
|
|
|
|
export default { |
|
|
|
data() { |
|
|
@ -97,125 +98,45 @@ export default { |
|
|
|
'setBottomEnd', |
|
|
|
]), |
|
|
|
|
|
|
|
/** |
|
|
|
* 初始化 |
|
|
|
* @param {object | null} options |
|
|
|
*/ |
|
|
|
init(options) { |
|
|
|
if (!this.token) { |
|
|
|
// 不论有没有token都直接从userId获取token |
|
|
|
// token有过期时间 从本地获取可能是过期 干脆直接从userId获取 |
|
|
|
if (!options || !options.u) { |
|
|
|
this.$t.ui.showToast('缺少用户信息参数'); // 参数里没有u (userId)提示 |
|
|
|
} else { |
|
|
|
this.getToken(options.u); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
// 参数里有项目名称 就设置标题里的项目名称 |
|
|
|
options && options.pname && this.setProjectName(options.pname); |
|
|
|
|
|
|
|
if (!options || !options.p) { |
|
|
|
this.$t.ui.showToast('缺少项目信息参数'); // 没有项目id参数 |
|
|
|
} else { |
|
|
|
this.getProjectById({ projectId: options.p }); // 根据项目id获取项目信息 |
|
|
|
} |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
|
* 通过项目id获取项目信息 |
|
|
|
* @param {object} params 提交的参数 |
|
|
|
*/ |
|
|
|
async getProjectById(params) { |
|
|
|
try { |
|
|
|
const data = await uni.$u.api.findProjectById(params); |
|
|
|
this.setProject(data); |
|
|
|
// 根据项目id获取角色列表 |
|
|
|
this.getRoles(params); |
|
|
|
} catch (error) { |
|
|
|
console.log('error: ', error || '获取项目信息失败'); |
|
|
|
} |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
|
* 通过项目id获取角色信息 |
|
|
|
* @param {string} projectId |
|
|
|
* @param {object} params 提交的参数 |
|
|
|
*/ |
|
|
|
getRoles(params) { |
|
|
|
this.$t.$q.findShowRole(params, (err, data) => { |
|
|
|
if (err) { |
|
|
|
console.error('err: ', err || '获取角色信息失败'); |
|
|
|
} else { |
|
|
|
this.setInvisibleRoles(data ? data.invisibleList : []); |
|
|
|
this.setVisibleRoles(data ? data.visibleList : []); |
|
|
|
this.setInitialRoleId(data ? data.visibleList : []); |
|
|
|
} |
|
|
|
}); |
|
|
|
}, |
|
|
|
|
|
|
|
// 清除已有的任务数据 |
|
|
|
clearTasksData() { |
|
|
|
// 清空日常任务的数据 |
|
|
|
this.setPermanents([]); |
|
|
|
this.setDailyTasks([]); |
|
|
|
// 清空定期任务数据 |
|
|
|
this.clearTasks(); |
|
|
|
// 到顶的标志复位 |
|
|
|
// 到底的标志复位 |
|
|
|
this.clearEndFlag(); |
|
|
|
}, |
|
|
|
|
|
|
|
// 初始化 时间轴 |
|
|
|
// 初始化 定期任务 |
|
|
|
async initPlanTasks() { |
|
|
|
this.setPrevTasks(); // 向上加载空数据 |
|
|
|
this.setNextTasks(); // 向下加载空数据 |
|
|
|
this.setPrevPlaceholderTasks(); // 向上加载空数据 |
|
|
|
this.setNextPlaceholderTasks(); // 向下加载空数据 |
|
|
|
this.$nextTick(() => this.$refs.timeLine.setScrollPosition()); // 滚动到对应位置 |
|
|
|
|
|
|
|
await this.getInitTasks(); |
|
|
|
await this.getInitTasks(); // 获取初始数据 |
|
|
|
this.$nextTick(() => this.$refs.timeLine.setScrollPosition()); // 滚动到对应位置 |
|
|
|
}, |
|
|
|
|
|
|
|
// 切换了 颗粒度 || 角色时候 获取初始定期任务 |
|
|
|
getInitTasks() { |
|
|
|
// 预加载 上下的定期任务 |
|
|
|
function fn(that) { |
|
|
|
function preloadFn(that) { |
|
|
|
const detailId = that.tasks.findIndex(task => task.detailId); |
|
|
|
console.log('detailId: ', detailId); |
|
|
|
if (detailId !== -1) { |
|
|
|
// 只要有1个真实的任务 就预加载上下周期的任务 |
|
|
|
const { pageCount } = that.$t.task; |
|
|
|
that.$nextTick(() => { |
|
|
|
// 向上拿数据 |
|
|
|
const { tasks, timeGranularity } = that; |
|
|
|
that.getTasks({ timeNode: +tasks[0].planStart, queryType: 0, queryNum: 6 }); |
|
|
|
that.getTasks({ timeNode: +tasks[0].planStart, queryType: 0, queryNum: pageCount }); |
|
|
|
// 向下拿数据 |
|
|
|
const nextQueryTime = +that.$t.time.add(+tasks[tasks.length - 1].planStart, 1, timeGranularity); |
|
|
|
that.getTasks({ timeNode: nextQueryTime, queryType: 1, queryNum: 6 }); |
|
|
|
that.getTasks({ timeNode: nextQueryTime, queryType: 1, queryNum: pageCount }); |
|
|
|
}); |
|
|
|
} else { |
|
|
|
// 没有任务 上下显示时间刻度 |
|
|
|
// 向上加载 |
|
|
|
// that.setPrevTasks(); |
|
|
|
// that.setPrevPlaceholderTasks(); |
|
|
|
// // 向下加载 |
|
|
|
// that.setNextTasks(); |
|
|
|
// that.setNextPlaceholderTasks(); |
|
|
|
} |
|
|
|
} |
|
|
|
// 根据时间基准点和角色查找定期任务 |
|
|
|
this.getTasks({ queryType: 0 }); // 向上获取定期任务数据 |
|
|
|
|
|
|
|
// 根据项目id获取角色列表 |
|
|
|
this.getTasks({ queryType: 1 }, fn); // 向下获取定期任务数据 |
|
|
|
}, |
|
|
|
|
|
|
|
// 设置 初始显示角色信息 |
|
|
|
setInitialRoleId(visibleList) { |
|
|
|
if (!visibleList || !visibleList.length) return; |
|
|
|
const index = visibleList.findIndex(item => +item.mine === 1); |
|
|
|
const currentRole = index > 0 ? visibleList[index] : visibleList[0]; |
|
|
|
const storageRoleId = this.$t.storage.getStorageSync('roleId'); |
|
|
|
const currentRoleId = storageRoleId ? storageRoleId : currentRole ? currentRole.id : ''; |
|
|
|
this.setRoleId(currentRoleId); |
|
|
|
// 清空storage |
|
|
|
this.$t.storage.setStorageSync('roleId', ''); |
|
|
|
this.getTasks({ queryType: 1 }, preloadFn); // 向下获取定期任务数据 |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
@ -225,19 +146,11 @@ export default { |
|
|
|
* @param {string} query.timeNode 时间基准点 默认当前 |
|
|
|
* @param {string} query.timeUnit 时间颗粒度 默认天 |
|
|
|
* @param {string} query.queryNum 查找颗粒度数量 默认3个 |
|
|
|
* @param {string} query.queryType 0向上查找 1向下查找(默认) 下查包含自己,上查不包含 |
|
|
|
* @param {number} query.queryType 0向上查找 1向下查找(默认) 下查包含自己,上查不包含 |
|
|
|
*/ |
|
|
|
getTasks(query, fn) { |
|
|
|
this.setShowSkeleton(true); |
|
|
|
const { roleId, timeNode, timeUnit, projectId } = this; |
|
|
|
const params = { |
|
|
|
roleId, |
|
|
|
timeNode: query.timeNode || timeNode, |
|
|
|
timeUnit: query.timeUnit || timeUnit, |
|
|
|
queryNum: query.queryNum || 3, |
|
|
|
queryType: query.queryType, |
|
|
|
projectId, |
|
|
|
}; |
|
|
|
const params = this.generateGetTaskParam(query); |
|
|
|
|
|
|
|
this.$t.$q.getRegularTask(params, (err, data) => { |
|
|
|
this.setShowSkeleton(false); |
|
|
@ -245,14 +158,17 @@ export default { |
|
|
|
// TODO: 提示错误 |
|
|
|
console.error('err: ', err); |
|
|
|
} else { |
|
|
|
// 0 -> 向上 1 -> 向下 |
|
|
|
// 有数据用数据替换刻度 |
|
|
|
// 没有数据 继续加载刻度 |
|
|
|
if (data && data.length) { |
|
|
|
this.replacePrevData(data, params.queryType); |
|
|
|
} else { |
|
|
|
params.queryType === 0 ? this.setPrevTasks() : this.setNextTasks(); |
|
|
|
} |
|
|
|
|
|
|
|
// else { |
|
|
|
// TODO: 0 -> 向上 1 -> 向下 |
|
|
|
|
|
|
|
// params.queryType === 0 ? this.setPrevPlaceholderTasks() : this.setNextPlaceholderTasks(); |
|
|
|
// } |
|
|
|
if (this.tasks.length && fn) { |
|
|
|
fn(this); |
|
|
|
} |
|
|
@ -260,15 +176,24 @@ export default { |
|
|
|
}); |
|
|
|
}, |
|
|
|
|
|
|
|
// 获取可变全局任务 |
|
|
|
getGlobalData() { |
|
|
|
/** |
|
|
|
* 生成getTasks所用的参数 |
|
|
|
* @param {object} query getTasks传递的参数 |
|
|
|
*/ |
|
|
|
generateGetTaskParam(query) { |
|
|
|
const { roleId, timeNode, timeUnit, projectId } = this; |
|
|
|
const param = { roleId, timeNode, timeUnit, projectId }; |
|
|
|
this.getGlobal(param); |
|
|
|
return { |
|
|
|
roleId, |
|
|
|
timeNode: query.timeNode || timeNode, |
|
|
|
timeUnit: query.timeUnit || timeUnit, |
|
|
|
queryNum: query.queryNum || 3, |
|
|
|
queryType: query.queryType, |
|
|
|
projectId, |
|
|
|
}; |
|
|
|
}, |
|
|
|
|
|
|
|
// 设置时间轴向上的空数据 |
|
|
|
setPrevTasks() { |
|
|
|
setPrevPlaceholderTasks() { |
|
|
|
this.setTopEnd(true); |
|
|
|
let startTime = ''; |
|
|
|
const { tasks } = this; |
|
|
@ -282,72 +207,146 @@ export default { |
|
|
|
}, |
|
|
|
|
|
|
|
// 设置时间轴向下的空数据 |
|
|
|
setNextTasks() { |
|
|
|
setNextPlaceholderTasks() { |
|
|
|
this.setBottomEnd(true); |
|
|
|
let sTime = ''; |
|
|
|
let startTime = ''; |
|
|
|
if (!this.tasks || !this.tasks.length) { |
|
|
|
sTime = +new Date().getTime(); |
|
|
|
startTime = Date.now(); |
|
|
|
} else { |
|
|
|
sTime = +this.tasks[this.tasks.length - 1].planStart - 0; |
|
|
|
startTime = +this.tasks[this.tasks.length - 1].planStart; |
|
|
|
} |
|
|
|
const initData = setPlaceholderTasks(sTime, false, this.timeGranularity); |
|
|
|
const initData = setPlaceholderTasks(startTime, false, this.timeGranularity); |
|
|
|
this.setDownTasks(initData); |
|
|
|
}, |
|
|
|
|
|
|
|
// 筛选相同的时间替换数据 |
|
|
|
/** |
|
|
|
* 用拿到的新数据 替换 时间刻度/旧数据 |
|
|
|
* 先对比 新旧数据的 始末时间 补齐刻度 |
|
|
|
* 再遍历对比 用任务替换刻度 |
|
|
|
* @param {array} data 服务端返回的新数据 上边已经处理过空值 |
|
|
|
* @param {number} type 0 -> 向上 1->向下 |
|
|
|
*/ |
|
|
|
replacePrevData(data, type) { |
|
|
|
let newTasks = [...this.tasks]; |
|
|
|
let newDate = this.$u.deepClone(data); |
|
|
|
|
|
|
|
console.log('newDate: ', newDate); |
|
|
|
|
|
|
|
if (newDate && newDate.length) { |
|
|
|
for (let i = 0; i < newTasks.length; i++) { |
|
|
|
// const task = newTasks[i]; |
|
|
|
let arr = []; |
|
|
|
for (let j = 0; j < newDate.length; j++) { |
|
|
|
const item = newDate[j]; |
|
|
|
if (newTasks[i].id === item.id) { |
|
|
|
console.log('j', j, i); |
|
|
|
newTasks[i] = item; |
|
|
|
break; |
|
|
|
} |
|
|
|
// 查找有没有超出时间刻度的时间 |
|
|
|
if (+newTasks[0].planStart > +newDate[0].planStart) { |
|
|
|
this.setPrevTasks(); |
|
|
|
newTasks = [...this.tasks]; |
|
|
|
i--; |
|
|
|
break; |
|
|
|
} else if (+newDate[newDate.length - 1].planStart > +newTasks[newTasks.length - 1].planStart) { |
|
|
|
this.setNextTasks(); |
|
|
|
newTasks = [...this.tasks]; |
|
|
|
i--; |
|
|
|
break; |
|
|
|
} else { |
|
|
|
// 筛选相同的时间替换数据 |
|
|
|
const taskItem = this.$t.time.isSame(+newTasks[i].planStart, +item.planStart, this.timeGranularity); |
|
|
|
if (taskItem) { |
|
|
|
arr.push(item); |
|
|
|
if (newTasks[i].detailId) { |
|
|
|
newTasks.splice(i, 0, item); |
|
|
|
} else { |
|
|
|
if (arr.length === 1) { |
|
|
|
newTasks.splice(i, 1, item); |
|
|
|
} |
|
|
|
if (arr.length > 1) { |
|
|
|
newTasks.splice(i, 0, item); |
|
|
|
} |
|
|
|
} |
|
|
|
i++; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
const { timeGranularity } = this; |
|
|
|
let oldTasks = this.fillPlaceholderTask({ tasks: this.tasks, data, timeGranularity }); // 已经上下补齐时间刻度的 |
|
|
|
|
|
|
|
// 遍历对比 用任务替换刻度 |
|
|
|
// TODO: tasks越来越多 遍历越来越多 需要优化 |
|
|
|
oldTasks.forEach((taskItem, index) => { |
|
|
|
const arr = data.filter(dataItem => this.$moment(+dataItem.planStart).isSame(+taskItem.planStart, timeGranularity)); |
|
|
|
if (arr && arr.length) { |
|
|
|
oldTasks.splice(index, 1, [...arr]); // 这里加入的数据是array类型的, [{},{},[],[],{}] |
|
|
|
} |
|
|
|
}); |
|
|
|
|
|
|
|
oldTasks = flatten(oldTasks); // 1维拍平 |
|
|
|
|
|
|
|
this.clearTasks(); // setUpTasks setUpTasks 的限制 需要清空 |
|
|
|
type === 0 ? this.setUpTasks(oldTasks) : this.setUpTasks(oldTasks); |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
|
* 超出旧数据上、下限 补齐时间刻度到新数据的起始时间颗粒度 |
|
|
|
*/ |
|
|
|
fillPlaceholderTask({ tasks, data, timeGranularity }) { |
|
|
|
const { prev, next } = computeFillPlaceholderTaskCount({ tasks, data, timeGranularity }); |
|
|
|
if (prev) { |
|
|
|
const newTasks = setPlaceholderTasks(+tasks[0].planStart, true, timeGranularity, prev); |
|
|
|
this.setUpTasks(newTasks); |
|
|
|
} |
|
|
|
if (next) { |
|
|
|
const newTasks = setPlaceholderTasks(+tasks[tasks.length - 1].planStart, false, timeGranularity, next); |
|
|
|
this.setDownTasks(newTasks); |
|
|
|
} |
|
|
|
return this.tasks; |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
|
* 初始化 |
|
|
|
* @param {object | null} options |
|
|
|
*/ |
|
|
|
init(options) { |
|
|
|
if (!this.token) { |
|
|
|
// 不论有没有token都直接从userId获取token |
|
|
|
// token有过期时间 从本地获取可能是过期 干脆直接从userId获取 |
|
|
|
if (!options || !options.u) { |
|
|
|
this.$t.ui.showToast('缺少用户信息参数'); // 参数里没有u (userId)提示 |
|
|
|
} else { |
|
|
|
this.getToken(options.u); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
// 参数里有项目名称 就设置标题里的项目名称 |
|
|
|
options && options.pname && this.setProjectName(options.pname); |
|
|
|
|
|
|
|
if (!options || !options.p) { |
|
|
|
this.$t.ui.showToast('缺少项目信息参数'); // 没有项目id参数 |
|
|
|
} else { |
|
|
|
this.getProjectById({ projectId: options.p }); // 根据项目id获取项目信息 |
|
|
|
} |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
|
* 通过项目id获取项目信息 |
|
|
|
* @param {object} params 提交的参数 |
|
|
|
*/ |
|
|
|
async getProjectById(params) { |
|
|
|
try { |
|
|
|
const data = await uni.$u.api.findProjectById(params); |
|
|
|
this.setProject(data); |
|
|
|
// 根据项目id获取角色列表 |
|
|
|
this.getRoles(params); |
|
|
|
} catch (error) { |
|
|
|
console.log('error: ', error || '获取项目信息失败'); |
|
|
|
} |
|
|
|
}, |
|
|
|
|
|
|
|
/** |
|
|
|
* 通过项目id获取角色信息 |
|
|
|
* @param {string} projectId |
|
|
|
* @param {object} params 提交的参数 |
|
|
|
*/ |
|
|
|
getRoles(params) { |
|
|
|
this.$t.$q.findShowRole(params, (err, data) => { |
|
|
|
if (err) { |
|
|
|
console.error('err: ', err || '获取角色信息失败'); |
|
|
|
} else { |
|
|
|
this.setInvisibleRoles(data ? data.invisibleList : []); |
|
|
|
this.setVisibleRoles(data ? data.visibleList : []); |
|
|
|
this.setInitialRoleId(data ? data.visibleList : []); |
|
|
|
} |
|
|
|
}); |
|
|
|
}, |
|
|
|
|
|
|
|
// 设置 初始显示角色信息 |
|
|
|
setInitialRoleId(visibleList) { |
|
|
|
if (!visibleList || !visibleList.length) return; |
|
|
|
const index = visibleList.findIndex(item => +item.mine === 1); |
|
|
|
const currentRole = index > 0 ? visibleList[index] : visibleList[0]; |
|
|
|
const storageRoleId = this.$t.storage.getStorageSync('roleId'); |
|
|
|
const currentRoleId = storageRoleId ? storageRoleId : currentRole ? currentRole.id : ''; |
|
|
|
this.setRoleId(currentRoleId); |
|
|
|
// 清空storage |
|
|
|
this.$t.storage.setStorageSync('roleId', ''); |
|
|
|
}, |
|
|
|
|
|
|
|
// 获取可变全局任务 |
|
|
|
getGlobalData() { |
|
|
|
const { roleId, timeNode, timeUnit, projectId } = this; |
|
|
|
const param = { roleId, timeNode, timeUnit, projectId }; |
|
|
|
this.getGlobal(param); |
|
|
|
}, |
|
|
|
|
|
|
|
// 清除已有的任务数据 |
|
|
|
clearTasksData() { |
|
|
|
// 清空日常任务的数据 |
|
|
|
this.setPermanents([]); |
|
|
|
this.setDailyTasks([]); |
|
|
|
// 清空定期任务数据 |
|
|
|
this.clearTasks(); |
|
|
|
type === 0 ? this.setUpTasks(newTasks) : this.setDownTasks(newTasks); |
|
|
|
console.log('newTasks: ', newTasks.length); |
|
|
|
console.log('end'); |
|
|
|
// 到顶的标志复位 |
|
|
|
// 到底的标志复位 |
|
|
|
this.clearEndFlag(); |
|
|
|
}, |
|
|
|
}, |
|
|
|
}; |
|
|
|