ContainerController.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. <?php
  2. namespace App\Http\Controllers\API;
  3. use App\Http\Controllers\Controller;
  4. use App\Http\Requests\API\Container\ApprovalMoveFileRequest;
  5. use App\Http\Requests\API\Container\CreateOrUpdateRequest;
  6. use App\Http\Resources\API\BimFileResource;
  7. use App\Http\Resources\API\ContainerDetailResource;
  8. use App\Http\Resources\API\ContainerReportResource;
  9. use App\Http\Resources\API\ContainerResource;
  10. use App\Http\Resources\API\FileAttachmentResource;
  11. use App\Http\Resources\API\FileByObjectResource;
  12. use App\Http\Resources\API\LibraryReportResource;
  13. use App\Models\Approval;
  14. use App\Models\BimFile;
  15. use App\Models\Container;
  16. use App\Models\ContainerContent;
  17. use App\Models\Enums\ActionObjectType;
  18. use App\Models\Enums\FileObjectType;
  19. use App\Models\Enums\ObjectAction;
  20. use App\Models\Enums\ObjectApprovalStatus;
  21. use App\Models\File;
  22. use App\Models\Folder;
  23. use App\Models\Library;
  24. use App\Repositories\ActionRepository;
  25. use App\Repositories\CustomFieldRepository;
  26. use App\Services\File\FileAssociationService;
  27. use App\Services\File\ImageUrlService;
  28. use App\Services\History\ModelChangeDetector;
  29. use Carbon\Carbon;
  30. use Illuminate\Http\Request;
  31. use Illuminate\Support\Facades\Auth;
  32. class ContainerController extends Controller
  33. {
  34. /**
  35. * Display a listing of the resource.
  36. */
  37. public function index(Request $request)
  38. {
  39. $containers = Container::query()->allowed()->filter($request->all())->paginate();
  40. $foldsCount = Folder::query()
  41. ->where('object_type','container')
  42. ->where('parent_id',0)
  43. ->whereIn("object_id", $containers->pluck("id")->toArray())
  44. ->selectRaw("count(*) as cut, object_id")
  45. ->groupBy("object_id")
  46. ->pluck("cut", "object_id");
  47. $fileCount = File::query()
  48. ->where('object_type','container')
  49. ->whereIn('object_id',$containers->pluck("id")->toArray())
  50. ->where("folder_id", 0)
  51. ->where("is_latest_version", 1)
  52. ->selectRaw("count(*) as cut, object_id")
  53. ->groupBy("object_id")
  54. ->pluck("cut", "object_id");
  55. $index=1;
  56. $containers = $containers->map(function (Container $container) use ($foldsCount,$fileCount,&$index) {
  57. $container->itemCount =$foldsCount->get($container->id, 0)+$fileCount->get($container->id, 0);
  58. $container->type = 'container';
  59. $container->uniId = $container->type . '_' . $container->id;
  60. $container->display_id=(string)$index++;
  61. return $container;
  62. });
  63. return $this->success([
  64. 'data' => $containers
  65. ]);
  66. //return ContainerResource::collection($containers);
  67. }
  68. /**
  69. * Store a newly created resource in storage.
  70. */
  71. public function store(
  72. CreateOrUpdateRequest $request,
  73. ImageUrlService $imageUrlService,
  74. FileAssociationService $service,
  75. CustomFieldRepository $customFieldRepo
  76. )
  77. {
  78. $formData = [
  79. ...$request->all(),
  80. 'company_id' => Auth::user()->company_id,
  81. 'created_by' => Auth::id(),
  82. 'whitelist' => $request->whitelist ? sprintf(",%s,", implode(',', $request->whitelist)) : null,
  83. ];
  84. $service->check(
  85. $request->get("file_ids", []),
  86. FileObjectType::CONTAINER,
  87. $request->get("file_uuid"),
  88. );
  89. $service->fileNameNamingRuleCheckByContainer($request->library_id, (int)$request->naming_rule_id);
  90. if ($request->has("naming_rule_id") && $request->get("naming_rule_id") > 0) {
  91. $keys = $customFieldRepo->keysByGroup($request->get("naming_rule_id"));
  92. $formData['naming_rules'] = $request->only($keys);
  93. }
  94. $container = new Container();
  95. $container->mergeFillable(['company_id', 'created_by']);
  96. $container->fill($formData);
  97. $container->save();
  98. $service->association($container->id);
  99. $files = File::query()->where('object_id', $container->id)
  100. ->where('object_type', ActionObjectType::CONTAINER)
  101. ->where('source', 1)
  102. ->pluck("id")
  103. ->sort();
  104. $contentFormData = [
  105. 'description' => $imageUrlService->interceptImageUrl($request->description),
  106. 'container_id' => $container->id,
  107. 'created_by' => Auth::id(),
  108. 'name' => $request->name,
  109. 'files' => $files->implode(",") ?: null
  110. ];
  111. ContainerContent::query()->create($contentFormData);
  112. ActionRepository::createByContainer($container, ObjectAction::CREATED);
  113. return $this->created();
  114. }
  115. /**
  116. * Display the specified resource.
  117. */
  118. public function show(string $id)
  119. {
  120. $container = Container::query()->allowed()
  121. ->when(\request("version") > 0, fn($query) => $query->where("version", ">=", \request("version")))
  122. ->findOrFail($id);
  123. return new ContainerDetailResource($container);
  124. }
  125. /**
  126. * Update the specified resource in storage.
  127. */
  128. public function update(
  129. CreateOrUpdateRequest $request,
  130. ImageUrlService $imageUrlService,
  131. CustomFieldRepository $customFieldRepo,
  132. string $id
  133. )
  134. {
  135. $container = Container::query()->allowed()->findOrFail($id);
  136. $formData = [
  137. ...$request->all(),
  138. 'whitelist' => $request->whitelist ? sprintf(",%s,", implode(',', $request->whitelist)) : null,
  139. 'description' => $imageUrlService->interceptImageUrl($request->description) ,
  140. ];
  141. if ($request->has("naming_rule_id") && $request->get("naming_rule_id") > 0) {
  142. $keys = $customFieldRepo->keysByGroup($request->get("naming_rule_id"));
  143. $formData['naming_rules'] = $request->only($keys);
  144. }
  145. $container->fill($formData);
  146. $changes = ModelChangeDetector::detector(ActionObjectType::CONTAINER, $container);
  147. $files = File::query()->where('object_id', $container->id)
  148. ->where('object_type', ActionObjectType::CONTAINER)
  149. ->where('source', 1)
  150. ->pluck("id")
  151. ->sort();
  152. $contentFormData = [
  153. 'description' => $imageUrlService->interceptImageUrl($request->description),
  154. 'name' => $request->name,
  155. 'files' => $files->implode(",") ?: null
  156. ];
  157. $containerContent = $container->content;
  158. $containerContent->fill($contentFormData);
  159. $contentChange = ModelChangeDetector::detector(ActionObjectType::CONTAINER_CONTENT, $containerContent);
  160. if ($contentChange) {
  161. $container->version++;
  162. $changes = array_merge($changes, $contentChange);
  163. ContainerContent::query()->create([
  164. ...$contentFormData,
  165. 'container_id' => $container->id,
  166. 'created_by' => Auth::id(),
  167. 'version' => $container->version,
  168. ]);
  169. }
  170. $container->save();
  171. ActionRepository::createByContainer($container, ObjectAction::EDITED, objectChanges: $changes);
  172. return $this->noContent();
  173. }
  174. /**
  175. * Remove the specified resource from storage.
  176. */
  177. public function destroy(string $id)
  178. {
  179. $container = Container::query()->allowed()->findOrFail($id);
  180. $container->delete();
  181. Approval::query()->where('object_type','container')->where('object_id',$id)->delete();
  182. ActionRepository::createByContainer($container, ObjectAction::DELETED);
  183. return $this->noContent();
  184. }
  185. public function linkage(string $libraryId)
  186. {
  187. $items = Container::query()->allowed()->where("library_id", $libraryId)->get(['id', 'name'])->each(function ($items) {
  188. // 设置固定的type值
  189. $items->type = 'container';
  190. $items->uniId=$items->type.'_'.$items->id;
  191. });
  192. return $this->success([
  193. 'data' => $items
  194. ]);
  195. }
  196. public function attachments(Request $request){
  197. $folderObjectType = FileObjectType::from($request->object_type);
  198. $object=$folderObjectType->modelBuilderAllowed()->findOrFail($request->object_id);
  199. $pageSize = $request->get('page_size') ?? 10;
  200. if($folderObjectType->value==='asset'){
  201. $object->load('requirements');
  202. $requirementsId=$object->requirements->pluck('id');
  203. if(!$requirementsId->isEmpty()){
  204. $filesWithRequirements = File::query()->where('object_type','requirement')
  205. ->whereIn('object_id',$requirementsId)
  206. ->where('is_latest_version',1)
  207. ->where('source',1)
  208. ->paginate($pageSize);
  209. $filesWithRequirements->transform(function (File $file) use ($object) {
  210. $requirement = $object->requirements->where('id',$file->object_id)->first();
  211. $file->source_name = $requirement ? $requirement->title : null;
  212. $file->type = 'requirement';
  213. return $file;
  214. });
  215. make_display_id($filesWithRequirements,$pageSize);
  216. return FileAttachmentResource::collection($filesWithRequirements);
  217. }
  218. }else if($folderObjectType->value==='project'){
  219. $object->load('tasks');
  220. $tasksId=$object->tasks->pluck('id');
  221. if(!$tasksId->isEmpty()){
  222. $filesWithTasks=File::query()->where('object_type','task')
  223. ->whereIn('object_id',$tasksId)
  224. ->where('is_latest_version',1)
  225. ->where('source',1)
  226. ->paginate($pageSize);
  227. $filesWithTasks->transform(function (File $file) use ($object) {
  228. $task = $object->tasks->where('id',$file->object_id)->first();
  229. $file->source_name = $task ? $task->name : null;
  230. $file->project_id=$task ?$task->project_id:null;
  231. $file->type = 'tasks';
  232. return $file;
  233. });
  234. make_display_id($filesWithTasks,$pageSize);
  235. return FileAttachmentResource::collection($filesWithTasks);
  236. }
  237. }
  238. return $this->noContent();
  239. }
  240. public function containerReport(Request $request){
  241. $companyId=Auth::user()->company_id;
  242. $librarys=Library::query()->where('company_id',$companyId)->withCount('container') ->get(['id', 'name', 'created_at', 'updated_at', 'container_count as container_count']);
  243. $libraryCount=$librarys->count();
  244. $containerCount = $librarys->sum('container_count');
  245. $MyLibrarys=Library::query()->allowed()->filter($request->all())->orderBy('updated_at', 'desc')->get(['id', 'name', 'type','created_at', 'updated_at']);
  246. $MyContainers = Container::query()->allowed()->when($request->type && $request->asset,function ($query) use ($request){
  247. $query->whereHas('library', function ($query) use ($request) {
  248. $query->where('type', $request->type)
  249. ->where('asset_id',$request->asset);
  250. });
  251. })->orderBy('updated_at', 'desc')->get( ['id', 'name','library_id', 'created_at', 'updated_at']);
  252. return $this->success([
  253. 'data' =>[
  254. 'libraryCount'=>$libraryCount,
  255. 'containerCount'=>$containerCount,
  256. 'MyLibrarysCount'=>$MyLibrarys->count(),
  257. 'MyContainersCount'=>$MyContainers->count(),
  258. 'MyContainers'=>ContainerReportResource::collection($MyContainers),
  259. 'MyLibrarys'=>LibraryReportResource::collection($MyLibrarys)
  260. ]
  261. ]);
  262. }
  263. /**
  264. * @param ApprovalMoveFileRequest $request
  265. * @param string $id
  266. * @return \Illuminate\Http\Response
  267. */
  268. public function approvalMoveFile(ApprovalMoveFileRequest $request, string $id)
  269. {
  270. $container = Container::query()->findOrFail($id);
  271. throw_validation_if(
  272. $container->approval_status != ObjectApprovalStatus::APPROVED->value,
  273. 'Please wait for the container to be approved.'
  274. );
  275. $targetFolder = Folder::query()->findOrFail($request->get("target_folder_id"));
  276. $topFolderIds = $request->get("top_folder_ids", []);
  277. if ($topFolderIds) {
  278. File::query()
  279. ->where("object_type", FileObjectType::CONTAINER)
  280. ->where("object_id", $id)
  281. ->whereIn("folder_id", $topFolderIds)
  282. ->update([
  283. 'folder_id' => $targetFolder->id,
  284. ]);
  285. foreach ($topFolderIds as $topFolderId) {
  286. $topFolder = Folder::query()->find($topFolderId);
  287. $childrenFolders = Folder::query()
  288. ->where("object_type", FileObjectType::CONTAINER)
  289. ->where("object_id", $id)
  290. ->where("id", "!=", $topFolderId)
  291. ->where("path", "like", "," . $topFolderId . ",%")
  292. ->get();
  293. foreach ($childrenFolders as $child) {
  294. $path = "," . $targetFolder->id . ",";
  295. $child->fill([
  296. 'path' => str_replace($topFolder->path, $path, $child->path),
  297. ]);
  298. $child->save();
  299. }
  300. }
  301. }
  302. $topFileIds = $request->get("top_files_ids", []);
  303. if ($topFileIds) {
  304. File::query()
  305. ->where("object_type", FileObjectType::CONTAINER)
  306. ->where("object_id", $id)
  307. ->whereIn("id", $topFileIds)
  308. ->update([
  309. 'folder_id' => $targetFolder->id,
  310. ]);
  311. }
  312. return $this->noContent();
  313. }
  314. /**
  315. * 容器下的所有模型列表
  316. */
  317. public function models(Request $request)
  318. {
  319. $containerId = $request->container_id;
  320. $pageSize = $request->get('page_size') ?? 10;
  321. $container = Container::query()->allowed()->findOrFail($containerId);
  322. $bimFiles = BimFile::query()
  323. ->whereHas('file', function ($query) use ($container) {
  324. return $query->containerLibraryId($container->library_id);
  325. })
  326. ->with('file')
  327. ->latest('id')
  328. ->paginate($pageSize);
  329. return BimFileResource::collection($bimFiles);
  330. }
  331. }