detail.vue 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. <template>
  2. <!-- 商品详情页面 -->
  3. <view v-show="!isLoading" class="container" :style="appThemeStyle">
  4. <!-- 商品图片轮播 -->
  5. <SlideImage v-if="!isLoading" :video="goods.video" :videoCover="goods.videoCover" :images="goods.goods_images" />
  6. <!-- 商品信息 -->
  7. <view v-if="!isLoading" class="goods-info m-top20">
  8. <!-- 价格、销量 -->
  9. <view class="info-item info-item__top dis-flex flex-x-between flex-y-end">
  10. <view class="block-left dis-flex flex-y-center">
  11. <!-- 商品售价 -->
  12. <text class="floor-price__samll">¥</text>
  13. <text class="floor-price">{{ goods.goods_price_min }}</text>
  14. <!-- 会员价标签 -->
  15. <view v-if="goods.is_user_grade" class="user-grade">
  16. <text>会员价</text>
  17. </view>
  18. <!-- 划线价 -->
  19. <text v-if="goods.line_price_min > 0" class="original-price">¥{{ goods.line_price_min }}</text>
  20. </view>
  21. <view class="block-right dis-flex">
  22. <!-- 销量 -->
  23. <view class="goods-sales">
  24. <text>已售{{ goods.goods_sales }}件</text>
  25. </view>
  26. </view>
  27. </view>
  28. <!-- 标题、分享 -->
  29. <view class="info-item info-item__name dis-flex flex-y-center">
  30. <view class="goods-name flex-box">
  31. <text class="twoline-hide">{{ goods.goods_name }}</text>
  32. </view>
  33. <view class="goods-share__line"></view>
  34. <view class="goods-share">
  35. <button class="share-btn dis-flex flex-dir-column" @click="onShowShareSheet()">
  36. <text class="share__icon iconfont icon-fenxiang"></text>
  37. <text class="f-24">分享</text>
  38. </button>
  39. </view>
  40. </view>
  41. <!-- 商品卖点 -->
  42. <view v-if="goods.selling_point" class="info-item info-item_selling-point">
  43. <text>{{ goods.selling_point }}</text>
  44. </view>
  45. </view>
  46. <!-- 选择商品规格 -->
  47. <view v-if="goods.spec_type == 20" class="goods-choice m-top20 b-f" @click="onShowSkuPopup(1)">
  48. <view class="spec-list">
  49. <view class="flex-box">
  50. <text class="col-8">选择:</text>
  51. <text class="spec-name" v-for="(item, index) in goods.specList" :key="index">{{ item.spec_name }}</text>
  52. </view>
  53. <view class="f-26 col-9 t-r">
  54. <text class="iconfont icon-arrow-right"></text>
  55. </view>
  56. </view>
  57. </view>
  58. <!-- 商品服务 -->
  59. <Service v-if="!isLoading" :goods-id="goodsId" />
  60. <!-- 商品SKU弹窗 -->
  61. <SkuPopup v-if="!isLoading" v-model="showSkuPopup" :skuMode="skuMode" :goods="goods" @addCart="onAddCart" />
  62. <!-- 商品评价 -->
  63. <Comment v-if="!isLoading" :goods-id="goodsId" :limit="2" />
  64. <!-- 商品描述 -->
  65. <view v-if="!isLoading" class="goods-content m-top20">
  66. <view class="item-title b-f">
  67. <text>商品描述</text>
  68. </view>
  69. <view v-if="goods.content != ''" class="goods-content__detail b-f">
  70. <mp-html :content="goods.content" />
  71. </view>
  72. </view>
  73. <!-- 底部选项卡 -->
  74. <view class="footer-fixed">
  75. <view class="footer-container">
  76. <!-- 导航图标 -->
  77. <view class="foo-item-fast">
  78. <!-- 首页 -->
  79. <view class="fast-item fast-item--home" @click="onTargetHome">
  80. <view class="fast-icon">
  81. <text class="iconfont icon-shouye"></text>
  82. </view>
  83. <view class="fast-text">
  84. <text>首页</text>
  85. </view>
  86. </view>
  87. <!-- 客服 -->
  88. <customer-btn v-if="isShowCustomerBtn" :showCard="true" :cardTitle="goods.goods_name" :cardImage="goods.goods_image">
  89. <view class="fast-item">
  90. <view class="fast-icon">
  91. <text class="iconfont icon-kefu1"></text>
  92. </view>
  93. <view class="fast-text">
  94. <text>客服</text>
  95. </view>
  96. </view>
  97. </customer-btn>
  98. <!-- 购物车 -->
  99. <view class="fast-item fast-item--cart" @click="onTargetCart">
  100. <view v-if="cartTotal > 0" class="fast-badge fast-badge--fixed">{{ cartTotal > 99 ? '99+' : cartTotal }} </view>
  101. <view class="fast-icon">
  102. <text class="iconfont icon-gouwuche"></text>
  103. </view>
  104. <view class="fast-text">
  105. <text>购物车</text>
  106. </view>
  107. </view>
  108. </view>
  109. <!-- 操作按钮 -->
  110. <view class="foo-item-btn">
  111. <view class="btn-wrapper">
  112. <view v-if="isEnableCart" class="btn-item btn-item-deputy" @click="onShowSkuPopup(2)">
  113. <text>加入购物车</text>
  114. </view>
  115. <view class="btn-item btn-item-main" @click="onShowSkuPopup(3)">
  116. <text>立即购买</text>
  117. </view>
  118. </view>
  119. </view>
  120. </view>
  121. </view>
  122. <!-- 快捷导航 -->
  123. <!-- <shortcut bottom="120rpx" /> -->
  124. <!-- 分享菜单 -->
  125. <share-sheet v-model="showShareSheet" :shareTitle="goods.goods_name" :shareImageUrl="goods.goods_image" />
  126. </view>
  127. </template>
  128. <script>
  129. import {getSceneData} from '@/core/app'
  130. import * as GoodsApi from '@/api/goods'
  131. import * as CartApi from '@/api/cart'
  132. import SettingModel from '@/common/model/Setting'
  133. import {GoodsTypeEnum} from '@/common/enum/goods'
  134. import ShareSheet from '@/components/share-sheet'
  135. import CustomerBtn from '@/components/customer-btn'
  136. import SlideImage from './components/SlideImage'
  137. import SkuPopup from './components/SkuPopup'
  138. import Comment from './components/Comment'
  139. import Service from './components/Service'
  140. export default {
  141. components: {
  142. ShareSheet,
  143. CustomerBtn,
  144. SlideImage,
  145. SkuPopup,
  146. Comment,
  147. Service
  148. },
  149. data() {
  150. return {
  151. // 正在加载
  152. isLoading: true,
  153. // 当前商品ID
  154. goodsId: null,
  155. // 商品详情
  156. goods: {},
  157. // 购物车总数量
  158. cartTotal: 0,
  159. // 显示/隐藏SKU弹窗
  160. showSkuPopup: false,
  161. // 模式 1:都显示 2:只显示购物车 3:只显示立即购买
  162. skuMode: 1,
  163. // 显示/隐藏分享菜单
  164. showShareSheet: false,
  165. // 是否支持加入购物车
  166. isEnableCart: false,
  167. // 是否显示在线客服按钮
  168. isShowCustomerBtn: false
  169. }
  170. },
  171. /**
  172. * 生命周期函数--监听页面加载
  173. */
  174. async onLoad(options) {
  175. // 记录query参数
  176. this.onRecordQuery(options)
  177. // 加载页面数据
  178. this.onRefreshPage()
  179. // 是否显示在线客服按钮
  180. // this.isShowCustomerBtn = await SettingModel.isShowCustomerBtn()
  181. },
  182. methods: {
  183. // 记录query参数
  184. onRecordQuery(query) {
  185. const scene = getSceneData(query)
  186. this.goodsId = query.goodsId ? parseInt(query.goodsId) : parseInt(scene.gid)
  187. },
  188. // 刷新页面数据
  189. onRefreshPage() {
  190. const app = this
  191. app.isLoading = true
  192. Promise.all([app.getGoodsDetail(), app.getCartTotal()]).finally(() => (app.isLoading = false))
  193. },
  194. // 获取商品信息
  195. getGoodsDetail() {
  196. const app = this
  197. return new Promise((resolve, reject) => {
  198. GoodsApi.detail(app.goodsId)
  199. .then((result) => {
  200. app.goods = result.data.detail
  201. if (app.goods.goods_type == GoodsTypeEnum.PHYSICAL.value) {
  202. app.isEnableCart = true
  203. }
  204. resolve(result)
  205. })
  206. .catch(reject)
  207. })
  208. },
  209. // 获取购物车总数量
  210. getCartTotal() {
  211. const app = this
  212. return new Promise((resolve, reject) => {
  213. CartApi.total()
  214. .then((result) => {
  215. app.cartTotal = result.data.cartTotal
  216. resolve(result)
  217. })
  218. .catch(reject)
  219. })
  220. },
  221. // 更新购物车数量
  222. onAddCart(total) {
  223. this.cartTotal = total
  224. },
  225. /**
  226. * 显示/隐藏SKU弹窗
  227. * @param {skuMode} 模式 1:都显示 2:只显示购物车 3:只显示立即购买
  228. */
  229. onShowSkuPopup(skuMode = 1) {
  230. const app = this
  231. if (app.isEnableCart) {
  232. app.skuMode = skuMode
  233. } else {
  234. app.skuMode = 3
  235. }
  236. app.showSkuPopup = !app.showSkuPopup
  237. },
  238. // 显示隐藏分享菜单
  239. onShowShareSheet() {
  240. this.showShareSheet = !this.showShareSheet
  241. },
  242. // 跳转到首页
  243. onTargetHome(e) {
  244. this.$navTo('pages/index/index')
  245. },
  246. // 跳转到购物车页
  247. onTargetCart() {
  248. this.$navTo('pages/cart/index')
  249. }
  250. },
  251. /**
  252. * 分享当前页面
  253. */
  254. onShareAppMessage() {
  255. const app = this
  256. // 构建页面参数
  257. const params = app.$getShareUrlParams({
  258. goodsId: app.goodsId
  259. })
  260. return {
  261. title: app.goods.goods_name,
  262. path: `/pages/goods/detail?${params}`
  263. }
  264. },
  265. /**
  266. * 分享到朋友圈
  267. * 本接口为 Beta 版本,暂只在 Android 平台支持,详见分享到朋友圈 (Beta)
  268. * https://developers.weixin.qq.com/miniprogram/dev/framework/open-ability/share-timeline.html
  269. */
  270. onShareTimeline() {
  271. const app = this
  272. // 构建页面参数
  273. const params = app.$getShareUrlParams({
  274. goodsId: app.goodsId
  275. })
  276. return {
  277. title: app.goods.goods_name,
  278. path: `/pages/goods/detail?${params}`
  279. }
  280. }
  281. }
  282. </script>
  283. <style>
  284. page {
  285. background: #fafafa;
  286. }
  287. </style>
  288. <style lang="scss" scoped>
  289. @import './detail.scss';
  290. </style>