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

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