Goods.php 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428
  1. <?php
  2. /**
  3. * Niushop商城系统 - 团队十年电商经验汇集巨献!
  4. * =========================================================
  5. * Copy right 2019-2029 杭州牛之云科技有限公司, 保留所有权利。
  6. * ----------------------------------------------
  7. * 官方网址: https://www.niushop.com
  8. * =========================================================
  9. */
  10. namespace addon\shopcomponent\model;
  11. use app\model\BaseModel;
  12. class Goods extends BaseModel
  13. {
  14. /**
  15. * 商品状态
  16. * @var string[]
  17. */
  18. private $status = [
  19. 0 => '未上架',
  20. 5 => '上架中',
  21. 11 => '已下架',
  22. 13 => '违规下架'
  23. ];
  24. /**
  25. * 审核状态
  26. * @var string[]
  27. */
  28. private $editStatus = [
  29. 0 => '审核中',
  30. 1 => '编辑中',
  31. 2 => '审核中',
  32. 3 => '审核失败',
  33. 4 => '审核成功',
  34. ];
  35. /**
  36. * 获取商品列表
  37. * @param array $condition
  38. * @param bool $field
  39. * @param string $order
  40. * @param int $page
  41. * @param int $list_rows
  42. * @param string $alias
  43. * @param array $join
  44. * @return array
  45. */
  46. public function getGoodsPageList($condition = [], $field = true, $order = '', $page = 1, $list_rows = PAGE_LIST_ROWS, $alias = 'a', $join = [])
  47. {
  48. $field = 'sg.*,
  49. g.goods_name,g.goods_image,g.price,g.goods_stock,g.recommend_way';
  50. $alias = 'sg';
  51. $join = [
  52. [ 'goods g', 'g.goods_id = sg.out_product_id', 'inner' ],
  53. ];
  54. $data = model('shopcompoent_goods')->pageList($condition, $field, $order, $page, $list_rows, $alias, $join);
  55. if (!empty($data[ 'list' ])) {
  56. foreach ($data[ 'list' ] as $k => $item) {
  57. $data[ 'list' ][ $k ][ 'goods_stock' ] = numberFormat($data[ 'list' ][ $k ][ 'goods_stock' ]);
  58. $data[ 'list' ][ $k ][ 'status_name' ] = $this->status[ $item[ 'status' ] ] ?? '';
  59. $data[ 'list' ][ $k ][ 'edit_status_name' ] = $this->editStatus[ $item[ 'edit_status' ] ] ?? '';
  60. $arr_img = explode(',', $item[ 'goods_image' ]);
  61. $data[ 'list' ][ $k ][ 'cover_img' ] = $arr_img[ 0 ] ?? '';
  62. $data[ 'list' ][ $k ][ 'create_time' ] = $item[ 'create_time' ] > 0 ? time_to_date($item[ 'create_time' ]) : '--';
  63. $data[ 'list' ][ $k ][ 'audit_time' ] = $item[ 'audit_time' ] > 0 ? time_to_date($item[ 'audit_time' ]) : '--';
  64. $data[ 'list' ][ $k ][ 'reject_reason' ] = str_replace('https://developers.weixin.qq.com/miniprogram/dev/framework/ministore/minishopopencomponent/API/spu/add_spu.html', '', $item[ 'reject_reason' ]);
  65. }
  66. }
  67. return $this->success($data);
  68. }
  69. /**
  70. * 同步商品库商品
  71. */
  72. public function syncGoods($start, $limit, $site_id, $status = 5)
  73. {
  74. $weapp = new Weapp($site_id);
  75. $sync_res = $weapp->getSpuPage([ 'page' => $start, 'page_size' => $limit, 'status' => $status ]);
  76. if ($sync_res[ 'code' ] < 0) return $sync_res;
  77. if (!empty($sync_res[ 'data' ][ 'list' ])) {
  78. foreach ($sync_res[ 'data' ][ 'list' ] as $goods_item) {
  79. $count = model('shopcompoent_goods')->getCount([ [ 'out_product_id', '=', $goods_item[ 'out_product_id' ] ], [ 'site_id', '=', $site_id ] ]);
  80. if ($count) {
  81. model('shopcompoent_goods')->update([
  82. 'status' => $goods_item[ 'status' ],
  83. 'edit_status' => $goods_item[ 'edit_status' ],
  84. 'audit_time' => empty($goods_item[ 'audit_info' ]) ? 0 : strtotime($goods_item[ 'audit_info' ][ 'audit_time' ]),
  85. 'reject_reason' => $goods_item[ 'edit_status' ] == 3 ? $goods_item[ 'audit_info' ][ 'reject_reason' ] : ''
  86. ], [
  87. [ 'out_product_id', '=', $goods_item[ 'out_product_id' ] ]
  88. ]);
  89. } else {
  90. $category = ( new Category() )->getCategoryInfo($goods_item[ 'third_cat_id' ], $site_id);
  91. model('shopcompoent_goods')->add([
  92. 'site_id' => $site_id,
  93. 'product_id' => $goods_item[ 'product_id' ],
  94. 'out_product_id' => $goods_item[ 'out_product_id' ],
  95. 'third_cat_id' => $goods_item[ 'third_cat_id' ],
  96. 'brand_id' => $goods_item[ 'brand_id' ],
  97. 'info_version' => $goods_item[ 'info_version' ],
  98. 'status' => $goods_item[ 'status' ],
  99. 'edit_status' => $goods_item[ 'edit_status' ],
  100. 'create_time' => strtotime($goods_item[ 'create_time' ]),
  101. 'update_time' => strtotime($goods_item[ 'update_time' ]),
  102. 'audit_time' => empty($goods_item[ 'audit_info' ]) ? 0 : strtotime($goods_item[ 'audit_info' ][ 'audit_time' ]),
  103. 'reject_reason' => $goods_item[ 'edit_status' ] == 3 ? $goods_item[ 'audit_info' ][ 'reject_reason' ] : '',
  104. 'cat_name' => "{$category['first_cat_name']}>{$category['second_cat_name']}>{$category['third_cat_name']}"
  105. ]);
  106. }
  107. }
  108. $total_page = ceil($sync_res[ 'data' ][ 'total' ] / $limit);
  109. return $this->success([ 'page' => $start, 'total_page' => $total_page ]);
  110. } else {
  111. return $this->success([ 'page' => $start, 'total_page' => 1 ]);
  112. }
  113. }
  114. /**
  115. * 添加商品
  116. * @param $param
  117. */
  118. public function addGoods($param)
  119. {
  120. $goods_list = model('goods')->getList([ [ 'goods_id', 'in', explode(',', $param[ 'goods_ids' ]) ], [ 'site_id', '=', $param[ 'site_id' ] ] ], 'goods_id,goods_name,goods_image,sku_id,goods_content');
  121. if (!empty($goods_list)) {
  122. $category = ( new Category() )->getCategoryInfo($param[ 'third_cat_id' ], $param[ 'site_id' ]);
  123. if (empty($category) || $category[ 'status' ] == 0) return $this->error('', '该类目不存在或未审核通过');
  124. $weapp = new Weapp($param[ 'site_id' ]);
  125. foreach ($goods_list as $goods_item) {
  126. // 需加到库中的商品数据
  127. $goods_data = [
  128. 'out_product_id' => $goods_item[ 'goods_id' ],
  129. 'third_cat_id' => $param[ 'third_cat_id' ],
  130. 'brand_id' => $param[ 'brand_id' ] ?? 2100000000,
  131. 'info_version' => '0.0.1',
  132. ];
  133. //处理图片
  134. $goods_image_arr = $this->handleImg($goods_item[ 'goods_image' ]);
  135. $goods_image = [];
  136. foreach ($goods_image_arr as $img_k => $img_y) {
  137. $res = $weapp->getImg($img_y);
  138. if ($res[ 'code' ] == 0) {
  139. $goods_image[] = $res[ 'data' ][ 'img_info' ][ 'temp_img_url' ];
  140. }
  141. }
  142. // 同步商品所需数据
  143. $spu_data = [
  144. 'title' => $goods_item[ 'goods_name' ],
  145. 'path' => 'pages/goods/detail?sku_id=' . $goods_item[ 'sku_id' ],
  146. 'head_img' => $goods_image,
  147. 'qualification_pics' => $category[ 'product_qualification_type' ] == 0 ? '' : explode(',', $category[ 'qualification_pics' ]),
  148. 'desc_info' => [
  149. 'desc' => $goods_item[ 'goods_content' ]
  150. ],
  151. 'skus' => []
  152. ];
  153. $sku_list = model('goods_sku')->getList([ [ 'goods_id', '=', $goods_item[ 'goods_id' ] ] ], 'sku_id,sku_no,sku_image,discount_price,market_price,stock,sku_spec_format');
  154. foreach ($sku_list as $sku_item) {
  155. //图片处理
  156. $sku_res = $weapp->getImg($this->handleImg($sku_item[ 'sku_image' ])[ 0 ]);
  157. $sku_image = "";
  158. if ($sku_res[ 'code' ] == 0) {
  159. $sku_image = $sku_res[ 'data' ][ 'img_info' ][ 'temp_img_url' ];
  160. }
  161. $sku_data = [
  162. 'out_product_id' => $goods_item[ 'goods_id' ],
  163. 'out_sku_id' => $sku_item[ 'sku_id' ],
  164. 'thumb_img' => $sku_image,
  165. 'sale_price' => $sku_item[ 'discount_price' ] * 100,
  166. 'market_price' => $sku_item[ 'market_price' ] * 100,
  167. 'stock_num' => numberFormat($sku_item[ 'stock' ]),
  168. 'sku_code' => $sku_item[ 'sku_no' ],
  169. 'sku_attrs' => []
  170. ];
  171. if (!empty($sku_item[ 'sku_spec_format' ])) {
  172. foreach (json_decode($sku_item[ 'sku_spec_format' ], true) as $spec_item) {
  173. array_push($sku_data[ 'sku_attrs' ], [
  174. 'attr_key' => $spec_item[ 'spec_name' ],
  175. 'attr_value' => $spec_item[ 'spec_value_name' ]
  176. ]);
  177. }
  178. } else {
  179. $sku_data[ 'sku_attrs' ] = [
  180. [
  181. 'attr_key' => '',
  182. 'attr_value' => ''
  183. ]
  184. ];
  185. }
  186. array_push($spu_data[ 'skus' ], $sku_data);
  187. }
  188. // 添加商品到小程序
  189. $add_res = $weapp->addSpu(array_merge($goods_data, $spu_data));
  190. if ($add_res[ 'code' ] != 0) return $add_res;
  191. $goods_data[ 'product_id' ] = $add_res[ 'data' ][ 'product_id' ];
  192. $goods_data[ 'create_time' ] = !empty($add_res[ 'data' ][ 'create_time' ]) ? strtotime($add_res[ 'data' ][ 'create_time' ]) : time();
  193. $goods_data[ 'site_id' ] = $param[ 'site_id' ];
  194. $goods_data[ 'cat_name' ] = "{$category['first_cat_name']}>{$category['second_cat_name']}>{$category['third_cat_name']}";
  195. model('shopcompoent_goods')->add($goods_data);
  196. }
  197. return $this->success();
  198. } else {
  199. return $this->error('', '未获取到要添加的商品');
  200. }
  201. }
  202. /**
  203. * 更新商品
  204. * @param $param
  205. * @return array
  206. */
  207. public function updateGoods($param)
  208. {
  209. $shopcompoent_goods_info = model('shopcompoent_goods')->getInfo([ [ 'out_product_id', '=', $param[ 'goods_id' ] ], [ 'site_id', '=', $param[ 'site_id' ] ] ], '*');
  210. $goods_info = model('goods')->getInfo([ [ 'goods_id', '=', $param[ 'goods_id' ] ], [ 'site_id', '=', $param[ 'site_id' ] ] ], 'goods_id,goods_name,goods_image,sku_id,goods_content');
  211. if (!empty($shopcompoent_goods_info) && !empty($goods_info)) {
  212. $third_cat_id = $param[ 'third_cat_id' ] ?? $shopcompoent_goods_info[ 'third_cat_id' ];
  213. $category = ( new Category() )->getCategoryInfo($third_cat_id, $param[ 'site_id' ]);
  214. if (empty($category) || $category[ 'status' ] == 0) return $this->error('', '该类目不存在或未审核通过');
  215. $weapp = new Weapp($param[ 'site_id' ]);
  216. //处理图片
  217. $goods_image_arr = $this->handleImg($goods_info[ 'goods_image' ]);
  218. $goods_image = [];
  219. foreach ($goods_image_arr as $img_k => $img_y) {
  220. $res = $weapp->getImg($img_y);
  221. if ($res[ 'code' ] == 0) {
  222. $goods_image[] = $res[ 'data' ][ 'img_info' ][ 'temp_img_url' ];
  223. }
  224. }
  225. // 同步商品所需数据
  226. $spu_data = [
  227. 'out_product_id' => $goods_info[ 'goods_id' ],
  228. 'third_cat_id' => $third_cat_id,
  229. 'brand_id' => $param[ 'brand_id' ] ?? $shopcompoent_goods_info[ 'brand_id' ],
  230. 'info_version' => '0.0.1',
  231. 'title' => $goods_info[ 'goods_name' ],
  232. 'path' => 'pages/goods/detail?sku_id=' . $goods_info[ 'sku_id' ],
  233. 'head_img' => $goods_image,
  234. 'qualification_pics' => $category[ 'product_qualification_type' ] == 0 ? '' : explode(',', $category[ 'qualification_pics' ]),
  235. 'desc_info' => [
  236. 'desc' => $goods_info[ 'goods_content' ]
  237. ],
  238. 'skus' => []
  239. ];
  240. $sku_list = model('goods_sku')->getList([ [ 'goods_id', '=', $goods_info[ 'goods_id' ] ] ], 'sku_id,sku_no,sku_image,discount_price,market_price,stock,sku_spec_format');
  241. foreach ($sku_list as $sku_item) {
  242. //图片处理
  243. $sku_res = $weapp->getImg($this->handleImg($sku_item[ 'sku_image' ])[ 0 ]);
  244. $sku_image = "";
  245. if ($sku_res[ 'code' ] == 0) {
  246. $sku_image = $sku_res[ 'data' ][ 'img_info' ][ 'temp_img_url' ];
  247. }
  248. $sku_data = [
  249. 'out_product_id' => $goods_info[ 'goods_id' ],
  250. 'out_sku_id' => $sku_item[ 'sku_id' ],
  251. 'thumb_img' => $sku_image,
  252. 'sale_price' => $sku_item[ 'discount_price' ] * 100,
  253. 'market_price' => $sku_item[ 'market_price' ] * 100,
  254. 'stock_num' => numberFormat($sku_item[ 'stock' ]),
  255. 'sku_code' => $sku_item[ 'sku_no' ],
  256. 'sku_attrs' => []
  257. ];
  258. if (!empty($sku_item[ 'sku_spec_format' ])) {
  259. foreach (json_decode($sku_item[ 'sku_spec_format' ], true) as $spec_item) {
  260. array_push($sku_data[ 'sku_attrs' ], [
  261. 'attr_key' => $spec_item[ 'spec_name' ],
  262. 'attr_value' => $spec_item[ 'spec_value_name' ]
  263. ]);
  264. }
  265. } else {
  266. $sku_data[ 'sku_attrs' ] = [
  267. [
  268. 'attr_key' => '',
  269. 'attr_value' => ''
  270. ]
  271. ];
  272. }
  273. array_push($spu_data[ 'skus' ], $sku_data);
  274. }
  275. // 更新商品
  276. $update_res = $weapp->updateSpu($spu_data);
  277. if ($update_res[ 'code' ] != 0) return $update_res;
  278. $goods_data = [
  279. 'edit_status' => 0,
  280. 'update_time' => time(),
  281. 'third_cat_id' => $third_cat_id,
  282. 'cat_name' => "{$category['first_cat_name']}>{$category['second_cat_name']}>{$category['third_cat_name']}"
  283. ];
  284. model('shopcompoent_goods')->update($goods_data, [ [ 'out_product_id', '=', $param[ 'goods_id' ] ], [ 'site_id', '=', $param[ 'site_id' ] ] ]);
  285. return $this->success();
  286. } else {
  287. return $this->error('', '未获取到要更新的商品');
  288. }
  289. }
  290. /**
  291. * 处理图片
  292. * @param $images
  293. * @return false|string[]
  294. */
  295. private function handleImg($images)
  296. {
  297. $img_arr = explode(',', $images);
  298. foreach ($img_arr as $k => $v) {
  299. $img_arr[ $k ] = img($v);
  300. }
  301. return $img_arr;
  302. }
  303. /**
  304. * 删除商品
  305. * @param $id
  306. * @param $site_id
  307. */
  308. public function deleteGoods($goods_ids, $site_id)
  309. {
  310. if (!empty($goods_ids)) {
  311. $array_goodsIds = explode(',', $goods_ids);
  312. }
  313. foreach ($array_goodsIds as $k => $goods_id) {
  314. $res[ $k ] = ( new Weapp($site_id) )->delSpu([ 'out_product_id' => $goods_id ]);
  315. if ($res[ $k ][ 'code' ] != 0) return $res[ $k ];
  316. }
  317. model('shopcompoent_goods')->delete([ [ 'site_id', '=', $site_id ], [ 'out_product_id', 'in', $goods_ids ] ]);
  318. return $this->success();
  319. }
  320. /**
  321. * 商品上架
  322. * @param $goods_ids
  323. * @param $site_id
  324. * @return array|mixed
  325. * @throws \GuzzleHttp\Exception\GuzzleException
  326. */
  327. public function goodsListing($goods_ids, $site_id)
  328. {
  329. if (!empty($goods_ids)) {
  330. $array_goodsIds = explode(',', $goods_ids);
  331. }
  332. foreach ($array_goodsIds as $k => $goods_id) {
  333. $res[ $k ] = ( new Weapp($site_id) )->listing([ 'out_product_id' => $goods_id ]);
  334. if ($res[ $k ][ 'code' ] != 0) return $res[ $k ];
  335. }
  336. model('shopcompoent_goods')->update([ 'status' => 5 ], [ [ 'site_id', '=', $site_id ], [ 'out_product_id', 'in', $goods_ids ] ]);
  337. return $this->success();
  338. }
  339. /**
  340. * 商品下架
  341. * @param $goods_ids
  342. * @param $site_id
  343. * @return array|mixed
  344. * @throws \GuzzleHttp\Exception\GuzzleException
  345. */
  346. public function goodsDelisting($goods_ids, $site_id)
  347. {
  348. if (!empty($goods_ids)) {
  349. $array_goodsIds = explode(',', $goods_ids);
  350. }
  351. foreach ($array_goodsIds as $k => $goods_id) {
  352. $res[ $k ] = ( new Weapp($site_id) )->delisting([ 'out_product_id' => $goods_id ]);
  353. if ($res[ $k ][ 'code' ] != 0) return $res[ $k ];
  354. }
  355. model('shopcompoent_goods')->update([ 'status' => 11 ], [ [ 'site_id', '=', $site_id ], [ 'out_product_id', 'in', $goods_ids ] ]);
  356. return $this->success();
  357. }
  358. /**
  359. * @param $site_id
  360. */
  361. public function getOrderPayInfo($site_id)
  362. {
  363. $join = [
  364. [ 'goods g', 'g.goods_id = sg.out_product_id', 'left' ]
  365. ];
  366. $condition = [
  367. [ 'sg.site_id', '=', $site_id ],
  368. [ 'sg.status', '=', 5 ],
  369. [ 'sg.edit_status', '=', 4 ],
  370. [ 'g.is_delete', '=', 0 ],
  371. [ 'g.goods_state', '=', 1 ]
  372. ];
  373. $goods_info = model('shopcompoent_goods')->getInfo($condition, 'g.goods_name,g.price,g.goods_id,g.sku_id', 'sg', $join);
  374. if (empty($goods_info)) return $this->error('', '请先将商品同步到微信侧,等待商品审核通过并已上架');
  375. $qrcode_res = event('Qrcode', [
  376. 'site_id' => $site_id,
  377. 'app_type' => 'weapp',
  378. 'type' => 'create',
  379. 'data' => [
  380. 'goods_id' => $goods_info[ 'goods_id' ],
  381. 'is_test' => 1
  382. ],
  383. 'page' => '/pages/goods/detail',
  384. 'qrcode_path' => 'upload/qrcode/goods',
  385. 'qrcode_name' => "goods_qrcode_" . $goods_info[ 'sku_id' ]
  386. ], true);
  387. if ($qrcode_res[ 'code' ] != 0) return $qrcode_res;
  388. $goods_info[ 'qrcode_path' ] = $qrcode_res[ 'data' ][ 'path' ];
  389. return $this->success($goods_info);
  390. }
  391. }