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

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