海南旅游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.

383 lines
14 KiB

4 years ago
4 years ago
  1. <?php
  2. namespace App\AdminAgent\Controllers;
  3. use App\AdminAgent\Extensions\Grid\SupplierShowQrcode;
  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 Dcat\Admin\Widgets\Alert;
  17. use Dcat\Admin\Widgets\Modal;
  18. use Dcat\Admin\Widgets\Table;
  19. use EasyWeChat\Factory;
  20. use EasyWeChat\Kernel\Http\StreamResponse;
  21. use Illuminate\Support\Facades\DB;
  22. use Illuminate\Support\Facades\Storage;
  23. class IndustryOrderController extends AdminController
  24. {
  25. /**
  26. * Make a grid builder.
  27. *
  28. * @return Grid
  29. */
  30. protected function grid()
  31. {
  32. return Grid::make(new IndustryOrder(['supplier:id,company_name', 'spec']), function (Grid $grid) {
  33. $grid->disableRowSelector();
  34. $grid->disableCreateButton();
  35. $grid->disableActions();
  36. $grid->model()->where('agent_id', Admin::user()->id);
  37. $grid->column('id')->sortable();
  38. $grid->column('supplier.company_name', '供应商')->limit(10);
  39. $grid->column('order_no')->limit(10);
  40. $grid->column('num');
  41. $grid->column('price');
  42. $grid->column('info', '客户信息')
  43. ->display('查看')
  44. ->modal('客户信息', function ($modal) {
  45. $info = $this->info ?? [];
  46. $info = array_map(function($v) {
  47. if (isset($v['value'], $v['type'])) {
  48. if ($v['type'] == 'image') {
  49. if (is_array($v['value'])) {
  50. return array_reduce($v['value'], fn($v2, $v3) => $v2 . '<img data-action="preview-img" src="' . $v3 . '" style="max-width:120px;max-height:200px;cursor:pointer" class="img img-thumbnail"> &nbsp;');
  51. } else {
  52. return '<img data-action="preview-img" src="' . $v['value'] . '" style="max-width:120px;max-height:200px;cursor:pointer" class="img img-thumbnail">';
  53. }
  54. } else {
  55. return is_string($v['value']) ? $v['value'] : join(',', $v['value']);
  56. }
  57. }
  58. return is_string($v) ? $v : json_encode($v);
  59. }, $info);
  60. return Table::make([], $info);
  61. })->xl();
  62. $grid->column('购买信息')
  63. ->display('查看')
  64. ->modal('购买信息', function ($model) {
  65. $info = [
  66. ['预留姓名', $this->name ?? ''],
  67. ['预留手机', $this->mobile ?? ''],
  68. ['产品名称', $this->title],
  69. ['规格名称', $this->spec->name ?? ''],
  70. ['规格日期', $this->spec->date ?? ''],
  71. ];
  72. return Table::make([], $info);
  73. });
  74. $grid->column('pay_type')->using(PayType::array());
  75. $grid->column('status')
  76. ->using(OrderStatus::array())
  77. ->if(fn() => $this->status == OrderStatus::UNPAID)
  78. ->display(fn() => '<a class="btn btn-sm btn-primary" href="' . admin_url('industry_order/list', $this->id) . '">付款</a>')
  79. ->if(fn() => $this->status == OrderStatus::PAY_EARNEST)
  80. ->display(fn() => '<a class="btn btn-sm btn-primary" href="' . admin_url('industry_order/list', $this->id) . '">付尾款</a>')
  81. ->if(fn() => $this->audit_status == -1)
  82. ->then(function ($column) {
  83. $column->display('')
  84. ->append(function () {
  85. return Modal::make()
  86. ->lg()
  87. ->title('审核信息')
  88. ->body($this->audit_opinion)
  89. ->button('<button class="btn btn-sm btn-info">被拒绝</button>');
  90. })
  91. ->append(function () {
  92. return ' <a class="btn btn-sm btn-warning" href="' . admin_url('industry_order/list', [$this->id, 'edit']) . '">编辑</a>';
  93. });
  94. })
  95. ->if(fn() => $this->audit_status == 0)
  96. ->then(function ($column) {
  97. $column->display('待审核')->label(Admin::color()->blueDarker());
  98. });
  99. $grid->column('paid_at')->width(100);
  100. $grid->column('verify_qrcode', '核销二维码')
  101. ->if(fn() => $this->verify_code)
  102. ->then(function (Grid\Column $column) {
  103. $verify_code = $this->id . '-' . $this->verify_code;
  104. $column->append(admin_url('industry_order/qrcode', $verify_code))->image('', 60, 60);
  105. $column->append('<br>' . $verify_code . '<br>');
  106. if (empty($this->show_qrcode)) {
  107. $column->append((new SupplierShowQrcode)->setKey($this->id));
  108. }
  109. })
  110. ->else()
  111. ->display('');
  112. $grid->column('created_at')->width(100);
  113. $grid->filter(function (Grid\Filter $filter) {
  114. $filter->equal('id')->width(2);
  115. $filter->equal('order_no')->width(3);
  116. });
  117. });
  118. }
  119. //生成核销二维码,行业产品订单使用支付小程序核销
  120. public function qrcode()
  121. {
  122. $verify_code = request()->route('verify_code');
  123. $qrcode = storage_path("app/public/industry_verify_code/$verify_code.jpg");
  124. if (file_exists($qrcode)) {
  125. return redirect(Storage::disk('public')->url("industry_verify_code/$verify_code.jpg"));
  126. }
  127. $setting = AdminSetting::val(['payee_appid', 'payee_appsecret']);
  128. $config = [
  129. 'app_id' => $setting['payee_appid'],
  130. 'secret' => $setting['payee_appsecret'],
  131. ];
  132. $app = Factory::miniProgram($config);
  133. //由于参数最多只能32个字符,故通过下面这种方式传参
  134. //pt表示使用普通订单,使用api/verification/verify接口核销;
  135. //hy表示行业产品订单,使用api/verification/industry_verify接口核销
  136. $response = $app->app_code->getUnlimit('hy' . $verify_code, ['page' => 'pages/verification/index']);
  137. if ($response instanceof StreamResponse) {
  138. $filename = $response->saveAs(storage_path('app/public/industry_verify_code'), $verify_code); //保存二维码
  139. // $qrcode = Storage::disk('public')->url('industry_verify_code/' . $filename); //获取前端路径
  140. header("Content-Type: " . $response->getHeaderLine('Content-Type'));
  141. return $response; //输出图片
  142. }
  143. }
  144. /**
  145. * Make a show builder.
  146. *
  147. * @param mixed $id
  148. *
  149. * @return Show
  150. */
  151. protected function detail($id)
  152. {
  153. return Show::make($id, new IndustryOrder(['supplier:id,company_name', 'spec']), function (Show $show) {
  154. $show->disableEditButton();
  155. $show->disableDeleteButton();
  156. $show->field('id');
  157. $show->field('supplier.company_name', '供应商');
  158. $show->field('order_no');
  159. $show->field('status')->using(OrderStatus::array())->label();
  160. $show->field('pay_type')->using(PayType::array());
  161. $show->field('spec', '规格')->as(fn() => ($this->spec->name ?? '') . ' | ' . ($this->spec->date ?? ''));
  162. $show->field('num');
  163. $show->field('price');
  164. $show->field('name', '姓名');
  165. $show->field('mobile');
  166. $show->field('title');
  167. $show->field('picture')->image('', 80, 80);
  168. $show->field('paid_at');
  169. $show->field('created_at', '下单时间');
  170. //付款对话框 weixin://wxpay/bizpayurl?pr=sk9zOCwzz
  171. if (in_array($show->model()->status, [OrderStatus::UNPAID, OrderStatus::PAY_EARNEST])) {
  172. $pay_config = $this->payConfig($show->model()->id);
  173. if (empty($pay_config['code_url'])) {
  174. if (isset($pay_config['result_code'], $pay_config['err_code_des']) && $pay_config['result_code'] != 'SUCCESS') {
  175. $msg = $pay_config['err_code_des'];
  176. } else {
  177. $msg = $pay_config['return_msg'] ?? '获取支付信息失败';
  178. }
  179. Admin::script("Dcat.swal.info('支付:$msg', null);");
  180. } else {
  181. $status_text = $show->model()->status == OrderStatus::PAY_EARNEST ?
  182. '\'当前状态:<b style="color:red;">'.OrderStatus::array()[$show->model()->status].'</b>\''
  183. : 'null';
  184. $back_url = admin_url('industry_order/list');
  185. Admin::js('@qrcode');
  186. Admin::script(<<<JS
  187. Dcat.swal.info('<div id="qrcode" style="margin-top:1rem;"></div>', $status_text, {
  188. type: null,
  189. imageWidth: 240,
  190. imageHeight: 240,
  191. animation: false,
  192. confirmButtonText: '已支付,刷新',
  193. showCancelButton: true,
  194. cancelButtonText: '返回列表',
  195. allowOutsideClick: false,
  196. allowEscapeKey: false,
  197. onOpen: function () {
  198. $('#qrcode').qrcode({text:'{$pay_config['code_url']}', width:240, height:240});
  199. }
  200. }).then((res) => {
  201. if (res.dismiss === 'cancel') {
  202. window.location.href = '$back_url';
  203. } else {
  204. window.location.reload();
  205. }
  206. });
  207. JS
  208. );
  209. }
  210. } else {
  211. redirect(admin_url('industry_order/list'))->send();
  212. }
  213. });
  214. }
  215. //付款
  216. private function payConfig($order_id)
  217. {
  218. $order = \App\Models\IndustryOrder::where('agent_id', Admin::user()->id)
  219. ->whereIn('status', [OrderStatus::UNPAID, OrderStatus::PAY_EARNEST])->find($order_id);
  220. if (!$order) {
  221. Admin::exit('订单不存在或已支付');
  222. }
  223. $config = AdminSetting::val(['payee_appid', 'payee_mchid', 'payee_mchkey']);
  224. $config = [
  225. 'app_id' => $config['payee_appid'],
  226. 'mch_id' => $config['payee_mchid'],
  227. 'key' => $config['payee_mchkey'],
  228. 'notify_url' => route('wxpay_industry_product_notify'),
  229. ];
  230. $app = Factory::payment($config);
  231. //计算价格
  232. if ($order->status == OrderStatus::PAY_EARNEST) {
  233. $price = $order->price - $order->paid_money;
  234. } elseif (in_array($order->pay_type, [PayType::DEPOSIT_PAY, PayType::EARNEST_PAY])) {
  235. $price = $order->prepay_price;
  236. } else {
  237. $price = $order->price;
  238. }
  239. return $app->order->unify([
  240. 'product_id' => $order->industry_product_id,
  241. 'body' => mb_strcut($order->title, 0, 127),
  242. //后面加status,主要是为了方便微信支付回调时区分定金(首付款)和尾款支付。substr(time(), -6)主要为了防止订单号重复
  243. 'out_trade_no' => $order->order_no . '-' . $order->status . substr(time(), -6),
  244. 'total_fee' => round($price * 100), //支付金额单位为分
  245. 'trade_type' => 'NATIVE', // 请对应换成你的支付方式对应的值类型
  246. ]);
  247. }
  248. protected function form()
  249. {
  250. return Form::make(new IndustryOrder(), function (Form $form) {
  251. if ($form->model()->agent_id != Admin::user()->id) {
  252. Admin::exit('数据不存在');
  253. }
  254. if ($form->model()->audit_status != -1) {
  255. return redirect(admin_url('industry_order/list'))->send();
  256. }
  257. $form->number('num')->required()->min(IndustryProduct::where('id', $form->model()->industry_product_id)->value('min_sale') ?? 1);
  258. $form->text('name', '您的姓名')->required();
  259. $form->mobile('mobile', '您的手机号')->required();
  260. //支付信息
  261. $pay_type = [PayType::ONLINE, PayType::OFFLINE];
  262. if ((float)$form->model()->deposit) { //订金支付
  263. $pay_type = [...$pay_type, PayType::DEPOSIT_PAY];
  264. }
  265. if ((float)$form->model()->earnest) { //定金支付
  266. $pay_type = [...$pay_type, PayType::EARNEST_PAY];
  267. }
  268. $options = array_filter(PayType::array(), fn($k) => in_array($k, $pay_type), ARRAY_FILTER_USE_KEY);
  269. $form->select('pay_type')
  270. ->options($options)->default(PayType::ONLINE)->required()
  271. ->when(PayType::DEPOSIT_PAY, function () use ($form) {
  272. $form->display('deposit', '订金')->customFormat(fn() => $form->model()->deposit);
  273. })->when(PayType::EARNEST_PAY, function () use ($form) {
  274. $form->display('earnest', '定金')->customFormat(fn() => $form->model()->earnest);
  275. });
  276. //载入信息收集表单数据
  277. if (!empty($form->model()->info)) {
  278. $form->html(Alert::make(null, '客户信息收集表单')->warning())->width(12);
  279. $fields = $form->model()->info;
  280. foreach ($fields as $v) {
  281. if (!isset($v['type'], $v['field'], $v['value'])) {
  282. continue;
  283. }
  284. if ($v['type'] == 'radio' || $v['type'] == 'checkbox') {
  285. $form->{$v['type']}('info.' . $v['field'])
  286. ->options(array_combine($v['options'], $v['options']))
  287. ->required((bool)$v['required'])
  288. ->customFormat(fn() => $v['value']);
  289. } else if ($v['type'] == 'image') {
  290. $form->multipleImage('info.' . $v['field'])
  291. ->uniqueName()->saveFullUrl()
  292. ->required((bool)$v['required'])
  293. ->customFormat(fn() => $v['value']);
  294. } else {
  295. $form->{$v['type']}('info.' . $v['field'])
  296. ->required((bool)$v['required'])
  297. ->customFormat(fn() => $v['value']);
  298. }
  299. }
  300. }
  301. })->saving(function (Form $form) {
  302. //信息收集表处理,保留字段类型等信息,便于后台显示
  303. $order_info = $form->info ?? [];
  304. if (!empty($order_info)) {
  305. $fields = array_column($form->model()->info, null, 'field');
  306. foreach ($fields as &$field) {
  307. if ($field['required'] && !isset($order_info[$field['field']])) { //判断是否必填
  308. return $form->response()->error($field['field'] . '不能为空');
  309. }
  310. $field['value'] = $order_info[$field['field']] ?? '';
  311. }
  312. $form->info = $fields;
  313. }
  314. $form->hidden(['audit_status', 'price', 'trade_deposit', 'single_price']);
  315. $form->audit_status = 0;
  316. //产品规格
  317. $spec = IndustryProductSpec::where([
  318. ['industry_product_id', '=', $form->model()->industry_product_id],
  319. ['stock', '>=', $form->num],
  320. ])->find($form->model()->industry_product_spec_id);
  321. if (!$spec) {
  322. return $form->response()->error('产品规格不存在或库存不足');
  323. }
  324. $industry_product = IndustryProduct::where([
  325. ['status', '=', ProductStatus::ON_SALE],
  326. ['stock', '>=', $form->num]
  327. ])->find($form->model()->industry_product_id);
  328. if (empty($industry_product)) {
  329. return $form->response()->error('产品不存在或库存不足');
  330. }
  331. $form->price = $form->num * $spec['price'];
  332. $form->trade_deposit = $industry_product->single_deposit * $form->num;
  333. $form->single_price = $industry_product->single_deposit;
  334. DB::beginTransaction();
  335. try {
  336. # 产品表减库存
  337. $industry_product->stock -= $form->num;
  338. $industry_product->save();
  339. if ($industry_product->stock < 0) {
  340. throw new \Exception('产品库存足');
  341. }
  342. # 规格减库存
  343. $spec->stock -= $form->num;
  344. $spec->save();
  345. if ($spec->stock < 0) {
  346. throw new \Exception('产品规格库存足');
  347. }
  348. DB::commit();
  349. } catch (\Exception $exception) {
  350. DB::rollBack();
  351. return $form->response()->error($exception->getMessage());
  352. }
  353. });
  354. }
  355. }