MenuRepository.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2016~2024 https://www.crmeb.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  8. // +----------------------------------------------------------------------
  9. // | Author: CRMEB Team <admin@crmeb.com>
  10. // +----------------------------------------------------------------------
  11. namespace app\common\repositories\system\auth;
  12. //附件
  13. use app\common\dao\BaseDao;
  14. use app\common\dao\system\menu\MenuDao;
  15. use app\common\repositories\BaseRepository;
  16. use app\common\repositories\system\config\ConfigRepository;
  17. use crmeb\traits\SpecialConfig;
  18. use FormBuilder\Exception\FormBuilderException;
  19. use FormBuilder\Factory\Elm;
  20. use FormBuilder\Form;
  21. use think\db\exception\DataNotFoundException;
  22. use think\db\exception\DbException;
  23. use think\db\exception\ModelNotFoundException;
  24. use think\Exception;
  25. use think\facade\Db;
  26. use think\facade\Route;
  27. use think\Model;
  28. /**
  29. * 菜单
  30. */
  31. class MenuRepository extends BaseRepository
  32. {
  33. use SpecialConfig;
  34. /**
  35. * MenuRepository constructor.
  36. * @param MenuDao $dao
  37. */
  38. protected $styles = array(
  39. 'success' => "\033[0;32m%s\033[0m",
  40. 'error' => "\033[31;31m%s\033[0m",
  41. 'info' => "\033[33;33m%s\033[0m"
  42. );
  43. public $prompt = 'all';
  44. public function __construct(MenuDao $dao)
  45. {
  46. /**
  47. * @var MenuDao
  48. */
  49. $this->dao = $dao;
  50. }
  51. /**
  52. * 获取菜单列表
  53. * @param array $where
  54. * @param int $merId
  55. * @return array
  56. * @throws DataNotFoundException
  57. * @throws DbException
  58. * @throws ModelNotFoundException
  59. * @author wuhaotian
  60. * @email 442384644@qq.com
  61. * @date 2024/7/23
  62. */
  63. public function getList(array $where, $merId = 0)
  64. {
  65. $query = $this->dao->search($where, $merId);
  66. $count = $query->count();
  67. $list = $query->hidden(['update_time', 'path'])->select()->toArray();
  68. return compact('count', 'list');
  69. }
  70. /**
  71. * 新增菜单
  72. * @param array $data
  73. * @return BaseDao|Model
  74. * @author wuhaotian
  75. * @email 442384644@qq.com
  76. * @date 2024/7/23
  77. */
  78. public function create(array $data)
  79. {
  80. $data['path'] = '/';
  81. if ($data['pid']) {
  82. $data['path'] = $this->getPath($data['pid']) . $data['pid'] . '/';
  83. }
  84. return $this->dao->create($data);
  85. }
  86. /**
  87. * 更新菜单
  88. * @param int $id
  89. * @param array $data
  90. * @return int
  91. * @throws DbException
  92. * @author xaboy
  93. * @day 2020-04-09
  94. */
  95. public function update(int $id, array $data)
  96. {
  97. $menu = $this->dao->get($id);
  98. if ($menu->pid != $data['pid']) {
  99. Db::transaction(function () use ($menu, $data) {
  100. $data['path'] = '/';
  101. if ($data['pid']) {
  102. $data['path'] = $this->getPath($data['pid']) . $data['pid'] . '/';
  103. }
  104. $this->dao->updatePath($menu->path . $menu->menu_id . '/', $data['path'] . $menu->menu_id . '/');
  105. $menu->save($data);
  106. });
  107. } else {
  108. unset($data['path']);
  109. $this->dao->update($id, $data);
  110. }
  111. }
  112. /**
  113. * 获取菜单树形列表
  114. * @param bool $is_mer
  115. * @return array
  116. * @author xaboy
  117. * @day 2020-04-18
  118. */
  119. public function getTree($merType = 0)
  120. {
  121. if (!$merType) {
  122. $options = $this->dao->getAllOptions();
  123. } else {
  124. $options = $this->dao->merchantTypeByOptions($merType);
  125. }
  126. return formatTree($options, 'menu_name');
  127. }
  128. /**
  129. * 添加菜单表单
  130. * @param int $isMer
  131. * @param int|null $id
  132. * @param array $formData
  133. * @return Form
  134. * @throws FormBuilderException
  135. * @author xaboy
  136. * @day 2020-04-16
  137. */
  138. public function menuForm(int $isMer = 0, ?int $id = null, array $formData = []): Form
  139. {
  140. $action = $isMer == 0 ? (is_null($id) ? Route::buildUrl('systemMenuCreate')->build() : Route::buildUrl('systemMenuUpdate', ['id' => $id])->build())
  141. : (is_null($id) ? Route::buildUrl('systemMerchantMenuCreate')->build() : Route::buildUrl('systemMerchantMenuUpdate', ['id' => $id])->build());
  142. $form = Elm::createForm($action);
  143. $form->setRule([
  144. Elm::cascader('pid', '父级分类:')->options(function () use ($id, $isMer) {
  145. $menus = $this->dao->getAllOptions($isMer, true);
  146. if ($id && isset($menus[$id])) unset($menus[$id]);
  147. $menus = formatCascaderData($menus, 'menu_name');
  148. array_unshift($menus, ['label' => '顶级分类:', 'value' => 0]);
  149. return $menus;
  150. })->placeholder('请选择分级分类')->props(['props' => ['checkStrictly' => true, 'emitPath' => false]]),
  151. Elm::select('is_menu', '权限类型:', 1)->options([
  152. ['value' => 1, 'label' => '菜单'],
  153. ['value' => 0, 'label' => '权限'],
  154. ])->control([
  155. [
  156. 'value' => 0,
  157. 'rule' => [
  158. Elm::input('menu_name', '路由名称:')->placeholder('请输入路由名称')->required(),
  159. Elm::textarea('params', '参数:')->placeholder("路由参数:\r\nkey1:value1\r\nkey2:value2"),
  160. ]
  161. ], [
  162. 'value' => 1,
  163. 'rule' => [
  164. Elm::switches('is_show', '是否显示:', 1)->inactiveValue(0)->activeValue(1)->inactiveText('关')->activeText('开'),
  165. Elm::frameInput('icon', '菜单图标:', '/' . config('admin.admin_prefix') . '/setting/icons?field=icon')->icon('el-icon-circle-plus-outline')->height('338px')->width('700px')->modal(['modal' => false]),
  166. Elm::input('menu_name', '菜单名称:')->placeholder('请输入菜单名称')->required(),
  167. ]
  168. ]
  169. ]),
  170. Elm::input('route', '路由:')->placeholder('请输入路由'),
  171. Elm::number('sort', '排序:', 0)->precision(0)->max(99999)
  172. ]);
  173. return $form->setTitle(is_null($id) ? '添加菜单' : '编辑菜单')->formData($formData);
  174. }
  175. /**
  176. * 更新菜单表单
  177. * @param int $id
  178. * @param int $merId
  179. * @return Form
  180. * @throws DataNotFoundException
  181. * @throws DbException
  182. * @throws FormBuilderException
  183. * @throws ModelNotFoundException
  184. * @author xaboy
  185. * @day 2020-04-16
  186. */
  187. public function updateMenuForm(int $id, $merId = 0)
  188. {
  189. return $this->menuForm($merId, $id, $this->dao->get($id)->toArray());
  190. }
  191. /**
  192. * 格式化数据
  193. * @param string $params
  194. * @return array
  195. * @author xaboy
  196. * @day 2020-04-22
  197. */
  198. public function tidyParams(?string $params)
  199. {
  200. return $params ? array_reduce(explode('|', $params), function ($initial, $val) {
  201. $data = explode(':', $val, 2);
  202. if (count($data) != 2) return $initial;
  203. $initial[$data[0]] = $data[1];
  204. return $initial;
  205. }, []) : [];
  206. }
  207. /**
  208. * 检测数据
  209. * @param array $params
  210. * @param array $routeParams
  211. * @return bool
  212. * @author xaboy
  213. * @day 2020-04-23
  214. */
  215. public function checkParams(array $params, array $routeParams)
  216. {
  217. foreach ($routeParams as $k => $param) {
  218. if (isset($params[$k]) && $params[$k] != $param)
  219. return false;
  220. }
  221. return true;
  222. }
  223. /**
  224. * 格式化路径。
  225. * 该方法用于处理一类特定的路径格式化任务,可以通过传入参数来指定处理的类别。
  226. * 默认情况下,它处理的是一般路径,但可以通过设置$is_mer参数来处理特定的商户路径。
  227. *
  228. * @param int $is_mer 标识是否为商户路径,0表示一般路径,非0表示商户路径。
  229. */
  230. public function formatPath($is_mer = 0)
  231. {
  232. // 获取所有相关路径选项,根据$is_mer参数决定是一般路径还是商户路径。
  233. $options = $this->getAll($is_mer);
  234. // 对获取的路径选项进行预处理,以'menu_id'为基准进行排序或格式化。
  235. $options = formatCategory($options, 'menu_id');
  236. // 开启数据库事务,以确保路径格式化的原子性。
  237. Db::transaction(function () use ($options) {
  238. // 遍历处理后的路径选项,调用内部方法逐个格式化路径。
  239. foreach ($options as $option) {
  240. $this->_formatPath($option);
  241. }
  242. });
  243. }
  244. /**
  245. * 格式化菜单项的路径。
  246. * 该方法递归地更新菜单项的路径,确保每个菜单项都有一个正确的路径标识。
  247. * 路径是基于其父项的路径和自身的ID构建的,这有助于快速定位和组织菜单结构。
  248. *
  249. * @param array $parent 父菜单项的信息,包含菜单ID和子菜单项。
  250. * @param string $path 父菜单项的路径,默认为根路径'/'。
  251. */
  252. protected function _formatPath($parent, $path = '/')
  253. {
  254. // 更新父菜单项的路径
  255. $this->dao->update($parent['menu_id'], ['path' => $path]);
  256. // 遍历父菜单项的子菜单项
  257. foreach ($parent['children'] ?? [] as $item) {
  258. // 构建当前子菜单项的路径
  259. $itemPath = $path . $item['pid'] . '/';
  260. // 递归调用,对子菜单项进行路径格式化
  261. $this->_formatPath($item, $itemPath);
  262. }
  263. }
  264. /**
  265. * 根据给定的数据生成命令行菜单。
  266. * 该方法主要用于处理系统(sys)和商户(mer)类型的菜单数据,通过给定的菜单标识符,从数据库中获取菜单的父ID,
  267. * 如果不存在,则尝试创建一个新的菜单项。
  268. *
  269. * @param string $type 菜单的类型,决定是系统菜单还是商户菜单。
  270. * @param array $data 菜单数据,包含需要处理的菜单标识符及其对应显示名称。
  271. * @param string $prompt命令行提示信息,用于优化用户交互体验。
  272. * @return int 返回处理的菜单项数量。
  273. */
  274. public function commandMenu($type, $data, $prompt)
  275. {
  276. $res = [];
  277. // 根据$type设置是否为商户菜单的标志
  278. $isMer = ($type == 'sys') ? 0 : 1;
  279. foreach ($data as $key => $value) {
  280. try {
  281. // 尝试根据$key获取菜单的父ID,如果失败则处理特别情况
  282. $result = $this->dao->getMenuPid($key, $isMer, 0);
  283. if (!$result) {
  284. // 处理新增菜单项的特殊情况,包括附加权限和自定义路由
  285. $route = $key;
  286. $isAppend = 0;
  287. if (substr($key, 0, 7) === 'append_') {
  288. $isAppend = 1;
  289. $route = substr($key, 7);
  290. }
  291. // 再次尝试获取菜单的父ID,如果仍然失败且$key不是'self',则记录未找到菜单的信息并跳过当前循环
  292. $result = $this->dao->getMenuPid($route, $isMer, 1);
  293. if (!$result && $key !== 'self') {
  294. printf($this->styles['info'], '未找到菜单: ' . $key);
  295. echo PHP_EOL;
  296. continue;
  297. } else {
  298. // 创建新的菜单项
  299. $result = $this->dao->create([
  300. 'pid' => $key == 'self' ? 0 : $result['menu_id'],
  301. 'path' => $key == 'self' ? '/' : $result['path'] . $result['menu_id'] . '/',
  302. 'menu_name' => $isAppend ? '附加权限' : '权限',
  303. 'route' => $key,
  304. 'is_mer' => $isMer,
  305. 'is_menu' => 0
  306. ]);
  307. }
  308. }
  309. // 将处理后的菜单项数据合并到结果集中
  310. $res = array_merge($res, $this->createSlit($isMer, $result['menu_id'], $result['path'], $value));
  311. } catch (\Exception $exception) {
  312. // 抛出异常,处理菜单创建过程中的错误
  313. throw new Exception($key);
  314. }
  315. }
  316. // 插入处理后的菜单项数据到数据库,并返回处理的菜单项数量
  317. if (!empty($res)) $this->dao->insertAll($res);
  318. return count($res);
  319. }
  320. /**
  321. * 新增权限数据整理
  322. * @param int $isMer
  323. * @param int $menuId
  324. * @param string $path
  325. * @param array $data
  326. * @return array
  327. * @author Qinii
  328. * @day 3/18/22
  329. */
  330. public function createSlit(int $isMer, int $menuId, string $path, array $data)
  331. {
  332. $arr = [];
  333. try {
  334. foreach ($data as $k => $v) {
  335. $result = $this->dao->getWhere(['route' => $v['route'], 'pid' => $menuId]);
  336. if (!$result) {
  337. $arr[] = [
  338. 'pid' => $menuId,
  339. 'path' => $path . $menuId . '/',
  340. 'menu_name' => $v['menu_name'],
  341. 'route' => $v['route'],
  342. 'is_mer' => $isMer,
  343. 'is_menu' => 0,
  344. 'params' => $v['params'] ?? [],
  345. ];
  346. if ($this->prompt == 's') {
  347. printf($this->styles['success'], '新增权限: ' . $v['menu_name'] . ' [' . $v['route'] . ']');
  348. echo PHP_EOL;
  349. }
  350. }
  351. }
  352. return $arr;
  353. } catch (\Exception $exception) {
  354. halt($isMer, $menuId, $path, $data);
  355. }
  356. }
  357. /**
  358. * 快捷搜索
  359. * @param int $isMer 是否商户
  360. * @param string $keyWord 关键词
  361. * @return array
  362. * @throws DataNotFoundException
  363. * @throws DbException
  364. * @throws ModelNotFoundException
  365. * FerryZhao 2024/4/3
  366. */
  367. public function getMenusList($isMer, $keyWord)
  368. {
  369. $pre = '/' . config('admin.' . ($isMer ? 'merchant' : 'admin') . '_prefix');
  370. $configRepository = app()->make(ConfigRepository::class);
  371. $configData = $configRepository->getSearch([])
  372. ->where('user_type', $isMer)
  373. ->whereLike('config_name', '%' . $keyWord . '%')
  374. ->field('config_id,config_name,config_classify_id')
  375. ->with(['classConfig.parent'])->select()->toArray();
  376. $configMenus = [];
  377. $configMenusPath = [];
  378. $param = [];
  379. if ($configData) {
  380. foreach ($configData as $configDatum) {
  381. if ($_p = $this->valSpecial($isMer, $configDatum['config_name'])) {
  382. $configMenus[$_p] = [
  383. 'title' => $configDatum['config_name'],
  384. 'path' => $_p,
  385. ];
  386. } else if ($classConfig = $configDatum['classConfig']) {
  387. $classConfigId = !empty($classConfig['parent']) ? $classConfig['parent']['config_classify_id'] : $classConfig['config_classify_id'];
  388. $classifyKey = !empty($classConfig['parent']) ? $classConfig['parent']['classify_key'] : $classConfig['classify_key'];
  389. $title = !empty($classConfig['parent']) ? $classConfig['parent']['classify_name'] : $classConfig['classify_name'];
  390. $path = '/systemForm/Basics/' . $classifyKey;
  391. $configMenus[$classConfigId] = [
  392. 'title' => $title,
  393. 'path' => $path,
  394. ];
  395. $param[$path] = '/' . $classConfig['config_classify_id'];
  396. }
  397. }
  398. $configMenus = array_values($configMenus);
  399. $configMenusPath = array_column($configMenus, 'path');
  400. }
  401. $menuData = $this->dao->getSearch([])
  402. ->where('is_mer', $isMer)
  403. ->where('is_menu', 1)
  404. ->where('is_show', 1)
  405. ->where(function ($query) use ($keyWord, $configMenusPath) {
  406. $query->whereLike('menu_name', '%' . $keyWord . '%')->whereOr('route', 'in', $configMenusPath);
  407. })->select()->append(['parents', 'child'])->toArray();
  408. $menus = [];
  409. foreach ($menuData as $datum) {
  410. if (in_array($datum['route'], $this->unsetConfigArray)) {
  411. continue;
  412. }
  413. if (!empty($datum['child'])) {
  414. continue;
  415. }
  416. $title = '';
  417. if ($parents = $datum['parents']) {
  418. $path = explode('/', trim($datum['path'], '/'));
  419. $parents = array_combine(array_column($parents, 'pid'), $parents);
  420. $menu = $parents[0]['menu_name'] ?? '';
  421. foreach ($path as $item) {
  422. if (isset($parents[$item])) {
  423. $menu .= ' - ' . $parents[$item]['menu_name'];
  424. }
  425. }
  426. $menu .= ' - ' . $datum['menu_name'];
  427. $title = trim($menu, '-');
  428. }
  429. $menus[] = [
  430. 'path' => $pre . $datum['route'] . ($param[$datum['route']] ?? ''),
  431. 'title' => $title ?: $datum['menu_name'],
  432. ];
  433. }
  434. return compact('menus');
  435. }
  436. }