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

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