未完成项目概况

This commit is contained in:
kuaifan 2022-01-04 20:23:01 +08:00
parent f1f1d784ff
commit 186290e355
6 changed files with 487 additions and 192 deletions

View File

@ -26,21 +26,52 @@ use Request;
class ProjectController extends AbstractController class ProjectController extends AbstractController
{ {
/** /**
* 获取项目列表 * @api {get} api/project/lists 获取项目列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName lists
* *
* @apiParam {String} [all] 是否查看所有项目(限制管理员) * @apiParam {String} [all] 是否查看所有项目(限制管理员)
* @apiParam {String} [archived] 归档状态 * @apiParam {String} [archived] 归档状态
* - all全部 * - all全部
* - no未归档默认 * - no未归档默认
* - yes已归档 * - yes已归档
* @apiParam {String} [andcolumn] 同时取项目列表 * @apiParam {String} [getcolumn] 同时取项目列表
* - no不取默认 * - no不取默认
* - yes取列表 * - yes取列表
* @apiParam {Object} [keys] 搜索条件 * @apiParam {Object} [keys] 搜索条件
* - keys.name 项目名称 * - keys.name: 项目名称
* *
* @apiParam {Number} [page] 当前页,默认:1 * @apiParam {Number} [page] 当前页,默认:1
* @apiParam {Number} [pagesize] 每页显示数量,默认:50,最大:100 * @apiParam {Number} [pagesize] 每页显示数量,默认:50,最大:100
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
* @apiSuccessExample {json} dataDemo:
{
"data": [
{
"id": 7,
"name": "🏢 产品官网项目",
"desc": "设置各小组成员的工作列表,各自领取或领导分配任务,将做好的任务分期归档,方便复盘!",
"userid": 1,
"dialog_id": 15,
"archived_at": null,
"archived_userid": 0,
"created_at": "2022-01-02 06:23:15",
"updated_at": "2022-01-02 07:12:33",
"owner": 1, // 是否项目负责人
"owner_userid": 1 // 项目负责人ID
},
],
"current_page": 1, // 当前页数
"last_page": 1, // 下一页数
"total": 6, // 总计数(当前查询条件)
"total_all": 6 // 总计数(全部)
}
*/ */
public function lists() public function lists()
{ {
@ -48,16 +79,16 @@ class ProjectController extends AbstractController
// //
$all = Request::input('all'); $all = Request::input('all');
$archived = Request::input('archived', 'no'); $archived = Request::input('archived', 'no');
$andcolumn = Request::input('andcolumn', 'no'); $getcolumn = Request::input('getcolumn', 'no');
// //
if ($all) { if ($all) {
$user->identity('admin'); $user->identity('admin');
$builder = Project::select('projects.*'); $builder = Project::allData();
} else { } else {
$builder = Project::select(Project::projectSelect)->authData(); $builder = Project::authData();
} }
// //
if ($andcolumn == 'yes') { if ($getcolumn == 'yes') {
$builder->with(['projectColumn']); $builder->with(['projectColumn']);
} }
// //
@ -74,7 +105,11 @@ class ProjectController extends AbstractController
$builder->where("projects.name", "like", "%{$keys['name']}%"); $builder->where("projects.name", "like", "%{$keys['name']}%");
} }
} }
//
$list = $builder->orderByDesc('projects.id')->paginate(Base::getPaginate(100, 50)); $list = $builder->orderByDesc('projects.id')->paginate(Base::getPaginate(100, 50));
$list->transform(function (Project $project) {
return array_merge($project->toArray(), $project->getTaskStatistics());
});
// //
$data = $list->toArray(); $data = $list->toArray();
if (isset($buildClone)) { if (isset($buildClone)) {
@ -87,9 +122,32 @@ class ProjectController extends AbstractController
} }
/** /**
* 获取一个项目信息 * @api {get} api/project/one 获取一个项目信息
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName one
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
* @apiSuccessExample {json} dataDemo:
{
"id": 7,
"name": "🏢 产品官网项目",
"desc": "设置各小组成员的工作列表,各自领取或领导分配任务,将做好的任务分期归档,方便复盘!",
"userid": 1,
"dialog_id": 15,
"archived_at": null,
"archived_userid": 0,
"created_at": "2022-01-02 06:23:15",
"updated_at": "2022-01-02 07:12:33",
"owner": 1, // 是否项目负责人
"owner_userid": 1 // 项目负责人ID
}
*/ */
public function one() public function one()
{ {
@ -98,16 +156,26 @@ class ProjectController extends AbstractController
$project_id = intval(Request::input('project_id')); $project_id = intval(Request::input('project_id'));
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id);
$data = array_merge($project->toArray(), $project->getTaskStatistics());
// //
return Base::retSuccess('success', $project); return Base::retSuccess('success', $data);
} }
/** /**
* 添加项目 * @api {get} api/project/add 添加项目
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName add
* *
* @apiParam {String} name 项目名称 * @apiParam {String} name 项目名称
* @apiParam {String} [desc] 项目介绍 * @apiParam {String} [desc] 项目介绍
* @apiParam {String} [columns] 列表格式列表名称1,列表名称2 * @apiParam {String} [columns] 列表格式列表名称1,列表名称2
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function add() public function add()
{ {
@ -177,11 +245,20 @@ class ProjectController extends AbstractController
} }
/** /**
* 修改项目(限:项目负责人) * @api {get} api/project/update 修改项目
*
* @apiDescription 需要token身份项目负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName update
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {String} name 项目名称 * @apiParam {String} name 项目名称
* @apiParam {String} [desc] 项目介绍 * @apiParam {String} [desc] 项目介绍
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function update() public function update()
{ {
@ -199,10 +276,7 @@ class ProjectController extends AbstractController
return Base::retError('项目介绍最多只能设置255个字'); return Base::retError('项目介绍最多只能设置255个字');
} }
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id, true, true);
if (!$project->owner) {
return Base::retError('仅限项目负责人修改');
}
// //
if ($project->name != $name) { if ($project->name != $name) {
$project->addLog("修改项目名称:{$project->name} => {$name}"); $project->addLog("修改项目名称:{$project->name} => {$name}");
@ -219,10 +293,19 @@ class ProjectController extends AbstractController
} }
/** /**
* 修改项目成员(限:项目负责人) * @api {get} api/project/user 修改项目成员
*
* @apiDescription 需要token身份项目负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName user
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {Number} userid 成员ID 成员ID组 * @apiParam {Number} userid 成员ID 成员ID组
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function user() public function user()
{ {
@ -232,10 +315,7 @@ class ProjectController extends AbstractController
$userid = Request::input('userid'); $userid = Request::input('userid');
$userid = is_array($userid) ? $userid : [$userid]; $userid = is_array($userid) ? $userid : [$userid];
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id, true, true);
if (!$project->owner) {
return Base::retError('仅限项目负责人修改');
}
// //
$deleteUser = AbstractModel::transaction(function() use ($project, $userid) { $deleteUser = AbstractModel::transaction(function() use ($project, $userid) {
$array = []; $array = [];
@ -260,12 +340,21 @@ class ProjectController extends AbstractController
} }
/** /**
* 获取邀请链接(限:项目负责人) * @api {get} api/project/invite 获取邀请链接
*
* @apiDescription 需要token身份项目负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName invite
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {String} refresh 刷新链接 * @apiParam {String} refresh 刷新链接
* - no: 只获取(默认) * - no: 只获取(默认)
* - yes: 刷新链接,之前的将失效 * - yes: 刷新链接,之前的将失效
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function invite() public function invite()
{ {
@ -274,10 +363,7 @@ class ProjectController extends AbstractController
$project_id = intval(Request::input('project_id')); $project_id = intval(Request::input('project_id'));
$refresh = Request::input('refresh', 'no'); $refresh = Request::input('refresh', 'no');
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id, true, true);
if (!$project->owner) {
return Base::retError('仅限项目负责人查看');
}
// //
$invite = Base::settingFind('system', 'project_invite'); $invite = Base::settingFind('system', 'project_invite');
if ($invite == 'close') { if ($invite == 'close') {
@ -304,9 +390,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 通过邀请链接code获取项目信息 * @api {get} api/project/invite/info 通过邀请链接code获取项目信息
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName invite__info
* *
* @apiParam {String} code * @apiParam {String} code
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function invite__info() public function invite__info()
{ {
@ -327,9 +422,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 通过邀请链接code加入项目 * @api {get} api/project/invite/join 通过邀请链接code加入项目
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName invite__join
* *
* @apiParam {String} code * @apiParam {String} code
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function invite__join() public function invite__join()
{ {
@ -364,10 +468,19 @@ class ProjectController extends AbstractController
} }
/** /**
* 移交项目(限:项目负责人) * @api {get} api/project/transfer 移交项目
*
* @apiDescription 需要token身份项目负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName transfer
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {Number} owner_userid 新的项目负责人ID * @apiParam {Number} owner_userid 新的项目负责人ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function transfer() public function transfer()
{ {
@ -376,10 +489,7 @@ class ProjectController extends AbstractController
$project_id = intval(Request::input('project_id')); $project_id = intval(Request::input('project_id'));
$owner_userid = intval(Request::input('owner_userid')); $owner_userid = intval(Request::input('owner_userid'));
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id, true, true);
if (!$project->owner) {
return Base::retError('你不是项目负责人');
}
// //
if (!User::whereUserid($owner_userid)->exists()) { if (!User::whereUserid($owner_userid)->exists()) {
return Base::retError('会员不存在'); return Base::retError('会员不存在');
@ -402,11 +512,20 @@ class ProjectController extends AbstractController
} }
/** /**
* 排序任务 * @api {get} api/project/sort 排序任务
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName sort
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {Object} sort 排序数据 * @apiParam {Object} sort 排序数据
* @apiParam {Number} [only_column] 仅更新列表 * @apiParam {Number} [only_column] 仅更新列表
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function sort() public function sort()
{ {
@ -455,9 +574,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 退出项目 * @api {get} api/project/exit 退出项目
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName exit
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function exit() public function exit()
{ {
@ -465,14 +593,11 @@ class ProjectController extends AbstractController
// //
$project_id = intval(Request::input('project_id')); $project_id = intval(Request::input('project_id'));
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id, true, false);
if ($project->owner) {
return Base::retError('项目负责人无法退出项目');
}
// //
AbstractModel::transaction(function() use ($user, $project) { AbstractModel::transaction(function() use ($user, $project) {
$row = ProjectUser::whereProjectId($project->id)->whereUserid($user->userid)->first(); $row = ProjectUser::whereProjectId($project->id)->whereUserid($user->userid)->first();
$row && $row->exitProject(); $row?->exitProject();
$project->syncDialogUser(); $project->syncDialogUser();
$project->addLog("会员ID" . $user->userid . " 退出项目"); $project->addLog("会员ID" . $user->userid . " 退出项目");
$project->pushMsg('delete', null, $user->userid); $project->pushMsg('delete', null, $user->userid);
@ -481,12 +606,21 @@ class ProjectController extends AbstractController
} }
/** /**
* 归档项目(限:项目负责人) * @api {get} api/project/archived 归档项目
*
* @apiDescription 需要token身份项目负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName archived
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {String} [type] 类型 * @apiParam {String} [type] 类型
* - add归档默认 * - add归档默认
* - recovery还原归档 * - recovery还原归档
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function archived() public function archived()
{ {
@ -495,10 +629,7 @@ class ProjectController extends AbstractController
$project_id = intval(Request::input('project_id')); $project_id = intval(Request::input('project_id'));
$type = Request::input('type', 'add'); $type = Request::input('type', 'add');
// //
$project = Project::userProject($project_id, false); $project = Project::userProject($project_id, false, true);
if (!$project->owner) {
return Base::retError('仅限项目负责人操作');
}
// //
if ($type == 'recovery') { if ($type == 'recovery') {
$project->archivedProject(null); $project->archivedProject(null);
@ -509,9 +640,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 删除项目(限:项目负责人) * @api {get} api/project/remove 删除项目
*
* @apiDescription 需要token身份项目负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName remove
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function remove() public function remove()
{ {
@ -519,22 +659,28 @@ class ProjectController extends AbstractController
// //
$project_id = intval(Request::input('project_id')); $project_id = intval(Request::input('project_id'));
// //
$project = Project::userProject($project_id); $project = Project::userProject($project_id, true, true);
if (!$project->owner) {
return Base::retError('仅限项目负责人删除');
}
// //
$project->deleteProject(); $project->deleteProject();
return Base::retSuccess('删除成功', ['id' => $project->id]); return Base::retSuccess('删除成功', ['id' => $project->id]);
} }
/** /**
* 获取任务列表 * @api {get} api/project/column/lists 获取任务列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName column__lists
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* *
* @apiParam {Number} [page] 当前页,默认:1 * @apiParam {Number} [page] 当前页,默认:1
* @apiParam {Number} [pagesize] 每页显示数量,默认:100,最大:200 * @apiParam {Number} [pagesize] 每页显示数量,默认:100,最大:200
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function column__lists() public function column__lists()
{ {
@ -553,10 +699,19 @@ class ProjectController extends AbstractController
} }
/** /**
* 添加任务列表 * @api {get} api/project/column/add 添加任务列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName column__add
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {String} name 列表名称 * @apiParam {String} name 列表名称
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function column__add() public function column__add()
{ {
@ -585,11 +740,20 @@ class ProjectController extends AbstractController
} }
/** /**
* 修改任务列表 * @api {get} api/project/column/update 修改任务列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName column__update
* *
* @apiParam {Number} column_id 列表ID * @apiParam {Number} column_id 列表ID
* @apiParam {String} [name] 列表名称 * @apiParam {String} [name] 列表名称
* @apiParam {String} [color] 颜色 * @apiParam {String} [color] 颜色
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function column__update() public function column__update()
{ {
@ -619,9 +783,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 删除任务列表 * @api {get} api/project/column/remove 删除任务列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName column__remove
* *
* @apiParam {Number} column_id 列表ID留空为添加列表 * @apiParam {Number} column_id 列表ID留空为添加列表
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function column__remove() public function column__remove()
{ {
@ -641,7 +814,12 @@ class ProjectController extends AbstractController
} }
/** /**
* 任务列表 * @api {get} api/project/task/lists 任务列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__lists
* *
* @apiParam {Number} [project_id] 项目ID * @apiParam {Number} [project_id] 项目ID
* @apiParam {Number} [parent_id] 主任务ID填写此项时 project_id 参数无效) * @apiParam {Number} [parent_id] 主任务ID填写此项时 project_id 参数无效)
@ -655,6 +833,10 @@ class ProjectController extends AbstractController
* @apiParam {String} [archived] 归档状态 * @apiParam {String} [archived] 归档状态
* - yes已归档 * - yes已归档
* - no未归档默认 * - no未归档默认
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__lists() public function task__lists()
{ {
@ -672,12 +854,12 @@ class ProjectController extends AbstractController
// //
if ($parent_id > 0) { if ($parent_id > 0) {
ProjectTask::userTask($parent_id); ProjectTask::userTask($parent_id);
$builder->ownerData()->where('project_tasks.parent_id', $parent_id); $builder->allData()->where('project_tasks.parent_id', $parent_id);
} elseif ($project_id > 0) { } elseif ($project_id > 0) {
Project::userProject($project_id); Project::userProject($project_id);
$builder->ownerData()->where('project_tasks.project_id', $project_id); $builder->allData()->where('project_tasks.project_id', $project_id);
} else { } else {
$builder->joinData(); $builder->authData();
} }
// //
if ($name) { if ($name) {
@ -712,9 +894,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 获取单个任务信息 * @api {get} api/project/task/one 获取单个任务信息
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__one
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__one() public function task__one()
{ {
@ -724,16 +915,26 @@ class ProjectController extends AbstractController
// //
$task = ProjectTask::userTask($task_id, ['taskUser', 'taskTag'], true, $project); $task = ProjectTask::userTask($task_id, ['taskUser', 'taskTag'], true, $project);
// //
$task->project_name = $project?->name; $data = $task->toArray();
$task->column_name = ProjectColumn::whereId($task->column_id)->value('name'); $data['project_name'] = $project?->name;
$data['column_name'] = ProjectColumn::whereId($task->column_id)->value('name');
// //
return Base::retSuccess('success', $task); return Base::retSuccess('success', $data);
} }
/** /**
* 获取任务详细描述 * @api {get} api/project/task/content 获取任务详细描述
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__content
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__content() public function task__content()
{ {
@ -747,9 +948,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 获取任务文件列表 * @api {get} api/project/task/files 获取任务文件列表
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__files
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__files() public function task__files()
{ {
@ -763,9 +973,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 删除任务文件(限:项目、任务负责人) * @api {get} api/project/task/filedelete 删除任务文件
*
* @apiDescription 需要token身份项目、任务负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__filedelete
* *
* @apiParam {Number} file_id 文件ID * @apiParam {Number} file_id 文件ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__filedelete() public function task__filedelete()
{ {
@ -790,7 +1009,12 @@ class ProjectController extends AbstractController
} }
/** /**
* {post} 添加任务 * @api {post} api/project/task/add 添加任务
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__add
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {mixed} [column_id] 列表ID任意值自动创建留空取第一个 * @apiParam {mixed} [column_id] 列表ID任意值自动创建留空取第一个
@ -800,6 +1024,10 @@ class ProjectController extends AbstractController
* @apiParam {Number} [owner] 负责人 * @apiParam {Number} [owner] 负责人
* @apiParam {Array} [subtasks] 子任务(格式:[{name,owner,times}] * @apiParam {Array} [subtasks] 子任务(格式:[{name,owner,times}]
* @apiParam {Number} [top] 添加的任务排到列表最前面 * @apiParam {Number} [top] 添加的任务排到列表最前面
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__add() public function task__add()
{ {
@ -830,8 +1058,8 @@ class ProjectController extends AbstractController
$column->sort = intval(ProjectColumn::whereProjectId($project->id)->orderByDesc('sort')->value('sort')) + 1; $column->sort = intval(ProjectColumn::whereProjectId($project->id)->orderByDesc('sort')->value('sort')) + 1;
$column->save(); $column->save();
$column->addLog("创建列表:" . $column->name); $column->addLog("创建列表:" . $column->name);
$newColumn = $column->find($column->id); $newColumn = $column->find($column->id)->toArray();
$newColumn->project_task = []; $newColumn['project_task'] = [];
} }
if (empty($column)) { if (empty($column)) {
return Base::retError('任务列表不存在或已被删除'); return Base::retError('任务列表不存在或已被删除');
@ -851,10 +1079,19 @@ class ProjectController extends AbstractController
} }
/** /**
* 添加子任务(限:项目、任务负责人) * @api {get} api/project/task/addsub 添加子任务
*
* @apiDescription 需要token身份项目、任务负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__addsub
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
* @apiParam {String} name 任务描述 * @apiParam {String} name 任务描述
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__addsub() public function task__addsub()
{ {
@ -886,7 +1123,12 @@ class ProjectController extends AbstractController
} }
/** /**
* {post} 修改任务、子任务(限:项目、任务负责人) * @api {post} api/project/task/update 修改任务、子任务
*
* @apiDescription 需要token身份项目、任务负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__update
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
* @apiParam {String} [name] 任务描述 * @apiParam {String} [name] 任务描述
@ -901,6 +1143,10 @@ class ProjectController extends AbstractController
* @apiParam {String} [p_color] 优先级相关(子任务不支持) * @apiParam {String} [p_color] 优先级相关(子任务不支持)
* *
* @apiParam {String|false} [complete_at] 完成时间2020-01-01 00:00false表示未完成 * @apiParam {String|false} [complete_at] 完成时间2020-01-01 00:00false表示未完成
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__update() public function task__update()
{ {
@ -948,12 +1194,21 @@ class ProjectController extends AbstractController
} }
/** /**
* {post} 上传文件(限:项目、任务负责人) * @api {post} api/project/task/upload 上传文件
*
* @apiDescription 需要token身份项目、任务负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__upload
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
* @apiParam {String} [filename] post-文件名称 * @apiParam {String} [filename] post-文件名称
* @apiParam {String} [image64] post-base64图片二选一 * @apiParam {String} [image64] post-base64图片二选一
* @apiParam {File} [files] post-文件对象(二选一) * @apiParam {File} [files] post-文件对象(二选一)
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__upload() public function task__upload()
{ {
@ -1008,9 +1263,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 创建/获取聊天室 * @api {get} api/project/task/dialog 创建/获取聊天室
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__dialog
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__dialog() public function task__dialog()
{ {
@ -1046,12 +1310,21 @@ class ProjectController extends AbstractController
} }
/** /**
* 归档任务(限:项目、任务负责人) * @api {get} api/project/task/archived 归档任务
*
* @apiDescription 需要token身份项目、任务负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__archived
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
* @apiParam {String} [type] 类型 * @apiParam {String} [type] 类型
* - add归档默认 * - add归档默认
* - recovery还原归档 * - recovery还原归档
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__archived() public function task__archived()
{ {
@ -1078,9 +1351,18 @@ class ProjectController extends AbstractController
} }
/** /**
* 删除任务(限:项目、任务负责人) * @api {get} api/project/task/remove 删除任务
*
* @apiDescription 需要token身份项目、任务负责人
* @apiVersion 1.0.0
* @apiGroup project
* @apiName task__remove
* *
* @apiParam {Number} task_id 任务ID * @apiParam {Number} task_id 任务ID
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function task__remove() public function task__remove()
{ {
@ -1098,13 +1380,22 @@ class ProjectController extends AbstractController
} }
/** /**
* 获取项目、任务日志 * @api {get} api/project/log/lists 获取项目、任务日志
*
* @apiDescription 需要token身份
* @apiVersion 1.0.0
* @apiGroup project
* @apiName log__lists
* *
* @apiParam {Number} project_id 项目ID * @apiParam {Number} project_id 项目ID
* @apiParam {Number} task_id 任务ID 项目ID 二选一任务ID优先 * @apiParam {Number} task_id 任务ID 项目ID 二选一任务ID优先
* *
* @apiParam {Number} [page] 当前页,默认:1 * @apiParam {Number} [page] 当前页,默认:1
* @apiParam {Number} [pagesize] 每页显示数量,默认:20,最大:100 * @apiParam {Number} [pagesize] 每页显示数量,默认:20,最大:100
*
* @apiSuccess {Number} ret 返回状态码1正确、0错误
* @apiSuccess {String} msg 返回信息(错误描述)
* @apiSuccess {Object} data 返回数据
*/ */
public function log__lists() public function log__lists()
{ {

View File

@ -5,6 +5,7 @@ namespace App\Models;
use App\Exceptions\ApiException; use App\Exceptions\ApiException;
use App\Tasks\PushTask; use App\Tasks\PushTask;
use Carbon\Carbon; use Carbon\Carbon;
use DB;
use Hhxsv5\LaravelS\Swoole\Task\Task; use Hhxsv5\LaravelS\Swoole\Task\Task;
use Illuminate\Database\Eloquent\SoftDeletes; use Illuminate\Database\Eloquent\SoftDeletes;
use Request; use Request;
@ -36,7 +37,9 @@ use Request;
* @property-read int|null $project_log_count * @property-read int|null $project_log_count
* @property-read \Illuminate\Database\Eloquent\Collection|\App\Models\ProjectUser[] $projectUser * @property-read \Illuminate\Database\Eloquent\Collection|\App\Models\ProjectUser[] $projectUser
* @property-read int|null $project_user_count * @property-read int|null $project_user_count
* @method static \Illuminate\Database\Eloquent\Builder|Project allData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|Project authData($userid = null) * @method static \Illuminate\Database\Eloquent\Builder|Project authData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|Project ownerData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|Project newModelQuery() * @method static \Illuminate\Database\Eloquent\Builder|Project newModelQuery()
* @method static \Illuminate\Database\Eloquent\Builder|Project newQuery() * @method static \Illuminate\Database\Eloquent\Builder|Project newQuery()
* @method static \Illuminate\Database\Query\Builder|Project onlyTrashed() * @method static \Illuminate\Database\Query\Builder|Project onlyTrashed()
@ -59,99 +62,14 @@ class Project extends AbstractModel
{ {
use SoftDeletes; use SoftDeletes;
const projectSelect = [ protected $hidden = [
'projects.*', 'deleted_at',
'project_users.owner',
]; ];
protected $appends = [ protected $appends = [
'task_num',
'task_complete',
'task_percent',
'task_my_num',
'task_my_complete',
'task_my_percent',
'owner_userid', 'owner_userid',
]; ];
/**
* 生成任务数据
*/
private function generateTaskData()
{
if (!isset($this->appendattrs['task_num'])) {
$builder = ProjectTask::whereProjectId($this->id)->whereParentId(0)->whereNull('archived_at');
$this->appendattrs['task_num'] = $builder->count();
$this->appendattrs['task_complete'] = $builder->whereNotNull('complete_at')->count();
$this->appendattrs['task_percent'] = $this->appendattrs['task_num'] ? intval($this->appendattrs['task_complete'] / $this->appendattrs['task_num'] * 100) : 0;
//
$builder = ProjectTask::whereProjectId($this->id)->whereParentId(0)->authData(User::userid())->whereNull('archived_at');
$this->appendattrs['task_my_num'] = $builder->count();
$this->appendattrs['task_my_complete'] = $builder->whereNotNull('complete_at')->count();
$this->appendattrs['task_my_percent'] = $this->appendattrs['task_my_num'] ? intval($this->appendattrs['task_my_complete'] / $this->appendattrs['task_my_num'] * 100) : 0;
}
}
/**
* 任务数量
* @return int
*/
public function getTaskNumAttribute()
{
$this->generateTaskData();
return $this->appendattrs['task_num'];
}
/**
* 任务完成数量
* @return int
*/
public function getTaskCompleteAttribute()
{
$this->generateTaskData();
return $this->appendattrs['task_complete'];
}
/**
* 任务完成率
* @return int
*/
public function getTaskPercentAttribute()
{
$this->generateTaskData();
return $this->appendattrs['task_percent'];
}
/**
* 任务数量(我的)
* @return int
*/
public function getTaskMyNumAttribute()
{
$this->generateTaskData();
return $this->appendattrs['task_my_num'];
}
/**
* 任务完成数量(我的)
* @return int
*/
public function getTaskMyCompleteAttribute()
{
$this->generateTaskData();
return $this->appendattrs['task_my_complete'];
}
/**
* 任务完成率(我的)
* @return int
*/
public function getTaskMyPercentAttribute()
{
$this->generateTaskData();
return $this->appendattrs['task_my_percent'];
}
/** /**
* 负责人会员ID * 负责人会员ID
* @return int * @return int
@ -159,7 +77,7 @@ class Project extends AbstractModel
public function getOwnerUseridAttribute() public function getOwnerUseridAttribute()
{ {
if (!isset($this->appendattrs['owner_userid'])) { if (!isset($this->appendattrs['owner_userid'])) {
$ownerUser = $this->projectUser->where('owner', 1)->first(); $ownerUser = ProjectUser::whereProjectId($this->id)->whereOwner(1)->first();
$this->appendattrs['owner_userid'] = $ownerUser ? $ownerUser->userid : 0; $this->appendattrs['owner_userid'] = $ownerUser ? $ownerUser->userid : 0;
} }
return $this->appendattrs['owner_userid']; return $this->appendattrs['owner_userid'];
@ -190,7 +108,29 @@ class Project extends AbstractModel
} }
/** /**
* 查询自己的项目 * 查询所有项目与正常查询多返回owner字段
* @param self $query
* @param null $userid
* @return self
*/
public function scopeAllData($query, $userid = null)
{
$userid = $userid ?: User::userid();
$query
->select([
'projects.*',
'project_users.owner',
])
->leftJoin('project_users', function ($leftJoin) use ($userid) {
$leftJoin
->on('project_users.userid', '=', DB::raw($userid))
->on('projects.id', '=', 'project_users.project_id');
});
return $query;
}
/**
* 查询自己参与的项目
* @param self $query * @param self $query
* @param null $userid * @param null $userid
* @return self * @return self
@ -198,11 +138,51 @@ class Project extends AbstractModel
public function scopeAuthData($query, $userid = null) public function scopeAuthData($query, $userid = null)
{ {
$userid = $userid ?: User::userid(); $userid = $userid ?: User::userid();
$query->join('project_users', 'projects.id', '=', 'project_users.project_id') $query
->select([
'projects.*',
'project_users.owner',
])
->join('project_users', 'projects.id', '=', 'project_users.project_id')
->where('project_users.userid', $userid); ->where('project_users.userid', $userid);
return $query; return $query;
} }
/**
* 查询自己负责的项目
* @param self $query
* @param null $userid
* @return self
*/
public function scopeOwnerData($query, $userid = null)
{
$userid = $userid ?: User::userid();
$query
->select([
'projects.*',
'project_users.owner',
])
->join('project_users', 'projects.id', '=', 'project_users.project_id')
->where('project_users.userid', $userid)
->where('project_users.owner', 1);
return $query;
}
/**
* 获取任务统计
* @return array
*/
public function getTaskStatistics()
{
$array = [];
$builder = ProjectTask::whereProjectId($this->id)->whereParentId(0)->whereNull('archived_at');
$array['task_num'] = $builder->count();
$array['task_complete'] = $builder->whereNotNull('complete_at')->count();
$array['task_percent'] = $array['task_num'] ? intval($array['task_complete'] / $array['task_num'] * 100) : 0;
//
return $array;
}
/** /**
* 加入项目 * 加入项目
* @param int $userid 加入的会员ID * @param int $userid 加入的会员ID
@ -368,18 +348,28 @@ class Project extends AbstractModel
/** /**
* 根据用户获取项目信息(用于判断会员是否存在项目内) * 根据用户获取项目信息(用于判断会员是否存在项目内)
* @param int $project_id * @param int $project_id
* @param bool $ignoreArchived 排除已归档 * @param null|bool $ignoreArchived 排除已归档
* @param null|bool $mustOwner 是否仅限项目负责人
* @return self * @return self
*/ */
public static function userProject($project_id, $ignoreArchived = true) public static function userProject($project_id, $ignoreArchived = true, $mustOwner = null)
{ {
$project = self::select(self::projectSelect)->authData()->where('projects.id', intval($project_id))->first(); $project = self::authData()->where('projects.id', intval($project_id))->first();
if (empty($project)) { if (empty($project)) {
throw new ApiException('项目不存在或不在成员列表内', [ 'project_id' => $project_id ], -4001); throw new ApiException('项目不存在或不在成员列表内', [ 'project_id' => $project_id ], -4001);
} }
if ($ignoreArchived && $project->archived_at != null) { if ($ignoreArchived === true && $project->archived_at != null) {
throw new ApiException('项目已归档', [ 'project_id' => $project_id ], -4001); throw new ApiException('项目已归档', [ 'project_id' => $project_id ], -4001);
} }
if ($ignoreArchived === false && $project->archived_at == null) {
throw new ApiException('项目未归档', [ 'project_id' => $project_id ]);
}
if ($mustOwner === true && !$project->owner) {
throw new ApiException('仅限项目负责人操作', [ 'project_id' => $project_id ]);
}
if ($mustOwner === false && $project->owner) {
throw new ApiException('禁止项目负责人操作', [ 'project_id' => $project_id ]);
}
return $project; return $project;
} }
} }

View File

@ -54,8 +54,8 @@ use Request;
* @property-read int|null $task_tag_count * @property-read int|null $task_tag_count
* @property-read \Illuminate\Database\Eloquent\Collection|\App\Models\ProjectTaskUser[] $taskUser * @property-read \Illuminate\Database\Eloquent\Collection|\App\Models\ProjectTaskUser[] $taskUser
* @property-read int|null $task_user_count * @property-read int|null $task_user_count
* @method static \Illuminate\Database\Eloquent\Builder|ProjectTask allData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|ProjectTask authData($userid = null) * @method static \Illuminate\Database\Eloquent\Builder|ProjectTask authData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|ProjectTask joinData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|ProjectTask ownerData($userid = null) * @method static \Illuminate\Database\Eloquent\Builder|ProjectTask ownerData($userid = null)
* @method static \Illuminate\Database\Eloquent\Builder|ProjectTask betweenTime($start, $end) * @method static \Illuminate\Database\Eloquent\Builder|ProjectTask betweenTime($start, $end)
* @method static \Illuminate\Database\Eloquent\Builder|ProjectTask newModelQuery() * @method static \Illuminate\Database\Eloquent\Builder|ProjectTask newModelQuery()
@ -257,17 +257,27 @@ class ProjectTask extends AbstractModel
} }
/** /**
* 查询自己负责的任务 * 查询所有任务与正常查询多返回owner字段
* @param self $query * @param self $query
* @param null $userid * @param null $userid
* @return self * @return self
*/ */
public function scopeAuthData($query, $userid = null) public function scopeAllData($query, $userid = null)
{ {
DB::statement("SET SQL_MODE=''");
$pre = DB::connection()->getTablePrefix();
$userid = $userid ?: User::userid(); $userid = $userid ?: User::userid();
$query->whereIn('id', function ($qy) use ($userid) { $query
$qy->select('task_pid')->from('project_task_users')->where('userid', $userid)->where('owner', 1); ->select([
}); 'project_tasks.*',
DB::raw("MAX({$pre}project_task_users.owner) as owner")
])
->leftJoin('project_task_users', function ($leftJoin) use ($userid) {
$leftJoin
->on('project_task_users.userid', '=', DB::raw($userid))
->on('project_tasks.id', '=', 'project_task_users.task_id');
})
->groupBy('project_tasks.id');
return $query; return $query;
} }
@ -277,7 +287,7 @@ class ProjectTask extends AbstractModel
* @param null $userid * @param null $userid
* @return self * @return self
*/ */
public function scopeJoinData($query, $userid = null) public function scopeAuthData($query, $userid = null)
{ {
DB::statement("SET SQL_MODE=''"); DB::statement("SET SQL_MODE=''");
$pre = DB::connection()->getTablePrefix(); $pre = DB::connection()->getTablePrefix();
@ -294,7 +304,7 @@ class ProjectTask extends AbstractModel
} }
/** /**
* 查询自己参与的任务(参与条件非必须) * 查询自己负责的任务
* @param self $query * @param self $query
* @param null $userid * @param null $userid
* @return self * @return self
@ -309,11 +319,9 @@ class ProjectTask extends AbstractModel
'project_tasks.*', 'project_tasks.*',
DB::raw("MAX({$pre}project_task_users.owner) as owner") DB::raw("MAX({$pre}project_task_users.owner) as owner")
]) ])
->leftJoin('project_task_users', function ($leftJoin) use ($userid) { ->join('project_task_users', 'project_tasks.id', '=', 'project_task_users.task_id')
$leftJoin ->where('project_task_users.userid', $userid)
->on('project_task_users.userid', '=', DB::raw($userid)) ->where('project_task_users.owner', 1)
->on('project_tasks.id', '=', 'project_task_users.task_id');
})
->groupBy('project_tasks.id'); ->groupBy('project_tasks.id');
return $query; return $query;
} }

View File

@ -42,12 +42,21 @@ class ProjectUser extends AbstractModel
*/ */
public function exitProject() public function exitProject()
{ {
$tasks = ProjectTask::whereProjectId($this->project_id)->authData($this->userid)->get(); ProjectTaskUser::whereProjectId($this->project_id)
->whereUserid($this->userid)
->chunk(100, function ($list) {
$tastIds = [];
foreach ($list as $item) {
if (!in_array($item->task_pid, $tastIds)) {
$tastIds[] = $item->task_pid;
}
$item->delete();
}
$tasks = ProjectTask::whereIn('id', $tastIds)->get();
foreach ($tasks as $task) { foreach ($tasks as $task) {
if (ProjectTaskUser::whereTaskId($task->id)->whereUserid($this->userid)->delete()) {
$task->syncDialogUser(); $task->syncDialogUser();
} }
} });
$this->delete(); $this->delete();
} }
} }

View File

@ -433,7 +433,7 @@ export default {
keys: { keys: {
name: this.cascaderValue, name: this.cascaderValue,
}, },
andcolumn: 'yes' getcolumn: 'yes'
}).then(() => { }).then(() => {
this.cascaderLoading--; this.cascaderLoading--;
this.initCascaderData(); this.initCascaderData();

View File

@ -84,9 +84,6 @@ export default {
if (data.complete_at) { if (data.complete_at) {
return false; return false;
} }
if (!data.end_at) {
return false;
}
return data.owner; return data.owner;
}); });
if (index > -1) { if (index > -1) {