海南旅游SAAS
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

342 lines
13 KiB

4 years ago
4 years ago
  1. <?php
  2. namespace App\AdminAgent\Controllers;
  3. use App\AdminAgent\Renderable\SelectIndustryProductSpec;
  4. use App\AdminAgent\Repositories\IndustryOrder;
  5. use App\Common\OrderStatus;
  6. use App\Common\PayType;
  7. use App\Common\ProductStatus;
  8. use App\Models\AdminSetting;
  9. use App\Models\IndustryProduct;
  10. use App\Models\IndustryProductSpec;
  11. use Dcat\Admin\Admin;
  12. use Dcat\Admin\Form;
  13. use Dcat\Admin\Grid;
  14. use Dcat\Admin\Show;
  15. use Dcat\Admin\Http\Controllers\AdminController;
  16. use EasyWeChat\Factory;
  17. use EasyWeChat\Kernel\Http\StreamResponse;
  18. use Illuminate\Support\Facades\Storage;
  19. class IndustryOrderController extends AdminController
  20. {
  21. /**
  22. * Make a grid builder.
  23. *
  24. * @return Grid
  25. */
  26. protected function grid()
  27. {
  28. return Grid::make(new IndustryOrder(['supplier:id,company_name,contact_phone']), function (Grid $grid) {
  29. $grid->disableRowSelector();
  30. $grid->disableCreateButton();
  31. $grid->disableActions();
  32. $grid->model()->where('agent_id', Admin::user()->id);
  33. $grid->column('id')->sortable();
  34. $grid->column('supplier.company_name', '供应商')->limit(10);
  35. $grid->column('supplier.contact_phone', '供应商电话');
  36. $grid->column('order_no')->limit(10);
  37. $grid->column('num');
  38. $grid->column('price');
  39. $grid->column('name', '预留姓名');
  40. $grid->column('mobile', '预留手机');
  41. $grid->column('title')->limit(15);
  42. $grid->column('picture')->image('', 80, 80);
  43. $grid->column('status')
  44. ->using(OrderStatus::array())
  45. ->if(fn() => $this->status == OrderStatus::UNPAID)
  46. ->display(fn() => '<a class="btn btn-sm btn-primary" href="' . admin_url('industry_order/list', $this->id) . '">付款</a>')
  47. ->if(fn() => $this->status == OrderStatus::PAY_EARNEST)
  48. ->display(fn() => '<a class="btn btn-sm btn-primary" href="' . admin_url('industry_order/list', $this->id) . '">付尾款</a>');
  49. $grid->column('paid_at');
  50. $grid->column('verify_qrcode', '核销二维码')
  51. ->if(fn() => $this->verify_code)
  52. ->then(function ($column) {
  53. $verify_code = $this->id . '-' . $this->verify_code;
  54. $column->append(admin_url('industry_order/qrcode', $verify_code))->image('', 60, 60);
  55. $column->append('<br>' . $verify_code);
  56. })
  57. ->else()
  58. ->display('');
  59. $grid->column('created_at');
  60. $grid->filter(function (Grid\Filter $filter) {
  61. $filter->equal('id')->width(2);
  62. $filter->equal('order_no')->width(3);
  63. });
  64. });
  65. }
  66. //生成核销二维码,行业产品订单使用支付小程序核销
  67. public function qrcode()
  68. {
  69. $verify_code = request()->route('verify_code');
  70. $qrcode = storage_path("app/public/industry_verify_code/$verify_code.jpg");
  71. if (file_exists($qrcode)) {
  72. return redirect(Storage::disk('public')->url("industry_verify_code/$verify_code.jpg"));
  73. }
  74. $setting = AdminSetting::val(['payee_appid', 'payee_appsecret']);
  75. $config = [
  76. 'app_id' => $setting['payee_appid'],
  77. 'secret' => $setting['payee_appsecret'],
  78. ];
  79. $app = Factory::miniProgram($config);
  80. //由于参数最多只能32个字符,故通过下面这种方式传参
  81. //pt表示使用普通订单,使用api/verification/verify接口核销;
  82. //hy表示行业产品订单,使用api/verification/industry_verify接口核销
  83. $response = $app->app_code->getUnlimit('hy' . $verify_code, ['page' => 'pages/verification/index']);
  84. if ($response instanceof StreamResponse) {
  85. $filename = $response->saveAs(storage_path('app/public/industry_verify_code'), $verify_code); //保存二维码
  86. // $qrcode = Storage::disk('public')->url('industry_verify_code/' . $filename); //获取前端路径
  87. header("Content-Type: " . $response->getHeaderLine('Content-Type'));
  88. exit($response); //输出图片
  89. }
  90. }
  91. /**
  92. * Make a show builder.
  93. *
  94. * @param mixed $id
  95. *
  96. * @return Show
  97. */
  98. protected function detail($id)
  99. {
  100. return Show::make($id, new IndustryOrder(['supplier:id,company_name', 'spec']), function (Show $show) {
  101. $show->disableEditButton();
  102. $show->disableDeleteButton();
  103. $show->field('id');
  104. $show->field('supplier.company_name', '供应商');
  105. $show->field('order_no');
  106. $show->field('status')->using(OrderStatus::array())->label();
  107. $show->field('pay_type')->using(PayType::array());
  108. $show->field('spec', '规格')->as(fn() => ($this->spec->name ?? '') . ' | ' . ($this->spec->date ?? ''));
  109. $show->field('num');
  110. $show->field('price');
  111. $show->field('name', '姓名');
  112. $show->field('mobile');
  113. $show->field('title');
  114. $show->field('picture')->image('', 80, 80);
  115. $show->field('paid_at');
  116. $show->field('created_at', '下单时间');
  117. //付款对话框 weixin://wxpay/bizpayurl?pr=sk9zOCwzz
  118. if (in_array($show->model()->status, [OrderStatus::UNPAID, OrderStatus::PAY_EARNEST])) {
  119. $pay_config = $this->payConfig($show->model()->id);
  120. if (empty($pay_config['code_url'])) {
  121. if (isset($pay_config['result_code'], $pay_config['err_code_des']) && $pay_config['result_code'] != 'SUCCESS') {
  122. $msg = $pay_config['err_code_des'];
  123. } else {
  124. $msg = $pay_config['return_msg'] ?? '获取支付信息失败';
  125. }
  126. Admin::script("Dcat.swal.info('支付:$msg', null);");
  127. } else {
  128. $status_text = $show->model()->status == OrderStatus::PAY_EARNEST ?
  129. '\'当前状态:<b style="color:red;">'.OrderStatus::array()[$show->model()->status].'</b>\''
  130. : 'null';
  131. $back_url = admin_url('industry_order/list');
  132. Admin::js('@qrcode');
  133. Admin::script(<<<JS
  134. Dcat.swal.info('<div id="qrcode" style="margin-top:1rem;"></div>', $status_text, {
  135. type: null,
  136. imageWidth: 240,
  137. imageHeight: 240,
  138. animation: false,
  139. confirmButtonText: '已支付,刷新',
  140. showCancelButton: true,
  141. cancelButtonText: '返回列表',
  142. allowOutsideClick: false,
  143. allowEscapeKey: false,
  144. onOpen: function () {
  145. $('#qrcode').qrcode({text:'{$pay_config['code_url']}', width:240, height:240});
  146. }
  147. }).then((res) => {
  148. if (res.dismiss === 'cancel') {
  149. window.location.href = '$back_url';
  150. } else {
  151. window.location.reload();
  152. }
  153. });
  154. JS
  155. );
  156. }
  157. } else {
  158. redirect(admin_url('industry_order/list'))->send();
  159. }
  160. });
  161. }
  162. //付款
  163. private function payConfig($order_id)
  164. {
  165. $order = \App\Models\IndustryOrder::where('agent_id', Admin::user()->id)
  166. ->whereIn('status', [OrderStatus::UNPAID, OrderStatus::PAY_EARNEST])->find($order_id);
  167. if (!$order) {
  168. Admin::exit('订单不存在或已支付');
  169. }
  170. $config = AdminSetting::val(['payee_appid', 'payee_mchid', 'payee_mchkey']);
  171. $config = [
  172. 'app_id' => $config['payee_appid'],
  173. 'mch_id' => $config['payee_mchid'],
  174. 'key' => $config['payee_mchkey'],
  175. 'notify_url' => route('industry_product_wxpay_notify'),
  176. ];
  177. $app = Factory::payment($config);
  178. //计算价格
  179. if ($order->status == OrderStatus::PAY_EARNEST) {
  180. $price = $order->price - $order->paid_money;
  181. } elseif (in_array($order->pay_type, [PayType::DEPOSIT_PAY, PayType::EARNEST_PAY])) {
  182. $price = $order->prepay_price;
  183. } else {
  184. $price = $order->price;
  185. }
  186. return $app->order->unify([
  187. 'product_id' => $order->industry_product_id,
  188. 'body' => mb_strcut($order->title, 0, 127),
  189. 'out_trade_no' => $order->order_no . '-' . $order->status, //后面加status,主要是为了方便微信支付回调时区分定金(首付款)和尾款支付
  190. 'total_fee' => round($price * 100), //支付金额单位为分
  191. 'trade_type' => 'NATIVE', // 请对应换成你的支付方式对应的值类型
  192. ]);
  193. }
  194. /**
  195. * Make a form builder.
  196. *
  197. * @return Form
  198. */
  199. /*protected function form()
  200. {
  201. $pid = request()->input('pid');
  202. $industry = IndustryProduct::with('diyForm.fields')
  203. ->where([
  204. ['status', '=', ProductStatus::ON_SALE],
  205. ['stock', '>', 0],
  206. ])->find($pid);
  207. return Form::make(new IndustryOrder(), function (Form $form) use ($industry) {
  208. if (!$industry) {
  209. Admin::exit('订单不允许编辑');
  210. }
  211. $form->selectTable('industry_product_spec_id', '选择产品规格')
  212. ->required()
  213. ->title('选择产品规格')
  214. ->dialogWidth('80%;min-width:825px;')
  215. ->from(SelectIndustryProductSpec::make(['industry_product_id' => request()->input('pid')]))
  216. ->model(IndustryProductSpec::class);
  217. $form->hidden('pid')->value($industry->id); //pid要跟上面的request中的一样,否则提交出错
  218. $form->number('num')
  219. ->min($industry->min_sale)->required()
  220. ->default($industry->min_sale);
  221. $form->text('name')->default(Admin::user()->director)->required();
  222. $form->mobile('mobile')->default(Admin::user()->contact_phone)->required();
  223. $pay_type = [PayType::ONLINE, PayType::OFFLINE];
  224. if ($industry->deposit) { //订金支付
  225. $pay_type = [...$pay_type, PayType::DEPOSIT_PAY];
  226. }
  227. if ($industry->earnest) { //定金支付
  228. $pay_type = [...$pay_type, PayType::EARNEST_PAY];
  229. }
  230. $options = array_filter(PayType::array(), fn($k) => in_array($k, $pay_type), ARRAY_FILTER_USE_KEY);
  231. $form->select('pay_type')->options($options)->default(PayType::ONLINE)->required();
  232. //信息收集表单 TODO 信息收集表单文件上传不了,不能用事务
  233. if (!empty($industry->diyForm->fields)) {
  234. $form->divider();
  235. $fields = $industry->diyForm->fields->toArray();
  236. foreach ($fields as $v) {
  237. if ($v['type'] == 'radio' || $v['type'] == 'checkbox') {
  238. $form->{$v['type']}('info.' . $v['field'])->options($v['options'])->required((bool)$v['required']);
  239. } else {
  240. $form->{$v['type']}('info.' . $v['field'])->required((bool)$v['required']);
  241. }
  242. }
  243. }
  244. $form->divider();
  245. $form->text('', '购买产品')->default($industry->title)->disable();
  246. $form->text('', '单价')->default($industry->price)->disable();
  247. $form->text('', '库存')->default($industry->stock)->disable();
  248. $form->text('', '起购数量')->default($industry->min_sale)->disable();
  249. $form->image('picture', '产品图')->default($industry->pictures)->disable();
  250. $form->display('', '旅游须知')->default(fn() => preg_replace('/<script.*?>.*?<\/script>/is', '', $industry->know))->disable();
  251. $form->display('', '产品详情')->default(fn() => preg_replace('/<script.*?>.*?<\/script>/is', '', $industry->content))->disable();
  252. })->saving(function (Form $form) use ($industry) {
  253. //禁止编辑
  254. if ($form->isEditing()) {
  255. return $form->response()->error('操作禁止');
  256. }
  257. //判断最小起购数
  258. if ($form->num < $industry->min_sale) {
  259. return $form->response()->error('购买数量不能小于最低起购数:' . $industry->min_sale);
  260. }
  261. //判断产品状态和库存
  262. if (!$industry || $industry->status != ProductStatus::ON_SALE || $industry->stock < $form->num) {
  263. return $form->response()->error('产品已下架或库存不足');
  264. }
  265. //生成订单号
  266. list($micro, $sec) = explode(' ', microtime());
  267. $micro = str_pad(floor($micro * 1000000), 6, 0, STR_PAD_LEFT);
  268. $order_no = date('ymdHis', $sec) . $micro . mt_rand(1000, 9999);
  269. //产品规格处理
  270. $spec = IndustryProductSpec::where('industry_product_id', $form->pid)->find($form->industry_product_spec_id);
  271. if (!$spec) {
  272. return $form->response()->error('您选择的产品规格不存在');
  273. }
  274. $form->deleteInput(['pid', 'picture']);
  275. $form->hidden(['industry_product_id', 'supplier_id', 'agent_id', 'order_no', 'price', 'title', 'picture', 'status', 'pay_type', 'paid_at', 'verify_code', 'trade_deposit', 'timeout']);
  276. $form->name = $form->name ?? Admin::user()->director;
  277. $form->mobile = $form->mobile ?? Admin::user()->contact_phone;
  278. $form->industry_product_id = $industry->id;
  279. $form->supplier_id = $industry->supplier_id;
  280. $form->agent_id = Admin::user()->id;
  281. $form->order_no = $order_no;
  282. $form->price = $form->num * $spec->price;
  283. $form->title = $industry->title;
  284. $form->picture = $industry->pictures[0] ?? '' ;
  285. $form->status = $form->pay_type == PayType::OFFLINE ? OrderStatus::OFFLINE_UNPAID : OrderStatus::UNPAID;
  286. $form->paid_at = null;
  287. $form->verify_code = '';
  288. $form->trade_deposit = $form->num * $industry->single_deposit;
  289. $form->single_price = $industry->single_deposit;
  290. $form->timeout = null;
  291. if ($form->pay_type == PayType::DEPOSIT_PAY) {
  292. $form->prepay_price = $industry->deposit * $form->num;
  293. } else if ($form->pay_type == PayType::EARNEST_PAY) {
  294. $form->prepay_price = $industry->earnest * $form->num;
  295. } else {
  296. $form->prepay_price = 0;
  297. }
  298. //产品规格表减库存
  299. $spec->stock = $spec->stock - $form->num;
  300. $spec->save();
  301. })->saved(function (Form $form) {
  302. return $form->response()->success('下单成功,请等待供应商审核订单')->redirect(admin_url('industry_order/list'));
  303. })->deleting(function (Form $form) {
  304. return $form->response()->error('操作禁止');
  305. });
  306. }*/
  307. }