Area.php 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. <?php
  2. namespace app\common\model;
  3. use think\Cache;
  4. use think\db\Query;
  5. use think\Model;
  6. /**
  7. * 地区数据模型
  8. * @property bool has_air_coordinate
  9. * @method static static shi($name=null)
  10. * @method static static area()
  11. */
  12. class Area extends Model
  13. {
  14. /**
  15. * 根据经纬度获取当前地区信息
  16. *
  17. * @param string $lng 经度
  18. * @param string $lat 纬度
  19. * @return Area 城市信息
  20. */
  21. public static function getAreaFromLngLat($lng, $lat, $level = 3)
  22. {
  23. $namearr = [1 => 'geo:province', 2 => 'geo:city', 3 => 'geo:district'];
  24. $rangearr = [1 => 15000, 2 => 1000, 3 => 200];
  25. $geoname = isset($namearr[$level]) ? $namearr[$level] : $namearr[3];
  26. $georange = isset($rangearr[$level]) ? $rangearr[$level] : $rangearr[3];
  27. // 读取范围内的ID
  28. $redis = Cache::store('redis')->handler();
  29. $georadiuslist = [];
  30. if (method_exists($redis, 'georadius')) {
  31. $georadiuslist = $redis->georadius($geoname, $lng, $lat, $georange, 'km', ['WITHDIST', 'COUNT' => 5, 'ASC']);
  32. }
  33. if ($georadiuslist) {
  34. list($id, $distance) = $georadiuslist[0];
  35. }
  36. $id = isset($id) && $id ? $id : 3;
  37. return self::get($id);
  38. }
  39. /**
  40. * 根据经纬度获取省份
  41. *
  42. * @param string $lng 经度
  43. * @param string $lat 纬度
  44. * @return Area
  45. */
  46. public static function getProvinceFromLngLat($lng, $lat)
  47. {
  48. $provincedata = null;
  49. $citydata = self::getCityFromLngLat($lng, $lat);
  50. if ($citydata) {
  51. $provincedata = self::get($citydata['pid']);
  52. }
  53. return $provincedata;
  54. }
  55. /**
  56. * 根据经纬度获取城市
  57. *
  58. * @param string $lng 经度
  59. * @param string $lat 纬度
  60. * @return Area
  61. */
  62. public static function getCityFromLngLat($lng, $lat)
  63. {
  64. $citydata = null;
  65. $districtdata = self::getDistrictFromLngLat($lng, $lat);
  66. if ($districtdata) {
  67. $citydata = self::get($districtdata['pid']);
  68. }
  69. return $citydata;
  70. }
  71. /**
  72. * 根据经纬度获取地区
  73. *
  74. * @param string $lng 经度
  75. * @param string $lat 纬度
  76. * @return Area
  77. */
  78. public static function getDistrictFromLngLat($lng, $lat)
  79. {
  80. $districtdata = self::getAreaFromLngLat($lng, $lat, 3);
  81. return $districtdata;
  82. }
  83. public function scopeLevel(Query $query,$level){
  84. $query->where('level',$level);
  85. }
  86. public function scopePro(Query $query){
  87. $query->level(1);
  88. }
  89. public function scopeShi(Query $query,$name=null){
  90. $query->level(2);
  91. if($name!==null){
  92. $query->where('name|shortname','like',"%{$name}%");
  93. }
  94. }
  95. public function scopeArea(Query $query){
  96. $query->level(3);
  97. }
  98. public function province(){
  99. return $this->hasMany(User::class,'province_id');
  100. }
  101. public function city(){
  102. return $this->hasMany(User::class,'city_id');
  103. }
  104. public function county(){
  105. return $this->hasMany(User::class,'county_id');
  106. }
  107. public function getMergenameAttr($a){
  108. return str_replace(',','',str_replace('中国,','',$a));
  109. }
  110. public static function getIdByName($name,$level=null){
  111. $map=[];
  112. if($level){
  113. $map['level']=$level;
  114. }
  115. return self::where('name|shortname','like',"%{$name}%")->where($map)->value('id');
  116. }
  117. public static function getModelByName($name,$level=null){
  118. $map=[];
  119. if($level){
  120. $map['level']=$level;
  121. }
  122. return Cache::remember("getModelByName_{$name}_{$level}",function ()use ($name,$map){
  123. return self::where('name|shortname','like',"%{$name}%")->where($map)->find();
  124. },0);
  125. }
  126. public static function getTreeId($id){
  127. if(is_numeric($id)) {
  128. $county = self::where('id', $id)->area()->find();
  129. }else{
  130. $county= self::where('name|shortname',$id)->find();
  131. }
  132. if(!$county){
  133. throw_user('区县不存在');
  134. }
  135. $cityId=$county['pid'];
  136. $provinceId=self::where('id',$cityId)->value('pid');
  137. if(!$provinceId){
  138. throw_user('省不存在');
  139. }
  140. return [$provinceId,$cityId,$county['id']];
  141. }
  142. public static function getNameString($id,$sp=''){
  143. if(!$id){
  144. return '';
  145. }
  146. $cacheName='area_'.serialize($id).md5($sp);
  147. $res=Cache::get($cacheName);
  148. if(!$res){
  149. $res=implode($sp,self::whereIn('id',$id)->column('name'));
  150. Cache::set($cacheName,$res,0);
  151. }
  152. return $res;
  153. }
  154. public static function shouldSend($county_id){
  155. $config=config('site.disable_send_province');
  156. $config=explode(',',$config)?:[];
  157. $provinceId=self::getTreeId($county_id)[0];
  158. if(in_array($provinceId,$config)){
  159. throw_user('此区域无法购买此产品');
  160. }
  161. }
  162. }