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

325 lines
14 KiB

4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
  1. <?php
  2. namespace App\AdminSupplier\Controllers;
  3. use App\AdminSupplier\Repositories\IndustryProduct;
  4. use App\Common\ProductStatus;
  5. use App\Models\Category;
  6. use App\Models\Order;
  7. use App\Models\Supplier;
  8. use App\Models\SystemSetting;
  9. use Dcat\Admin\Admin;
  10. use Dcat\Admin\Form;
  11. use Dcat\Admin\Form\NestedForm;
  12. use Dcat\Admin\Grid;
  13. use Dcat\Admin\Show;
  14. use Dcat\Admin\Http\Controllers\AdminController;
  15. use Illuminate\Support\Facades\DB;
  16. class IndustryProductController extends AdminController
  17. {
  18. /**
  19. * Make a grid builder.
  20. *
  21. * @return Grid
  22. */
  23. protected function grid()
  24. {
  25. return Grid::make(new IndustryProduct(['category:id,name']), function (Grid $grid) {
  26. $grid->disableDeleteButton();
  27. $grid->disableRowSelector();
  28. $grid->model()->where('supplier_id', Admin::user()->id);
  29. $grid->column('id')->sortable();
  30. $grid->column('type')->using(admin_trans('product.options.publish_type'));
  31. $grid->column('category.name', '分类');
  32. $grid->column('title')->limit(15);
  33. $grid->column('picture')->image('', 60,60);
  34. $grid->column('price');
  35. $grid->column('original_price');
  36. $grid->column('stock');
  37. $grid->column('sale');
  38. $grid->column('status')
  39. /*->if(fn() => in_array($this->status, [ProductStatus::SOLD_OUT, ProductStatus::ON_SALE]))
  40. ->using([ProductStatus::SOLD_OUT => 0, ProductStatus::ON_SALE => 1])
  41. ->switch()
  42. ->else()*/
  43. ->using(ProductStatus::array());
  44. $grid->column('service_persons');
  45. $grid->column('min_sale');
  46. $grid->column('created_at');
  47. $grid->filter(function (Grid\Filter $filter) {
  48. $filter->equal('id')->width(2);
  49. });
  50. });
  51. }
  52. /**
  53. * Make a show builder.
  54. *
  55. * @param mixed $id
  56. *
  57. * @return Show
  58. */
  59. protected function detail($id)
  60. {
  61. return Show::make($id, new IndustryProduct(['category:id,name']), function (Show $show) {
  62. $show->disableDeleteButton();
  63. //不允许查看非自己的数据
  64. if ($show->model()->supplier_id != Admin::user()->id) {
  65. Admin::exit('数据不存在');
  66. }
  67. $show->field('id');
  68. $show->field('type')->using(admin_trans('product.options.publish_type'));
  69. $show->field('category.name', '分类');
  70. $show->field('title');
  71. $show->field('pictures')->image('', 80, 80);
  72. $show->field('price');
  73. $show->field('original_price');
  74. $show->field('stock');
  75. $show->field('sale');
  76. $show->field('status')->using(ProductStatus::array());
  77. $show->field('know')->unescape()->as(fn($v) => preg_replace('/<script.*?>.*?<\/script>/is', '', $v));
  78. $show->field('content')->unescape()->as(fn($v) => preg_replace('/<script.*?>.*?<\/script>/is', '', $v));
  79. $show->field('service_persons');
  80. $show->field('min_sale');
  81. $show->field('verify_mobile');
  82. $show->field('created_at');
  83. $show->field('updated_at');
  84. });
  85. }
  86. /**
  87. * Make a form builder.
  88. *
  89. * @return Form
  90. */
  91. protected function form()
  92. {
  93. Admin::user()->publish_type = json_decode(Admin::user()->publish_type, true);
  94. //注:这2个变量需要通过引用方式传值才可以
  95. $change_deposit = 0; //变动的
  96. $old_deposit = 0; //原来交易金数
  97. return Form::make(new IndustryProduct(), function (Form $form) {
  98. $form->disableDeleteButton();
  99. $form->display('id');
  100. $options = Category::selectOptions(fn($query) => $query->where('agent_id', 0));
  101. $form->select('category_id')->options(array_slice($options, 1, null, true))->required();
  102. $form->text('title')->required();
  103. $form->currency('price')->required();
  104. $form->currency('original_price')->required();
  105. $form->number('service_persons')->min(0)->required()->help('所需交易金 = 单库存服务用户数 x 库存 x ' . SystemSetting::val('single', 'price'));
  106. $form->number('stock')->required()->help('所需交易金 = 单库存服务用户数 x 库存 x ' . SystemSetting::val('single', 'price'));
  107. $form->number('min_sale')->min(1)->required();
  108. if ($form->isEditing() && in_array($form->model()->status, [ProductStatus::SOLD_OUT, ProductStatus::ON_SALE])) {
  109. $form->radio('status')->options([1 => '上架', -2 => '下架'])->default(1);
  110. }
  111. $form->multipleImage('pictures')->required();
  112. $form->editor('know');
  113. $form->editor('content')->required();
  114. $form->mobile('verify_mobile')->required();
  115. //扩展字段
  116. $publish_type = array_intersect_key(
  117. admin_trans('product.options.publish_type'),
  118. array_flip(Admin::user()->publish_type)
  119. );
  120. //0:旅游线路、1:酒店、2:景区、3:餐厅、4:车队、5:单项
  121. $form->radio('type', '产品类型')
  122. ->options($publish_type)->disable($form->isEditing())
  123. ->default(current(Admin::user()->publish_type))
  124. ->when(0, function (Form $form) { //旅游线路
  125. if ($form->isEditing() && $form->model()->type != 0) {
  126. return;
  127. }
  128. $form->text('extends.field_0_departure_place', '出发地');
  129. $form->map('extends.field_0_departure_place_latitude', 'extends.field_0_departure_place_longitude', '出发地位置');
  130. $form->text('extends.field_0_destination', '目的地');
  131. $form->map('extends.field_0_destination_latitude', 'extends.field_0_destination_longitude', '目的地位置');
  132. $form->table('extends.field_0_project', '包含项目', function (NestedForm $table) {
  133. $table->text('name', '字段1');
  134. $table->text('num', '字段2');
  135. $table->text('price', '字段3');
  136. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  137. $form->dateRange('extends.field_0_date.start', 'extends.field_0_date.end', '行程时间');
  138. })->when(1, function (Form $form) { //酒店
  139. if ($form->isEditing() && $form->model()->type != 1) {
  140. return;
  141. }
  142. $default = [
  143. ['tag' => '行李寄存'], ['tag' => '24小时前台'], ['tag' => '前台保险柜'], ['tag' => '唤醒服务'],
  144. ['tag' => '早餐'], ['tag' => '送餐服务'], ['tag' => '电梯'], ['tag' => '空调'],
  145. ['tag' => '新风系统'], ['tag' => '24小时热水'], ['tag' => '吹风机'], ['tag' => '加湿器'],
  146. ['tag' => '自动售货机'], ['tag' => '健身房'], ['tag' => '桌球室'], ['tag' => '洗衣服务']
  147. ];
  148. $form->table('extends.field_1_tags', '酒店设施', function (NestedForm $table) {
  149. $table->text('tag', '包含项目')->placeholder('如:24小时热水、干洗服务等');
  150. })->value($default)->help('首次创建时,系统会默认填充基本服务,请根据本酒店情况进行删减或新增');
  151. $form->text('extends.field_1_name', '酒店名');
  152. $form->text('extends.field_1_address', '地址');
  153. $form->map('extends.field_1_latitude', 'extends.field_1_longitude', '位置');
  154. })->when(2, function (Form $form) { //景区
  155. if ($form->isEditing() && $form->model()->type != 2) {
  156. return;
  157. }
  158. $form->table('extends.field_2_open_time', '开放时间', function (NestedForm $table) {
  159. $table->text('node', '字段1')->placeholder('如:周一至周五');
  160. $table->text('summer', '字段2')->placeholder('如:08:00~19:00');
  161. $table->text('winter', '字段3')->placeholder('如:08:00~18:00');
  162. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  163. $form->table('extends.field_2_project', '包含项目', function (NestedForm $table) {
  164. $table->text('name', '字段1');
  165. $table->text('num', '字段2');
  166. $table->text('price', '字段3');
  167. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  168. $form->text('extends.field_2_name', '景区名');
  169. $form->text('extends.field_2_address', '地址');
  170. $form->map('extends.field_2_latitude', 'extends.field_2_longitude', '位置');
  171. })->when(3, function (Form $form) { //餐厅
  172. if ($form->isEditing() && $form->model()->type != 3) {
  173. return;
  174. }
  175. $form->table('extends.field_3_open_time', '开放时间', function (NestedForm $table) {
  176. $table->text('week', '字段1')->placeholder('如:周一至周五');
  177. $table->text('section', '字段2')->placeholder('如:上午/下午');
  178. $table->text('time', '字段3')->placeholder('如:08:00~18:00');
  179. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  180. $form->table('extends.field_3_package', '包含套餐', function (NestedForm $table) {
  181. $table->text('name', '字段1')->placeholder('如:清蒸鱿鱼');
  182. $table->text('num', '字段2')->placeholder('如:1条');
  183. $table->text('price', '字段3')->placeholder('如:99元');
  184. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  185. $form->text('extends.field_3_name', '餐厅名');
  186. $form->text('extends.field_3_address', '地址');
  187. $form->map('extends.field_3_latitude', 'extends.field_3_longitude', '位置');
  188. })->when(4, function (Form $form) { //车队
  189. $form->text('extends.field_4_address', '地址');
  190. $form->map('extends.field_4_latitude', 'extends.field_4_longitude', '位置');
  191. })->when(5, function (Form $form) { //单项
  192. $form->text('extends.field_5_address', '地址');
  193. $form->map('extends.field_5_latitude', 'extends.field_5_longitude', '位置');
  194. });
  195. })->saving(function (Form $form) use (&$change_deposit, &$old_deposit) {
  196. //不允许编辑非自己数据
  197. if ($form->isEditing() && $form->model()->supplier_id != Admin::user()->id) {
  198. return $form->response()->error('数据不存在');
  199. }
  200. if ($form->isCreating()) {
  201. if (!Admin::user()->publish_type || !in_array($form->type, Admin::user()->publish_type)) {
  202. return $form->response()->error('对不起,你没有此类产品的发布、编辑权限');
  203. }
  204. } else if ($form->isEditing()) { //type不允许编辑
  205. $form->deleteInput('type');
  206. }
  207. //处理交易金
  208. $single = SystemSetting::val('single', 'price'); //单人头交易费
  209. if (!$single) {
  210. return $form->response()->error('未设置单人头交易费,请联系管理员在后台设置');
  211. }
  212. //用户可编辑的状态
  213. $user_status = [ProductStatus::SOLD_OUT, ProductStatus::ON_SALE];
  214. //目前逻辑不考虑总后台修改单人头交易费的情况
  215. if ($form->isCreating()) {
  216. $change_deposit = $single * $form->service_persons * $form->stock;
  217. $form->hidden(['status', 'supplier_id']);
  218. $form->supplier_id = Admin::user()->id;
  219. $form->status = ProductStatus::UNAUDITED;
  220. }
  221. //库存和单库存服务用户数变动时冻结或解冻相应的保证金
  222. else if ($form->isEditing()) {
  223. if (!in_array($form->model()->status, $user_status) || !in_array($form->status, $user_status)) {
  224. $form->deleteInput('status');
  225. }
  226. //只处理原来状态是下架或上架用户可编辑的状态,否则后台未审核又多次编辑将出错
  227. if (in_array($form->model()->status, $user_status)) {
  228. //原来的单库存服务用户数及库存计算出来的值,如果是未审核状态要返回去
  229. $old_deposit = $single * $form->model()->service_persons * $form->model()->stock;
  230. //编辑库存或单库存服务用户数
  231. if ($form->model()->stock != $form->stock || $form->model()->service_persons != $form->service_persons) {
  232. $new_deposit = $single * $form->service_persons * $form->stock; //新的单库存服务用户数及库存计算出来的值
  233. $change_deposit = $new_deposit - $old_deposit; //saved里面获取的$form->model()->stock是不对的
  234. }
  235. }
  236. //因extends是数组,具有一定的特殊性,不能直接用$form->saved方法中的wasChanged判断是否修改,所以只能在这里做特殊处理
  237. if (!is_null($form->extends)) {
  238. $old_extends = array_filter($form->model()->extends, fn($v, $k) => str_contains($k, 'field_' . $form->model()->type) && !empty($v), ARRAY_FILTER_USE_BOTH);
  239. $new_extends = array_filter($form->extends, fn($v, $k) => str_contains($k, 'field_' . $form->model()->type) && !empty($v), ARRAY_FILTER_USE_BOTH);
  240. $old = $new = [];
  241. array_walk_recursive($old_extends, function($v) use (&$old) {if ($v !== null) $old[] = $v;});
  242. array_walk_recursive($new_extends, function($v) use (&$new) {if ($v !== null) $new[] = $v;});
  243. if (array_diff($old, $new)) {
  244. $form->status = ProductStatus::UNAUDITED;
  245. }
  246. }
  247. }
  248. //判断可用交易金是否充足
  249. if (Admin::user()->deposit_normal < $change_deposit) {
  250. return $form->response()->error('当前可用交易金:' . Admin::user()->deposit_normal . ",你还需要充值{$change_deposit}交易金才能发布此产品");
  251. }
  252. //忽略字段
  253. $form->ignore(['id', 'sale', 'created_at', 'updated_at', 'deleted_at']);
  254. })->saved(function (Form $form, $result) use (&$change_deposit, &$old_deposit) {
  255. if ($form->isEditing()) {
  256. $supplier = Supplier::query()->find(Admin::user()->id); //不能使用Admin::user()修改,必须使用Supplier模型才能正确记录资金变动日志
  257. //如果修改了库存和单库存服务用户数
  258. $is_change = $form->model()->wasChanged(['title', 'price', 'original_price', 'pictures', 'know', 'content']); //有extends判断不准
  259. if ($is_change) {
  260. DB::beginTransaction();
  261. try {
  262. //将产品状态改为未审核
  263. $form->model()->update(['status' => ProductStatus::UNAUDITED]);
  264. //如果改变的交易金不是0,原来冻结过的交易金先全部返还,待总后台审核通过之后再做扣除处理
  265. $supplier->deposit_normal = $supplier->deposit_normal + $old_deposit; //正常交易金
  266. $supplier->deposit_frozen = $supplier->deposit_frozen - $old_deposit; //冻结交易金
  267. $supplier->save();
  268. DB::commit();
  269. } catch (\Exception $e) {
  270. DB::rollBack();
  271. }
  272. } else if ($change_deposit != 0) { //修改库存或单库存服务用户数,但不修改关键信息,增减交易金,$change_deposit无论是正数还是负数都一样计算
  273. $supplier->deposit_normal = $supplier->deposit_normal - $change_deposit; //正常交易金
  274. $supplier->deposit_frozen = $supplier->deposit_frozen + $change_deposit; //冻结交易金
  275. $supplier->save();
  276. }
  277. }
  278. })->deleting(function (Form $form) {
  279. //不允许删除非自己的数据
  280. if (array_filter($form->model()->toArray(), fn($v) => $v['supplier_id'] != Admin::user()->id)) {
  281. return $form->response()->error('数据不存在');
  282. }
  283. });
  284. }
  285. }