截流自动化的商城平台
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.

ShopContentLogic.php 36KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965
  1. <?php
  2. namespace app\api\logic;
  3. use app\common\basics\Logic;
  4. use app\common\enum\GoodsEnum;
  5. use app\common\enum\ShopAdEnum;
  6. use app\common\enum\ShopEnum;
  7. use app\common\logic\QrCodeLogic;
  8. use app\common\model\dev\DevRegion;
  9. use app\common\model\shop\Shop as shopModel;
  10. use app\common\model\shop\ShopAd;
  11. use app\common\model\shop\ShopGoods as shopGoodsModel;
  12. use app\common\server\ConfigServer;
  13. use app\common\server\UrlServer;
  14. use app\common\model\goods\Goods;
  15. use app\common\model\shop\Shop;
  16. use app\common\model\shop\ShopFollow;
  17. use app\common\model\content\ClosureCategory;
  18. use app\common\model\content\Closure;
  19. use app\common\model\content\EquiCategory as EquiCategoryModel;
  20. use app\common\model\shop\ShopHkLog;
  21. use think\facade\Db;
  22. use think\facade\Event;
  23. class ShopContentLogic extends Logic
  24. {
  25. /*
  26. * 获取机器秘钥
  27. */
  28. public static function codeInfo($code='',$mcode='')
  29. {
  30. if(empty($code) || empty($mcode)){
  31. $data = [
  32. 'status' => 105,
  33. 'msg' => "参数不正确,不能为空!"
  34. ];
  35. return $data;
  36. }
  37. $where = [
  38. "del" => 0,
  39. "is_show" => 1,
  40. "pid" => 0, //必须是电脑
  41. "code" => $code
  42. ];
  43. $item = EquiCategoryModel::where($where)->find();
  44. //判断商家服务
  45. if(!empty($item)) {
  46. //判断商户
  47. $shop = shopModel::where('id', $item['shop_id'])->find();
  48. if(empty($shop)){
  49. $data = [
  50. 'status' => 110,
  51. 'msg' => "商户信息不存在!"
  52. ];
  53. return $data;
  54. }
  55. //不需要转数组
  56. //echo $shop['id'];
  57. //$shop = $shop->toArray();
  58. //var_dump($shop);die;
  59. //判断 商家状态 is_freeze
  60. if((int)$shop['is_freeze'] === 1){
  61. $data = [
  62. 'status' => 111,
  63. 'msg' => "商家冻结中!"
  64. ];
  65. return $data;
  66. }
  67. //判断 营业状态 is_run
  68. if((int)$shop['is_run'] === 0){
  69. $data = [
  70. 'status' => 112,
  71. 'msg' => "商家暂停营业中!"
  72. ];
  73. return $data;
  74. }
  75. //模型调用出来的 与 直接查询字段出来的值 不一样
  76. $expire_time = strtotime($shop['expire_time']);
  77. //dump($expire_time);
  78. //echo $expire_time;die;
  79. /***
  80. * 原先这是 先判断是否到期 未到期 不扣次数 已到期 判断剩余数量
  81. *
  82. if($expire_time < time()){
  83. //判断到期时间
  84. $data = [
  85. 'status' => 107,
  86. 'msg' => "商户服务当前套餐到期,到期时间".date('Y-m-d H:i:s',$expire_time)."!"
  87. ];
  88. //当商家已到期
  89. if((int)$shop['hksy_count'] < 10){
  90. //判断剩余数量 hksy_count
  91. $data = [
  92. 'status' => 113,
  93. 'msg' => "商户服务可评论次数不足,剩余".$shop['hksy_count']."条,请充值后再使用 或 开通商户包年包月服务!"
  94. ];
  95. return $data;
  96. }
  97. //可评论次数大于10 即使到期 也继续执行
  98. $is_gi = 1;
  99. }else{
  100. $is_gi = 0;
  101. }
  102. //未到期 不需要扣次数 继续执行即可
  103. * */
  104. //判断类型
  105. if((int)$shop['hksy_type'] === 0){
  106. //包月
  107. if($expire_time < time()){
  108. //判断到期时间
  109. $data = [
  110. 'status' => 107,
  111. 'msg' => "商户服务当前套餐到期,到期时间".date('Y-m-d H:i:s',$expire_time)."!"
  112. ];
  113. return $data;
  114. }
  115. //包月 未过期 = 前端会按包月模式 运行
  116. $is_gi = 0;
  117. }else{
  118. //按量
  119. if((int)$shop['hksy_count'] < 5){
  120. //判断剩余数量 hksy_count
  121. $data = [
  122. 'status' => 113,
  123. 'msg' => "商户服务可评论次数不足,剩余".$shop['hksy_count']."条,请充值次数后再使用!"
  124. ];
  125. return $data;
  126. }
  127. //按量 统一按过期处理 = 前端会按按量模式 运行
  128. $is_gi = 1;
  129. }
  130. }
  131. if(empty($item))
  132. {
  133. $data = [
  134. 'status' => 101,
  135. 'msg' => "秘钥错误,找不到设备!"
  136. ];
  137. }else if(empty($item['machine_code']))
  138. {
  139. //验证秘钥是否合法
  140. $res = self::getCodeIs($item['shop_id'],$code);
  141. if($res === false){
  142. $data = [
  143. 'status' => 106,
  144. 'msg' => "超出套餐服务电脑数量!"
  145. ];
  146. }else {
  147. //未绑定机器码 那么进行绑定
  148. $row = EquiCategoryModel::where(['id' => $item['id']])->update([
  149. "machine_code" => $mcode,
  150. ]);
  151. if ($row == 1) {
  152. $data = [
  153. 'status' => 100,
  154. 'msg' => "绑定机器码成功!",
  155. 'is_gi' => $is_gi,
  156. 'count' => $shop['hksy_count']
  157. ];
  158. } else {
  159. $data = [
  160. 'status' => 102,
  161. 'msg' => "绑定机器码失败!"
  162. ];
  163. }
  164. }
  165. }else{
  166. //验证秘钥是否合法
  167. $res = self::getCodeIs($item['shop_id'],$code);
  168. if($res === false){
  169. $data = [
  170. 'status' => 106,
  171. 'msg' => "超出套餐服务电脑数量!"
  172. ];
  173. }else {
  174. //var_dump($mcode);
  175. //var_dump($item['machine_code']);
  176. //机器码已存在
  177. if($item['machine_code'] != $mcode){
  178. //机器码不一致
  179. $data = [
  180. 'status' => 103,
  181. 'msg' => "该秘钥已绑定其他设备,操作失败!"
  182. ];
  183. }else{
  184. $data = [
  185. 'status' => 100,
  186. 'msg' => "已绑定该机器码!",
  187. 'is_gi' => $is_gi,
  188. 'count' => $shop['hksy_count']
  189. ];
  190. }
  191. }
  192. }
  193. return $data;
  194. }
  195. public static function getCodeIs($shop_id,$code)
  196. {
  197. //首先获取合法的电脑数量是多少
  198. $tid = shopModel::where('id',$shop_id)->value('tid');
  199. if ((int)$tid === 0) {
  200. //未配置
  201. $count_pc = 1;
  202. $count_mobile = 3;
  203. $count_run = 3;
  204. } else if ((int)$tid > 0) {
  205. //查找配置
  206. $one = shopGoodsModel::where('id', $tid)->find();
  207. $count_pc = $one['pc_num'];
  208. $count_mobile = $one['mobile_num'];
  209. $count_run = $one['run_num'];
  210. } else {
  211. //其他
  212. $count_pc = 1;
  213. $count_mobile = 3;
  214. $count_run = 3;
  215. }
  216. //调用合法的秘钥是哪些
  217. $where = [
  218. "del" => 0, //未删除
  219. //"is_show" => 1,
  220. "pid" => 0, //必须是电脑
  221. "shop_id" => $shop_id
  222. ];
  223. $code_list = EquiCategoryModel::where($where)->order('id asc')->limit($count_pc)->column('code');
  224. //var_dump($code);
  225. if(in_array($code,$code_list)){
  226. return true;
  227. }
  228. return false;
  229. }
  230. /*
  231. * getComputer
  232. * 获取电脑 根据code
  233. */
  234. public static function getComputer($code){
  235. if(empty($code)){
  236. $data = [
  237. 'status' => 105,
  238. 'msg' => "参数不正确,不能为空!"
  239. ];
  240. return $data;
  241. }
  242. $where = [
  243. "del" => 0,
  244. "is_show" => 1,
  245. "pid" => 0, //必须是电脑
  246. "code" => $code
  247. ];
  248. $item = EquiCategoryModel::where($where)->find();
  249. if(empty($item))
  250. {
  251. $data = [
  252. 'status' => 101,
  253. 'msg' => "秘钥错误,找不到设备!"
  254. ];
  255. }else{
  256. $data = [
  257. 'status' => 100,
  258. 'msg' => "设备存在!",
  259. 'item' => $item
  260. ];
  261. }
  262. return $data;
  263. }
  264. /*
  265. * 获取手机配置
  266. */
  267. public static function getDefaultMobile($data){
  268. //判断是否绑定配置
  269. if(empty($data['pz'])){
  270. $data = [
  271. 'status' => 101,
  272. 'msg' => "请对该设备绑定默认配置!"
  273. ];
  274. }else{
  275. $pz = json_decode($data['pz'],true);
  276. //获取小红书appid
  277. $app_id = ClosureCategory::where([
  278. 'name' => '小红书',
  279. 'shop_id' => $data['shop_id']
  280. ])->value('id');
  281. if(empty($app_id)){
  282. $data = [
  283. 'status' => 102,
  284. 'msg' => "找不到名为小红书的appID!"
  285. ];
  286. }else{
  287. //读取默认配置
  288. foreach ($pz as $k=>$v){
  289. if((int)$v['cid'] === (int)$app_id){
  290. //读取具体配置对象 $v['val']
  291. $item = Closure::where(['id'=>$v['val']])->find();
  292. if(empty($item['json_data'])){
  293. $data = [
  294. 'status' => 103,
  295. 'msg' => "找不到配置对象记录!"
  296. ];
  297. }else{
  298. $data = [
  299. 'status' => 100,
  300. 'msg' => "存在默认配置!",
  301. 'data' => json_decode($item['json_data'],true)
  302. ];
  303. $data['data']['ver'] = $v['ver'];
  304. }
  305. }
  306. }
  307. }
  308. }
  309. //查询默认配置
  310. return $data;
  311. }
  312. /*
  313. * 获取手机列表
  314. */
  315. public static function getMobile($data)
  316. {
  317. //限制手机数量 首先获取合法的手机数量是多少
  318. $tid = shopModel::where('id',$data['shop_id'])->value('tid');
  319. if ((int)$tid === 0) {
  320. //未配置
  321. $count_pc = 1;
  322. $count_mobile = 3;
  323. $count_run = 3;
  324. } else if ((int)$tid > 0) {
  325. //查找配置
  326. $one = shopGoodsModel::where('id', $tid)->find();
  327. $count_pc = $one['pc_num'];
  328. $count_mobile = $one['mobile_num'];
  329. $count_run = $one['run_num'];
  330. } else {
  331. //其他
  332. $count_pc = 1;
  333. $count_mobile = 3;
  334. $count_run = 3;
  335. }
  336. //只调用允许数量 按id排序即可 后台不可更改排序
  337. $list = EquiCategoryModel::where(['pid'=>$data['id'],'is_show'=>1,'del'=>0,'is_user'=>1])
  338. ->order('id asc')
  339. ->limit($count_run)
  340. ->select();
  341. if(empty($list)){
  342. $data = [
  343. 'status' => 100,
  344. 'msg' => "找不到手机设备,全部使用默认配置!"
  345. ];
  346. }else{
  347. //获取小红书appid
  348. $app_id = ClosureCategory::where([
  349. 'name' => '小红书',
  350. 'shop_id' => $data['shop_id']
  351. ])->value('id');
  352. if(empty($app_id)){
  353. $data = [
  354. 'status' => 102,
  355. 'msg' => "找不到名为小红书的appID!"
  356. ];
  357. }else{
  358. $pz_list = [];
  359. $ids = [];
  360. //dump($list->toArray());die;
  361. foreach ($list as $k=>$v){
  362. //没有配置指定 就使用默认的配置
  363. if(!empty($v['pz'])){
  364. $pz = json_decode($v['pz'],true);
  365. foreach ($pz as $kk=>$vv){
  366. //小红书
  367. if((int)$vv['cid'] === (int)$app_id){
  368. $pz_list[] = $v['name'];
  369. $ids[] = $vv['val'];
  370. $ver[] = $vv['ver']; //版本号
  371. }
  372. }
  373. }
  374. }
  375. if(empty($ids)){
  376. $data = [
  377. 'status' => 100,
  378. 'msg' => "不存在手机配置,使用默认配置"
  379. ];
  380. }else{
  381. $data = [
  382. 'status' => 100,
  383. 'data' => $pz_list,
  384. 'ids' => $ids,
  385. 'ver' => $ver
  386. ];
  387. }
  388. }
  389. }
  390. return $data;
  391. }
  392. /*
  393. * 获取手机配置
  394. */
  395. public static function getMobileConfig($data,$ids,$ver)
  396. {
  397. $res = [];
  398. if(!empty($ids)){
  399. $list = Closure::where('id','in',$ids)->select();
  400. //用配置id 作 key 如果配置id相同 就会覆盖
  401. /* foreach ($list as $k=>$v){
  402. if(!empty($v['json_data'])){
  403. $res[$data[(int)$v['id']]] = json_decode($v['json_data'],true);
  404. $res[$data[(int)$v['id']]]['ver'] = $ver[(int)$v['id']]; //追加版本号
  405. }
  406. }*/
  407. //重写
  408. $pz_list = [];
  409. foreach ($list as $k=>$v){
  410. if(!empty($v['json_data'])) {
  411. $pz_list[$v['id']] = $v;
  412. }
  413. }
  414. foreach ($ids as $k=>$v){
  415. if(!empty($pz_list[$v])){
  416. $res[$data[$k]] = json_decode($pz_list[$v]['json_data'],true);
  417. $res[$data[$k]]['ver'] = $ver[$k]; //追加版本号
  418. }
  419. }
  420. }
  421. return $res;
  422. }
  423. /**
  424. * 获取店铺信息
  425. */
  426. public static function getShopInfo($shopId, $userId, $params = [])
  427. {
  428. // 记录统计信息(访问商铺用户量)
  429. Event::listen('ShopStat', 'app\common\listener\ShopStat');
  430. event('ShopStat', $shopId);
  431. $where = [
  432. 'del' => 0,
  433. 'id' => $shopId
  434. ];
  435. $field = [
  436. 'id', 'create_time', 'name', 'logo', 'background',
  437. 'type', 'score', 'star', 'intro',
  438. 'visited_num', 'cover', 'banner', 'is_freeze',
  439. 'is_run', 'expire_time',
  440. 'province_id', 'city_id', 'district_id', 'address',
  441. 'run_start_time', 'run_end_time', 'weekdays',
  442. ];
  443. $shop = Shop::field($field)
  444. ->where($where)
  445. ->append([ 'type_desc', 'is_expire' ])
  446. ->findOrEmpty();
  447. if($shop->isEmpty()) {
  448. return [];
  449. }else{
  450. $shop = $shop->toArray();
  451. }
  452. //
  453. $shop['logo'] = UrlServer::getFileUrl($shop['logo'] ? : ShopEnum::DEFAULT_LOGO);
  454. $shop['background'] = UrlServer::getFileUrl($shop['background'] ? : ShopEnum::DEFAULT_BG);
  455. $shop['cover'] = UrlServer::getFileUrl($shop['cover'] ? :ShopEnum::DEFAULT_COVER);
  456. $shop['banner'] = UrlServer::getFileUrl($shop['banner'] ? : ShopEnum::DEFAULT_BANNER);
  457. $shop['run_start_time'] = $shop['run_start_time'] ? date('H:i:s', $shop['run_start_time']) : '';
  458. $shop['run_end_time'] = $shop['run_end_time'] ? date('H:i:s', $shop['run_end_time']) : '';
  459. $shop['province'] = DevRegion::getAreaName($shop['province_id']);
  460. $shop['city'] = DevRegion::getAreaName($shop['city_id']);
  461. $shop['district'] = DevRegion::getAreaName($shop['district_id']);
  462. $shop['qr_code'] = (new QrCodeLogic)->shopQrCode($shop['id'], $params['terminal'] ?? '');
  463. // 在售商品
  464. // 销售中商品:未删除/审核通过/已上架
  465. $onSaleWhere = [
  466. ['del', '=', GoodsEnum::DEL_NORMAL], // 未删除
  467. ['status', '=', GoodsEnum::STATUS_SHELVES], // 上架中
  468. ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK], // 审核通过
  469. ];
  470. $shop['on_sale_count'] = Goods::where($onSaleWhere)->where('shop_id', $shopId)->count();
  471. // 店铺推荐商品
  472. $shop['goods_list'] = Goods::field('id,image,name,min_price,market_price')
  473. ->where($onSaleWhere)
  474. ->where([
  475. 'shop_id' => $shop['id'],
  476. 'is_recommend' => 1, // 推荐商品
  477. ])
  478. ->limit(9)
  479. ->select()
  480. ->toArray();
  481. // 用户是否关注店铺
  482. $shop['shop_follow_status'] = 0;
  483. if($userId) { // 用户已登录
  484. $shopFollow = ShopFollow::where(['user_id'=>$userId, 'shop_id'=>$shopId])->findOrEmpty();
  485. if(!$shopFollow->isEmpty()) {
  486. $shop['shop_follow_status'] = $shopFollow['status'];
  487. }
  488. }
  489. $shop['follow_num'] = ShopFollow::where(['shop_id' => $shopId,'status' => 1])->count('id');
  490. $image = ConfigServer::get('shop_customer_service', 'image', '', $shopId);
  491. $shop['customer_image'] = $image ? UrlServer::getFileUrl($image) : '';
  492. $shop['customer_wechat'] = ConfigServer::get('shop_customer_service', 'wechat', '', $shopId);
  493. $shop['customer_phone'] = ConfigServer::get('shop_customer_service', 'phone', '', $shopId);
  494. // 店铺广告
  495. $adWhere = [
  496. [ 'shop_id', '=', $shopId ],
  497. [ 'status', '=', 1 ],
  498. ];
  499. $shop['ad'] = [
  500. 'pc' => ShopAd::where($adWhere)->where('terminal', ShopAdEnum::TERMINAL_PC)->append([ 'link_path', 'link_query' ])->order('sort desc,id desc')->select()->toArray(),
  501. 'mobile' => ShopAd::where($adWhere)->where('terminal', ShopAdEnum::TERMINAL_MOBILE)->append([ 'link_path', 'link_query' ])->order('sort desc,id desc')->select()->toArray(),
  502. ];
  503. return $shop;
  504. }
  505. /**
  506. * 店铺列表
  507. */
  508. public static function getShopList($get)
  509. {
  510. $where = [
  511. ['is_freeze', '=', 0], // 未冻结
  512. ['del', '=', 0], // 未删除
  513. ['is_run', '=', 1], // 未暂停营业
  514. ];
  515. // 店铺名称
  516. if(isset($get['name']) && !empty($get['name'])) {
  517. $where[] = ['name', 'like', '%'. trim($get['name']. '%')];
  518. }
  519. // 主营类目
  520. if(isset($get['shop_cate_id']) && !empty($get['shop_cate_id'])) {
  521. $where[] = ['cid', '=', $get['shop_cate_id']];
  522. }
  523. $order = [
  524. 'weight' => 'asc',
  525. 'score' => 'desc',
  526. 'id' => 'desc'
  527. ];
  528. $list = Shop::field('id,type,name,logo,background,visited_num,cover,banner')
  529. ->where($where)
  530. // 无限期 或 未到期
  531. ->whereRaw('expire_time =0 OR expire_time > '. time())
  532. ->order($order)
  533. ->page($get['page_no'], $get['page_size'])
  534. ->select()
  535. ->toArray();
  536. $count = Shop::where($where)
  537. // 无限期 或 未到期
  538. ->whereRaw('expire_time =0 OR expire_time > '. time())
  539. ->count();
  540. $onSaleWhere = [
  541. ['del', '=', GoodsEnum::DEL_NORMAL], // 未删除
  542. ['status', '=', GoodsEnum::STATUS_SHELVES], // 上架中
  543. ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK], // 审核通过
  544. ];
  545. foreach($list as &$shop) {
  546. $shop['goods_list'] = Goods::field('id,image,name,min_price,market_price')
  547. ->where($onSaleWhere)
  548. ->where([
  549. 'shop_id' => $shop['id'],
  550. ])
  551. ->limit(10)
  552. ->select()
  553. ->toArray();
  554. $shop['on_sale_goods'] = count($shop['goods_list']);
  555. // logo及背景图
  556. $shop['logo'] = $shop['logo'] ? UrlServer::getFileUrl($shop['logo']) : UrlServer::getFileUrl(ShopEnum::DEFAULT_LOGO);
  557. $shop['background'] = $shop['background'] ? UrlServer::getFileUrl($shop['background']) : UrlServer::getFileUrl(ShopEnum::DEFAULT_BG);
  558. $shop['cover'] = $shop['cover'] ? UrlServer::getFileUrl($shop['cover']) : UrlServer::getFileUrl(ShopEnum::DEFAULT_COVER);
  559. $shop['banner'] = $shop['banner'] ? UrlServer::getFileUrl($shop['banner']) : '';
  560. }
  561. $more = is_more($count, $get['page_no'], $get['page_size']);
  562. $data = [
  563. 'list' => $list,
  564. 'count' => $count,
  565. 'more' => $more,
  566. 'page_no' => $get['page_no'],
  567. 'page_isze' => $get['page_size']
  568. ];
  569. return $data;
  570. }
  571. /**
  572. * @notes 附近店铺列表
  573. * @param $get
  574. * @return array
  575. * @throws \think\db\exception\DataNotFoundException
  576. * @throws \think\db\exception\DbException
  577. * @throws \think\db\exception\ModelNotFoundException
  578. * @author ljj
  579. * @date 2022/9/20 4:29 下午
  580. */
  581. public static function getNearbyShops($get)
  582. {
  583. $where = [
  584. ['is_freeze', '=', 0], // 未冻结
  585. ['del', '=', 0], // 未删除
  586. ['is_run', '=', 1], // 未暂停营业
  587. ['city_id', '=', $get['city_id']],
  588. ];
  589. // 店铺名称
  590. if(isset($get['name']) && !empty($get['name'])) {
  591. $where[] = ['name', 'like', '%'. trim($get['name']. '%')];
  592. }
  593. // 主营类目
  594. if(isset($get['shop_cate_id']) && !empty($get['shop_cate_id'])) {
  595. $where[] = ['cid', '=', $get['shop_cate_id']];
  596. }
  597. $city = DevRegion::where('id',$get['city_id'])->field('db09_lng,db09_lat')->findOrEmpty()->toArray();
  598. $list = Shop::field('id,name,logo,background,visited_num,cover,banner,st_distance_sphere(point('.$city['db09_lng'].','.$city['db09_lat'].'),point(longitude, latitude)) as distance')
  599. ->where($where)
  600. // 无限期 或 未到期
  601. ->whereRaw('expire_time =0 OR expire_time > '. time())
  602. ->order('distance asc')
  603. ->page($get['page_no'], $get['page_size'])
  604. ->select()
  605. ->toArray();
  606. $count = Shop::where($where)
  607. // 无限期 或 未到期
  608. ->whereRaw('expire_time =0 OR expire_time > '. time())
  609. ->count();
  610. $onSaleWhere = [
  611. ['del', '=', GoodsEnum::DEL_NORMAL], // 未删除
  612. ['status', '=', GoodsEnum::STATUS_SHELVES], // 上架中
  613. ['audit_status', '=', GoodsEnum::AUDIT_STATUS_OK], // 审核通过
  614. ];
  615. foreach($list as &$shop) {
  616. $shop['goods_list'] = Goods::field('id,image,name,min_price,market_price')
  617. ->where($onSaleWhere)
  618. ->where([
  619. 'shop_id' => $shop['id'],
  620. ])
  621. ->select()
  622. ->toArray();
  623. $shop['on_sale_goods'] = count($shop['goods_list']);
  624. // logo及背景图
  625. $shop['logo'] = $shop['logo'] ? UrlServer::getFileUrl($shop['logo']) : UrlServer::getFileUrl(ShopEnum::DEFAULT_LOGO);
  626. $shop['background'] = $shop['background'] ? UrlServer::getFileUrl($shop['background']) : UrlServer::getFileUrl(ShopEnum::DEFAULT_BG);
  627. $shop['cover'] = $shop['cover'] ? UrlServer::getFileUrl($shop['cover']) : UrlServer::getFileUrl(ShopEnum::DEFAULT_COVER);
  628. $shop['banner'] = $shop['banner'] ? UrlServer::getFileUrl($shop['banner']) : '';
  629. //转换距离单位
  630. if ($shop['distance'] < 1000) {
  631. $shop['distance'] = round($shop['distance']).'m';
  632. }else {
  633. $shop['distance'] = round($shop['distance'] / 1000,2).'km';
  634. }
  635. }
  636. $more = is_more($count, $get['page_no'], $get['page_size']);
  637. $data = [
  638. 'list' => $list,
  639. 'count' => $count,
  640. 'more' => $more,
  641. 'page_no' => $get['page_no'],
  642. 'page_isze' => $get['page_size']
  643. ];
  644. return $data;
  645. }
  646. public static function getXhsDefaultAppuim()
  647. {
  648. $data = [
  649. '默认' => '',
  650. "搜索列表" => '//androidx.recyclerview.widget.RecyclerView[@class="androidx.recyclerview.widget.RecyclerView"]/android.widget.FrameLayout[@class="android.widget.FrameLayout"]/android.widget.RelativeLayout[@class="android.widget.RelativeLayout"]/android.widget.TextView[@class="android.widget.TextView" and @text!=""]',
  651. "推荐列表" => '//androidx.recyclerview.widget.RecyclerView[@class="androidx.recyclerview.widget.RecyclerView"]/android.widget.LinearLayout[@class="android.widget.LinearLayout" and @content-desc!=""]',
  652. '推荐列表元素' => '//android.widget.LinearLayout[@content-desc="变量1"]',
  653. '搜索列表元素' => '//android.widget.TextView[@text="变量1"]',
  654. "私信发送按钮" => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.TextView[contains(@text,"发送")]',
  655. '判断当前是否视频' => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.Button[@content-desc="分享"]',
  656. '视频评论数' => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.Button[contains(@content-desc,"评论")]',
  657. '视频分享按钮' => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.Button[@content-desc="分享"]',
  658. '视频获取账号名称' => '//android.widget.Button[contains(@content-desc,"作者") and @class="android.widget.Button"]/android.widget.LinearLayout[@class="android.widget.LinearLayout"]/android.widget.TextView[@class="android.widget.TextView"]',
  659. "视频获取点赞按钮" => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.Button[contains(@content-desc,"点赞")]',
  660. "视频获取收藏按钮" => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.Button[contains(@content-desc,"收藏")]',
  661. "视频详情返回按钮" => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.Button[contains(@content-desc,"返回")]'
  662. ];
  663. return $data;
  664. }
  665. public static function getXhsDefaultU2()
  666. {
  667. $data = [
  668. '默认' => ''
  669. ];
  670. return $data;
  671. }
  672. public static function getXhsAppuim_1()
  673. {
  674. //相对位置 /parent::*[1]/
  675. //p推荐页 p搜索页 p搜索列表页_1 p搜索列表页_2 p图文页 p视频页 p旧版视频页 p用户主页 p私信页
  676. $data = [
  677. "默认" => '',
  678. //页面唯一标识 start
  679. "p推荐页" => '//android.view.ViewGroup[@class="android.view.ViewGroup" and @index="0"]/android.widget.RelativeLayout[@class="android.widget.RelativeLayout" and @index="2" and @content-desc="发布"]',
  680. "p搜索页" => '//android.widget.LinearLayout[@class="android.widget.LinearLayout"]/android.widget.ImageView[@class="android.widget.ImageView" and @content-desc="拍照搜索"]',
  681. "p搜索列表页_1" => '//android.widget.Button[@class="android.widget.Button"]/android.widget.TextView[@text="筛选"]',
  682. "p搜索列表页_2" => '//android.widget.TextView[@class="android.widget.TextView" and @text="全部"]/parent::*[1]/android.widget.ImageView[@class="android.widget.ImageView"]',
  683. "p图文页" => '//android.widget.RelativeLayout[@class="android.widget.RelativeLayout"]/android.widget.Button[@content-desc="分享"]',
  684. "p视频页" => '//android.view.ViewGroup[@class="android.view.ViewGroup"]//android.widget.ImageView[@class="android.widget.ImageView" and @content-desc="搜索"]',
  685. "p旧版视频页" => '//androidx.recyclerview.widget.RecyclerView[@class="androidx.recyclerview.widget.RecyclerView"]//android.view.ViewGroup[@class="android.view.ViewGroup"]//android.widget.ImageView[@class="android.widget.ImageView" and @index="5"]',
  686. "p用户主页" => '//android.widget.LinearLayout[@class="android.widget.LinearLayout"]/android.widget.TextView[contains(@text,"小红书号")]',
  687. "p私信页" => '//android.widget.FrameLayout[@class="android.widget.FrameLayout"]/android.widget.EditText[@class="android.widget.EditText"]',
  688. //页面唯一标识 end
  689. //页面多版本 start
  690. "搜索列表" => '//androidx.recyclerview.widget.RecyclerView[@class="androidx.recyclerview.widget.RecyclerView"]/android.widget.FrameLayout[@class="android.widget.FrameLayout"]/android.widget.RelativeLayout[@class="android.widget.RelativeLayout"]/android.widget.TextView[@class="android.widget.TextView" and @text!=""]',
  691. "推荐列表" => '//androidx.recyclerview.widget.RecyclerView[@class="androidx.recyclerview.widget.RecyclerView"]/android.widget.FrameLayout[@class="android.widget.FrameLayout" and @content-desc!=""]',
  692. '推荐列表元素' => '//android.widget.FrameLayout[@content-desc="变量1"]',
  693. '搜索列表元素' => '//android.widget.TextView[@text="变量1"]',
  694. "私信发送按钮" => '//android.widget.RelativeLayout[@class="android.widget.RelativeLayout"]/android.widget.TextView[contains(@text,"发送")]',
  695. '判断当前是否视频' => '//android.widget.FrameLayout[@class="android.widget.FrameLayout" and @index="1"]/android.widget.LinearLayout[@index="0"]/android.widget.LinearLayout[@index="3"]/android.widget.TextView[@class="android.widget.TextView"]',
  696. '视频评论数' => '//android.widget.FrameLayout[@class="android.widget.FrameLayout" and @index="1"]/android.widget.LinearLayout[@index="0"]/android.widget.LinearLayout[@index="1"]/android.widget.TextView[@class="android.widget.TextView"]',
  697. '视频分享按钮' => '//android.widget.FrameLayout[@class="android.widget.FrameLayout" and @index="1"]/android.widget.LinearLayout[@index="0"]/android.widget.LinearLayout[@index="4"]/android.widget.ImageView[@class="android.widget.ImageView"]',
  698. '视频获取账号名称' => '//android.widget.FrameLayout[@class="android.widget.FrameLayout"]/android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.LinearLayout[@class="android.widget.LinearLayout"]/android.widget.LinearLayout[@class="android.widget.LinearLayout"]/android.widget.TextView[@class="android.widget.TextView"]',
  699. "视频获取点赞按钮" => '//android.widget.FrameLayout[@class="android.widget.FrameLayout" and @index="1"]/android.widget.LinearLayout[@index="0"]/android.widget.LinearLayout[@index="0"]/android.widget.TextView[@class="android.widget.TextView"]',
  700. "视频获取收藏按钮" => '//android.widget.FrameLayout[@class="android.widget.FrameLayout" and @index="1"]/android.widget.LinearLayout[@index="0"]/android.widget.LinearLayout[@index="2"]/android.widget.TextView[@class="android.widget.TextView"]',
  701. "视频详情返回按钮" => '//android.view.ViewGroup[@class="android.view.ViewGroup"]/android.widget.ImageView[contains(@content-desc,"返回")]'
  702. //页面多版本 end
  703. ];
  704. return $data;
  705. }
  706. public static function getXhsU2_1()
  707. {
  708. $data = [
  709. '默认' => ''
  710. ];
  711. return $data;
  712. }
  713. /**
  714. * 加密 ZMH 2025-03-18
  715. * @param $data
  716. * @return bool|string
  717. */
  718. public static function encrypt($data)
  719. {
  720. $key = 'ABCDEFGHIJKLMNOP'; //16位
  721. $vi = '0102030405060708'; //16位
  722. if (is_array($data)) {
  723. $str = json_encode($data,JSON_UNESCAPED_UNICODE); //讲数组转为字符串
  724. } else if (is_string($data)) {
  725. $str = $data;
  726. } else {
  727. return false;
  728. }
  729. $sign = openssl_encrypt($str, 'AES-128-CBC', $key, OPENSSL_RAW_DATA, $vi);
  730. $sign = base64_encode($sign);
  731. return $sign;
  732. }
  733. /**
  734. * 获取商家信息 ZMH 2025-03-18
  735. * @param $code
  736. * @return model|bool
  737. */
  738. public static function getShopByCode($code)
  739. {
  740. if (empty($code)) {
  741. self::$error = "设备编码不能为空";
  742. self::$errCode = 2;
  743. return false;
  744. }
  745. $where = [
  746. "del" => 0,
  747. "is_show" => 1,
  748. "pid" => 0, //必须是电脑
  749. "code" => $code
  750. ];
  751. $item = EquiCategoryModel::where($where)->find();
  752. if (empty($item)) {
  753. self::$error = "没有找到对应的设备";
  754. self::$errCode = 3;
  755. return false;
  756. }
  757. $shop = ShopModel::where("id", $item["shop_id"])->find();
  758. if (empty($shop)) {
  759. self::$error = "用户信息不存在";
  760. self::$errCode = 4;
  761. return false;
  762. }
  763. return $shop;
  764. }
  765. /**
  766. * 获取剩余数量 ZMH 2025-03-18
  767. * @param $get
  768. * @return bool|array
  769. */
  770. public static function getHKSYCount($get) {
  771. $code = $get["code"] ?? '';
  772. $shop = self::getShopByCode($code);
  773. if (empty($shop)) {
  774. return false;
  775. }
  776. return ['count' => $shop->hksy_count, 'expire_time' => $shop->expire_time];
  777. }
  778. /**
  779. * 扣减剩余数量 ZMH 2025-03-18
  780. * @param $get
  781. * @return bool|number
  782. */
  783. public static function subHKSYCount($post) {
  784. $code = $post["code"] ?? '';
  785. $mcode = $post["mcode"] ?? '0000';
  786. //$count = $post["count"] ?? 1; //默认就是1
  787. $count = 1;
  788. $remark = $post["remark"] ?? '';
  789. if ($count <= 0 || empty($remark) || empty($mcode)) {
  790. self::$errCode = 1;
  791. self::$error = "参数错误";
  792. return false;
  793. }
  794. //没有必要验证机器码 后续需要对接口进行加密传输,解密操作,验证来源 防篡改,使用token也不安全,秘钥和机器码和token都会暴露;
  795. /*$where001 = [
  796. "del" => 0,
  797. "is_show" => 1,
  798. "pid" => 0, //必须是电脑
  799. "code" => $code
  800. ];
  801. $item = EquiCategoryModel::where($where001)->find();*/
  802. $shop = self::getShopByCode($code);
  803. if (empty($shop)) {
  804. self::$errCode = 9;
  805. self::$error = "商家不存在";
  806. return false;
  807. }
  808. if((int)$shop->is_freeze === 1){
  809. self::$errCode = 7;
  810. self::$error = "商家冻结中";
  811. return false;
  812. }
  813. //判断 营业状态 is_run
  814. if((int)$shop->is_run === 0){
  815. self::$errCode = 8;
  816. self::$error = "商家暂停营业中";
  817. return false;
  818. }
  819. /*
  820. // 判断是否过期
  821. $expire_time = $shop->expire_time;
  822. //echo $expire_time;die;
  823. $expire_time = strtotime($expire_time);
  824. if (time() < $expire_time) {
  825. //未过期 不进行扣除
  826. return true;
  827. }*/
  828. //判断类型
  829. if((int)$shop->hksy_type === 0){
  830. self::$errCode = 10;
  831. self::$error = "商家类型是包月,不允许使用次数!";
  832. return false;
  833. }
  834. if ($shop->hksy_count < $count) {
  835. self::$errCode = 5;
  836. self::$error = "剩余数量不足";
  837. return false;
  838. }
  839. Db::startTrans();
  840. try {
  841. //code...
  842. $old_hksy_count = $shop->hksy_count;
  843. $shop->hksy_count = $shop->hksy_count - $count;
  844. $shop->save();
  845. // 添加数量记录
  846. $hk = new ShopHkLog();
  847. $hk->shop_id = $shop['id'];
  848. $hk->source_type = 101;
  849. $hk->change_count = 0 - $count;
  850. $hk->left_count = $shop->hksy_count;
  851. $hk->remark = $remark;
  852. $hk->save();
  853. Db::commit();
  854. } catch (\Throwable $th) {
  855. //throw $th;
  856. Db::rollback();
  857. self::$errCode = 6;
  858. self::$error = "扣除失败";
  859. return false;
  860. }
  861. return $shop->hksy_count;
  862. }
  863. }