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

287 lines
8.9 KiB

4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
  1. <?php
  2. namespace App\Http\Controllers\Api;
  3. use App\Common\ProductStatus;
  4. use App\Http\Controllers\Controller;
  5. use App\Models\Advertising;
  6. use App\Models\AgentProduct;
  7. use App\Models\Product;
  8. use App\Models\UserFav;
  9. use Illuminate\Support\Facades\Storage;
  10. /**
  11. * 代理商产品
  12. * Class AgentProductController
  13. * @package App\Http\Controllers\Api
  14. */
  15. class AgentProductController extends Controller
  16. {
  17. // 代理商产品列表
  18. public function index()
  19. {
  20. $category_id = request()->input('category_id');
  21. $where = [];
  22. if ($category_id) {
  23. $where['category_id'] = $category_id;
  24. }
  25. $list = AgentProduct::list($this->agent_id)->where($where)->orderBy('id', 'DESC')->simplePaginate();
  26. $list = $this->paginatePicAddHost($list);
  27. $list = $this->insertAd($list);
  28. return $this->success($list);
  29. }
  30. //首页搜索框
  31. public function search()
  32. {
  33. $category_id = request()->input('category_id');
  34. $keywords = request()->input('keywords');
  35. $type = request()->input('type', 0);
  36. $by = request()->input('by', 0);
  37. $list = AgentProduct::list($this->agent_id);
  38. if ($category_id) {
  39. $list = $list->where('category_id', $category_id);
  40. }
  41. if ($keywords) {
  42. $list = $list->where('title', 'like', "%$keywords%");
  43. }
  44. $fields = ['id', 'sale', 'updated_at', 'price']; //排序字段 TODO 还有距离排序
  45. $field = $fields[$type] ?? $fields[0];
  46. $by = $by == 0 ? 'desc' : 'asc';
  47. $list = $list->orderBy($field, $by)->simplePaginate();
  48. $list = $this->paginatePicAddHost($list);
  49. return $this->success($list);
  50. }
  51. //旅游线路搜索
  52. public function travel_search()
  53. {
  54. $formData = request()->only(['departure_place', 'destination', 'type', 'by']);
  55. if (empty($formData['departure_place']) && empty($formData['destination'])) {
  56. return $this->error('请输入出发地和目的地');
  57. }
  58. $fields = ['id', 'sale', 'updated_at', 'price']; //排序字段
  59. $field = $fields[$formData['type']] ?? $fields[0];
  60. $by = $formData['by'] == 0 ? 'desc' : 'asc';
  61. $list = AgentProduct::list($this->agent_id)->whereHas('product', function($query) use ($formData) {
  62. //出发地
  63. if (!empty($formData['departure_place'])) {
  64. $query->whereRaw("extends->'$.field_0_departure_place' LIKE ?", ["%{$formData['departure_place']}%"]);
  65. }
  66. //目的地
  67. if (!empty($formData['destination'])) {
  68. $query->whereRaw("extends->'$.field_0_destination' LIKE ?", ["%{$formData['destination']}%"]);
  69. }
  70. })->orderBy($field, $by)->simplePaginate();
  71. $list = $this->paginatePicAddHost($list);
  72. return $this->success($list);
  73. }
  74. // 产品详情
  75. public function show()
  76. {
  77. $id = (int)request()->input('id');
  78. if (AgentProduct::where('agent_id', $this->agent_id)->withTrashed()->count() === 0) { //演示产品用
  79. $where = ['id' => $id, 'status' => ProductStatus::ON_SALE];
  80. } else {
  81. $where = ['id' => $id, 'agent_id' => $this->agent_id, 'status' => ProductStatus::ON_SALE];
  82. }
  83. $agent_product = AgentProduct::with([
  84. 'coupon:tag,agent_product_id',
  85. 'product' => function ($query) {
  86. $query->select(['id', 'type', 'extends', 'diy_form_id'])->with('diyForm', function ($query) {
  87. $query->select(['id', 'name'])->with('fields');
  88. });
  89. },
  90. 'spec' => function($query) {
  91. return $query->has('productSpec')->with('productSpec', function ($query) {
  92. $query->select(['id', 'name', 'date'])->where('date', '>=', date('Y-m-d'))->orderBy('date', 'asc');
  93. });
  94. }
  95. ])
  96. ->whereDoesntHave('agentProductItem', function ($query) {
  97. return $query->whereHas('product', function ($query) {
  98. return $query->where('stock', '<=', 0)->orWhere('status', '<>', ProductStatus::ON_SALE);
  99. });
  100. })
  101. ->where('stock', '>', 0)
  102. ->firstWhere($where);
  103. if (!$agent_product) {
  104. return $this->error('产品已下架或库存不足');
  105. }
  106. $prefix = Storage::disk('public')->url('');
  107. $agent_product->pictures = array_map(fn($item) => ($prefix . $item), $agent_product->pictures);
  108. if ($this->user_id) {
  109. $agent_product->is_collect = UserFav::query()->where(['user_id' => $this->user_id, 'agent_product_id' => $id])->exists() ? 1 : 0; //判断是否收藏
  110. } else {
  111. $agent_product->is_collect = 0;
  112. }
  113. //计算折扣
  114. if ($agent_product->price < $agent_product->original_price) {
  115. $agent_product->cost = round($agent_product->price / $agent_product->original_price * 10, 1);
  116. } else {
  117. $agent_product->cost = '';
  118. }
  119. //处理自定义字段
  120. if (!empty($agent_product->product->diyForm->fields) && !$agent_product->product->diyForm->fields->isEmpty()) {
  121. foreach ($agent_product->product->diyForm->fields as &$v) {
  122. if ($v['type'] == 'checkbox' && is_array($v['options'])) {
  123. $v['options'] = array_map(function ($v2) {
  124. $arr['checked'] = false;
  125. $arr['disabled'] = false;
  126. $arr['name'] = $v2;
  127. return $arr;
  128. }, $v['options']);
  129. }
  130. }
  131. }
  132. //处理规格
  133. if (!$agent_product->spec->isEmpty()) {
  134. $specs = $agent_product->spec->toArray();
  135. //二维数组转一维
  136. $specs = array_map(function ($v) {
  137. if (is_array($v['product_spec'])) {
  138. unset($v['product_spec']['id']);
  139. $v = array_merge($v, $v['product_spec']);
  140. }
  141. unset($v['agent_product_id'], $v['product_spec_id'], $v['product_spec'], $v['deleted_at']);
  142. return $v;
  143. }, $specs);
  144. //去年name和date为空的数组
  145. $specs = array_filter($specs, fn($v) => isset($v['name'], $v['date']));
  146. $names = array_column($specs, 'name');
  147. $names = array_values(array_unique($names));
  148. $result = [];
  149. foreach ($names as $name) {
  150. $list = array_filter($specs, fn($v) => $v['name'] == $name);
  151. $result[] = [
  152. 'name' => $name,
  153. 'date_start' => min(array_column($list, 'date')),
  154. 'date_end' => max(array_column($list, 'date')),
  155. 'original_price' => min(array_column($list, 'original_price')),
  156. 'price' => min(array_column($list, 'price')),
  157. 'list' => array_values($list),
  158. ];
  159. }
  160. unset($agent_product->spec);
  161. $agent_product->spec = $result;
  162. }
  163. unset($agent_product->agent_id, $agent_product->status, $agent_product->deleted_at);
  164. return $this->success($agent_product);
  165. }
  166. // 猜你喜欢
  167. public function guessLike()
  168. {
  169. $list = AgentProduct::list($this->agent_id)->orderBy('id', 'DESC')->simplePaginate();
  170. $list = $this->paginatePicAddHost($list);
  171. $list = $list->toArray();
  172. if (!empty($list['data']) && is_array($list['data'])) {
  173. shuffle($list['data']); //随机乱序
  174. }
  175. $list = $this->insertAd($list);
  176. return $this->success($list);
  177. }
  178. //【我的】页面下方推荐
  179. public function recommendList()
  180. {
  181. $list = AgentProduct::list($this->agent_id)->where(['is_rec' => 1])
  182. ->orderBy('id', 'DESC')->simplePaginate();
  183. $list = $this->paginatePicAddHost($list);
  184. $list = $this->insertAd($list);
  185. return $this->success($list);
  186. }
  187. //人气爆款列表,销量排序
  188. public function hotList()
  189. {
  190. $list = AgentProduct::list($this->agent_id)
  191. ->orderBy('sale', 'DESC')->orderBy('id', 'DESC')->simplePaginate();
  192. $list = $this->paginatePicAddHost($list);
  193. $list = $this->insertAd($list);
  194. return $this->success($list);
  195. }
  196. //分页列表产品图片加域名
  197. private function paginatePicAddHost($list)
  198. {
  199. //如果是新入驻代理商没有数据,且没有删除过的数据,则显示最早的几条
  200. if ($list->isEmpty()) {
  201. if (AgentProduct::where('agent_id', $this->agent_id)->withTrashed()->count() === 0 && request('page', 1) <= 2) { //只获取2页数据
  202. $list = AgentProduct::list($this->agent_id)->orWhere([['status', '=', ProductStatus::ON_SALE], ['price', '>', 500]])->orderBy('id')->simplePaginate();
  203. } else {
  204. return $list; //因为只获取2页数据,不return可能会导入下面的代码出错
  205. }
  206. }
  207. $prefix = Storage::disk('public')->url('');
  208. foreach ($list->items() as $k=>&$v) {
  209. $v->pictures = array_map(function($item) use ($prefix) {
  210. return strpos($item, $prefix) === false ? $prefix . $item : $item;
  211. }, $v->pictures);
  212. }
  213. return $list;
  214. }
  215. //插入瀑布流广告
  216. private function insertAd($list)
  217. {
  218. //插入瀑布流广告,分别在第8个和第16个插入,同时需要考虑到分页。当所有瀑布流广告插入完之后,再次循环插入
  219. if (is_object($list) && method_exists($list, 'toArray')) {
  220. $list = $list->toArray();
  221. }
  222. $ad_total = Advertising::where(['agent_id' => $this->agent_id, 'status' => 1, 'display' => 2])->count();
  223. if ($list['data'] && $ad_total > 0) {
  224. $page = (int)request()->input('page');
  225. $start = ($page ? $page - 1 : 0) * 2 % $ad_total;
  226. $ad = Advertising::where(['agent_id' => $this->agent_id, 'status' => 1, 'display' => 2])
  227. ->orderBy('sort')->orderBy('id', 'DESC')
  228. ->offset($start)->limit(2)->get(['title', 'picture', 'type', 'url'])->toArray();
  229. $prefix = Storage::disk('public')->url('');
  230. foreach ($ad as $k => &$v) {
  231. $v['is_ad'] = true;
  232. $v['picture'] = $prefix . $v['picture'];
  233. //每隔8个插入广告
  234. $temp = 8 * ($k+1);
  235. if (!empty($list['data'][$temp - 1]) && !empty($ad[$k])) {
  236. array_splice($list['data'], $temp + $k, 0, [$ad[$k]]);
  237. }
  238. }
  239. }
  240. return $list;
  241. }
  242. }