GoodsCategory.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391
  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 app\model\BaseModel;
  12. use think\facade\Db;
  13. /**
  14. * 商品分类
  15. */
  16. class GoodsCategory extends BaseModel
  17. {
  18. /**
  19. * 添加商品分类
  20. * @param $data
  21. * @return \multitype
  22. */
  23. public function addCategory($data)
  24. {
  25. $site_id = isset($data[ 'site_id' ]) ? $data[ 'site_id' ] : '';
  26. if ($site_id === '') {
  27. return $this->error('', 'REQUEST_SITE_ID');
  28. }
  29. $category_id = model('goods_category')->add($data);
  30. $common_data = array (
  31. 'category_id_1' => 0,
  32. 'category_id_2' => 0,
  33. 'category_id_3' => 0,
  34. );
  35. switch ( $data[ 'level' ] ) {
  36. case 1:
  37. $common_data[ 'category_id_1' ] = $category_id;
  38. break;
  39. case 2:
  40. $common_data[ 'category_id_1' ] = $data[ 'pid' ];//这让我并没有验证合法性,业务发生变动之后要注意(author:周)
  41. $common_data[ 'category_id_2' ] = $category_id;
  42. break;
  43. case 3:
  44. //错误数据纠正
  45. $parent_category_info = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'pid' ] ], [ 'site_id', '=', $site_id ] ], 'pid');//这让我并没有验证合法性,业务发生变动之后要注意(author:周)
  46. $common_data[ 'category_id_1' ] = $parent_category_info[ 'pid' ];//这让我并没有验证合法性,业务发生变动之后要注意(周)
  47. $common_data[ 'category_id_2' ] = $data[ 'pid' ];
  48. $common_data[ 'category_id_3' ] = $category_id;
  49. break;
  50. }
  51. $data = array_merge($data, $common_data);
  52. model('goods_category')->update([ 'category_id_' . $data[ 'level' ] => $category_id ], [ [ 'category_id', '=', $category_id ] ]);
  53. return $this->success($category_id);
  54. }
  55. /**
  56. * 验证分类是否可以修改
  57. */
  58. public function checkEditCategory($data)
  59. {
  60. $category_id = $data[ 'category_id' ];
  61. $pid = $data[ 'pid' ] ?? 0;
  62. $parent_category_info = model('goods_category')->getInfo([ [ 'category_id', '=', $pid ], [ 'site_id', '=', $data[ 'site_id' ] ] ]) ?? [];
  63. if (empty($parent_category_info)) return $this->success();
  64. $category_info = model('goods_category')->getInfo([ [ 'category_id', '=', $category_id ], [ 'site_id', '=', $data[ 'site_id' ] ] ]) ?? [];
  65. if ($parent_category_info[ 'category_id' ] == $category_info[ 'category_id' ]) {
  66. return $this->error('', '不能修改上级为自己');
  67. }
  68. if ($category_info[ 'level' ] < 3 && $parent_category_info[ 'level' ] == 2) {
  69. $child_list = model('goods_category')->getCount([ [ 'pid', '=', $category_info[ 'category_id' ] ] ], 'category_id');
  70. if ($child_list > 0) return $this->error('', '当前等级存在下级,不可修改为该上级');
  71. }
  72. if ($category_info[ 'level' ] == 1 && $parent_category_info[ 'level' ] == 1) {
  73. $child_list = model('goods_category')->getColumn([ [ 'pid', '=', $category_info[ 'category_id' ] ] ], 'category_id');
  74. if ($child_list) {
  75. $child_child_list = model('goods_category')->getCount([ [ 'pid', 'in', $child_list ] ], 'category_id');
  76. if ($child_child_list > 0) return $this->error('', '当前等级存在下下级,不可修改为该上级');
  77. }
  78. }
  79. return $this->success();
  80. }
  81. /**
  82. * 修改商品分类
  83. * @param $data
  84. * @return \multitype
  85. */
  86. public function editCategory($data)
  87. {
  88. $site_id = isset($data[ 'site_id' ]) ? $data[ 'site_id' ] : '';
  89. if ($site_id === '') {
  90. return $this->error('', 'REQUEST_SITE_ID');
  91. }
  92. $check_res = $this->checkEditCategory($data);
  93. if ($check_res[ 'code' ] < 0) return $check_res;
  94. model('goods_category')->startTrans();
  95. try {
  96. //仅限当分类不支持跨级别修改
  97. $pid = $data[ 'pid' ];
  98. $level = 1;
  99. $parent_category_info = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'pid' ] ], [ 'site_id', '=', $site_id ] ]);
  100. if ($parent_category_info) $level = (int) $parent_category_info[ 'level' ] + 1;
  101. $data[ 'level' ] = $level;
  102. //获取该分类信息
  103. $info = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id' ] ], [ 'site_id', '=', $site_id ] ]);
  104. $common_data = array (
  105. 'category_id_1' => 0,
  106. 'category_id_2' => 0,
  107. 'category_id_3' => 0,
  108. );
  109. switch ( $level ) {
  110. case 1:
  111. $common_data[ 'category_id_1' ] = $info[ 'category_id' ];
  112. break;
  113. case 2:
  114. $common_data[ 'category_id_1' ] = $data[ 'pid' ];
  115. $common_data[ 'category_id_2' ] = $info[ 'category_id' ];
  116. //二三级要同步改动
  117. model('goods_category')->update($common_data, [ [ 'pid', '=', $info[ 'category_id' ] ] ]);
  118. model('goods_category')->update([ 'category_id_3' => Db::raw('category_id') ], [ [ 'pid', '=', $info[ 'category_id' ] ], [ 'level', '=', '3' ] ]);
  119. break;
  120. case 3:
  121. //错误数据纠正
  122. $parent_category_info = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'pid' ] ], [ 'site_id', '=', $site_id ] ], 'pid');//这让我并没有验证合法性,业务发生变动之后要注意(author:周)
  123. $common_data[ 'category_id_1' ] = $parent_category_info[ 'pid' ];//这让我并没有验证合法性,业务发生变动之后要注意(周)
  124. $common_data[ 'category_id_2' ] = $data[ 'pid' ];
  125. $common_data[ 'category_id_3' ] = $info[ 'category_id' ];
  126. break;
  127. }
  128. $info = array_merge($info, $common_data);
  129. $data = array_merge($data, $common_data);
  130. if ($data[ 'is_show' ] == -1) {
  131. switch ( $level ) {
  132. case 1:
  133. model('goods_category')->update([ 'is_show' => -1 ], [ [ 'category_id_1', '=', $info[ 'category_id_1' ] ] ]);
  134. $data[ 'category_full_name' ] = $data[ 'category_name' ];
  135. break;
  136. case 2:
  137. model('goods_category')->update([ 'is_show' => -1 ], [ [ 'category_id_2', '=', $info[ 'category_id_2' ] ] ]);
  138. $info_1 = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  139. $category_full_name = $info_1[ 'category_name' ] . '/' . $data[ 'category_name' ];
  140. $data[ 'category_full_name' ] = $category_full_name;
  141. break;
  142. case 3:
  143. // model('goods_category')->update(['is_show' => -1], [['category_id', 'in', [$info['category_id_1'], $info['category_id_2']]]]);
  144. $info_1 = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  145. $info_2 = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_2' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  146. $category_full_name = $info_1[ 'category_name' ] . '/' . $info_2[ 'category_name' ] . '/' . $data[ 'category_name' ];
  147. $data[ 'category_full_name' ] = $category_full_name;
  148. break;
  149. }
  150. } else {
  151. switch ( $level ) {
  152. case 1:
  153. model('goods_category')->update([ 'is_show' => 0 ], [ [ 'category_id_1', '=', $info[ 'category_id_1' ] ] ]);
  154. $data[ 'category_full_name' ] = $data[ 'category_name' ];
  155. break;
  156. case 2:
  157. model('goods_category')->update([ 'is_show' => 0 ], [ [ 'category_id', '=', $info[ 'category_id_1' ] ] ]);
  158. $info_1 = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  159. $category_full_name = $info_1[ 'category_name' ] . '/' . $data[ 'category_name' ];
  160. $data[ 'category_full_name' ] = $category_full_name;
  161. break;
  162. case 3:
  163. // model('goods_category')->update(['is_show' => 0], [['category_id', 'in', [$info['category_id_1'], $info['category_id_2']]]]);
  164. $info_1 = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_1' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  165. $info_2 = model('goods_category')->getInfo([ [ 'category_id', '=', $data[ 'category_id_2' ] ], [ 'site_id', '=', $site_id ] ], 'category_id_1,category_id_2,category_id_3,level,category_name');
  166. $category_full_name = $info_1[ 'category_name' ] . '/' . $info_2[ 'category_name' ] . '/' . $data[ 'category_name' ];
  167. $data[ 'category_full_name' ] = $category_full_name;
  168. break;
  169. }
  170. }
  171. if ($info[ 'pid' ] != $data[ 'pid' ]) {
  172. if ($level == 2) {
  173. model('goods')->update([
  174. 'category_json' => Db::raw("REPLACE(category_json, '[\"{$info['pid']},{$data[ 'category_id' ]},', '[\"{$data['pid']},{$data[ 'category_id' ]},')")
  175. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  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_id' => Db::raw("REPLACE(category_id, ',{$info['pid']},{$data[ 'category_id' ]},', ',{$data['pid']},{$data[ 'category_id' ]},')")
  181. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  182. } else {
  183. model('goods')->update([
  184. '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']}\"]')")
  185. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  186. model('goods')->update([
  187. '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' ]},')")
  188. ], [ [ 'category_id', 'like', "%,{$info['pid']},{$data[ 'category_id' ]},%" ], [ 'site_id', '=', $site_id ] ]);
  189. }
  190. }
  191. $res = model('goods_category')->update($data, [ [ 'category_id', '=', $data[ 'category_id' ] ], [ 'site_id', '=', $site_id ] ]);
  192. //变更下级等级层级
  193. $child_list = model('goods_category')->getColumn([ [ 'site_id', '=', $site_id ], [ 'pid', '=', $data[ 'category_id' ] ] ], 'category_id');
  194. if ($child_list) {
  195. model('goods_category')->update([ 'level' => (int) $data[ 'level' ] + 1 ], [ [ 'category_id', 'in', $child_list ] ]);
  196. $child_child_list = model('goods_category')->getColumn([ [ 'site_id', '=', $site_id ], [ 'pid', 'in', $child_list ] ], 'category_id');
  197. model('goods_category')->update([ 'level' => (int) $data[ 'level' ] + 2 ], [ [ 'category_id', 'in', $child_child_list ] ]);
  198. }
  199. model('goods_category')->commit();
  200. return $this->success($res);
  201. } catch (\Exception $e) {
  202. model('goods_category')->rollback();
  203. return $this->error('', $e->getMessage() . $e->getFile() . $e->getLine());
  204. }
  205. }
  206. /**
  207. * 删除分类
  208. * @param $category_id
  209. * @return \multitype
  210. */
  211. public function deleteCategory($category_id, $site_id)
  212. {
  213. $site_id = isset($site_id) ? $site_id : '';
  214. if ($site_id === '') {
  215. return $this->error('', 'REQUEST_SITE_ID');
  216. }
  217. //判断该分类下是否存在商品
  218. /* $goods_count = model('goods')->getCount([ [ 'category_id', 'like', '%,' . $category_id . ',%' ], [ 'site_id', '=', $site_id ] ]);
  219. if ($goods_count > 0) {
  220. return $this->error('', '该分类下存在商品,暂不能删除');
  221. }*/
  222. $goods_category_info = $this->getCategoryInfo([
  223. [ 'category_id', '=', $category_id ], [ 'site_id', '=', $site_id ]
  224. ], "level")[ 'data' ];
  225. $field = "category_id_" . $goods_category_info[ 'level' ];
  226. $res = model('goods_category')->delete([ [ $field, '=', $category_id ], [ 'site_id', '=', $site_id ] ]);
  227. model('goods_category')->delete([ [ 'category_id', '=', $category_id ], [ 'site_id', '=', $site_id ] ]);
  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_recommend,is_show,sort,image,keywords,description,attr_class_id,attr_class_name,category_id_1,category_id_2,category_id_3,category_full_name,commission_rate,image_adv,link_url,icon')
  236. {
  237. $res = model('goods_category')->getInfo($condition, $field);
  238. return $this->success($res);
  239. }
  240. /**
  241. * 获取商品分类列表
  242. * @param array $condition
  243. * @param string $field
  244. * @param string $order
  245. * @param null $limit
  246. * @return \multitype
  247. */
  248. public function getCategoryList($condition = [], $field = 'category_id,category_name,short_name,pid,level,is_recommend,is_show,sort,image,attr_class_id,attr_class_name,category_id_1,category_id_2,category_id_3,commission_rate,image_adv,icon', $order = '', $limit = null)
  249. {
  250. $list = model('goods_category')->getList($condition, $field, $order, '', '', '', $limit);
  251. return $this->success($list);
  252. }
  253. /**
  254. * 获取商品分类树结构
  255. * @param array $condition
  256. * @param string $field
  257. * @param string $order
  258. * @param null $limit
  259. * @return \multitype
  260. */
  261. public function getCategoryTree($condition = [], $field = 'category_id,category_name,short_name,pid,level,is_recommend,is_show,sort,image,attr_class_name,category_id_1,category_id_2,category_id_3,commission_rate,icon', $order = 'sort asc,category_id desc', $limit = null)
  262. {
  263. $list = model('goods_category')->getList($condition, $field, $order, '', '', '', $limit);
  264. $goods_category_list = [];
  265. //遍历一级商品分类
  266. foreach ($list as $k => $v) {
  267. if ($v[ 'level' ] == 1) {
  268. $goods_category_list[] = $v;
  269. unset($list[ $k ]);
  270. }
  271. }
  272. $list = array_values($list);
  273. //遍历二级商品分类
  274. foreach ($list as $k => $v) {
  275. foreach ($goods_category_list as $ck => $cv) {
  276. if ($v[ 'level' ] == 2 && $cv[ 'category_id' ] == $v[ 'pid' ]) {
  277. $goods_category_list[ $ck ][ 'child_list' ][] = $v;
  278. unset($list[ $k ]);
  279. }
  280. }
  281. }
  282. $list = array_values($list);
  283. //遍历三级商品分类
  284. foreach ($list as $k => $v) {
  285. foreach ($goods_category_list as $ck => $cv) {
  286. if (!empty($cv[ 'child_list' ])) {
  287. foreach ($cv[ 'child_list' ] as $third_k => $third_v) {
  288. if ($v[ 'level' ] == 3 && $third_v[ 'category_id' ] == $v[ 'pid' ]) {
  289. $goods_category_list[ $ck ][ 'child_list' ][ $third_k ][ 'child_list' ][] = $v;
  290. unset($list[ $k ]);
  291. }
  292. }
  293. }
  294. }
  295. }
  296. return $this->success($goods_category_list);
  297. }
  298. /**
  299. * 获取商品分类分页列表
  300. * @param array $condition
  301. * @param int $page
  302. * @param int $page_size
  303. * @param string $order
  304. * @param string $field
  305. * @return \multitype
  306. */
  307. public function getCategoryPageList($condition = [], $page = 1, $page_size = PAGE_LIST_ROWS, $order = '', $field = 'category_id,category_name,short_name,pid,level,is_recommend,is_show,sort,image,category_id_1,category_id_2,category_id_3,category_full_name,commission_rate,icon')
  308. {
  309. $list = model('goods_category')->pageList($condition, $field, $order, $page, $page_size);
  310. return $this->success($list);
  311. }
  312. /**
  313. * 获取商品分类列表
  314. * @param array $condition
  315. * @param string $field
  316. * @param string $order
  317. * @param null $limit
  318. * @return \multitype
  319. */
  320. public function getCategoryByParent($condition = [], $field = 'category_id,category_name,short_name,pid,level,is_recommend,is_show,sort,image,attr_class_id,attr_class_name,category_id_1,category_id_2,category_id_3,commission_rate,icon', $order = 'sort asc,category_id desc', $limit = null)
  321. {
  322. $list = model('goods_category')->getList($condition, $field, $order, '', '', '', $limit);
  323. foreach ($list as $k => $v) {
  324. $child_count = model('goods_category')->getCount([ 'pid' => $v[ 'category_id' ] ]);
  325. $list[ $k ][ 'child_count' ] = $child_count;
  326. }
  327. return $this->success($list);
  328. }
  329. /**
  330. * 修改排序
  331. * @param int $sort
  332. * @param int $category_id
  333. */
  334. public function modifyGoodsCategorySort($sort, $category_id, $site_id)
  335. {
  336. $res = model('goods_category')->update([ 'sort' => $sort ], [ [ 'category_id', '=', $category_id ], [ 'site_id', '=', $site_id ] ]);
  337. return $this->success($res);
  338. }
  339. }