UserOrderService.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. <?php
  2. namespace app\service;
  3. use app\common\model\Area;
  4. use app\common\model\SysConfig;
  5. use app\common\model\User;
  6. use app\common\model\UserCoupon;
  7. use app\common\model\UserOrder;
  8. class UserOrderService{
  9. protected $order;
  10. protected $user;
  11. protected $coupon;
  12. /** @var UserCoupon */
  13. protected $user_coupon;
  14. protected $config=[];
  15. protected $cage_spec;
  16. /** @var Area */
  17. protected $from_area;
  18. /** @var Area */
  19. protected $to_area;
  20. /** @var array */
  21. protected $pet_category;
  22. /** @var UserOrderLog */
  23. protected $log;
  24. public function setLog(UserOrderLog $log)
  25. {
  26. $this->log = $log;
  27. return $this;
  28. }
  29. /**
  30. * @param UserCoupon $coupon
  31. */
  32. public function setCoupon($coupon)
  33. {
  34. if($coupon) {
  35. $c = $this->getUser()->coupon()->use()->find($coupon);
  36. if (!$c) {
  37. throw_user('优惠券不存在');
  38. }
  39. $this->user_coupon = $c;
  40. }
  41. return $this;
  42. }
  43. public function __construct()
  44. {
  45. bcscale(2);
  46. $this->config=SysConfig::look('price_config',config('process'));
  47. }
  48. /**
  49. * @return User
  50. */
  51. public function getUser()
  52. {
  53. return $this->user;
  54. }
  55. /**
  56. * @param mixed $user
  57. */
  58. public function setUser(User $user)
  59. {
  60. $this->user = $user;
  61. $this->coupon = $user->coupon()->use()->select();
  62. return $this;
  63. }
  64. public function checkArea($area){
  65. $city=tm()->getLocation($area['longitude'],$area['latitude']);
  66. return $city['city'];
  67. }
  68. protected function area(){
  69. $from_area=$this->checkArea($this->order['from_addr']);
  70. $to_area=$this->checkArea($this->order['to_addr']);
  71. $this->order['from_city']=$from_area;
  72. $this->order['to_city']=$to_area;
  73. $database_from_area=Area::city()->where('name',$this->order['from_city'])->find();
  74. if(!$database_from_area){
  75. throw_user("数据库未找到取宠城市{$this->order['from_city']}");
  76. }
  77. $this->from_area=$database_from_area;
  78. $database_to_area=Area::city()->where('name',$this->order['to_city'])->find();
  79. if(!$database_to_area){
  80. throw_user("数据库未找到送宠城市{$this->order['to_city']}");
  81. }
  82. $this->to_area=$database_to_area;
  83. $this->order['from_longitude']=$this->order['from_addr']['longitude'];
  84. $this->order['from_latitude']=$this->order['from_addr']['latitude'];
  85. $this->order['to_longitude']=$this->order['to_addr']['longitude'];
  86. $this->order['to_latitude']=$this->order['to_addr']['latitude'];
  87. $this->recordLog("[{$this->order['from_city']}]-[{$this->order['to_city']}]");
  88. #检查空运有没有设置机场坐标
  89. $noneAir=!$this->hasAir($this->order['pick_up']);
  90. if($this->order['freight']==UserOrder::FREIGHT_AIR && $noneAir){
  91. throw_user('未找到机场,无法选择此方式运输');
  92. }
  93. }
  94. protected function distance(){
  95. if($this->order['freight']=='air') {
  96. $this->order['distance'] = ll_distance(
  97. $this->order['from_longitude'],
  98. $this->order['from_latitude'],
  99. $this->order['to_longitude'],
  100. $this->order['to_latitude']
  101. );
  102. }else{
  103. $this->order['distance']=tm()->getDistance(
  104. [$this->order['from_longitude'],$this->order['from_latitude']],
  105. [$this->order['to_longitude'],$this->order['to_latitude']]
  106. );
  107. }
  108. $this->recordLog("距离:{$this->order['distance']}km");
  109. }
  110. public function import(&$data){
  111. list($data['agree_date'],$data['agree_time'])=explode(' ',$data['agree_time']);
  112. $this->pet_category=Pet::category()[$data['pet_category']]??null;
  113. $data['pet_category']=$this->pet_category['name']??throw_user("宠物类目不存在");
  114. $this->cage_spec=Pet::spec()[$data['spec']]??throw_user('宠具规格不存在');
  115. $data['spec']=$this->cage_spec['name'];
  116. if(!empty($data['protect_id'])) {
  117. $protect=Pet::protect()[$data['protect_id']];
  118. $data['protect_amount'] = $protect['price'];
  119. $data['protect_max'] = $protect['max'];
  120. }
  121. else{
  122. #防止报错
  123. $data['protect_id']=null;
  124. }
  125. #没有优惠券置为null
  126. if(empty($data['coupon_id'])){
  127. $data['coupon_id']=null;
  128. }
  129. #没有选配送方式置为null
  130. if(!isset($data['freight'])){
  131. $data['freight']=null;
  132. }
  133. $data['coupon_amount']=0;
  134. $data['discount_amount']=0;
  135. $data['total_amount']=0;
  136. $data['real_amount']=0;
  137. $this->order=$data;
  138. $this->area();
  139. $this->distance();
  140. return $this;
  141. }
  142. function prepare(){
  143. $items=[];
  144. $total_amount=0;
  145. $freight_amount=0;
  146. $freights=UserOrder::$freights;
  147. if(!$this->pet_category['allow_air']){
  148. unset($freights[UserOrder::FREIGHT_AIR]);
  149. if($this->order['freight']==UserOrder::FREIGHT_AIR){
  150. throw_user('该宠物不支持空运');
  151. }
  152. }
  153. foreach ($freights as $type=>$_){
  154. $item=[
  155. 'type'=>$type,//类型
  156. 'price'=>0,#实付
  157. 'total_price'=>0,#单个类型支付总价
  158. //'coupon'=>[],//可用的优惠券列表
  159. ];
  160. $item['total_price']=$this->{$type}();
  161. $item['price']=$item['total_price'];
  162. $items[$type]=$item;
  163. if($type==$this->order['freight']){
  164. $total_amount=$freight_amount=bcadd($total_amount,$item['total_price']);
  165. }
  166. }
  167. $protect=Pet::protect()[$this->order['protect_id']]??[];
  168. $protect_price=$protect['price']??0;
  169. $total_amount=bcadd($total_amount,$protect['price']??0);
  170. #用户折扣
  171. $user_level_discount_money=$freight_amount;
  172. if($this->order['freight']==UserOrder::FREIGHT_AIR){
  173. $user_level_discount_money=$this->order['first_air_amount'];
  174. }
  175. $level_amount=bcmul($this->user->level_discount,$user_level_discount_money);
  176. $this->order['level_amount']=$level_amount;
  177. $this->recordLog("运费:{$freight_amount},折扣:{$this->user->level_discount},用户折扣:{$level_amount}");
  178. $freight_amount=bcsub($freight_amount,$level_amount);
  179. #券金额
  180. if($this->user_coupon){
  181. if(!$this->user_coupon->fit($freight_amount)){
  182. throw_user('优惠券不满足使用条件');
  183. }
  184. if($this->user->is_vip && $this->order['freight']==UserOrder::FREIGHT_AIR){
  185. throw_user('您已是会员,当空运时无法使用优惠券折扣');
  186. }
  187. $this->order['coupon_amount']=$this->user_coupon->amount($freight_amount);
  188. $this->recordLog("优惠券:".json_encode($this->user_coupon,JSON_UNESCAPED_UNICODE));
  189. }
  190. $freight_amount=bcsub($freight_amount,$this->order['coupon_amount']);
  191. #优惠总金额
  192. $this->order['discount_amount']=bcadd($this->order['coupon_amount'],$this->order['level_amount']);
  193. $this->order['real_amount']=bcadd($freight_amount,$protect_price);
  194. $this->order['total_amount']=$total_amount;
  195. $this->order['freights']=$items;
  196. #可用的优惠券
  197. $coupon=[];
  198. if($this->order['freight']!=UserOrder::FREIGHT_AIR) {
  199. foreach ($this->coupon as $_coupon) {
  200. if ($_coupon->fit($this->order['total_amount'])) {
  201. $coupon[] = $_coupon;
  202. }
  203. }
  204. }
  205. $this->order['allow_coupons']=$coupon;
  206. return $this->order;
  207. }
  208. public function fast(){
  209. #距离
  210. $distance=$this->order['distance']??401;
  211. $start_kilo=$this->config['fast']['start_kilo'];
  212. $start_amount=$this->config['fast']['start_amount'];
  213. if($distance<=$start_kilo){
  214. $distance_price=$start_amount;
  215. }else{
  216. $distance_price=0;
  217. foreach ($this->config['fast']['detail'] as $k=>$detail){
  218. if($distance>$detail['start_kilo'] && $distance<=$detail['end_kilo']){
  219. $calc=str_replace('距离数',$distance,$detail['amount']);
  220. $distance_price=eval("return $calc;");
  221. break;
  222. }
  223. }
  224. }
  225. #笼子百分比算出运费
  226. $cage_float=$this->cage_spec['per']/100;
  227. $cage_float_price=bcmul($distance_price,$cage_float);
  228. #宠具+运费
  229. $cage_price=$this->cage();
  230. $total = bcadd($cage_float_price,$cage_price);
  231. #件数,多只按百分比算优惠,按笼子数量
  232. $num=$this->order['piece'];
  233. $per=$this->config['fast']['out_one_per'];
  234. $out_float=bcdiv($per,100);
  235. $num_price=bcmul(bcmul($out_float,$cage_float_price),$num-1);
  236. $total=bcadd($total,$num_price);
  237. $this->recordLog("快车-起步距离:{$start_kilo},起步价:{$start_amount},距离总价:{$distance_price},宠具百分比:{$cage_float}," .
  238. "笼子百分比算出的运费:{$cage_float_price},宠具价格:{$cage_price},".
  239. "只数:{$num},超出百分比:{$this->config['fast']['out_one_per']},超出一只总价:{$num_price}".
  240. "总价:{$total}"
  241. );
  242. return $total;
  243. }
  244. protected function air(){
  245. $total=0;
  246. #基础价格
  247. $base_get_price=$this->from_area['first_air_amount'];
  248. $base_send_price=0;
  249. $start_amount=$this->config['air']['start_amount']??null;
  250. if(is_null($base_get_price)){
  251. if(is_null($start_amount)){
  252. throw_user('未设置出港费');
  253. }
  254. $base_get_price=$start_amount;
  255. }
  256. $base_price=bcadd($base_get_price,$base_send_price);
  257. $total=bcadd($total,$base_price);
  258. #多只出港费半价,按宠物数量
  259. $num=$this->order['num'];
  260. $out_one_per=$this->config['air']['out_one_per'];
  261. $out_one_float=$out_one_per/100;
  262. $first_out_price=bcmul(bcmul($base_price,$out_one_float),$num-1);
  263. $total=bcadd($total,$first_out_price);
  264. $this->recordLog("空运-出港费设置比例:{$out_one_per}%,额外出港费:{$first_out_price}");
  265. $this->order['first_air_amount']=bcadd($first_out_price,$base_price);
  266. #距离
  267. #上门接
  268. $get_price=0;
  269. $send_price=0;
  270. $get_send_price=0;
  271. if($this->order['pick_up']==1 && $this->hasAir(1)) {
  272. $get_price=$this->pickup($this->from_area['air_longitude'],$this->from_area['air_latitude'],$this->order['from_longitude'],$this->order['from_latitude']);
  273. $total=bcadd($total,$get_price);
  274. $get_send_price=$get_price;
  275. }
  276. #上门送
  277. elseif ($this->order['pick_up']==2 && $this->hasAir(2)){
  278. $send_price=$this->pickup($this->to_area['air_longitude'],$this->to_area['air_latitude'],$this->order['to_longitude'],$this->order['to_latitude']);
  279. $total=bcadd($total,$send_price);
  280. $get_send_price=$send_price;
  281. }
  282. #上门接送
  283. elseif ($this->order['pick_up']==4 && $this->hasAir(4)){
  284. $get_price=$this->pickup($this->from_area['air_longitude'],$this->from_area['air_latitude'],$this->order['from_longitude'],$this->order['from_latitude']);
  285. $total=bcadd($total,$get_price);
  286. $send_price=$this->pickup($this->to_area['air_longitude'],$this->to_area['air_latitude'],$this->order['to_longitude'],$this->order['to_latitude']);
  287. $total=bcadd($total,$send_price);
  288. $get_send_price=bcadd($get_price,$send_price);
  289. }
  290. #机场代提费
  291. $take_price=0;
  292. if(in_array($this->order['pick_up'],[2,4])){
  293. //$take_price=$this->config['air']['send']['default_amount'];
  294. if($this->to_area['second_air_amount']){
  295. $take_price=$this->to_area['second_air_amount'];
  296. }
  297. $total=bcadd($total,$take_price);
  298. }
  299. #宠具
  300. $cage_price=$this->cage();
  301. $total=bcadd($total,$cage_price);
  302. #重量
  303. $weight=$this->order['weight'];
  304. $avgWeight=bcdiv($weight,$num);
  305. $base_weight=$this->config['air']['weight']['base_weight'];
  306. $out_weight_price=0;
  307. $out_price=0;
  308. if($avgWeight>$base_weight){
  309. $out_weight=bcsub($weight,$base_weight);
  310. $out_weight_price=$this->config['air']['weight']['out_amount'];
  311. $out_price=bcmul($out_weight,$out_weight_price);
  312. $total=bcadd($total,$out_price);
  313. }
  314. $this->recordLog("空运-从地区取基础价格(出港费):{$base_price},默认起步价:{$start_amount},上门接:{$get_price},上门送:{$send_price},".
  315. "上门接送:{$get_send_price},|地区{$this->to_area['second_air_amount']},".
  316. "宠具价格:{$cage_price},重量:{$weight},平均重量:{$avgWeight},基础重量:{$base_weight},超出后每KG价格:{$out_weight_price},重量总价:{$out_price},".
  317. "总价:{$total}"
  318. );
  319. return $total;
  320. }
  321. protected function special(){
  322. #距离
  323. $distance=$this->order['distance'];
  324. $start_kilo=$this->config['special']['start_kilo'];
  325. $start_amount=$this->config['special']['start_amount'];
  326. if($distance<=$start_kilo){
  327. $distance_price=$start_amount;
  328. }else{
  329. $distance_price=bcmul(bcsub($distance,$start_kilo),$this->config['special']['out_amount']);
  330. $distance_price=bcadd($start_amount,$distance_price);
  331. }
  332. #笼具
  333. $cage_price=$this->cage();
  334. $price=bcadd($distance_price,$cage_price);
  335. $this->recordLog("专车-起步距离:{$start_kilo},起步价:{$start_amount},距离价格:{$distance_price},宠具价格:{$cage_price},总价:{$price}");
  336. return $price;
  337. }
  338. protected function cage(){
  339. if($this->order['has_cage']){
  340. return 0;
  341. }
  342. $cage_num=$this->order['piece'];
  343. return bcmul($cage_num,$this->cage_spec['amount']);
  344. }
  345. protected function pickup($from_longitude,$from_latitude,$to_longitude,$to_latitude){
  346. $distance=ll_distance($from_longitude,$from_latitude,$to_longitude,$to_latitude);
  347. $this->recordLog("上门接送:$from_longitude,$from_latitude,$to_longitude,{$to_latitude},距离:{$distance}");
  348. $get_default_price = $this->config['air']['pickup']['start_amount'];
  349. $get_default_kilo = $this->config['air']['pickup']['start_distance'];
  350. if($distance<=$get_default_kilo){
  351. $get_price=$get_default_price;
  352. }else{
  353. $get_price=$get_default_price;
  354. $max_kilo=$this->config['air']['pickup']['out_kilo_num'];
  355. $every_kilo_amount=$this->config['air']['pickup']['every_kilo_amount'];
  356. $out_kilo_amount=$this->config['air']['pickup']['out_kilo_amount'];
  357. if($distance<=$max_kilo){
  358. $get_price=bcadd($get_price,bcmul($every_kilo_amount,$distance));
  359. }else{
  360. $get_price=bcadd($get_price,bcmul($every_kilo_amount,$max_kilo));
  361. $get_price=bcadd($get_price,bcmul($out_kilo_amount,bcsub($distance,$max_kilo)));
  362. }
  363. }
  364. return $get_price;
  365. }
  366. protected function hasAir($pickup){
  367. switch ($pickup){
  368. case 0:
  369. return true;
  370. case 1:
  371. return $this->from_area->has_air_coordinate;
  372. case 2:
  373. return $this->to_area->has_air_coordinate;
  374. case 4:
  375. return $this->from_area->has_air_coordinate && $this->to_area->has_air_coordinate;
  376. default:
  377. throw_user('上门接送类型有误');
  378. }
  379. throw_user('上门接送类型有误');
  380. }
  381. protected function recordLog($content){
  382. $this->log && $this->log->setContent($content);
  383. }
  384. public function __destruct()
  385. {
  386. $this->recordLog("orderInfo:".json_encode($this->order,JSON_UNESCAPED_UNICODE));
  387. $this->log && $this->log->write();
  388. }
  389. }