ProjectController.php 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. <?php
  2. namespace App\Http\Controllers\API;
  3. use App\Http\Controllers\Controller;
  4. use App\Http\Requests\API\Project\CreateOrUpdateRequest;
  5. use App\Http\Requests\API\Project\LinkRequirementByPlanRequest;
  6. use App\Http\Requests\API\Project\LinkRequirementRequest;
  7. use App\Http\Requests\API\Project\PostponeRequest;
  8. use App\Http\Requests\API\Project\UnlinkRequirementRequest;
  9. use App\Http\Requests\API\Project\UpdateLinkAssetsRequest;
  10. use App\Http\Resources\API\AssetRequirementGroupTreeResource;
  11. use App\Http\Resources\API\AssetRequirementResource;
  12. use App\Http\Resources\API\AssetParentResource;
  13. use App\Http\Resources\API\ProjectAssetResource;
  14. use App\Http\Resources\API\ProjectDetailResource;
  15. use App\Http\Resources\API\ProjectGroupViewTaskResource;
  16. use App\Http\Resources\API\ProjectKanbanRequirementResource;
  17. use App\Http\Resources\API\ProjectKanbanTaskResource;
  18. use App\Http\Resources\API\RequirementGroupParentResource;
  19. use App\Http\Resources\API\RequirementGroupResource;
  20. use App\Http\Resources\API\SimplePlanResource;
  21. use App\Http\Resources\API\ProjectRequirementResource;
  22. use App\Http\Resources\API\ProjectResource;
  23. use App\Models\Enums\ObjectAction;
  24. use App\Models\Enums\ProjectStatus;
  25. use App\Models\Enums\TaskStatus;
  26. use App\Models\Plan;
  27. use App\Models\Project;
  28. use App\Models\ProjectAsset;
  29. use App\Models\ProjectPlan;
  30. use App\Models\ProjectRequirement;
  31. use App\Models\Requirement;
  32. use App\Models\Task;
  33. use App\Models\User;
  34. use App\Models\RequirementGroup;
  35. use App\Repositories\ActionRepository;
  36. use App\Services\Project\ProjectKanbanService;
  37. use App\Services\Project\ProjectGanttService;
  38. use App\Services\Project\ProjectTaskGroupViewService;
  39. use Illuminate\Http\Request;
  40. use Illuminate\Support\Facades\Auth;
  41. use Illuminate\Support\Facades\DB;
  42. use function Nette\Utils\setAttribute;
  43. class ProjectController extends Controller
  44. {
  45. /**
  46. * Display a listing of the resource.
  47. */
  48. public function index(Request $request)
  49. {
  50. $projectAsset = Project::filter($request->all())->with('assets')->simplePaginate();
  51. return ProjectResource::collection($projectAsset);
  52. }
  53. public function treeIndex(string $id){
  54. $project = Project::findOrFail($id);
  55. //获取项目所关联的需求所在需求分组的id
  56. $requirementsGroupIds=$project->requirementsGroup ->pluck('id')->unique()->toArray();
  57. // 加载项目中的所有Asset及其关联的RequirementGroup
  58. $assetsWithRequirementGroups = $project->assets()->with('requirementGroups')->get();
  59. $filteredAssets = $assetsWithRequirementGroups->map(function ($asset) use ($requirementsGroupIds) {
  60. if (empty($requirementsGroupIds)) {
  61. // 如果$requirementsGroupIds为空,则清空requirementGroups集合
  62. $asset->setRelation('requirementGroups', collect());
  63. } else {
  64. // 否则,过滤requirementGroups集合
  65. $asset->setRelation('requirementGroups', $asset->requirementGroups->filter(function ($group) use ($requirementsGroupIds) {
  66. return in_array($group->id, $requirementsGroupIds);
  67. }));
  68. }
  69. return $asset;
  70. });
  71. return AssetRequirementGroupTreeResource::collection($filteredAssets);
  72. }
  73. /**
  74. * Store a newly created resource in storage.
  75. */
  76. public function store(CreateOrUpdateRequest $request)
  77. {
  78. $project = new Project();
  79. $project->mergeFillable([
  80. 'company_id', 'created_by'
  81. ]);
  82. $project->fill([
  83. ...$request->all(),
  84. 'company_id' => Auth::user()->company_id,
  85. 'created_by' => Auth::id(),
  86. 'whitelist' => $request->whitelist ? sprintf(",%s", implode(',', $request->whitelist)) : null,
  87. ]);
  88. $project->save();
  89. ActionRepository::createByProject($project, ObjectAction::CREATED);
  90. if ($request->has("assets")) {
  91. foreach ($request->get("assets", []) as $assetId) {
  92. ProjectAsset::create([
  93. 'project_id' => $project->id,
  94. 'asset_id' => $assetId,
  95. ]);
  96. }
  97. }
  98. if ($request->has("plans")) {
  99. foreach ($request->get("plans", []) as $planId) {
  100. ProjectPlan::create([
  101. 'project_id' => $project->id,
  102. 'plan_id' => $planId,
  103. ]);
  104. }
  105. }
  106. return $this->created();
  107. }
  108. /**
  109. * Display the specified resource.
  110. */
  111. public function show(string $id)
  112. {
  113. $project = Project::findOrFail($id);
  114. return new ProjectDetailResource($project);
  115. }
  116. /**
  117. * Update the specified resource in storage.
  118. */
  119. public function update(CreateOrUpdateRequest $request, string $id)
  120. {
  121. $project = Project::findOrFail($id);
  122. $project->fill([
  123. ...$request->all(),
  124. 'whitelist' => $request->whitelist ? sprintf(",%s", implode(',', $request->whitelist)) : null,
  125. ]);
  126. $project->save();
  127. ActionRepository::createByProject($project, ObjectAction::EDITED);
  128. if ($request->has("assets")) {
  129. ProjectAsset::where('project_id', $project->id)->delete();
  130. foreach ($request->get("assets", []) as $assetId) {
  131. ProjectAsset::create([
  132. 'project_id' => $project->id,
  133. 'asset_id' => $assetId,
  134. ]);
  135. }
  136. }
  137. if ($request->has("plans")) {
  138. ProjectPlan::where('project_id', $project->id)->delete();
  139. foreach ($request->get("plans", []) as $planId) {
  140. ProjectPlan::create([
  141. 'project_id' => $project->id,
  142. 'plan_id' => $planId,
  143. ]);
  144. }
  145. }
  146. return $this->noContent();
  147. }
  148. /**
  149. * Remove the specified resource from storage.
  150. */
  151. public function destroy(string $id)
  152. {
  153. $project = Project::findOrFail($id);
  154. $project->delete();
  155. ActionRepository::createByProject($project, ObjectAction::DELETED);
  156. return $this->noContent();
  157. }
  158. public function closed(Request $request, string $id)
  159. {
  160. $project = Project::findOrFail($id);
  161. $project->status = ProjectStatus::CLOSED->value;
  162. $project->save();
  163. ActionRepository::createByProject($project, ObjectAction::CLOSED, $request->get("comment"));
  164. return $this->noContent();
  165. }
  166. public function start(Request $request, string $id)
  167. {
  168. $project = Project::findOrFail($id);
  169. $project->status = ProjectStatus::DOING->value;
  170. $project->save();
  171. ActionRepository::createByProject($project, ObjectAction::STARTED, $request->get("comment"));
  172. return $this->noContent();
  173. }
  174. public function pause(Request $request, string $id)
  175. {
  176. $project = Project::findOrFail($id);
  177. $project->status = ProjectStatus::PAUSE->value;
  178. $project->save();
  179. ActionRepository::createByProject($project, ObjectAction::PAUSED, $request->get("comment"));
  180. return $this->noContent();
  181. }
  182. /**
  183. * 延期
  184. *
  185. * @param PostponeRequest $request
  186. * @param string $id
  187. * @return \Illuminate\Http\Response
  188. */
  189. public function postpone(PostponeRequest $request, string $id)
  190. {
  191. $project = Project::findOrFail($id);
  192. $project->fill($request->only([
  193. 'begin', 'end'
  194. ]));
  195. $project->save();
  196. ActionRepository::createByProject($project, ObjectAction::DELAY, $request->get("comment"));
  197. return $this->noContent();
  198. }
  199. public function linkRequirement(LinkRequirementRequest $request, string $id)
  200. {
  201. $requirementIds = $request->get("requirement_id",[]);
  202. $project = Project::findOrFail($id);
  203. if(! $requirementIds){
  204. return $this->forbidden("Please select the correct requirement");
  205. }
  206. // $requirement = Requirement::findOrFail($request->requirement_id);
  207. // $exists = ProjectAsset::query()
  208. // ->where('project_id', $project->id)
  209. // ->where('asset_id', $requirement->asset_id)
  210. // ->count();
  211. // if (! $exists) {
  212. // return $this->forbidden("Please select the correct requirement");
  213. // }
  214. DB::transaction(function () use ($requirementIds,$project) {
  215. foreach ($requirementIds as $requirementId) {
  216. $requirement=Requirement::query()->where('id',$requirementId)->first();
  217. ProjectRequirement::query()->firstOrCreate([
  218. 'project_id' => $project->id,
  219. 'requirement_id' => $requirement->id,
  220. 'asset_id' => $requirement->asset_id,
  221. 'requirement_group_id'=> $requirement->requirement_group_id,
  222. ]);
  223. }
  224. });
  225. return $this->noContent();
  226. }
  227. public function unlinkRequirement(UnlinkRequirementRequest $request, string $id)
  228. {
  229. $requirementIds = $request->get("requirement_id",[]);
  230. $project = Project::findOrFail($id);
  231. if (! $requirementIds) {
  232. return $this->forbidden("Please select the correct requirement");
  233. }
  234. if($project->requirements->isEmpty()){
  235. return $this->forbidden("Project has no associated requirements");
  236. }
  237. DB::transaction(function () use ($requirementIds,$project) {
  238. foreach ($requirementIds as $requirement) {
  239. ProjectRequirement::query()->where([
  240. 'project_id' => $project->id,
  241. 'requirement_id' => $requirement,
  242. ])->delete();
  243. }
  244. });
  245. return $this->noContent();
  246. }
  247. public function linkRequirementByPlan(LinkRequirementByPlanRequest $request, string $id)
  248. {
  249. $project = Project::findOrFail($id);
  250. $plan = Plan::findOrFail($request->plan_id);
  251. foreach ($plan->requirements as $requirement) {
  252. ProjectRequirement::query()->firstOrCreate([
  253. 'project_id' => $project->id, 'requirement_id' => $requirement->id,
  254. ]);
  255. }
  256. return $this->noContent();
  257. }
  258. public function plan(string $id)
  259. {
  260. $project = Project::findOrFail($id);
  261. return SimplePlanResource::collection($project->plans);
  262. }
  263. public function requirement(Request $request,string $id)
  264. {
  265. $project = Project::findOrFail($id);
  266. // 初始化 requirements 的查询构建器
  267. $requirementsQuery = $project->requirements();
  268. if ($request->has('asset')) {
  269. $asset_id = $request->input('asset');
  270. $requirementsQuery->where('requirements.asset_id', $asset_id);
  271. }
  272. if ($request->has('requirementGroup')) {
  273. $requirementGroup = $request->input('requirementGroup');
  274. $requirementsQuery->where('requirements.requirement_group_id', $requirementGroup);
  275. }
  276. if ($request->has('status')) {
  277. $status = $request->input('status');
  278. $requirementsQuery->where('requirements.status', $status);
  279. }
  280. $requirements = $requirementsQuery
  281. ->with(['createdBy']) // 预加载 createdBy 关联
  282. ->simplePaginate();
  283. return ProjectRequirementResource::collection($requirements);
  284. }
  285. public function notLinkAssetRequirement(string $id)
  286. {
  287. $project = Project::findOrFail($id);
  288. $requirements = $project->assets
  289. ? Requirement::query()->whereIn('asset_id', $project->assets?->pluck('id')->toArray())->whereNotIn('id', $project->requirements?->pluck('id')->toArray())->simplePaginate()
  290. : [];
  291. return AssetRequirementResource::collection($requirements);
  292. }
  293. public function updateLinkAssets(UpdateLinkAssetsRequest $request,string $project_id)
  294. {
  295. $project = Project::find($project_id);
  296. if (is_null($project)){
  297. return $this->badRequest('project does not exist');
  298. }
  299. $assetsIds = $request->assets;
  300. DB::transaction(function () use ($project_id,$assetsIds) {
  301. ProjectAsset::where('project_id', $project_id)->delete();
  302. foreach ($assetsIds as $assetId) {
  303. ProjectAsset::create([
  304. 'project_id' => $project_id,
  305. 'asset_id' => $assetId,
  306. ]);
  307. }
  308. });
  309. return $this->noContent();
  310. }
  311. public function dynamic(Request $request, string $id)
  312. {
  313. $project = Project::findOrFail($id);
  314. return $this->success([
  315. 'data' => ActionRepository::dynamic($project, $request->all())
  316. ]);
  317. }
  318. public function kanban(Request $request, ProjectKanbanService $service, string $id)
  319. {
  320. $project = Project::findOrFail($id);
  321. return $this->success([
  322. 'data' => $service->kanban($project, $request->get("group", "requirement_asc"))
  323. ]);
  324. }
  325. public function groupView(Request $request, string $id)
  326. {
  327. $project = Project::findOrFail($id);
  328. $group = in_array(
  329. $request->get("group"),
  330. ['requirement_id','status','assign','finished_by','closed_by','task_type']
  331. ) ? $request->get("group") : "requirement_id";
  332. return $this->success([
  333. 'data' => (new ProjectTaskGroupViewService())->groupView($project, $group, $request->all())
  334. ]);
  335. }
  336. public function gantt(Request $request, string $id)
  337. {
  338. $project = Project::findOrFail($id);
  339. $group = in_array(
  340. $request->get("group"),
  341. ['requirement_id','assign','task_type']
  342. ) ? $request->get("group") : "task_type";
  343. return $this->success([
  344. 'data' => (new ProjectGanttService())->gantt($project, $group)
  345. ]);
  346. }
  347. public function printKanban(string $id)
  348. {
  349. $project = Project::findOrFail($id);
  350. return $this->success([
  351. 'data' => [
  352. 'requirements' => ProjectKanbanRequirementResource::collection($project->requirements),
  353. 'tasks' => ProjectKanbanTaskResource::collection($project->tasks),
  354. 'task_status' => TaskStatus::cases(),
  355. ]
  356. ]);
  357. }
  358. public function latestDynamic(string $id)
  359. {
  360. $project = Project::findOrFail($id);
  361. return $this->success([
  362. 'data' => ActionRepository::latestDynamic($project)
  363. ]);
  364. }
  365. public function requirementsLinkGroup(string $id){
  366. $requirementsLinksGroup = Project::findOrFail($id)->requirementsGroup->unique('id');
  367. return RequirementGroupParentResource::collection($requirementsLinksGroup);
  368. }
  369. }