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

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