Store.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482
  1. <?php
  2. /**
  3. * @copyright Copyright (c) 2017 https://www.sapixx.com All rights reserved.
  4. * @license Licensed (http://www.apache.org/licenses/LICENSE-2.0).
  5. * @author pillar<ltmn@qq.com>
  6. * 小程序公共API服务
  7. */
  8. namespace app\popupshop\controller\api\v1;
  9. use app\popupshop\controller\api\Base;
  10. use app\popupshop\model\Store as AppStore;
  11. use app\popupshop\model\Sale;
  12. use app\popupshop\model\SaleUser;
  13. use app\popupshop\model\SaleOrder;
  14. use app\popupshop\model\SaleOrderCache;
  15. use app\popupshop\model\SaleCategory;
  16. use app\popupshop\model\SaleHouse;
  17. use app\popupshop\model\Order;
  18. use app\popupshop\model\OrderCache;
  19. use app\popupshop\model\Config;
  20. use app\popupshop\model\BankBill;
  21. use app\popupshop\widget\Reward;
  22. use app\popupshop\model\Bank;
  23. use app\common\facade\WechatPay;
  24. use app\common\model\SystemMemberPayment;
  25. use app\common\model\SystemUserAddress;
  26. use think\facade\Request;
  27. class Store extends Base{
  28. public function initialize() {
  29. parent::initialize();
  30. if(!$this->user){
  31. exit(json_encode(['code'=>401,'msg'=>'用户认证失败']));
  32. }
  33. }
  34. /**
  35. * 获取配置
  36. */
  37. public function isOpen(){
  38. $rel = AppStore::where(['uid' => $this->user->id ])->find();
  39. if($rel){
  40. return enjson(200,'已开通小店',['name' => $rel->name]);
  41. }else{
  42. return enjson(204);
  43. }
  44. }
  45. /**
  46. * 开通小店
  47. * @return void
  48. */
  49. public function regStore(){
  50. if (request()->isPost()) {
  51. $param['store_name'] = Request::param('store_name');
  52. $param['formId'] = Request::param('formId');
  53. $param['sign'] = Request::param('sign');
  54. $rel = $this->apiSign($param);
  55. if($rel['code'] != 200){
  56. return enjson(204,'签名失败');
  57. }
  58. if (empty($param['store_name'])) {
  59. return enjson(403,'必须输入小店名称');
  60. }
  61. $store = AppStore::where(['uid' => $this->user->id])->count();
  62. if($store){
  63. return enjson(204,'已开通小店不用重复申请');
  64. }
  65. AppStore::insert(['name' => $param['store_name'],'uid' => $this->user->id,'update_time' => time()]);
  66. return enjson(200,'小店开通成功');
  67. }
  68. }
  69. /**
  70. * 我的产品
  71. * @return void
  72. */
  73. public function item(){
  74. if (request()->isGet()) {
  75. $param['page'] = Request::param('page/d');
  76. $param['types'] = Request::param('types/d',0);
  77. $param['sign'] = Request::param('sign');
  78. $rel = $this->apiSign($param);
  79. if($rel['code'] != 200){
  80. return enjson(204,'签名失败');
  81. }
  82. $condition['user_id'] = $this->user->id;
  83. $condition['member_miniapp_id'] = $this->miniapp_id;
  84. switch ($param['types']) {
  85. case 1:
  86. $condition['is_sale'] = 1;
  87. $condition['is_rebate'] = 0;
  88. $condition['is_out'] = 0;
  89. break;
  90. case 2:
  91. $condition['is_sale'] = 0;
  92. $condition['is_rebate'] = 1;
  93. $condition['is_out'] = 0;
  94. break;
  95. case 3:
  96. $condition['is_out'] = 1;
  97. break;
  98. default:
  99. $condition['is_sale'] = 0;
  100. $condition['is_rebate'] = 0;
  101. $condition['is_out'] = 0;
  102. break;
  103. }
  104. $lists = SaleUser::with(['house' => function($query) {
  105. $query->field('id,title,name,img,note');
  106. }])->where($condition)->order('id desc')->paginate(10);
  107. $data = [];
  108. foreach ($lists as $key => $value) {
  109. $data[$key]['id'] = $value->id;
  110. $data[$key]['house'] = $value->house;
  111. $data[$key]['status_text'] = SaleUser::status($value);
  112. $data[$key]['create_time'] = date('Y-m-d H:i',$value->create_time);
  113. $data[$key]['is_out'] = $value->is_out;
  114. $data[$key]['is_sale'] = $value->is_sale;
  115. $data[$key]['is_rebate'] = $value->is_rebate;
  116. $data[$key]['rebate'] = $value->rebate;
  117. $data[$key]['user_price'] = $value->user_price;
  118. //计算配套产品
  119. $gift = [];
  120. if(isset($value->sale)){
  121. $house_ids = array_column(json_decode($value->sale->gift),'house_id');
  122. foreach ($house_ids as $i => $id) {
  123. $gift[$i] = SaleHouse::where(['id' => $id])->find()->toArray();
  124. }
  125. }
  126. $data[$key]['sale'] = $gift;
  127. }
  128. if(empty($data)){
  129. return enjson(204,'无内容');
  130. }
  131. return enjson(200,'成功',$data);
  132. }
  133. }
  134. /**
  135. * 上下架宝贝
  136. * @return void
  137. */
  138. public function onUnder(){
  139. if (request()->isPost()) {
  140. $param['id'] = Request::param('id/d');
  141. $param['sign'] = Request::param('sign');
  142. $rel = $this->apiSign($param);
  143. if($rel['code'] != 200){
  144. return enjson(204,'签名失败');
  145. }
  146. $sale = SaleUser::where(['user_id' => $this->user->id,'id' => $param['id'],'is_out' => 0])->find();
  147. if(empty($sale)){
  148. return enjson(403,'宝贝未找到或未下架');
  149. }
  150. if(empty($sale->sale)){
  151. return enjson(302,'初次上架,请先采购礼品',['url' => '/pages/store/house?id='.$sale->id]);
  152. }
  153. $is_sale = $sale->is_sale ? 0 : 1;
  154. //更新上下架
  155. Sale::where(['sales_user_id' => $sale->id])->update(['is_sale' => $is_sale,'update_time' => time()]);
  156. $sale->is_sale = $is_sale;
  157. $sale->update_time = time();
  158. $sale->save();
  159. $msg = $is_sale ? '宝贝上架成功' : '宝贝下架成功';
  160. return enjson(200,$msg);
  161. }
  162. }
  163. /**
  164. * 判断订单的赠品是否已经卖了还钱,如果没有就去卖了还钱
  165. * @return void
  166. */
  167. public function isOnSale(){
  168. if (request()->isPost()) {
  169. $param['id'] = Request::param('id/d');
  170. $param['order_no'] = Request::param('order_no/s');
  171. $param['sign'] = Request::param('sign');
  172. $rel = $this->apiSign($param);
  173. if($rel['code'] != 200){
  174. return enjson(403,'接口签名失败');
  175. }
  176. $store = AppStore::where(['uid' => $this->user->id])->find();
  177. if(empty($store)){
  178. return enjson(302,'请先开通小店',['url' => '/pages/store/index']);
  179. }
  180. $condition = [];
  181. $condition['member_miniapp_id'] = $this->miniapp_id;
  182. $condition['user_id'] = $this->user->id;
  183. $condition['order_no'] = $param['order_no'];
  184. $condition['paid_at'] = 1;
  185. $condition['is_entrust'] = 0;
  186. $condition['is_out'] = 0;
  187. $condition['status'] = 0;
  188. $order = SaleOrder::where($condition)->find();
  189. if(empty($order)){
  190. return enjson(204,'订单状态不可操作');
  191. }
  192. $orderList = $order->orderList()->where(['is_entrust' =>0,'is_sales' => 0])->select(); //查询订单中包含的宝贝泪奔
  193. if($orderList->isEmpty()){
  194. $order->is_entrust = 1;
  195. $order->save();
  196. return enjson(204,'禁止重复委托');
  197. }
  198. $data = [];
  199. $order_cache_id = 0;
  200. foreach ($orderList as $value) {
  201. if($value->id == $param['id']){
  202. $order_cache_id = $value->id;
  203. $data['member_miniapp_id'] = $this->miniapp_id;
  204. $data['user_id'] = $this->user->id;
  205. $data['house_id'] = $value->house_id;
  206. $data['order_no'] = $value->order_no;
  207. $data['user_price'] = $order->real_amount;
  208. $data['rebate'] = 0;
  209. $data['is_rebate'] = 0;
  210. $data['is_sale'] = 0;
  211. $data['create_time'] = time();
  212. $data['update_time'] = time();
  213. }
  214. }
  215. if($order_cache_id && !empty($data)){
  216. $sale_user_id = SaleUser::insertGetId($data);
  217. if($sale_user_id){
  218. SaleOrderCache::where(['id' => $order_cache_id])->update(['is_entrust' => 1]);
  219. $orderCachenum = SaleOrderCache::where(['order_no' => $param['order_no'],'is_sales' => 0,'is_entrust' => 0])->count();
  220. if(!$orderCachenum){
  221. SaleOrder::where(['order_no' => $param['order_no']])->update(['is_entrust' => 1]);
  222. }
  223. //判断结算利润
  224. if($order->is_settle == 0 && !empty($order->sale)){
  225. $config = Config::where(['member_miniapp_id' => $order->member_miniapp_id])->find();
  226. $rebate = saleUser::where(['id' =>$order->sale->sales_user_id,'is_rebate' => 1,'is_lock_rebate' => 0])->find();
  227. if(!empty($rebate)){
  228. $rebate->is_lock_rebate = 1;
  229. $rebate->save();
  230. //资金到账并创建日志
  231. Bank::setDueMoney($order->member_miniapp_id,$order->sale->user_id,$rebate->rebate);
  232. BankBill::add($order->member_miniapp_id,$order->sale->user_id,$rebate->rebate,'成交利润已结算',$order->user_id,$order->order_no);
  233. //创建分账与扣点
  234. Reward::agent($order,$config); //给代理结算费用
  235. }
  236. Reward::fees($order,$config); //扣除平台费用
  237. }
  238. //修改订单状态为已分账
  239. if($order->is_settle == 0){
  240. $order->is_settle = 1;
  241. $order->save();
  242. }
  243. return enjson(200,'去采购礼品',['url' => '/pages/store/house?id='.$sale_user_id]);
  244. }
  245. }
  246. return enjson(403,'卖了换钱操作失败,请稍后再试');
  247. }
  248. }
  249. /**
  250. * 初次上下架宝贝
  251. * @return void
  252. */
  253. public function onSale(){
  254. if (request()->isPost()) {
  255. $param['sale_ids'] = Request::param('sale_ids/s');
  256. $param['user_sale_id'] = Request::param('user_sale_id/d');
  257. $param['entrust_price'] = Request::param('entrust_price/f',0);
  258. $param['sign'] = Request::param('sign');
  259. $rel = $this->apiSign($param);
  260. if($rel['code'] != 200){
  261. return enjson(403,'接口签名失败');
  262. }
  263. $sale_ids = ids(json_decode($param['sale_ids'],true),true);
  264. $store = AppStore::where(['uid' => $this->user->id])->find();
  265. if(empty($store)){
  266. return enjson(302,'请先开通小店',['url' => '/pages/store/index']);
  267. }
  268. $info = SaleUser::where(['user_id' => $this->user->id,'id' => $param['user_sale_id'],'is_sale' => 0,'is_out' => 0])->find();
  269. if(empty($info)){
  270. return enjson(403,'要上架的产品已不存在');
  271. }
  272. if(!empty($info->sale)){
  273. return enjson(403,'禁止重复采购礼品');
  274. }
  275. if($info->is_out){
  276. return enjson(403,'上架产品已退出委托');
  277. }
  278. if($info->is_rebate){
  279. return enjson(403,'上架产品已成交,不用重复上架');
  280. }
  281. $house = SaleHouse::where(['id' => $sale_ids,'is_sale' => 1,'is_del' => 0])->select();
  282. if(empty($house)){
  283. return enjson(403,'未找到礼品');
  284. }
  285. //配置
  286. $config = Config::where(['member_miniapp_id' => $this->miniapp_id])->find();
  287. $cost_price = array_sum(array_column($house->toArray(),'cost_price'));
  288. //利润
  289. $service_fee = $param['entrust_price']*$config->profit/100; //服务费
  290. $rebate = $param['entrust_price']-$service_fee-$cost_price; //成交价-服务费-成本
  291. if($rebate <= 0){
  292. return enjson(403,'您设置的寄卖价过低.利润低于0元');
  293. }
  294. $gift = [];
  295. $entrust_price = $param['entrust_price'];
  296. foreach ($house as $key => $value) {
  297. $gift[$key]['house_id'] = $value->id;
  298. $gift[$key]['cost_price'] = $value->cost_price;
  299. $gift[$key]['entrust_price'] = $entrust_price;
  300. $gift[$key]['sale_price'] = $entrust_price;
  301. }
  302. if(count($sale_ids) != count($gift)){
  303. return enjson(204,'你采购的礼品数量不符');
  304. }
  305. $data = [
  306. 'member_miniapp_id' => $this->miniapp_id,
  307. 'sales_user_id' => $info->id,
  308. 'house_id' => $info->house_id,
  309. 'cost_price' => $info->house->cost_price,
  310. 'entrust_price' => $entrust_price,
  311. 'sale_price' => $entrust_price,
  312. 'store_id' => $store->id,
  313. 'user_id' => $this->user->id,
  314. 'is_sale' => 1,
  315. 'gift' => $gift,
  316. ];
  317. $rel = Sale::edit($data);
  318. if($rel){
  319. $info->is_sale = 1;
  320. $info->save();
  321. return enjson(200,'产品上架成功');
  322. }
  323. return enjson(200,'产品上架失败');
  324. }
  325. }
  326. /**
  327. * 确认提货
  328. * @return void
  329. */
  330. public function onOrder(){
  331. if (request()->isPost()) {
  332. $param['id'] = Request::param('id/d');
  333. $param['sign'] = Request::param('sign');
  334. $rel = $this->apiSign($param);
  335. if($rel['code'] != 200){
  336. return enjson(204,'签名失败');
  337. }
  338. $sale = SaleUser::where(['user_id' => $this->user->id,'id' => $param['id'],'is_rebate' => 0,'is_out' => 0])->find();
  339. if(empty($sale)){
  340. return enjson(403,'宝贝未找到或未下架');
  341. }
  342. //读取发货地址
  343. $address = SystemUserAddress::where(['user_id'=>$this->user->id,'is_first' => 1])->find();
  344. if(empty($address)){
  345. return enjson(403,'请重新选择收货地址');
  346. }
  347. //支付接口
  348. $payment = SystemMemberPayment::where(['apiname' => 'wepay','member_miniapp_id'=>$this->miniapp_id])->find();
  349. if(empty($payment)){
  350. return enjson(403,'未开通微信支付功能');
  351. }
  352. $amount = 0.01;
  353. $order_no = $this->user->invite_code.order_no(); //生成的订单号
  354. $order['payment_id'] = $payment['id']; //支付ID
  355. $order['express_name'] = $address['name'];
  356. $order['express_phone'] = $address['telphone'];
  357. $order['express_address'] = $address['address'];
  358. $order['order_amount'] = $amount;
  359. $order['real_amount'] = $amount;
  360. $order['real_freight'] = 0;
  361. $order['order_no'] = $order_no;
  362. $order['member_miniapp_id'] = $this->miniapp_id;
  363. $order['user_id'] = $this->user->id;
  364. $order['sales_user_id'] = $sale->id;
  365. $order['order_starttime'] = time();
  366. $order_id = Order::insertGetId($order);
  367. if(empty($order_id)){
  368. return enjson(403,'创建订单失败');
  369. }
  370. //保存订单产品到缓存数据表
  371. $item_data['order_id'] = $order_id ;
  372. $item_data['order_no'] = $order_no;
  373. $item_data['item_id'] = $sale->house->id;
  374. $item_data['name'] = $sale->house->name;
  375. $item_data['img'] = $sale->house->img;
  376. $item_data['buy_price'] = 0;
  377. $item_data['buy_nums'] = 1;
  378. OrderCache::insert($item_data);
  379. //已经创建的订单下架和退出
  380. Sale::where(['sales_user_id' => $sale->id])->update(['is_sale' => 0,'is_out' => 1,'update_time' => time()]);
  381. //把用户的产品强制下架并退出
  382. $sale->is_out = 1;
  383. $sale->is_rebate = 1;
  384. $sale->is_sale = 0;
  385. $sale->update_time = time();
  386. $sale->save();
  387. //去请求微信支付接口
  388. $payparm = [
  389. 'openid' => $this->user->miniapp_uid,
  390. 'miniapp_id' => $this->miniapp_id,
  391. 'name' => $this->miniapp->appname.'申请提货',
  392. 'order_no' => $order_no,
  393. 'total_fee' => $amount*100,
  394. 'notify_url' => api(1,'popupshop/notify/shop',$this->miniapp_id),
  395. ];
  396. $ispay = WechatPay::orderPay($payparm);
  397. if($ispay['code'] == 0){
  398. return enjson(403,$ispay['msg']);
  399. }
  400. return enjson(200,'成功',$ispay['data']);
  401. }
  402. }
  403. /**
  404. * 库存商品栏目
  405. */
  406. public function saleHouseCate(){
  407. if (request()->isGet()) {
  408. $param['signkey'] = Request::param('signkey');
  409. $param['sign'] = Request::param('sign');
  410. $rel = $this->apiSign($param);
  411. if($rel['code'] != 200){
  412. return enjson(204,'签名失败');
  413. }
  414. $info = SaleCategory::where(['member_miniapp_id' => $this->miniapp_id])->field('id,name,title,picture')->order(['sort'=>'desc','id'=>'desc'])->select();
  415. return enjson(200,'成功',$info->toArray());
  416. }
  417. }
  418. /**
  419. * 平台中的库存商品列表
  420. */
  421. public function salelHouse(){
  422. if (request()->isGet()) {
  423. $param['page'] = Request::param('page/d',1);
  424. $param['cate_id'] = Request::param('cate_id/d',0);
  425. $param['user_sale_id'] = Request::param('user_sale_id/d',0);
  426. $param['sign'] = Request::param('sign');
  427. $param['num'] = Request::param('num/d',10);
  428. $rel = $this->apiSign($param);
  429. if($rel['code'] != 200){
  430. return enjson(204,'签名失败');
  431. }
  432. $info = SaleUser::where(['member_miniapp_id' => $this->miniapp_id,'user_id' => $this->user->id,'id' => $param['user_sale_id']])->field('user_price')->find();
  433. $condition[] = ['is_sale','=',1];
  434. $condition[] = ['is_del','=',0];
  435. $condition[] = ['member_miniapp_id','=',$this->miniapp_id];
  436. $condition[] = ['sell_price','<=',$info->user_price];
  437. if($param['cate_id'] > 0){
  438. $condition[] = ['category_id','=',$param['cate_id']];
  439. }
  440. $list = SaleHouse::where($condition)->field('id,title,name,note,img,sell_price,cost_price')->order('id desc')->paginate(20)->toArray();
  441. if(empty($list['data'])){
  442. return enjson(204,'无内容');
  443. }
  444. return enjson(200,'成功',$list['data']);
  445. }
  446. }
  447. /**
  448. * 售卖库存的商品
  449. * @return void
  450. */
  451. public function saleUser(){
  452. if (request()->isGet()) {
  453. $param['user_sale_id'] = Request::param('user_sale_id/d',0);
  454. $param['sign'] = Request::param('sign');
  455. $rel = $this->apiSign($param);
  456. if($rel['code'] != 200){
  457. return enjson(204,'签名失败');
  458. }
  459. $info = SaleUser::with('house')->where(['member_miniapp_id' => $this->miniapp_id,'user_id' => $this->user->id,'id' => $param['user_sale_id']])->find();
  460. if($info->is_out){
  461. return enjson(403,'上架产品已退出委托');
  462. }
  463. if($info->is_rebate){
  464. return enjson(403,'上架产品已成交,不用重复上架');
  465. }
  466. return enjson(200,'成功',$info);
  467. }
  468. }
  469. }