Checkout.php 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | 萤火商城系统 [ 致力于通过产品和服务,帮助商家高效化开拓市场 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2017~2021 https://www.yiovo.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed 这不是一个自由软件,不允许对程序代码以任何形式任何目的的再发行
  8. // +----------------------------------------------------------------------
  9. // | Author: 萤火科技 <admin@yiovo.com>
  10. // +----------------------------------------------------------------------
  11. declare (strict_types=1);
  12. namespace app\index\controller;
  13. use app\index\model\Order;
  14. use app\index\model\Order as OrderModel;
  15. use app\index\model\User as UserModel;
  16. use app\index\service\order\PaySuccess;
  17. use app\index\service\User as UserService;
  18. use app\index\service\Cart as CartService;
  19. use app\index\service\order\Checkout as CheckoutService;
  20. use app\index\validate\order\Checkout as CheckoutValidate;
  21. use app\common\enum\order\PayType as OrderPayTypeEnum;
  22. use app\common\library\paypal\PayPal;
  23. use cores\exception\BaseException;
  24. use think\facade\Cache;
  25. use think\facade\Session;
  26. use think\response\Json;
  27. /**
  28. * 订单结算控制器
  29. * Class Checkout
  30. * @package app\api\controller
  31. */
  32. class Checkout extends Controller
  33. {
  34. // 结算台验证器
  35. /* @var CheckoutValidate $validate */
  36. private $validate;
  37. /**
  38. * 结算台订单信息
  39. * @param string $mode
  40. * @return Json
  41. * @throws BaseException
  42. * @throws \think\db\exception\DataNotFoundException
  43. * @throws \think\db\exception\DbException
  44. * @throws \think\db\exception\ModelNotFoundException
  45. */
  46. public function order(string $mode = 'buyNow'): Json
  47. {
  48. if ($mode === 'buyNow') {
  49. return $this->buyNow();
  50. } elseif ($mode === 'cart') {
  51. return $this->cart();
  52. }
  53. return $this->renderError('结算模式不合法');
  54. }
  55. /**
  56. * 订单提交
  57. * @param string $mode
  58. * @return Json
  59. * @throws BaseException
  60. * @throws \think\db\exception\DataNotFoundException
  61. * @throws \think\db\exception\DbException
  62. * @throws \think\db\exception\ModelNotFoundException
  63. */
  64. public function submit(string $mode = 'buyNow'): Json
  65. {
  66. $userId = Session::get('user_id');
  67. if (empty($userId)) {
  68. return $this->renderJson(config('status.not_logged'), 'Log in please!');
  69. }
  70. return $this->order($mode);
  71. }
  72. /**
  73. * 订单确认-立即购买
  74. * @return Json
  75. * @throws BaseException
  76. * @throws \think\db\exception\DataNotFoundException
  77. * @throws \think\db\exception\DbException
  78. * @throws \think\db\exception\ModelNotFoundException
  79. */
  80. private function buyNow(): Json
  81. {
  82. // 实例化结算台服务
  83. $Checkout = new CheckoutService;
  84. // 订单结算api参数
  85. $params = $Checkout->setParam($this->getParam([
  86. 'goodsId' => 0,
  87. 'goodsSkuId' => '',
  88. 'goodsNum' => 0,
  89. ]));
  90. // 表单验证
  91. if (!$this->getValidate()->scene('buyNow')->check($params)) {
  92. return $this->renderError($this->getValidate()->getError(), ['isCreated' => false]);
  93. }
  94. // 立即购买:获取订单商品列表
  95. $model = new OrderModel;
  96. $goodsList = $model->getOrderGoodsListByNow(
  97. (int)$params['goodsId'],
  98. (string)$params['goodsSkuId'],
  99. (int)$params['goodsNum']
  100. );
  101. // 获取订单确认信息
  102. $orderInfo = $Checkout->onCheckout($goodsList);
  103. if ($this->request->isGet()) {
  104. return $this->renderSuccess([
  105. 'order' => $orderInfo,
  106. 'personal' => $Checkout->getPersonal(),
  107. 'setting' => $Checkout->getSetting(),
  108. ]);
  109. }
  110. // 验证订单是否存在错误
  111. if ($Checkout->hasError()) {
  112. return $this->renderError($Checkout->getError(), ['is_created' => false]);
  113. }
  114. // 创建订单
  115. if (!$Checkout->createOrder($orderInfo)) {
  116. return $this->renderError($Checkout->getError() ?: '订单创建失败', ['is_created' => false]);
  117. }
  118. // 构建微信支付请求
  119. $payment = $model->onOrderPayment($Checkout->model, $params['payType']);
  120. // 返回结算信息
  121. return $this->renderSuccess([
  122. 'orderId' => $Checkout->model['order_id'], // 订单id
  123. 'payType' => $params['payType'], // 支付方式
  124. 'payment' => $payment // 微信支付参数
  125. ]);
  126. }
  127. /**
  128. * 订单确认-购物车结算
  129. * @return Json
  130. * @throws BaseException
  131. * @throws \think\db\exception\DataNotFoundException
  132. * @throws \think\db\exception\DbException
  133. * @throws \think\db\exception\ModelNotFoundException
  134. */
  135. private function cart(): Json
  136. {
  137. // 实例化结算台服务
  138. $Checkout = new CheckoutService;
  139. // 订单结算api参数
  140. $params = $Checkout->setParam($this->getParam());
  141. // 购物车ID集
  142. //$cartIds = $this->getCartIds();//不需要接口传过来
  143. $CartModel = new CartService;
  144. $cartIds = $CartModel->getCartIds();
  145. // 商品结算信息
  146. // 购物车商品列表
  147. $goodsList = $CartModel->getOrderGoodsList($cartIds);
  148. if (empty($goodsList)) {
  149. return $this->renderError('请选择商品结算');
  150. }
  151. // 获取订单结算信息
  152. $orderInfo = $Checkout->onCheckout($goodsList);
  153. if ($this->request->isGet()) {
  154. return $this->renderSuccess([
  155. 'order' => $orderInfo,
  156. 'personal' => $Checkout->getPersonal(),
  157. 'setting' => $Checkout->getSetting(),
  158. ]);
  159. }
  160. // 验证订单是否存在错误
  161. if ($Checkout->hasError()) {
  162. return $this->renderError($Checkout->getError(), ['is_created' => false]);
  163. }
  164. // 创建订单
  165. if (!$Checkout->createOrder($orderInfo)) {
  166. return $this->renderError($Checkout->getError() ?: '订单创建失败');
  167. }
  168. // 移出购物车中已下单的商品
  169. $CartModel->clear($cartIds);
  170. // 构建微信支付请求
  171. $payment = $Checkout->onOrderPayment();
  172. if ($params['payType'] == OrderPayTypeEnum::POINTS) {
  173. if (!$payment['flag']) {
  174. return $this->renderError($payment['message'] ?? '兑换失败');
  175. }
  176. }
  177. // 返回状态,如果是paypal支付,需要给用户跳转页面之外,还需要轮训查询订单的支付情况,如果支付成功,则弹窗提示,超过一分钟的话,提示支付超时,请重试
  178. return $this->renderSuccess([
  179. 'orderId' => $Checkout->model['order_id'], // 订单id
  180. 'payType' => $params['payType'], // 支付方式
  181. 'payment' => $payment // 微信支付参数
  182. ]);
  183. }
  184. /**
  185. * @return \think\response\View
  186. */
  187. public function payPayExecutePay($orderNo, $token)
  188. {
  189. if (empty($orderNo) || empty($token)) {
  190. return view('payError', ['notice' => lang('login success')]);
  191. }
  192. $oriToken = Cache::get(PayPal::PRE_STR . $orderNo);
  193. if ($token != $oriToken) {
  194. return view('payError', ['notice' => lang('login success')]);
  195. }
  196. $paymentId = $this->request->param('paymentId');
  197. //$token = $this->request->param('token');
  198. //$PayerID = $this->request->param('PayerID');
  199. $conf = config('paypal');
  200. //$orderNo = '';
  201. $pp = new PayPal($conf);
  202. $flag = $pp->executePayment($paymentId);
  203. if ($flag) {
  204. $orderModel = new PaySuccess($orderNo);
  205. $status = $orderModel->onPaySuccess(OrderPayTypeEnum::PAYPAL, ['transaction_id' => $paymentId]);
  206. if (!$status) {
  207. return view('payError', ['notice' => lang('login success')]);
  208. }
  209. Cache::delete(PayPal::PRE_STR . $orderNo);
  210. }
  211. //todo 等会测试
  212. /* $order = Order::detail(['order_no' => $orderNo]);
  213. //如果有分享人,给分享人送积分
  214. if (isset($order['user_id'])) {
  215. $key = Cache::get(UserModel::SHARE_PREFIX . $order['user_id']);
  216. if (!empty($key)) {
  217. $fromUserId = decrypt($key);
  218. $describe = "好友消费赠送的积分";
  219. $payPoints = intval(bcmul($order['pay_price'], '100', 0));
  220. UserModel::setIncPoints(intval($fromUserId), -$payPoints, $describe);
  221. }
  222. }*/
  223. return view('paySuccessful', ['notice' => lang('login success')]);
  224. }
  225. /**
  226. * 获取结算台验证器
  227. * @return CheckoutValidate
  228. */
  229. private function getValidate(): CheckoutValidate
  230. {
  231. if (!$this->validate) {
  232. $this->validate = new CheckoutValidate;
  233. }
  234. return $this->validate;
  235. }
  236. /**
  237. * 获取购物车ID集
  238. * @return false|string[]
  239. */
  240. private function getCartIds()
  241. {
  242. $cartIds = $this->request->param('cartIds');
  243. return explode(',', $cartIds);
  244. }
  245. /**
  246. * 订单结算提交的参数
  247. * @param array $define
  248. * @return array
  249. */
  250. private function getParam(array $define = []): array
  251. {
  252. return array_merge($define, $this->request->param());
  253. }
  254. }