detail.vue 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628
  1. <template>
  2. <view class="content">
  3. <block v-if="loading">
  4. <view class="skeleton">
  5. <view class="skeleton-line max animate" style="height: 200upx;"></view>
  6. <view class="skeleton-line animate"></view>
  7. <view class="skeleton-line min animate"></view>
  8. <view class="skeleton-line min animate"></view>
  9. <view class="skeleton-line max animate" style="height: 200upx;"></view>
  10. <view class="skeleton-line max animate" style="height: 200upx;"></view>
  11. </view>
  12. </block>
  13. <block v-else>
  14. <view class="home-banner" v-if="bannerData.length>0">
  15. <swiper class="swiper" circular :indicator-dots="indicatorDots" :style="{ height: swiperHeight + 'px' }" :current="currIndex" @change="changeSwiper" :autoplay="false" :interval="2000" :duration="200">
  16. <swiper-item class="swiper-item" v-for="(item,idex) in bannerData" :key="idex">
  17. <image :src="item.url" :id="'wrap' + idex" mode="widthFix"></image>
  18. </swiper-item>
  19. </swiper>
  20. </view>
  21. <view class="goods-top" v-if="goodsData">
  22. <view class="price">
  23. <view class="sale">{{$dialog.disFormat(goodsData.price,goodsData.line_price)}}折</view>
  24. <view class="price-sale">{{ goodsData.price?$dialog.formatMoney(goodsData.price):'0.00'}}</view>
  25. <view class="line-price">{{ goodsData.line_price ? $dialog.formatMoney(goodsData.line_price) : '0.00'}}</view>
  26. <view class="desc-num">已售{{goodsData.sale_stock}}</view>
  27. </view>
  28. <view class="title">{{goodsData.product_name}}</view>
  29. <view class="goods-rule" @click="openRule(1)">
  30. 购买须知<view class="right">
  31. <span class="mix-item" v-if="goodsData.use_date_type==1">
  32. 自购买之日起{{ goodsData.day_duration }}天有效
  33. </span>
  34. <span class="mix-item" v-if="goodsData.use_date_type==2">
  35. 有效期:{{ $dialog.dateFormat(goodsData.use_time_start,'yyyy-MM-dd') }}至{{ $dialog.dateFormat(goodsData.use_time_end,'yyyy-MM-dd') }}
  36. </span>
  37. <image src="/static/image/right-black.png"></image></view>
  38. </view>
  39. <view class="goods-safe">
  40. <view class="safe-item"><image src="/static/image/check.png"></image>过期退</view>
  41. <view class="safe-item"><image src="/static/image/check.png"></image>随时退</view>
  42. </view>
  43. </view>
  44. <view class="goods-store" v-if="goodsData.poi">
  45. <view class="title">适用门店</view>
  46. <view class="store-info">
  47. <view class="store-left">
  48. <view class="name">{{goodsData.poi?goodsData.poi.poi_name:'-'}}</view>
  49. <view class="address">{{goodsData.poi?goodsData.poi.poi_address:'-'}}</view>
  50. </view>
  51. <view class="store-right">
  52. <view class="right-item" @click="toMap">
  53. <image src="/static/image/amap.png"></image>
  54. <view class="desc">导航</view>
  55. </view>
  56. </view>
  57. </view>
  58. </view>
  59. <view class="service-popup popup-white" v-if="specs.length > 0">
  60. <view class="title">团购套餐</view>
  61. <div class="group-body" v-for="(item,indx) in specs" :key="indx">
  62. <div class="body-title">
  63. {{item.name}}
  64. <span v-if="item.num == 0">全部可选</span>
  65. <span v-else>{{item.list.length}}选{{item.num}}</span>
  66. </div>
  67. <div class="body-item">
  68. <div class="body-item-li" v-for="(its,idx) in item.list" :key="idx">
  69. {{its.name}}
  70. <div class="item-right">
  71. <span class="desc">({{its.number}}份)</span>
  72. <span>¥ {{ its.price }}</span>
  73. </div>
  74. </div>
  75. </div>
  76. </div>
  77. </view>
  78. <view class="service-popup popup-white" v-if="goodsData">
  79. <view class="title">购买须知</view>
  80. <view class="rule-list">
  81. <view class="rule-item">
  82. <view class="icon"><image src="/static/image/rule-date.png"></image></view>
  83. <view class="item-top">
  84. <view class="name">可用日期</view>
  85. <view class="desc">
  86. <span class="mix-item" v-if="goodsData.use_date_type==1">
  87. 自购买之日起{{ goodsData.day_duration }}天有效
  88. </span>
  89. <span class="mix-item" v-if="goodsData.use_date_type==2">
  90. 有效期:{{ $dialog.dateFormat(goodsData.use_time_start,'yyyy-MM-dd') }}至{{ $dialog.dateFormat(goodsData.use_time_end,'yyyy-MM-dd') }}
  91. </span>
  92. </view>
  93. </view>
  94. </view>
  95. <view class="rule-item">
  96. <view class="icon"><image src="/static/image/rule-time.png"></image></view>
  97. <view class="item-top">
  98. <view class="name">可用时间</view>
  99. <view class="desc" v-if="goodsData.can_no_use_date==1">商家营业时间可用</view>
  100. </view>
  101. </view>
  102. <view class="rule-item">
  103. <view class="icon"><image src="/static/image/rule-rule.png"></image></view>
  104. <view class="item-top">
  105. <view class="name">预约规则</view>
  106. <view class="desc" v-if="goodsData.booking_type==1">无需预约,高峰期可能需要排队</view>
  107. <view class="desc" v-if="goodsData.booking_type==2">
  108. 需要提前<span>{{ goodsData.booking_date }}</span>{{ goodsData.booking_unit==1?'天':'小时' }}预约
  109. </view>
  110. </view>
  111. </view>
  112. <view class="rule-item">
  113. <view class="icon"><image src="/static/image/rule-refund.png"></image></view>
  114. <view class="item-top">
  115. <view class="name">退款政策<view class="tips">随时退/过期退</view></view>
  116. <view class="desc">随时退款:未核销的商品支持随时申请退款;过期自动退:商品若在有效时间内未核销使用,到期之后系统将自动发起退款</view>
  117. </view>
  118. </view>
  119. </view>
  120. </view>
  121. <view class="service-popup popup-white" v-if="goodsData">
  122. <view class="rule-list">
  123. <view class="rule-content" v-html="goodsData.content"></view>
  124. </view>
  125. </view>
  126. <view class="line-footer"></view>
  127. <view class="footer">
  128. <view class="footer-op">
  129. <view class="footer-left">
  130. <view class="item" @click="toHome"><image src="/static/image/goods/home.png"></image>首页</view>
  131. <view class="item" @click="showService(1)"><image src="/static/image/goods/service.png"></image>客服</view>
  132. <view class="item" @click="toOrder"><image src="/static/image/goods/order.png"></image>订单</view>
  133. </view>
  134. <view class="footer-right" @click="buyOrder(1)" v-if="userToken">立即购买 {{ goodsData?$dialog.formatMoney(goodsData.price):'0.00'}}</view>
  135. <button class="footer-right" open-type="getPhoneNumber" @getphonenumber="toLogin" v-else>登录下单 {{ goodsData?$dialog.formatMoney(goodsData.price):'0.00'}}</button>
  136. </view>
  137. <view class="footer-safe"></view>
  138. </view>
  139. </block>
  140. <uni-popup ref="rulepopup" type="bottom" border-radius="10px 10px 0 0" background-color="#ffffff">
  141. <view class="service-popup popup-height" v-if="goodsData">
  142. <view class="title" @click="openRule(2)">购买须知 <view class="close"><image src="/static/image/close.png"></image></view></view>
  143. <view class="rule-list">
  144. <view class="rule-item">
  145. <view class="icon"><image src="/static/image/rule-date.png"></image></view>
  146. <view class="item-top">
  147. <view class="name">可用日期</view>
  148. <view class="desc">
  149. <span class="mix-item" v-if="goodsData.use_date_type==1">
  150. 自购买之日起{{ goodsData.day_duration }}天有效
  151. </span>
  152. <span class="mix-item" v-if="goodsData.use_date_type==2">
  153. 有效期:{{ $dialog.dateFormat(goodsData.use_time_start,'yyyy-MM-dd') }}至{{ $dialog.dateFormat(goodsData.use_time_end,'yyyy-MM-dd') }}
  154. </span>
  155. </view>
  156. </view>
  157. </view>
  158. <view class="rule-item">
  159. <view class="icon"><image src="/static/image/rule-time.png"></image></view>
  160. <view class="item-top">
  161. <view class="name">可用时间</view>
  162. <view class="desc" v-if="goodsData.can_no_use_date==1">商家营业时间可用</view>
  163. </view>
  164. </view>
  165. <view class="rule-item">
  166. <view class="icon"><image src="/static/image/rule-rule.png"></image></view>
  167. <view class="item-top">
  168. <view class="name">预约规则</view>
  169. <view class="desc" v-if="goodsData.booking_type==1">无需预约,高峰期可能需要排队</view>
  170. <view class="desc" v-if="goodsData.booking_type==2">
  171. 需要提前<span>{{ goodsData.booking_date }}</span>{{ goodsData.booking_unit==1?'天':'小时' }}预约
  172. </view>
  173. </view>
  174. </view>
  175. <view class="rule-item">
  176. <view class="icon"><image src="/static/image/rule-refund.png"></image></view>
  177. <view class="item-top">
  178. <view class="name">退款政策<view class="tips">随时退/过期退</view></view>
  179. <view class="desc">随时退款:未核销的商品支持随时申请退款;过期自动退:商品若在有效时间内未核销使用,到期之后系统将自动发起退款</view>
  180. </view>
  181. </view>
  182. </view>
  183. </view>
  184. </uni-popup>
  185. <uni-popup ref="maskpopup" type="bottom" border-radius="10px 10px 0 0" background-color="#ffffff" :mask-click="false">
  186. <view class="kefu-popup">
  187. <view class="top">
  188. <view class="item">
  189. 客服时间:{{storeData.start_at}}-{{storeData.end_at}}
  190. </view>
  191. <view class="item">
  192. <button class="btn" @click="$dialog.jumpUri('/pages/service/chat/chat?poi='+storeData.poi_id+'&source=goods&goods='+goodsId,1);showService(2)">在线咨询商家</button>
  193. </view>
  194. <view class="item" v-if="storeData.service_mobile" @click="callMobile(storeData.service_mobile)">商家客服电话</view>
  195. </view>
  196. <view class="foot" @click="showService(2)">取消</view>
  197. </view>
  198. </uni-popup>
  199. <uni-popup ref="buypopup" type="bottom" border-radius="10px 10px 0 0" background-color="#ffffff" :mask-click="false">
  200. <view class="service-popup popup-flex popup-height" v-if="goodsData">
  201. <view class="buy-godos-top">
  202. <view class="goods-top-info">
  203. <view class="img"><image :src="buyGoods.img" mode="widthFix"></image></view>
  204. <view class="price-body">
  205. <view class="price">
  206. <view class="price-sale">{{ buyGoods.price?$dialog.formatMoney(buyGoods.price):'0.00'}}</view>
  207. <view class="line-price">{{ buyGoods.line_price ? $dialog.formatMoney(buyGoods.line_price) : '0.00'}}</view>
  208. </view>
  209. <view class="sale">{{$dialog.disFormat(buyGoods.price,buyGoods.line_price)}}折</view>
  210. <view class="name">已选:{{ buyGoods.name }}</view>
  211. </view>
  212. </view>
  213. <view class="close" @click="buyOrder(2)"><image src="/static/image/close.png"></image></view>
  214. </view>
  215. <view class="buy-godos-mib">
  216. <view class="mib-title">规格</view>
  217. <view class="mib-body" v-if="goodsData.mult_sku==2">
  218. <view :class="indx==skuIndex?'mib-body-item active':'mib-body-item'" v-for="(item,indx) in skuSpecs" :key="indx" @click="checkSku(item,indx)">{{item.name}}</view>
  219. </view>
  220. <view class="mib-body" v-else>
  221. <view class="mib-body-item active">默认</view>
  222. </view>
  223. <view class="mib-number">
  224. <view class="num-name">数量 </view>
  225. <view class="numb-box">
  226. <uni-number-box v-model="buyGoods.number" :min="1" :max="10" v-if="goodsData.limit_rule_type==1"></uni-number-box>
  227. <uni-number-box v-model="buyGoods.number" :min="1" :max="goodsData.limit_rule" v-if="goodsData.limit_rule_type==2" ></uni-number-box>
  228. </view>
  229. </view>
  230. </view>
  231. <view class="buy-godos-footer">
  232. <view class="btn-item">
  233. <button type="warn" @click="buyOrder(2)">再看看</button>
  234. </view>
  235. <view class="btn-item fex1">
  236. <pay-button-sdk
  237. class="pay-button"
  238. :mode="2"
  239. :goods-type="1"
  240. :goods-id="goodsData.goods_id"
  241. bind:getgoodsinfo="getGoodsInfo"
  242. bind:placeorder="userLogin"
  243. bind:error="onErrors"
  244. bind:pay="onPay"
  245. ></pay-button-sdk>
  246. <!-- <button type="primary" :disabled="buyDis" @click="buyNow">确认购买</button> -->
  247. </view>
  248. </view>
  249. </view>
  250. </uni-popup>
  251. </view>
  252. </template>
  253. <script>
  254. var app;
  255. import * as Api from "@/static/api/goods.js";
  256. export default {
  257. data() {
  258. return {
  259. buyDis:false,
  260. loading:true,
  261. goodsId:"101088609520561356",
  262. bannerData:[],
  263. indicatorDots: true,
  264. goodsData:null,
  265. storeData:null,
  266. spm:0,
  267. specs:[],
  268. skuSpecs:[],
  269. buyGoods:null,
  270. skuIndex:0,
  271. loadingPayBtn:0,
  272. currIndex: 0, // 当前索引
  273. swiperHeight: 200,
  274. userToken:null
  275. }
  276. },
  277. created() {
  278. this.$scope.getGoodsInfo = this.getGoodsInfo;
  279. this.$scope.userLogin = this.userLogin;
  280. this.$scope.onErrors = this.onErrors;
  281. this.$scope.onPay = this.onPay;
  282. },
  283. onLoad({goods,spm}) {
  284. app = this;
  285. app.storeData = uni.getStorageSync("contact");
  286. app.userToken = uni.getStorageSync("user_token");
  287. app.goodsId = goods;
  288. app.spm = spm?spm:'0';
  289. app.getGoods();
  290. },
  291. methods: {
  292. toLogin(e){
  293. uni.login({
  294. force:true,
  295. success(resp){
  296. var detail = e.detail;
  297. detail.login = resp.code;
  298. Api.login(detail).then((res)=>{
  299. if (res.code == 0) {
  300. return app.$dialog.showSuccess(res.msg);
  301. }
  302. uni.setStorageSync("user_token", res.data.access_token)
  303. app.userToken = res.data.access_token
  304. })
  305. }
  306. })
  307. },
  308. callMobile(mobile){
  309. uni.makePhoneCall({
  310. phoneNumber:mobile,
  311. success(res) {
  312. console.log(res)
  313. },fail(res) {
  314. console.log(res)
  315. }
  316. })
  317. },
  318. /* 切换 swiper 滑块 */
  319. changeSwiper(e) {
  320. app.currIndex = e.detail.current;
  321. app.$nextTick(() => {
  322. app.setSwiperHeight(); // 动态设置 swiper 的高度
  323. });
  324. },
  325. /* 动态设置 swiper 的高度 */
  326. setSwiperHeight() {
  327. const element = "#wrap" + app.currIndex;
  328. const query = uni.createSelectorQuery().in(this);
  329. query.select(element).boundingClientRect();
  330. query.exec(res => {
  331. console.log("动态高低",res)
  332. if (res && res[0]) this.swiperHeight = res[0].height==0?res[0].width:res[0].height;
  333. });
  334. },
  335. getGoodsInfo(event){
  336. console.log("getGoodsInfo",event)
  337. const { goodsId } = event.detail;
  338. return new Promise(resolve => {
  339. // 在此处开发者可以进行商品数据请求,获取商品信息
  340. // 然后将商品信息传入 resolve 函数
  341. resolve({
  342. minLimits: 1,
  343. maxLimits: app.goodsData.limit_rule_type==1?99:app.goodsData.limit_rule,
  344. dateRule: '周一至周日可用',
  345. relationType: 1,
  346. validation: {
  347. phoneNumber: {
  348. required: app.goodsData.customer_reserved_info==2?true:false // 手机号是否必填
  349. }
  350. },
  351. extra:app.buyGoods,
  352. // 在 bind:getgoodsinfo 返回的 promise 的 resolve 函数中新增 marketingVersion 字段
  353. marketingVersion: 2,
  354. });
  355. })
  356. },
  357. onErrors(e){
  358. console.log("错误",e)
  359. // if (app.loadingPayBtn == 1) {
  360. // app.buyNow()
  361. // return ;
  362. // }
  363. // app.loadingPayBtn ++;
  364. },
  365. userLogin(event) {
  366. return new Promise((resolve, reject) => {
  367. tt.login({
  368. success(){
  369. console.log("next")
  370. resolve()
  371. },
  372. fail: () => reject()
  373. });
  374. });
  375. },
  376. onPay(event){
  377. console.log("event",event)
  378. const { status, outOrderNo, result } = event.detail;
  379. console.log(status, outOrderNo, result)
  380. const { code } = result;
  381. if (code === 0) { // 支付成功
  382. app.$dialog.showSuccess("支付成功","none",function(){
  383. uni.navigateTo({
  384. url:"usr://pages/order/detail?order="+outOrderNo
  385. })
  386. })
  387. } else {
  388. app.$dialog.showSuccess("支付失败","none",function(){
  389. uni.navigateTo({
  390. url:"usr://pages/order/detail?order="+outOrderNo
  391. })
  392. })
  393. // 继续支付失败(超时、取消、关闭)
  394. }
  395. },
  396. buyNow(){
  397. var formData = app.buyGoods;
  398. formData.parent = app.goodsId;
  399. formData.spm = app.spm;
  400. app.buyDis = true;
  401. Api.add(formData).then((res)=>{
  402. app.buyDis = false;
  403. if (res.code == 0) {
  404. return app.$dialog.showSuccess(res.msg)
  405. }
  406. uni.navigateTo({
  407. url:"/pages/order/confirm?order="+res.data.order
  408. })
  409. })
  410. },
  411. checkSku(data,index){
  412. app.skuIndex = index;
  413. app.buyGoods = {
  414. img:data.image,
  415. price:data.price,
  416. line_price:data.line_price,
  417. name:data.name,
  418. product_id:data.product_id,
  419. number:1,
  420. sku_id:data.sku_id
  421. }
  422. },
  423. toGoods(id){
  424. uni.navigateTo({
  425. url:"/pages/goods/detail?goods="+id
  426. })
  427. },
  428. toMap(){
  429. var poi = app.goodsData.poi;
  430. uni.openLocation({
  431. latitude:poi.latitude,
  432. longitude:poi.longitude,
  433. name:poi.poi_name,
  434. address:poi.poi_address,
  435. success() {
  436. console.log("open success")
  437. },
  438. fail(err) {
  439. app.$dialog.showSuccess("打开导航失败")
  440. console.log(err)
  441. }
  442. })
  443. },
  444. /**
  445. * 直接购买
  446. */
  447. buyOrder(type){
  448. var userToken = uni.getStorageSync("user_token");
  449. if (!userToken) {
  450. return ;
  451. }
  452. if (type == 1) {
  453. app.$refs.buypopup.open()
  454. } else {
  455. app.$refs.buypopup.close()
  456. }
  457. return ;
  458. },
  459. toHome(){
  460. uni.switchTab({
  461. url:"/pages/index/index"
  462. })
  463. },
  464. toOrder(){
  465. uni.switchTab({
  466. url:"/pages/order/order"
  467. })
  468. },
  469. showService(type){
  470. if (type == 1) {
  471. app.$refs.maskpopup.open()
  472. } else {
  473. app.$refs.maskpopup.close()
  474. }
  475. },
  476. openRule(type){
  477. if (type == 1) {
  478. app.$refs.rulepopup.open()
  479. } else {
  480. app.$refs.rulepopup.close()
  481. }
  482. },
  483. getGoods(){
  484. app.loading = true;
  485. Api.data({"goods":app.goodsId,"spm":app.spm}).then((res)=>{
  486. app.loading = false;
  487. if (res.code && res.code == 0) {
  488. return app.$dialog.showSuccess(res.msg);
  489. }
  490. console.log(res.data)
  491. if (!app.storeData) {
  492. app.storeData = res.data.poi;
  493. uni.setStorageSync("contact",res.data.poi);
  494. }
  495. app.goodsData = res.data;
  496. app.bannerData = res.data.image_list
  497. app.specs = res.data.specs;
  498. app.skuSpecs = res.data.skuSpecs;
  499. if (res.data.skuSpecs.length > 0) {
  500. var skuSpecs = app.skuSpecs[0]
  501. var buyGoods = {
  502. img:skuSpecs.image,
  503. price:skuSpecs.price,
  504. line_price:skuSpecs.line_price,
  505. name:skuSpecs.name,
  506. number:1,
  507. product_id:skuSpecs.product_id,
  508. sku_id:skuSpecs.sku_id,
  509. goods_id:skuSpecs.goods_id
  510. }
  511. } else {
  512. var buyGoods = {
  513. img:res.data.image_list[0].url,
  514. price:res.data.price,
  515. line_price:res.data.line_price,
  516. name:"默认",
  517. number:1,
  518. product_id:res.data.product_id,
  519. sku_id:res.data.product_id,
  520. goods_id:res.data.goods_id
  521. }
  522. }
  523. app.buyGoods = buyGoods
  524. app.$nextTick(function(){
  525. app.setSwiperHeight()
  526. })
  527. })
  528. }
  529. }
  530. }
  531. </script>
  532. <style>
  533. page{background-color: #f8f8f8;}
  534. .popup-flex{display: flex;flex-direction: column;}
  535. .buy-godos-footer{padding: 20upx;display: flex;align-items: center;justify-content: space-between;gap: 20upx;}
  536. .buy-godos-footer .btn-item.fex1{flex: 1;}
  537. .buy-godos-footer .btn-item .pay-button{width: 100% !important;}
  538. .buy-godos-mib{padding: 20upx;flex: 1;}
  539. .buy-godos-mib .mib-number{margin-top: 20upx;display: flex;align-items: center;justify-content: space-between;font-size: 28upx;color: #666;}
  540. .buy-godos-mib .mib-title{font-size: 28upx;color: #666;}
  541. .buy-godos-mib .mib-body{margin-top: 15upx;display: flex;align-items: center;flex-wrap: wrap;gap: 20upx;}
  542. .buy-godos-mib .mib-body .mib-body-item{font-size: 24upx;color: #666;padding: 10upx;background-color: #f8f8f8;border-radius: 5upx;}
  543. .buy-godos-mib .mib-body .mib-body-item.active{background-color: #007aff;color: #fff;}
  544. .buy-godos-top{display: flex;padding: 15upx;align-items: start;}
  545. .buy-godos-top .close{margin-left: auto;margin-right: 0;display: flex;align-items: center;justify-content: center;}
  546. .buy-godos-top .close image{width: 40upx;height: 40upx;}
  547. .buy-godos-top .img{height: 200upx;overflow: hidden;}
  548. .buy-godos-top .img image{width: 200upx;border-radius: 10upx;}
  549. .buy-godos-top .goods-top-info{display: flex;align-items: center;gap: 10upx;}
  550. .buy-godos-top .goods-top-info .price{display: flex;align-items: center;flex: 1;}
  551. .buy-godos-top .goods-top-info .price .line-price{color: #999999;font-size: 24rpx;text-decoration-line: line-through;margin-right: 20upx;}
  552. .buy-godos-top .goods-top-info .price .price-sale{color: #FF4747;font-size: 32rpx;font-weight: bold;margin-right: 10upx;}
  553. .buy-godos-top .goods-top-info .price .desc-num{color: #999;font-size: 24upx;margin-left: auto;margin-right: 0;}
  554. .buy-godos-top .goods-top-info .sale{background-color: #FDEBE5;border: 4rpx;font-size: 22rpx;color: #FF4747;padding: 10rpx;display: inline-block;}
  555. .buy-godos-top .goods-top-info .name{font-size: 22rpx;margin-top: 15upx;color: #666;}
  556. .skeleton{width: 90%;margin: 20upx auto;}
  557. .line-footer{height: 200upx;}
  558. .footer{position: fixed;bottom: 0;background-color: #fff;padding: 20upx 20upx 0 20upx;border-top: 2upx solid #f8f8f8;left: 0;right: 0;}
  559. .footer .footer-op{display: flex;align-items: center;}
  560. .footer .footer-op .footer-left{display: flex;align-items: center;gap: 40upx;justify-content: center;}
  561. .footer .footer-op .footer-left .item{flex-direction: column;display: flex;font-size: 24rpx;color: #333;align-items: center;}
  562. .footer .footer-op .footer-left image{width: 40upx;height: 40upx;}
  563. .footer .footer-op .footer-right{background-color: #007aff;color: #fff;font-size: 28rpx;flex: 1;margin-left: 40upx;height: 80upx;border-radius: 10upx;line-height: 80upx;text-align: center;}
  564. .footer .footer-safe {height: env(safe-area-inset-bottom);width: 100%;}
  565. .kefu-popup .item{height: 100upx;line-height: 100upx;text-align: center;font-size: 28upx;color: #333;border-bottom: 2upx solid #f8f8f8;}
  566. .kefu-popup .item .btn{background-color: #fff;border: 0;height: 100upx;line-height: 100upx;text-align: center;font-size: 28upx;color: #333;}
  567. .kefu-popup .item .btn::after{border: 0;}
  568. .kefu-popup .foot{height: 100upx;line-height: 100upx;text-align: center;font-size: 28upx;color: #333;border-top: 20upx solid #f8f8f8;}
  569. .goods-store{background-color: #fff;padding: 0 20upx;margin-top: 20upx;}
  570. .goods-store .title{line-height: 80upx;font-size: 32rpx;color: #333;font-weight: bold;display: flex;align-items: center;border-bottom: 2upx solid #f8f8f8;}
  571. .goods-store .store-info{display: flex;align-items: center;padding: 20upx 0;}
  572. .goods-store .store-info .store-right{margin-left: auto;margin-right: 0;font-size: 24rpx;color: #999;}
  573. .goods-store .store-info .store-right image{width: 50upx;height: 50upx;margin-bottom: 10upx;}
  574. .goods-store .store-info .store-left .name{font-size: 30rpx;color: #333;font-weight: bold;line-height: 80upx;}
  575. .goods-store .store-info .store-left .address{font-size: 24rpx;color: #999;}
  576. .service-popup.popup-height{height: 60vh;}
  577. .service-popup.popup-white{background-color: #fff;margin-top: 20upx;}
  578. .service-popup.popup-white .title{border-bottom: 2upx solid #f8f8f8;}
  579. .service-popup .title{line-height: 80upx;font-size: 32rpx;color: #333;font-weight: bold;padding: 0 20upx;display: flex;align-items: center;}
  580. .service-popup .title .close{margin-left: auto;margin-right: 0;display: flex;align-items: center;justify-content: center;}
  581. .service-popup .title .close image{width: 40upx;height: 40upx;}
  582. .service-popup.popup-white .rule-list{padding-bottom: 20upx;}
  583. .service-popup .rule-item{display: flex;margin-top: 40upx;align-items: start;padding: 0 20upx;}
  584. .service-popup .rule-item image{width: 50upx;height: 50upx;}
  585. .service-popup .rule-item .item-top{margin-left: 20upx;}
  586. .service-popup .rule-item .name{font-size: 28rpx;color: #333;font-weight: bold;margin-bottom: 10upx;display: flex;align-items: center;}
  587. .service-popup .rule-item .name .tips{color: #333;border: 2upx solid #ddd;border-radius: 5upx;font-size: 22rpx;font-weight: normal;margin-left: 40upx;background-color: #fff;padding: 5upx 10upx;}
  588. .service-popup .rule-item .desc{color: #999;font-size: 24rpx;line-height: 40upx;}
  589. .home-banner image{width: 100%;height: auto;}
  590. .home-banner .swiper{height: 380upx;}
  591. .goods-top{background-color: #fff;padding: 20upx;border-radius: 30upx 30upx 0 0;}
  592. .goods-top .price{display: flex;align-items: center;flex: 1;}
  593. .goods-top .price .line-price{color: #999999;font-size: 24rpx;text-decoration-line: line-through;margin-right: 20upx;}
  594. .goods-top .price .price-sale{color: #FF4747;font-size: 32rpx;font-weight: bold;margin-right: 10upx;}
  595. .goods-top .price .desc-num{color: #999;font-size: 24upx;margin-left: auto;margin-right: 0;}
  596. .goods-top .title{margin: 20upx 0;font-size: 32rpx;color: #333;}
  597. .goods-top .goods-safe{display: flex;align-items: center;font-size: 24rpx;color: #999;gap: 40upx;}
  598. .goods-top .goods-safe .safe-item{display: flex;align-items: center;}
  599. .goods-top .goods-safe image{width: 30upx;height: 30upx;margin-right: 10upx;}
  600. .goods-top .goods-rule{margin: 20upx 0;display: flex;align-items: center;font-size: 24rpx;color: #999;height: 80upx;line-height: 80upx;}
  601. .goods-top .goods-rule .right{display: flex;align-items: center;flex: 1;color: #000;margin-left: 40upx;}
  602. .goods-top .goods-rule .right image{width: 30upx;height: 30upx;margin-left: auto;margin-right: 0;}
  603. .goods-top .price .sale{background-color: #FDEBE5;border: 4rpx;font-size: 22rpx;color: #FF4747;padding: 10rpx;margin-right: 20upx;}
  604. .popup-height-max{max-height: 60vh;}
  605. .group-body{padding: 20upx;}
  606. .group-body .body-title{display: flex;align-items: center;line-height: 48upx;font-size: 28upx;color: #333;font-weight: bold;gap: 20upx;justify-content: space-between;}
  607. .group-body .body-title span{font-size: 24upx;font-weight: normal;color: #666;}
  608. .group-body .body-item-li{display: flex;align-items: center;justify-content: space-between;line-height: 48upx;font-size: 24upx;color: #333;font-weight: bold;}
  609. .group-body .body-item-li .item-right{display: flex;align-items: center;gap: 20upx;width: 40%;justify-content: end;}
  610. .group-body .body-item-li .item-right .desc{color: #999;font-weight: normal;}
  611. </style>