MenuService.php 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. <?php
  2. /**
  3. * @Name
  4. * @Description
  5. * @Author 刘学玺
  6. * @Date 2024/8/15 14:30
  7. */
  8. namespace App\Http\Services\Backend\Server\System;
  9. use App\Enums\Common\ErrorMessage;
  10. use App\Enums\Common\Status;
  11. use App\Enums\System\MenuType;
  12. use App\Exceptions\ApiException;
  13. use App\Http\Services\Service;
  14. use App\Models\System\Menu;
  15. use App\Models\System\Role;
  16. use Exception;
  17. use Illuminate\Support\Facades\DB;
  18. use Spatie\Permission\Guard;
  19. use Spatie\Permission\Models\Permission;
  20. use Symfony\Component\HttpFoundation\Response;
  21. class MenuService extends Service
  22. {
  23. protected array $selectColumn = ['id', 'name', 'type','permission', 'sort', 'parent_id as parentId', 'path', 'icon', 'component', 'component_name as componentName', 'status', 'visible', 'keep_alive as keepAlive', 'always_show as alwaysShow'];
  24. // protected array $selectAppendColumn = ['created_at as createTime'];
  25. public function getMenuList($params = []): array
  26. {
  27. $menu = Menu::query();
  28. isset($params['name']) && filled($params['name']) && $menu->whereLike('name', "%{$params['name']}%");
  29. isset($params['status']) && filled($params['status']) && $menu->where('status', $params['status']);
  30. return $menu->orderBy('sort')->select($this->selectColumn)->get()->toArray();
  31. }
  32. public function getMenu($id): array
  33. {
  34. $menu = Menu::query();
  35. return $menu->select($this->selectColumn)->find($id)->toArray();
  36. }
  37. /**
  38. * @throws ApiException
  39. */
  40. public function createMenu(array $data)
  41. {
  42. // 校验父菜单存在
  43. self::validateParentMenu($data['parentId'], null);
  44. // 校验菜单(自己)
  45. self::validateMenu($data['parentId'], $data['name'], null);
  46. // 校验权限标识
  47. $permissions = isset($data['permission']) ? self::convertPermissions($data['permission']) : [];
  48. $this->validatePermissions($permissions);
  49. // 插入数据库
  50. $menu = self::toModel($data, Menu::class);
  51. self::initMenuProperty($menu);
  52. DB::beginTransaction();
  53. try {
  54. $menu = $menu->create($menu->getAttributes());
  55. self::createPermissions($permissions);
  56. self::syncMenuHasPermissions($menu, $permissions);
  57. DB::commit();
  58. return $menu->id;
  59. } catch (Exception) {
  60. DB::rollBack();
  61. self::error('添加失败');
  62. }
  63. }
  64. /**
  65. * @throws ApiException
  66. */
  67. public function updateMenu($data): void
  68. {
  69. // 校验更新的菜单是否存在
  70. !self::isExistMenu($data['id']) && self::error('MENU_NOT_EXISTS', Response::HTTP_UNPROCESSABLE_ENTITY);
  71. // 校验父菜单存在
  72. self::validateParentMenu($data['parentId'], $data['id']);
  73. // 校验菜单(自己)
  74. self::validateMenu($data['parentId'], $data['name'], $data['id']);
  75. // 校验权限标识
  76. $permissions = isset($data['permission']) ? self::convertPermissions($data['permission']) : [];
  77. $this->validatePermissions($permissions, $data['id']);
  78. // 更新到数据库
  79. $menu = self::toModel($data, Menu::class);
  80. self::initMenuProperty($menu);
  81. DB::beginTransaction();
  82. try {
  83. $menu->update($menu->getAttributes());
  84. self::detachMenuHasPermissions($menu);
  85. self::createPermissions($permissions);
  86. self::syncMenuHasPermissions($menu, $permissions);
  87. DB::commit();
  88. } catch (Exception) {
  89. DB::rollBack();
  90. self::error('修改失败');
  91. }
  92. }
  93. /**
  94. * @throws ApiException
  95. */
  96. public function deleteMenu($id)
  97. {
  98. // 校验是否还有子菜单
  99. self::validateChildrenMenu($id);
  100. // 校验删除的菜单是否存在
  101. !self::isExistMenu($id) && self::error('MENU_NOT_EXISTS', Response::HTTP_UNPROCESSABLE_ENTITY);
  102. $menu = self::toModel(['id' => $id], Menu::class);
  103. try {
  104. self::detachMenuHasPermissions($menu);
  105. // 标记删除
  106. $res = $menu->delete();
  107. DB::commit();
  108. return $res;
  109. } catch (Exception) {
  110. DB::rollBack();
  111. self::error('删除失败');
  112. }
  113. }
  114. public function getSimpleMenuList(): array
  115. {
  116. $menus = $this->getMenuList(['status' => 0]);
  117. // list.sort(Comparator . comparing(MenuDO::getSort));
  118. return self::filterDisableMenus($menus);
  119. }
  120. protected static function isExistMenu(int|array $condition): bool
  121. {
  122. $menu = Menu::query();
  123. is_array($condition) && $menu->where($condition);
  124. is_numeric($condition) && $menu->where('id', $condition);
  125. return $menu->exists();
  126. }
  127. /**
  128. * @throws ApiException
  129. */
  130. protected static function validateChildrenMenu($parentId): void
  131. {
  132. Menu::query()->where('parent_id', $parentId)->count() && self::error('MENU_EXISTS_CHILDREN', Response::HTTP_UNPROCESSABLE_ENTITY);
  133. }
  134. /**
  135. * @throws ApiException
  136. */
  137. protected static function validateParentMenu($parentId, $childId): void
  138. {
  139. if (!$parentId) return;
  140. // 不能设置自己为父菜单
  141. $parentId === $childId && self::error('MENU_PARENT_ERROR', Response::HTTP_UNPROCESSABLE_ENTITY);
  142. $menu = Menu::query()->select('type')->find($parentId);
  143. // 父菜单不存在
  144. !$menu && self::error('MENU_PARENT_NOT_EXISTS', Response::HTTP_UNPROCESSABLE_ENTITY);
  145. // 父菜单必须是目录或者菜单类型
  146. $menu->type != MenuType::DIR && $menu->type != MenuType::MENU && self::error('MENU_PARENT_NOT_DIR_OR_MENU', Response::HTTP_UNPROCESSABLE_ENTITY);
  147. }
  148. /**
  149. * @throws ApiException
  150. */
  151. protected static function validateMenu($parentId, $name, $id): void
  152. {
  153. $where = ['parent_id' => $parentId, 'name' => $name];
  154. $menu = Menu::query()->where($where)->first();
  155. if (is_null($menu)) return;
  156. // 如果 id 为空,说明不用比较是否为相同 id 的菜单
  157. !$id && self::error('MENU_NAME_DUPLICATE', Response::HTTP_UNPROCESSABLE_ENTITY);
  158. $menu->id !== $id && self::error('MENU_NAME_DUPLICATE', Response::HTTP_UNPROCESSABLE_ENTITY);
  159. }
  160. protected static function convertPermissions(string $permission): array
  161. {
  162. if (empty($permission)) return [];
  163. return collect(explode('|', $permission))->map(function ($item) {
  164. return trim($item);
  165. })->all();
  166. }
  167. /**
  168. * @throws ApiException
  169. */
  170. protected function validatePermissions(array $permissions, int $menu_id = null): void
  171. {
  172. if (empty($permissions)) return;
  173. $isExistsPermissions = Permission::query()->whereIn('name', $permissions)->where('guard_name', Guard::getDefaultName(static::class))->get();
  174. count($isExistsPermissions) && empty($menu_id) && self::error(ErrorMessage::PERMISSION_NAME_DUPLICATE);
  175. if (empty($menu_id)) return;
  176. $menu = Menu::query()->find($menu_id);
  177. $isHas = $menu->hasAllPermissions($isExistsPermissions);
  178. !$isHas && self::error(ErrorMessage::PERMISSION_NAME_DUPLICATE);
  179. }
  180. protected static function syncMenuHasPermissions(Menu $menu, array $permissions): void
  181. {
  182. if (empty($permissions)) return;
  183. $permissions_ids = Permission::query()->whereIn('name', $permissions)->pluck('id');
  184. empty($permissions_ids) && self::error(ErrorMessage::PERMISSION_NAME_NOT_EXISTS);;
  185. $menu->permissions()->sync($permissions_ids);
  186. }
  187. protected static function detachMenuHasPermissions(Menu $menu)
  188. {
  189. $permission_ids = $menu->permissions()->pluck('id');
  190. Permission::query()->whereIn('id', $permission_ids)->delete();
  191. }
  192. protected static function createPermissions(array $permissions): void
  193. {
  194. if (empty($permissions)) return;
  195. foreach ($permissions as $permissionName) {
  196. Permission::create(['name' => $permissionName]);
  197. }
  198. // $menu->givePermissionTo($permissions);
  199. // $menu->flushPermissionCache();
  200. }
  201. /**
  202. * 初始化菜单的通用属性。
  203. * <p>
  204. * 例如说,只有目录或者菜单类型的菜单,才设置 icon
  205. *
  206. */
  207. private static function initMenuProperty(Menu &$menu): void
  208. {
  209. // 菜单为按钮类型时,无需 component、icon、path 属性,进行置空
  210. if ($menu['type'] === MenuType::BUTTON) {
  211. $menu['component'] = null;
  212. $menu['component_name'] = null;
  213. $menu['icon'] = null;
  214. $menu['path'] = null;
  215. }
  216. (empty($menu['permission'])) && ($menu['permission'] = null);
  217. }
  218. private static function filterDisableMenus($menus): array
  219. {
  220. if (empty($menus)) return [];
  221. // 遍历 menu 菜单,查找不是禁用的菜单,添加到 enabledMenus 结果
  222. $enabledMenus = [];
  223. // $disabledMenuCache = []; // 存下递归搜索过被禁用的菜单,防止重复的搜索
  224. foreach ($menus as $menu) {
  225. if ($menu['status'] === Status::DISABLE) continue;
  226. $enabledMenus[] = collect($menu)->only(['id', 'status', 'name', 'parentId', 'type']); //['id' => $menu['id'], 'status' => $menu['status'], 'name' => $menu['name'], 'parentId' => $menu['parentId'], 'type' => $menu['type']];
  227. }
  228. return $enabledMenus;
  229. }
  230. }