MenuService.php 9.6 KB

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