Checkout.php 9.9 KB

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