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

423 lines
17 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
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
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\Product;
  4. use App\Common\OrderStatus;
  5. use App\Common\ProductStatus;
  6. use App\Models\AgentProduct;
  7. use App\Models\AgentProductItem;
  8. use App\Models\AgentProductSpec;
  9. use App\Models\Category;
  10. use App\Models\DiyForm;
  11. use App\Models\ProductSpec;
  12. use Dcat\Admin\Admin;
  13. use Dcat\Admin\Form;
  14. use Dcat\Admin\Form\NestedForm;
  15. use Dcat\Admin\Grid;
  16. use Dcat\Admin\Show;
  17. use Dcat\Admin\Http\Controllers\AdminController;
  18. class ProductController extends AdminController
  19. {
  20. protected $title = '产品';
  21. /**
  22. * Make a grid builder.
  23. *
  24. * @return Grid
  25. */
  26. protected function grid()
  27. {
  28. return Grid::make(new Product(['category:id,name']), function (Grid $grid) {
  29. $grid->model()->where('supplier_id', Admin::user()->id);
  30. $category_id = request()->input('cid');
  31. if ($category_id) {
  32. $grid->model()->whereIn('category_id', [$category_id, ...$this->get_category_child_ids($category_id)]);
  33. }
  34. $grid->column('id')->sortable();
  35. $grid->column('type')->using(admin_trans('product.options.publish_type'));
  36. $grid->column('category.name', '产品分类');
  37. $grid->column('title')->limit(15);
  38. $grid->column('picture')->image('', 60, 60);
  39. $grid->column('price');
  40. $grid->column('original_price');
  41. $grid->column('stock');
  42. $grid->column('sale');
  43. $grid->column('status')->help('切换开关可改变上下架状态')
  44. ->if(fn() => in_array($this->status, [ProductStatus::SOLD_OUT, ProductStatus::ON_SALE]))
  45. ->using([ProductStatus::SOLD_OUT => 0, ProductStatus::ON_SALE => 1])
  46. ->switch()
  47. ->else()
  48. ->using(ProductStatus::array())
  49. ->dot([
  50. ProductStatus::ON_SALE => 'success',
  51. ProductStatus::UNAUDITED => '',
  52. ProductStatus::REFUSE => 'danger',
  53. ProductStatus::SOLD_OUT => 'warning',
  54. ], 'primary');
  55. $grid->column('single_deposit');
  56. $grid->column('verify_mobile','核销员手机');
  57. $grid->column('created_at');
  58. $grid->column('updated_at');
  59. $grid->filter(function (Grid\Filter $filter) {
  60. $filter->panel();
  61. $filter->equal('id')->width(2);
  62. $filter->like('title')->width(3);
  63. $filter->equal('status')->select(ProductStatus::array())->width(2);
  64. $options = array_slice(Category::selectOptions(fn($query) => $query->where('agent_id', 0)), 1, null, true);
  65. $filter->equal('cid', '产品分类')->ignore()->select($options)->width(3);
  66. $filter->equal('type')->select(admin_trans('product.options.publish_type'))->width(2);
  67. });
  68. });
  69. }
  70. //递归获取指定分类下的所有子分类
  71. private function get_category_child_ids($category_id): array
  72. {
  73. static $category = null;
  74. if ($category === null) {
  75. $category = Category::where('agent_id', 0)->get()->toArray();
  76. }
  77. $child = [];
  78. foreach ($category as $cat) {
  79. if ($cat['pid'] == $category_id) {
  80. $child[] = $cat['id'];
  81. $child = array_merge($child, $this->get_category_child_ids($cat['id']));
  82. }
  83. }
  84. return $child;
  85. }
  86. /**
  87. * Make a show builder.
  88. *
  89. * @param mixed $id
  90. *
  91. * @return Show
  92. */
  93. protected function detail($id)
  94. {
  95. return Show::make($id, new Product(), function (Show $show) {
  96. //不允许查看非自己的数据
  97. if ($show->model()->supplier_id != Admin::user()->id) {
  98. Admin::exit('数据不存在');
  99. }
  100. $show->field('id');
  101. $show->field('supplier_id');
  102. $show->field('category_id');
  103. $show->field('title');
  104. $show->field('price');
  105. $show->field('original_price');
  106. $show->field('pictures')->image('', 80, 80);
  107. $show->field('stock');
  108. $show->field('sale');
  109. $show->field('single_deposit');
  110. $show->field('status');
  111. $show->field('know')->unescape()->as(fn($v) => preg_replace('/<script.*?>.*?<\/script>/is', '', $v));
  112. $show->field('content')->unescape()->as(fn($v) => preg_replace('/<script.*?>.*?<\/script>/is', '', $v));
  113. $show->field('verify_mobile','核销员手机');
  114. $show->field('created_at');
  115. $show->field('updated_at');
  116. });
  117. }
  118. /**
  119. * Make a form builder.
  120. *
  121. * @return Form
  122. */
  123. protected function form()
  124. {
  125. Admin::user()->publish_type = json_decode(Admin::user()->publish_type, true);
  126. return Form::make(new Product(['spec']), function (Form $form) {
  127. //不允许编辑非自己数据
  128. if ($form->isEditing() && $form->model()->supplier_id != Admin::user()->id) {
  129. return $form->response()->error('数据不存在');
  130. }
  131. $form->display('id');
  132. $options = Category::selectOptions(fn($query) => $query->where('agent_id', 0));
  133. $form->select('category_id')->options(array_slice($options, 1, null, true))->required();
  134. $cat2publish_type = json_encode(Category::where('agent_id', 0)->pluck('publish_type', 'id'));
  135. Admin::script(<<<JS
  136. var PublishType = $cat2publish_type;
  137. $('select[name="category_id"]').change(function () {
  138. var SelectId = $(this).val();
  139. $('input[name="type"][value="'+PublishType[SelectId]+'"]').click();
  140. });
  141. JS);
  142. //信息收集表单
  143. $options = DiyForm::where(['merchant_id' => Admin::user()->id, 'type' => 1])->pluck('name', 'id');
  144. if ($options->isEmpty()) {
  145. $form->select('diy_form_id', '信息收集表单')
  146. ->help('提示:信息收集表单为空,请前往“<a href="' . admin_url('diy_form/create') . '">信息收集表单</a>”处新增')
  147. ->options($options)->required();
  148. } else {
  149. $form->select('diy_form_id', '信息收集表单')->options($options)->required();
  150. }
  151. $form->text('title')->required();
  152. $form->hasMany('spec', function (NestedForm $form) {
  153. $form->hidden('id');
  154. $form->text('name', '规格')->required()->readonly();
  155. $form->date('date', '日期')->required();
  156. $form->text('stock')->required();
  157. $form->text('original_price')->required();
  158. $form->text('price')->required();
  159. $form->text('cost_price')->required();
  160. Admin::style('.field_date{width:100px!important;}
  161. .has-many-spec .col-md-12{padding:0;}
  162. .has-many-spec .add.btn{display:none;}
  163. .has-many-spec .input-group-prepend{display:none;}
  164. .has-many-spec .form-group{margin-bottom:0;}
  165. .has-many-spec .input-group>.form-control:not(:first-child){border-radius:.25rem;}');
  166. Admin::script(file_get_contents(resource_path('js/supplier-batch-add-spec.js')));
  167. })->useTable()->required();
  168. if ($form->isEditing() && in_array($form->model()->status, [ProductStatus::SOLD_OUT, ProductStatus::ON_SALE])) {
  169. $form->radio('status')->options([1 => '上架', -2 => '下架'])->default(1);
  170. }
  171. $form->multipleImage('pictures')->required()->removable(false)->uniqueName();
  172. $form->editor('know');
  173. $form->editor('content')->required();
  174. $form->mobile('verify_mobile')->required();
  175. //扩展字段
  176. $publish_type = array_intersect_key(
  177. admin_trans('product.options.publish_type'),
  178. array_flip(Admin::user()->publish_type)
  179. );
  180. //0:旅游线路、1:酒店、2:景区、3:餐厅、4:车队、5:单项
  181. $form->radio('type')
  182. ->options($publish_type)->disable($form->isEditing())
  183. ->default(current(Admin::user()->publish_type))
  184. ->when(0, function (Form $form) { //旅游线路
  185. if ($form->isEditing() && $form->model()->type != 0) {
  186. return;
  187. }
  188. $form->text('extends.field_0_departure_place', '出发地');
  189. $form->map('extends.field_0_departure_place_latitude', 'extends.field_0_departure_place_longitude', '出发地位置');
  190. $form->text('extends.field_0_destination', '目的地');
  191. $form->map('extends.field_0_destination_latitude', 'extends.field_0_destination_longitude', '目的地位置');
  192. $form->table('extends.field_0_project', '包含项目', function (NestedForm $table) {
  193. $table->text('name', '字段1');
  194. $table->text('num', '字段2');
  195. $table->text('price', '字段3');
  196. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  197. $form->dateRange('extends.field_0_date.start', 'extends.field_0_date.end', '行程时间');
  198. })->when(1, function (Form $form) { //酒店
  199. if ($form->isEditing() && $form->model()->type != 1) {
  200. return;
  201. }
  202. $default = [
  203. ['tag' => '行李寄存'], ['tag' => '24小时前台'], ['tag' => '前台保险柜'], ['tag' => '唤醒服务'],
  204. ['tag' => '早餐'], ['tag' => '送餐服务'], ['tag' => '电梯'], ['tag' => '空调'],
  205. ['tag' => '新风系统'], ['tag' => '24小时热水'], ['tag' => '吹风机'], ['tag' => '加湿器'],
  206. ['tag' => '自动售货机'], ['tag' => '健身房'], ['tag' => '桌球室'], ['tag' => '洗衣服务']
  207. ];
  208. $form->table('extends.field_1_tags', '酒店设施', function (NestedForm $table) {
  209. $table->text('tag', '包含项目')->placeholder('如:24小时热水、干洗服务等');
  210. })->default($default)->help('首次创建时,系统会默认填充基本服务,请根据本酒店情况进行删减或新增');
  211. $form->text('extends.field_1_name', '酒店名');
  212. $form->text('extends.field_1_address', '地址');
  213. $form->map('extends.field_1_latitude', 'extends.field_1_longitude', '位置');
  214. })->when(2, function (Form $form) { //景区
  215. if ($form->isEditing() && $form->model()->type != 2) {
  216. return;
  217. }
  218. $form->table('extends.field_2_open_time', '开放时间', function (NestedForm $table) {
  219. $table->text('node', '字段1')->placeholder('如:周一至周五');
  220. $table->text('summer', '字段2')->placeholder('如:08:00~19:00');
  221. $table->text('winter', '字段3')->placeholder('如:08:00~18:00');
  222. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  223. $form->table('extends.field_2_project', '包含项目', function (NestedForm $table) {
  224. $table->text('name', '字段1');
  225. $table->text('num', '字段2');
  226. $table->text('price', '字段3');
  227. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  228. $form->text('extends.field_2_name', '景区名');
  229. $form->text('extends.field_2_address', '地址');
  230. $form->map('extends.field_2_latitude', 'extends.field_2_longitude', '位置');
  231. })->when(3, function (Form $form) { //餐厅
  232. if ($form->isEditing() && $form->model()->type != 3) {
  233. return;
  234. }
  235. $form->table('extends.field_3_open_time', '开放时间', function (NestedForm $table) {
  236. $table->text('week', '字段1')->placeholder('如:周一至周五');
  237. $table->text('section', '字段2')->placeholder('如:上午/下午');
  238. $table->text('time', '字段3')->placeholder('如:08:00~18:00');
  239. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  240. $form->table('extends.field_3_package', '包含套餐', function (NestedForm $table) {
  241. $table->text('name', '字段1')->placeholder('如:清蒸鱿鱼');
  242. $table->text('num', '字段2')->placeholder('如:1条');
  243. $table->text('price', '字段3')->placeholder('如:99元');
  244. })->help('第一行数据默认是表头,如:项目名称、数量、额外费用');
  245. $form->text('extends.field_3_name', '餐厅名');
  246. $form->text('extends.field_3_address', '地址');
  247. $form->map('extends.field_3_latitude', 'extends.field_3_longitude', '位置');
  248. })->when(4, function (Form $form) { //车队
  249. if ($form->isEditing() && $form->model()->type != 4) {
  250. return;
  251. }
  252. $form->text('extends.field_4_address', '地址');
  253. $form->map('extends.field_4_latitude', 'extends.field_4_longitude', '位置');
  254. })->when(5, function (Form $form) { //单项
  255. if ($form->isEditing() && $form->model()->type != 5) {
  256. return;
  257. }
  258. $form->text('extends.field_5_address', '地址');
  259. $form->map('extends.field_5_latitude', 'extends.field_5_longitude', '位置');
  260. });
  261. })->saving(function (Form $form) {
  262. //不允许编辑非自己数据
  263. if ($form->isEditing() && $form->model()->supplier_id != Admin::user()->id) {
  264. return $form->response()->error('数据不存在');
  265. }
  266. if ($form->isCreating()) {
  267. if (!Admin::user()->publish_type || !in_array($form->type, Admin::user()->publish_type)) {
  268. return $form->response()->error('对不起,你没有此类产品的发布、编辑权限');
  269. }
  270. } else if ($form->isEditing()) {
  271. $form->type = $form->model()->type; //type不允许编辑
  272. //如果存在未核销的订单不允许编辑
  273. $exists = \App\Models\Order::where('product_id', $form->model()->id)
  274. ->whereIn('status', [OrderStatus::PAID, OrderStatus::PAID_RETAINAGE, OrderStatus::OFFLINE_PAID, OrderStatus::REFUSED_REFUND])
  275. ->exists();
  276. if ($exists) {
  277. return $form->response()->error('该产品还有未核销的订单,不允许编辑');
  278. }
  279. }
  280. //不允许编辑的字段,忽略字段不起作用?
  281. $form->ignore(['id', 'supplier_id', 'sale', 'created_at', 'updated_at', 'deleted_at']);
  282. //null字段转为''
  283. foreach ($form->input() as $k => $v) {
  284. if (is_null($v)) {
  285. $form->$k = '';
  286. }
  287. }
  288. //用户可编辑的状态
  289. $user_status = [ProductStatus::SOLD_OUT, ProductStatus::ON_SALE];
  290. //列表切换上下架按钮
  291. if ($form->isEditing() && !is_null($form->status) && is_null($form->title)) {
  292. if (empty($form->model()->spec->toArray())) {
  293. return $form->response()->error('请先设置产品规格')->refresh();
  294. } else if (in_array($form->model()->status, $user_status)) {
  295. $form->status = $form->status == 1 ? ProductStatus::ON_SALE : ProductStatus::SOLD_OUT;
  296. $form->model()->update(['status' => $form->status]);
  297. return $form->response()->success('更新成功')->refresh();
  298. }
  299. }
  300. //规格处理
  301. if (!$form->spec || !$spec = array_filter($form->spec, fn($v) => !$v['_remove_'])) {
  302. return $form->response()->error('请输入产品规格');
  303. }
  304. $form->hidden(['stock', 'original_price', 'price', 'longitude', 'latitude', 'address']);
  305. //处理库存、市场价、销售价
  306. $form->stock = array_sum(array_column($spec, 'stock'));
  307. $form->original_price = min(array_column($spec, 'original_price'));
  308. $form->price = min(array_column($spec, 'price'));
  309. //经度,纬度,地址
  310. if ($form->type == 0) { //旅游线路用出发地保存
  311. $form->longitude = $form->extends['field_0_departure_place_longitude'] ?? 0;
  312. $form->latitude = $form->extends['field_0_departure_place_latitude'] ?? 0;
  313. $form->address = $form->extends['field_0_departure_place'] ?? '';
  314. } else {
  315. $form->longitude = $form->extends['field_'.$form->type.'_longitude'] ?? 0;
  316. $form->latitude = $form->extends['field_'.$form->type.'_latitude'] ?? 0;
  317. $form->address = $form->extends['field_'.$form->type.'_address'] ?? '';
  318. }
  319. //特殊字段处理
  320. if ($form->isCreating()) {
  321. $form->hidden(['status', 'supplier_id']); //表单没有的字段,必须加上这句才能重置值
  322. $form->supplier_id = Admin::user()->id;
  323. $form->status = ProductStatus::UNAUDITED;
  324. } else if ($form->isEditing()) {
  325. //如果原来是下架或上架状态才允许修改
  326. if (in_array($form->status, $user_status) && in_array($form->model()->status, $user_status)) {
  327. $form->status = $form->status == ProductStatus::ON_SALE ? ProductStatus::ON_SALE : ProductStatus::SOLD_OUT;
  328. } else {
  329. $form->deleteInput('status');
  330. }
  331. }
  332. })->saved(function (Form $form, $result) {
  333. if ($form->isEditing() && $result) {
  334. $ap_ids = AgentProductItem::where('product_id', $form->getKey())->pluck('agent_product_id')->toArray();
  335. if ($ap_ids) {
  336. //修改信息同步信息到代理商产品,注:组合产品不同步
  337. AgentProduct::whereIn('id', $ap_ids)->where('type', 0)->update([
  338. 'title' => $form->title,
  339. 'pictures' => explode(',', $form->pictures),
  340. 'know' => $form->know,
  341. 'content' => $form->content,
  342. ]);
  343. }
  344. $delete_specs = array_filter($form->spec, fn($v) => $v['_remove_'] !== null); //删除的规格
  345. //同步删除代理商规格
  346. AgentProductSpec::whereIn('product_spec_id', array_keys($delete_specs))->delete();
  347. //最新销售价同步到代理商规格
  348. $product_id = $form->getKey();
  349. $agent_product_ids = AgentProduct::where('product_id', $product_id)->pluck('id')->toArray(); //获取当前供应商产品的所有供应商产品ID
  350. $product_specs = ProductSpec::where('product_id', $product_id)->get();
  351. $product_spec_old_ids = $form->model()->spec->pluck('id')->toArray(); //旧的规格ID(不包括新增的)
  352. if ($agent_product_ids && $product_spec_old_ids) {
  353. //将代理商规格价格小于供应商销售价的,设置代理商规格的价格设置为供应商的售价
  354. foreach ($product_specs->whereIn('id', $product_spec_old_ids) as $v) {
  355. AgentProductSpec::whereIn('agent_product_id', $agent_product_ids)
  356. ->where([['product_spec_id', '=', $v->id], ['price', '<', $v->price]])
  357. ->update(['price' => $v->price]);
  358. }
  359. }
  360. /*if ($product_spec_old_ids) {
  361. $product_spec_new_ids = $product_specs->whereNotIn('id', $product_spec_old_ids)->get()->toArray(); //新增的规格
  362. if ($product_spec_new_ids) {
  363. //暂时不处理新增的规格
  364. }
  365. }*/
  366. }
  367. })->deleting(function (Form $form) {
  368. //不允许删除非自己的数据
  369. if (array_filter($form->model()->toArray(), fn($v) => $v['supplier_id'] != Admin::user()->id)) {
  370. return $form->response()->error('数据不存在');
  371. }
  372. });
  373. }
  374. }