ServiceCategory.php 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. <?php
  2. /**
  3. * Niushop商城系统 - 团队十年电商经验汇集巨献!
  4. * =========================================================
  5. * Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
  6. * ----------------------------------------------
  7. * 官方网址: https://www.niushop.com
  8. * =========================================================
  9. */
  10. namespace app\model\goods;
  11. use think\facade\Cache;
  12. use app\model\BaseModel;
  13. use think\facade\Db;
  14. /**
  15. * 项目分类
  16. */
  17. class ServiceCategory extends BaseModel
  18. {
  19. /**
  20. * 添加项目分类
  21. * @param $data
  22. * @return \multitype
  23. */
  24. public function addCategory($data)
  25. {
  26. $site_id = isset($data[ 'site_id' ]) ? $data[ 'site_id' ] : '';
  27. if ($site_id === '') {
  28. return $this->error('', 'REQUEST_SITE_ID');
  29. }
  30. $category_id = model('service_category')->add($data);
  31. $info = model('service_category')->getInfo([ [ 'category_id', '=', $category_id ] ]);
  32. $common_data = array (
  33. 'category_id_1' => 0,
  34. 'category_id_2' => 0,
  35. 'category_id_3' => 0,
  36. );
  37. switch ( $data[ 'level' ] ) {
  38. case 1:
  39. $common_data[ 'category_id_1' ] = $category_id;
  40. break;
  41. case 2:
  42. $common_data[ 'category_id_1' ] = $data[ 'pid' ];//这让我并没有验证合法性,业务发生变动之后要注意(author:周)
  43. $common_data[ 'category_id_2' ] = $category_id;
  44. break;
  45. case 3:
  46. //错误数据纠正
  47. $parent_category_info = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'pid' ] ], [ 'site_id', '=', $site_id ] ], 'pid');//这让我并没有验证合法性,业务发生变动之后要注意(author:周)
  48. $common_data[ 'category_id_1' ] = $parent_category_info[ 'pid' ];//这让我并没有验证合法性,业务发生变动之后要注意(周)
  49. $common_data[ 'category_id_2' ] = $data[ 'pid' ];
  50. $common_data[ 'category_id_3' ] = $category_id;
  51. break;
  52. }
  53. $data = array_merge($data, $common_data);
  54. model('service_category')->update([ 'category_id_' . $data[ 'level' ] => $category_id ], [ [ 'category_id', '=', $category_id ] ]);
  55. Cache::tag("service_category_" . $site_id)->clear();
  56. return $this->success($category_id);
  57. }
  58. /**
  59. * 验证分类是否可以修改
  60. */
  61. public function checkEditCategory($data)
  62. {
  63. $category_id = $data[ 'category_id' ];
  64. $pid = $data[ 'pid' ] ?? 0;
  65. $parent_category_info = model('service_category')->getInfo([ [ 'category_id', '=', $pid ], [ 'site_id', '=', $data[ 'site_id' ] ] ]) ?? [];
  66. if (empty($parent_category_info)) return $this->success();
  67. $category_info = model('service_category')->getInfo([ [ 'category_id', '=', $category_id ], [ 'site_id', '=', $data[ 'site_id' ] ] ]) ?? [];
  68. if ($parent_category_info[ 'category_id' ] == $category_info[ 'category_id' ]) {
  69. return $this->error('', '不能修改上级为自己');
  70. }
  71. if ($category_info[ 'level' ] < 3 && $parent_category_info[ 'level' ] == 2) {
  72. $child_list = model('service_category')->getCount([ [ 'pid', '=', $category_info[ 'category_id' ] ] ], 'category_id');
  73. if ($child_list > 0) return $this->error('', '当前等级存在下级,不可修改为该上级');
  74. }
  75. if ($category_info[ 'level' ] == 1 && $parent_category_info[ 'level' ] == 1) {
  76. $child_list = model('service_category')->getColumn([ [ 'pid', '=', $category_info[ 'category_id' ] ] ], 'category_id');
  77. if ($child_list) {
  78. $child_child_list = model('service_category')->getCount([ [ 'pid', 'in', $child_list ] ], 'category_id');
  79. if ($child_child_list > 0) return $this->error('', '当前等级存在下下级,不可修改为该上级');
  80. }
  81. }
  82. return $this->success();
  83. }
  84. /**
  85. * 修改项目分类
  86. * @param $data
  87. * @return \multitype
  88. */
  89. public function editCategory($data)
  90. {
  91. $site_id = isset($data[ 'site_id' ]) ? $data[ 'site_id' ] : '';
  92. if ($site_id === '') {
  93. return $this->error('', 'REQUEST_SITE_ID');
  94. }
  95. $check_res = $this->checkEditCategory($data);
  96. if ($check_res[ 'code' ] < 0) return $check_res;
  97. model('service_category')->startTrans();
  98. try {
  99. //仅限当分类不支持跨级别修改
  100. $pid = $data[ 'pid' ];
  101. $level = 1;
  102. $parent_category_info = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'pid' ] ], [ 'site_id', '=', $site_id ] ]);
  103. if ($parent_category_info) $level = (int) $parent_category_info[ 'level' ] + 1;
  104. $data[ 'level' ] = $level;
  105. //获取该分类信息
  106. $info = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id' ] ], [ 'site_id', '=', $site_id ] ]);
  107. $common_data = array (
  108. 'category_id_1' => 0,
  109. 'category_id_2' => 0,
  110. 'category_id_3' => 0,
  111. );
  112. switch ( $level ) {
  113. case 1:
  114. $common_data[ 'category_id_1' ] = $info[ 'category_id' ];
  115. break;
  116. case 2:
  117. $common_data[ 'category_id_1' ] = $data[ 'pid' ];
  118. $common_data[ 'category_id_2' ] = $info[ 'category_id' ];
  119. //二三级要同步改动
  120. model('service_category')->update($common_data, [ [ 'pid', '=', $info[ 'category_id' ] ] ]);
  121. model('service_category')->update([ 'category_id_3' => Db::raw('category_id') ], [ [ 'pid', '=', $info[ 'category_id' ] ], [ 'level', '=', '3' ] ]);
  122. break;
  123. case 3:
  124. //错误数据纠正
  125. $parent_category_info = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'pid' ] ], [ 'site_id', '=', $site_id ] ], 'pid');//这让我并没有验证合法性,业务发生变动之后要注意(author:周)
  126. $common_data[ 'category_id_1' ] = $parent_category_info[ 'pid' ];//这让我并没有验证合法性,业务发生变动之后要注意(周)
  127. $common_data[ 'category_id_2' ] = $data[ 'pid' ];
  128. $common_data[ 'category_id_3' ] = $info[ 'category_id' ];
  129. break;
  130. }
  131. $info = array_merge($info, $common_data);
  132. $data = array_merge($data, $common_data);
  133. if ($data[ 'is_show' ] == -1) {
  134. switch ( $level ) {
  135. case 1:
  136. model('service_category')->update([ 'is_show' => -1 ], [ [ 'category_id_1', '=', $info[ 'category_id_1' ] ] ]);
  137. $data[ 'category_full_name' ] = $data[ 'category_name' ];
  138. break;
  139. case 2:
  140. model('service_category')->update([ 'is_show' => -1 ], [ [ 'category_id_2', '=', $info[ 'category_id_2' ] ] ]);
  141. $info_1 = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  142. $category_full_name = $info_1[ 'category_name' ] . '/' . $data[ 'category_name' ];
  143. $data[ 'category_full_name' ] = $category_full_name;
  144. break;
  145. case 3:
  146. model('service_category')->update([ 'is_show' => -1 ], [ [ 'category_id', 'in', [ $info[ 'category_id_1' ], $info[ 'category_id_2' ] ] ] ]);
  147. $info_1 = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  148. $info_2 = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_2' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  149. $category_full_name = $info_1[ 'category_name' ] . '/' . $info_2[ 'category_name' ] . '/' . $data[ 'category_name' ];
  150. $data[ 'category_full_name' ] = $category_full_name;
  151. break;
  152. }
  153. } else {
  154. switch ( $level ) {
  155. case 1:
  156. model('service_category')->update([ 'is_show' => 0 ], [ [ 'category_id_1', '=', $info[ 'category_id_1' ] ] ]);
  157. $data[ 'category_full_name' ] = $data[ 'category_name' ];
  158. break;
  159. case 2:
  160. model('service_category')->update([ 'is_show' => 0 ], [ [ 'category_id', '=', $info[ 'category_id_1' ] ] ]);
  161. $info_1 = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  162. $category_full_name = $info_1[ 'category_name' ] . '/' . $data[ 'category_name' ];
  163. $data[ 'category_full_name' ] = $category_full_name;
  164. break;
  165. case 3:
  166. model('service_category')->update([ 'is_show' => 0 ], [ [ 'category_id', 'in', [ $info[ 'category_id_1' ], $info[ 'category_id_2' ] ] ] ]);
  167. $info_1 = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  168. $info_2 = model('service_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_2' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  169. $category_full_name = $info_1[ 'category_name' ] . '/' . $info_2[ 'category_name' ] . '/' . $data[ 'category_name' ];
  170. $data[ 'category_full_name' ] = $category_full_name;
  171. break;
  172. }
  173. }
  174. if ($info[ 'pid' ] != $data[ 'pid' ]) {
  175. if ($level == 2) {
  176. model('goods')->update([
  177. 'category_json' => Db::raw("REPLACE(category_json, '[\"{$info['pid']},{$data[ 'category_id' ]},', '[\"{$data['pid']},{$data[ 'category_id' ]},')")
  178. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  179. model('goods')->update([
  180. 'category_json' => Db::raw("REPLACE(category_json, '[\"{$info['pid']},{$data[ 'category_id' ]}\"]', '[\"{$data['pid']},{$data[ 'category_id' ]}\"]')")
  181. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  182. model('goods')->update([
  183. 'category_id' => Db::raw("REPLACE(category_id, ',{$info['pid']},{$data[ 'category_id' ]},', ',{$data['pid']},{$data[ 'category_id' ]},')")
  184. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  185. } else {
  186. model('goods')->update([
  187. 'category_json' => Db::raw("REPLACE(category_json, '[\"{$info['category_id_1']},{$info['category_id_2']},{$info['category_id_3']}\"]', '[\"{$data['category_id_1']},{$data['category_id_2']},{$data['category_id_3']}\"]')")
  188. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  189. model('goods')->update([
  190. 'category_id' => Db::raw("REPLACE(category_id, ',{$info['category_id_1']},{$info[ 'category_id_2' ]},{$info[ 'category_id_3' ]},', ',{$data['category_id_1']},{$data[ 'category_id_2' ]},{$data[ 'category_id_3' ]},')")
  191. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  192. }
  193. }
  194. $res = model('service_category')->update($data, [ [ 'category_id', '=', $data[ 'category_id' ] ], [ 'site_id', '=', $site_id ] ]);
  195. //变更下级等级层级
  196. $child_list = model('service_category')->getColumn([ [ 'site_id', '=', $site_id ], [ 'pid', '=', $data[ 'category_id' ] ] ], 'category_id');
  197. if ($child_list) {
  198. model('service_category')->update([ 'level' => (int) $data[ 'level' ] + 1 ], [ [ 'category_id', 'in', $child_list ] ]);
  199. $child_child_list = model('service_category')->getColumn([ [ 'site_id', '=', $site_id ], [ 'pid', 'in', $child_list ] ], 'category_id');
  200. model('service_category')->update([ 'level' => (int) $data[ 'level' ] + 2 ], [ [ 'category_id', 'in', $child_child_list ] ]);
  201. }
  202. Cache::tag("service_category_" . $site_id)->clear();
  203. model('service_category')->commit();
  204. return $this->success($res);
  205. } catch (\Exception $e) {
  206. model('service_category')->rollback();
  207. return $this->error('', $e->getMessage() . $e->getFile() . $e->getLine());
  208. }
  209. }
  210. /**
  211. * 删除分类
  212. * @param $category_id
  213. * @return \multitype
  214. */
  215. public function deleteCategory($category_id, $site_id)
  216. {
  217. $site_id = isset($site_id) ? $site_id : '';
  218. if ($site_id === '') {
  219. return $this->error('', 'REQUEST_SITE_ID');
  220. }
  221. $service_category_info = $this->getCategoryInfo([
  222. [ 'category_id', '=', $category_id ], [ 'site_id', '=', $site_id ]
  223. ], "level")[ 'data' ];
  224. $field = "category_id_" . $service_category_info[ 'level' ];
  225. $res = model('service_category')->delete([ [ $field, '=', $category_id ], [ 'site_id', '=', $site_id ] ]);
  226. model('service_category')->delete([ [ 'category_id', '=', $category_id ], [ 'site_id', '=', $site_id ] ]);
  227. Cache::tag("service_category_" . $site_id)->clear();
  228. return $this->success($res);
  229. }
  230. /**
  231. * 获取项目分类信息
  232. * @param array $condition
  233. * @param string $field
  234. */
  235. public function getCategoryInfo($condition, $field = 'category_id,category_name,short_name,pid,level,is_show,sort,image,category_id_1,category_id_2,category_id_3,category_full_name,image_adv,link_url')
  236. {
  237. $check_condition = array_column($condition, 2, 0);
  238. $site_id = isset($check_condition[ 'site_id' ]) ? $check_condition[ 'site_id' ] : '';
  239. if ($site_id === '') {
  240. return $this->error('', 'REQUEST_SITE_ID');
  241. }
  242. $data = json_encode([ $condition, $field ]);
  243. $cache = Cache::get("service_category_getCategoryInfo_" . $site_id . "_" . $data);
  244. if (!empty($cache)) {
  245. return $this->success($cache);
  246. }
  247. $res = model('service_category')->getInfo($check_condition, $field);
  248. Cache::tag("service_category_" . $site_id)->set("service_category_getCategoryInfo_" . $site_id . "_" . $data, $res);
  249. return $this->success($res);
  250. }
  251. /**
  252. * 获取项目分类列表
  253. * @param array $condition
  254. * @param string $field
  255. * @param string $order
  256. * @param null $limit
  257. * @return \multitype
  258. */
  259. public function getCategoryList($condition = [], $field = 'category_id,category_name,short_name,pid,level,is_show,sort,image,category_id_1,category_id_2,category_id_3,image_adv', $order = 'sort asc', $limit = null)
  260. {
  261. $check_condition = array_column($condition, 2, 0);
  262. $site_id = isset($check_condition[ 'site_id' ]) ? $check_condition[ 'site_id' ] : '';
  263. if ($site_id === '') {
  264. return $this->error('', 'REQUEST_SITE_ID');
  265. }
  266. $data = json_encode([ $condition, $field, $order, $limit ]);
  267. $cache = Cache::get("service_category_getCategoryList_" . $site_id . "_" . $data);
  268. if (!empty($cache)) {
  269. return $this->success($cache);
  270. }
  271. $list = model('service_category')->getList($condition, $field, $order, '', '', '', $limit);
  272. Cache::tag("service_category_" . $site_id)->set("service_category_getCategoryList_" . $site_id . "_" . $data, $list);
  273. return $this->success($list);
  274. }
  275. /**
  276. * 获取项目分类树结构
  277. * @param array $condition
  278. * @param string $field
  279. * @param string $order
  280. * @param null $limit
  281. * @return \multitype
  282. */
  283. public function getCategoryTree($condition = [], $field = 'category_id,category_name,short_name,pid,level,is_show,sort,image,category_id_1,category_id_2,category_id_3', $order = 'sort asc,category_id desc', $limit = null)
  284. {
  285. $check_condition = array_column($condition, 2, 0);
  286. $site_id = isset($check_condition[ 'site_id' ]) ? $check_condition[ 'site_id' ] : '';
  287. if ($site_id === '') {
  288. return $this->error('', 'REQUEST_SITE_ID');
  289. }
  290. $data = json_encode([ $condition, $field, $order, $limit ]);
  291. $cache = Cache::get("service_category_getCategoryTree_" . $site_id . "_" . $data);
  292. if (!empty($cache)) {
  293. return $this->success($cache);
  294. }
  295. $list = model('service_category')->getList($condition, $field, $order, '', '', '', $limit);
  296. $service_category_list = [];
  297. //遍历一级项目分类
  298. foreach ($list as $k => $v) {
  299. if ($v[ 'level' ] == 1) {
  300. $service_category_list[] = $v;
  301. unset($list[ $k ]);
  302. }
  303. }
  304. $list = array_values($list);
  305. //遍历二级项目分类
  306. foreach ($list as $k => $v) {
  307. foreach ($service_category_list as $ck => $cv) {
  308. if ($v[ 'level' ] == 2 && $cv[ 'category_id' ] == $v[ 'pid' ]) {
  309. $service_category_list[ $ck ][ 'child_list' ][] = $v;
  310. unset($list[ $k ]);
  311. }
  312. }
  313. }
  314. $list = array_values($list);
  315. //遍历三级项目分类
  316. foreach ($list as $k => $v) {
  317. foreach ($service_category_list as $ck => $cv) {
  318. if (!empty($cv[ 'child_list' ])) {
  319. foreach ($cv[ 'child_list' ] as $third_k => $third_v) {
  320. if ($v[ 'level' ] == 3 && $third_v[ 'category_id' ] == $v[ 'pid' ]) {
  321. $service_category_list[ $ck ][ 'child_list' ][ $third_k ][ 'child_list' ][] = $v;
  322. unset($list[ $k ]);
  323. }
  324. }
  325. }
  326. }
  327. }
  328. Cache::tag("service_category_" . $site_id)->set("service_category_getCategoryTree_" . $site_id . "_" . $data, $service_category_list);
  329. return $this->success($service_category_list);
  330. }
  331. /**
  332. * 获取项目分类分页列表
  333. * @param array $condition
  334. * @param int $page
  335. * @param int $page_size
  336. * @param string $order
  337. * @param string $field
  338. * @return \multitype
  339. */
  340. public function getCategoryPageList($condition = [], $page = 1, $page_size = PAGE_LIST_ROWS, $order = '', $field = 'category_id,category_name,short_name,pid,level,is_show,sort,image,category_id_1,category_id_2,category_id_3,category_full_name,commission_rate')
  341. {
  342. $check_condition = array_column($condition, 2, 0);
  343. $site_id = isset($check_condition[ 'site_id' ]) ? $check_condition[ 'site_id' ] : '';
  344. if ($site_id === '') {
  345. return $this->error('', 'REQUEST_SITE_ID');
  346. }
  347. $data = json_encode([ $condition, $field, $order, $page, $page_size ]);
  348. $cache = Cache::get("service_category_getCategoryPageList_" . $site_id . "_" . $data);
  349. if (!empty($cache)) {
  350. return $this->success($cache);
  351. }
  352. $list = model('service_category')->pageList($condition, $field, $order, $page, $page_size);
  353. Cache::tag("service_category_" . $site_id)->set("service_category_getCategoryPageList_" . $site_id . "_" . $data, $list);
  354. return $this->success($list);
  355. }
  356. /**
  357. * 获取项目分类列表
  358. * @param array $condition
  359. * @param string $field
  360. * @param string $order
  361. * @param null $limit
  362. * @return \multitype
  363. */
  364. public function getCategoryByParent($condition = [], $field = 'category_id,category_name,short_name,pid,level,is_show,sort,image,category_id_1,category_id_2,category_id_3', $order = 'sort asc,category_id desc', $limit = null)
  365. {
  366. $check_condition = array_column($condition, 2, 0);
  367. $site_id = isset($check_condition[ 'site_id' ]) ? $check_condition[ 'site_id' ] : '';
  368. if ($site_id === '') {
  369. return $this->error('', 'REQUEST_SITE_ID');
  370. }
  371. $data = json_encode([ $condition, $field, $order, $limit ]);
  372. $cache = Cache::get("service_category_getCategoryByParent_" . $site_id . "_" . $data);
  373. if (!empty($cache)) {
  374. return $this->success($cache);
  375. }
  376. $list = model('service_category')->getList($condition, $field, $order, '', '', '', $limit);
  377. foreach ($list as $k => $v) {
  378. $child_count = model('service_category')->getCount([ 'pid' => $v[ 'category_id' ] ]);
  379. $list[ $k ][ 'child_count' ] = $child_count;
  380. }
  381. Cache::tag("service_category_" . $site_id)->set("service_category_getCategoryByParent_" . $site_id . "_" . $data, $list);
  382. return $this->success($list);
  383. }
  384. /**
  385. * 修改排序
  386. * @param int $sort
  387. * @param int $category_id
  388. */
  389. public function modifyGoodsCategorySort($sort, $category_id, $site_id)
  390. {
  391. $site_id = isset($site_id) ? $site_id : '';
  392. if ($site_id === '') {
  393. return $this->error('', 'REQUEST_SITE_ID');
  394. }
  395. $res = model('service_category')->update([ 'sort' => $sort ], [ [ 'category_id', '=', $category_id ], [ 'site_id', '=', $site_id ] ]);
  396. Cache::tag("service_category_" . $site_id)->clear();
  397. return $this->success($res);
  398. }
  399. }