index.vue 52 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348
  1. <template>
  2. <view>
  3. <!-- 顶部的 nav tab -->
  4. <view class='navbar' :style="{height:navH+'rpx',opacity:opacity}">
  5. <view class='navbarH' :style='"height:"+navH+"rpx;"'>
  6. <view class='navbarCon acea-row row-center-wrapper'>
  7. <view class="header acea-row row-center-wrapper">
  8. <view class="item" :class="navActive === index ? 'on' : ''" v-for="(item,index) in navList" :key='index' @tap="tap( index)">
  9. {{ item }}
  10. </view>
  11. </view>
  12. </view>
  13. </view>
  14. </view>
  15. <!-- 返回键 -->
  16. <view id="home" class="home-nav acea-row row-center-wrapper iconfont icon-xiangzuo" :class="opacity>0.5?'on':''" :style="{ top: homeTop + 'rpx' }" v-if="returnShow" @tap="returns">
  17. </view>
  18. <!-- 详情 -->
  19. <view class='product-con'>
  20. <scroll-view :scroll-top="scrollTop" scroll-y='true' scroll-with-animation="true" :style="'height:'+height+'px;'"
  21. @scroll="scroll">
  22. <view id="past0">
  23. <productConSwiper :imgUrls='spu.sliderPicUrls' />
  24. <!-- 价格、库存、销量 -->
  25. <view class='nav acea-row row-between-wrapper mb30'>
  26. <view class='money'>¥
  27. <text class='num'>{{ fen2yuan(spu.price) }}</text>
  28. <text class='y-money'>¥{{ fen2yuan(spu.marketPrice) }}</text>
  29. </view>
  30. <!-- 活动状态 -->
  31. <view class='acea-row row-middle'>
  32. <view class='time' v-if="status === 2">
  33. <view>距秒杀结束仅剩</view>
  34. <countDown :bgColor="bgColor" :is-day="false" :datatime="activity.endTime / 1000"
  35. :tip-text="' '" :day-text="' '" :hour-text="' : '" :minute-text="' : '" :second-text="' '" />
  36. </view>
  37. </view>
  38. </view>
  39. <view class="pad30 mb30">
  40. <view class='wrapper borRadius14 mb30'>
  41. <view class='introduce acea-row row-between'>
  42. <view class='infor'> {{ spu.name}}</view>
  43. <view class='iconfont icon-fenxiang' @click="listenerActionSheet"></view>
  44. </view>
  45. <view class='label acea-row row-middle'>
  46. <view class='stock'>累计销售:{{ spu.salesCount}} {{ spu.unitName }}</view>
  47. <view>限量: {{ activity.stock || 0 }} {{ spu.unitName }}</view>
  48. </view>
  49. </view>
  50. <!-- SKU 选择 -->
  51. <view class='attribute acea-row row-between-wrapper mb30 borRadius14' @tap='openAttr'>
  52. <view class="line1">{{ attrValue.length > 0 ? "已选择" : "请选择" }}:
  53. <text class='atterTxt'>{{attrValue}}</text>
  54. </view>
  55. <view class='iconfont icon-jiantou'></view>
  56. </view>
  57. <!-- 评论 -->
  58. <view class='userEvaluation' id="past1">
  59. <view class='title acea-row row-between-wrapper'
  60. :style="replyCount ===0?'border-bottom-left-radius:14rpx;border-bottom-right-radius:14rpx;':''">
  61. <view>用户评价({{ replyCount }})</view>
  62. <navigator class='praise' hover-class='none' :url="'/pages/users/goods_comment_list/index?productId='+ activity.spuId">
  63. <text class='font-color'>{{ replyChance }}%</text>好评率
  64. <text class='iconfont icon-jiantou'></text>
  65. </navigator>
  66. </view>
  67. <userEvaluation :reply="reply" />
  68. </view>
  69. </view>
  70. </view>
  71. <view class='product-intro' id="past2">
  72. <view class='title'>
  73. <image src="../../../static/images/xzuo.png"></image>
  74. <span class="sp">产品详情</span>
  75. <image src="../../../static/images/xyou.png"></image>
  76. </view>
  77. <view class='conter'>
  78. <jyf-parser :html="spu.description" ref="article" :tag-style="tagStyle"></jyf-parser>
  79. </view>
  80. </view>
  81. <view style='height:120rpx;'></view>
  82. </scroll-view>
  83. <view class='footer acea-row row-between-wrapper'>
  84. <!-- 客服 TODO 芋艿:待完成 -->
  85. <!-- #ifdef MP -->
  86. <button open-type="contact" hover-class='none' class='item'>
  87. <view class='iconfont icon-kefu'></view>
  88. <view>客服</view>
  89. </button>
  90. <!-- #endif -->
  91. <!-- #ifndef MP -->
  92. <navigator hover-class="none" class="item" @click="kefuClick">
  93. <view class="iconfont icon-kefu"></view>
  94. <view>客服</view>
  95. </navigator>
  96. <!-- #endif -->
  97. <!-- 收藏 -->
  98. <view @tap='setCollect' class='item'>
  99. <view class='iconfont icon-shoucang1' v-if="userCollect"></view>
  100. <view class='iconfont icon-shoucang' v-else></view>
  101. <view>收藏</view>
  102. </view>
  103. <!-- 购买操作 -->
  104. <view class="bnt acea-row" v-if="status === 0">
  105. <view class="buy bnts bg-color-hui">已关闭</view>
  106. </view>
  107. <view class="bnt acea-row" v-else-if="status === 1">
  108. <view class="joinCart bnts" @tap="openAlone">单独购买</view>
  109. <view class="buy bnts bg-color-hui">未开始</view>
  110. </view>
  111. <view class="bnt acea-row" v-else-if="status === 2 && attr.productSelect.stock > 0">
  112. <view class="joinCart bnts" @tap="openAlone">单独购买</view>
  113. <view class="buy bnts" @tap="goBuy">立即购买</view>
  114. </view>
  115. <view class="bnt acea-row" v-else-if="status === 2 && (attr.productSelect.stock <= 0)">
  116. <view class="joinCart bnts" @tap="openAlone">单独购买</view>
  117. <view class="buy bnts bg-color-hui">已售罄</view>
  118. </view>
  119. <view class="bnt acea-row" v-else-if="status === 3">
  120. <view class="joinCart bnts" @tap="openAlone">单独购买</view>
  121. <view class="buy bnts bg-color-hui">已结束</view>
  122. </view>
  123. <view class="bnt acea-row" v-else> <!-- 兜底 -->
  124. <view class="joinCart bnts" @tap="openAlone">单独购买</view>
  125. <view class="buy bnts bg-color-hui">未开始</view>
  126. </view>
  127. </view>
  128. </view>
  129. <!-- SKU 弹窗 -->
  130. <product-window
  131. :attr='attr'
  132. @ChangeAttr="ChangeAttr"
  133. @ChangeCartNum="ChangeCartNum"
  134. @iptCartNum="iptCartNum"
  135. @close="closeAttr"
  136. />
  137. <home></home>
  138. <!-- 分享按钮 -->
  139. <view class="generate-posters acea-row row-middle" :class="posters ? 'on' : ''">
  140. <!-- #ifndef MP -->
  141. <button class="item" hover-class='none' v-if="weixinStatus === true" @click="H5ShareBox = true">
  142. <view class="iconfont icon-weixin3"></view>
  143. <view class="">发送给朋友</view>
  144. </button>
  145. <!-- #endif -->
  146. <!-- #ifdef MP -->
  147. <button class="item" open-type="share" hover-class='none' @click="closePosters">
  148. <view class="iconfont icon-weixin3"></view>
  149. <view class="">发送给朋友</view>
  150. </button>
  151. <!-- #endif -->
  152. <button class="item" hover-class='none' @tap="goPoster">
  153. <view class="iconfont icon-haibao"></view>
  154. <view class="">生成海报</view>
  155. </button>
  156. </view>
  157. <view class="mask" v-if="posters" @click="closePosters"></view>
  158. <view class="mask" v-if="canvasStatus" @click="closePosters"></view>
  159. <!-- 海报展示 -->
  160. <view class='poster-pop' v-if="canvasStatus">
  161. <image src='/static/images/poster-close.png' class='close' @click="posterImageClose"></image>
  162. <image :src='posterImage'></image>
  163. <!-- #ifndef H5 -->
  164. <view class='save-poster' @click="savePosterPath">保存到手机</view>
  165. <!-- #endif -->
  166. <!-- #ifdef H5 -->
  167. <view class="keep">长按图片可以保存到手机</view>
  168. <!-- #endif -->
  169. </view>
  170. <view class="canvas" v-else>
  171. <canvas style="width:750px;height:1190px;" canvas-id="firstCanvas"></canvas>
  172. <canvas canvas-id="qrcode" :style="{width: `${qrcodeSize}px`, height: `${qrcodeSize}px`}"/>
  173. </view>
  174. <!-- 发送给朋友图片 -->
  175. <view class="share-box" v-if="H5ShareBox">
  176. <image src="/static/images/share-info.png" @click="H5ShareBox = false"></image>
  177. </view>
  178. </view>
  179. </template>
  180. <script>
  181. const app = getApp();
  182. import uQRCode from '@/js_sdk/Sansnn-uQRCode/uqrcode.js'
  183. import { mapGetters } from "vuex";
  184. import productConSwiper from '@/components/productConSwiper/index.vue'
  185. import productWindow from '@/components/productWindow/index.vue'
  186. import userEvaluation from '@/components/userEvaluation/index.vue'
  187. // #ifdef MP
  188. import { base64src } from '@/utils/base64src.js'
  189. import { getQrcode } from '@/api/api.js';
  190. // #endif
  191. import parser from "@/components/jyf-parser/jyf-parser";
  192. import home from '@/components/home/index.vue'
  193. import countDown from '@/components/countDown';
  194. import { imageBase64 } from "@/api/public";
  195. import { toLogin } from '@/libs/login.js';
  196. import { silenceBindingSpread } from "@/utils";
  197. import * as ProductSpuApi from '@/api/product/spu.js';
  198. import * as ProductFavoriteApi from '@/api/product/favorite.js';
  199. import * as ProductCommentApi from '@/api/product/comment.js';
  200. import * as SeckillApi from '@/api/promotion/seckill.js';
  201. import * as BrokerageAPI from '@/api/trade/brokerage.js'
  202. import * as Util from '@/utils/util.js';
  203. import * as ProductUtil from '@/utils/product.js';
  204. export default {
  205. data() {
  206. return {
  207. // ========== 秒杀活动相关变量 ==========
  208. id: 0, // 秒杀活动的编号
  209. activity: {}, // 秒杀活动的信息
  210. status: 1, // 0 - 已禁用;1 - 未开始;2 - 进行中;3 - 已结束
  211. bgColor: {
  212. 'bgColor': '#333333',
  213. 'Color': '#fff',
  214. 'isDay': true,
  215. 'width': '44rpx',
  216. 'timeTxtwidth': '16rpx',
  217. },
  218. // ========== 商品相关变量 ==========
  219. spu: {}, // 商品 SPU 详情
  220. skuMap: [], // 商品 SKU Map
  221. attrValue: '', // 已选属性名的拼接,例如说 红色,大 这样的格式
  222. attr: { // productWindow 组件,使用该属性
  223. cartAttr: false,
  224. // ↓↓↓ 属性数组,结构为:id = 属性编号;name = 属性编号的名字;values[].id = 属性值的编号,values[].name = 属性值的名字;index = 选中的属性值的名字
  225. properties: [],
  226. productSelect: {} // 选中的 SKU
  227. },
  228. tagStyle: {
  229. img: 'width:100%;display:block;',
  230. table: 'width:100%',
  231. video: 'width:100%'
  232. },
  233. // ========== 评价相关的变量 ==========
  234. replyCount: 0, // 总评论数量
  235. replyChance: 0, // 好评率
  236. reply: [], // 评论列表
  237. // ========== 收藏相关的变量 ==========
  238. userCollect: false,
  239. // ========== 分销相关的变量 ==========
  240. qrcodeSize: 600, // 二维码的大小
  241. promotionCode: '', // 二维码图片
  242. imgTop: '', // 商品图片的 base64 码
  243. errT: '', // 获得小程序码失败的提示文本
  244. posters: false, // 分享弹窗的开关
  245. weixinStatus: false, // 微信分享是否打开
  246. canvasStatus: false, // 是否显示海报
  247. H5ShareBox: false, // 公众号分享的弹出
  248. posterbackgd: '/static/images/posterbackgd.png', // 海报的背景,用于海报的生成
  249. storeImage: '', // 下载商品图片后的文件地址
  250. actionSheetHidden: true, // 微信小程序的右上角分享的弹出
  251. posterImage: '', // 海报路径
  252. // ========== 顶部 nav + scroll 相关的变量 ==========
  253. returnShow: true, // 判断顶部 [返回] 是否出现
  254. homeTop: 20, // 头部的 top 位置
  255. height: 0, // 窗口 height 高度
  256. scrollY: 0, // 滚动的 Y 轴
  257. scrollTop: 0, // 滚动条的 top 位置
  258. lock: false, // 是否锁定 scroll 下
  259. topArr: [], // 每个 nav 的 top 位置
  260. heightArr: [], // 每个 nav 的 height 高度
  261. navH: "", // 头部 nav 高度
  262. opacity: 0, // 头部 nav 的透明度
  263. navList: ['商品', '评价', '详情'],
  264. navActive: 0, // 选中的 navList 下标
  265. }
  266. },
  267. components: {
  268. productConSwiper,
  269. productWindow,
  270. userEvaluation,
  271. "jyf-parser": parser,
  272. home,
  273. countDown
  274. },
  275. computed: mapGetters(['isLogin','uid','chatUrl']),
  276. watch:{
  277. isLogin:{
  278. handler: function(newV,oldV) {
  279. if (newV) {
  280. this.getSeckillDetail();
  281. }
  282. },
  283. deep:true
  284. }
  285. },
  286. onLoad(options) {
  287. this.$store.commit("PRODUCT_TYPE", 'normal');
  288. // 设置商品列表高度
  289. uni.getSystemInfo({
  290. success: res => {
  291. this.height = res.windowHeight
  292. //res.windowHeight:获取整个窗口高度为px,*2为rpx;98为头部占据的高度;
  293. },
  294. });
  295. // #ifndef APP-PLUS
  296. this.navH = app.globalData.navHeight
  297. // #endif
  298. // #ifdef APP-PLUS
  299. this.navH = 90
  300. // #endif
  301. // #ifdef MP || APP-NVUE
  302. // 小程序链接进入获取绑定关系id
  303. // 绑定分销关系
  304. setTimeout(()=>{
  305. if (options.spread) {
  306. app.globalData.spread = options.spread;
  307. BrokerageAPI.bindBrokerageUser(options.spread).then(res => {})
  308. }
  309. },2000)
  310. // #endif
  311. // 校验参数是否正确
  312. if (!options.scene && !options.id){
  313. this.$util.Tips({
  314. title: '缺少参数无法查看商品'
  315. }, {
  316. url: '/pages/index/index'
  317. });
  318. return;
  319. }
  320. // 解析 id 商品编号
  321. if (options.scene) { // 仅仅小程序扫码进入
  322. // TODO 芋艿:code 是啥
  323. let qrCodeValue = this.$util.getUrlParams(decodeURIComponent(options.scene));
  324. let mapeMpQrCodeValue = this.$util.formatMpQrCodeData(qrCodeValue);
  325. app.globalData.spread = mapeMpQrCodeValue.spread;
  326. this.id = mapeMpQrCodeValue.id;
  327. // 绑定分销用户
  328. setTimeout(()=>{
  329. BrokerageAPI.bindBrokerageUser(mapeMpQrCodeValue.spread).then(res => {}).catch(res => {})
  330. },2000)
  331. } else {
  332. this.id = options.id;
  333. }
  334. // 获得秒杀详情
  335. this.getSeckillDetail();
  336. },
  337. //#ifdef MP
  338. onShareAppMessage() {
  339. return {
  340. title: this.spu.name || '',
  341. imageUrl: this.spu.picUrl,
  342. path: app.globalData.openPages
  343. };
  344. },
  345. //#endif
  346. onReady() {
  347. this.$nextTick(() => {
  348. // 设置微信的头部 top 位置
  349. // #ifdef MP
  350. const menuButton = uni.getMenuButtonBoundingClientRect();
  351. const query = uni.createSelectorQuery().in(this);
  352. query.select('#home')
  353. .boundingClientRect(data => {
  354. this.homeTop = menuButton.top * 2 + menuButton.height - data.height;
  355. })
  356. .exec();
  357. // #endif
  358. });
  359. silenceBindingSpread();
  360. },
  361. methods: {
  362. // ========== 秒杀活动相关 ==========
  363. getSeckillDetail: function() {
  364. SeckillApi.getSeckillActivity(this.id).then(res => {
  365. this.activity = res.data;
  366. // 计算活动状态
  367. if (this.activity) {
  368. const now = new Date().getTime();
  369. if (this.activity.status === 1) {
  370. this.status = 0;
  371. } else {
  372. if (this.activity.startTime > now) {
  373. this.status = 1;
  374. } else if (now <= this.activity.endTime) {
  375. this.status = 2;
  376. } else {
  377. this.status = 3;
  378. }
  379. }
  380. }
  381. // 获得商品详情
  382. this.getGoodsDetails();
  383. // 获得商品收藏
  384. this.isFavoriteExists();
  385. // 获得商品评价列表
  386. this.getProductReplyList();
  387. this.getProductReplyCount();
  388. app.globalData.openPages = '/pages/activity/goods_seckill_details/index?id='
  389. + this.id + '&spread=' + this.uid;
  390. }).catch(err => {
  391. this.$util.Tips({
  392. title:err
  393. },{
  394. tab:3
  395. })
  396. });
  397. },
  398. // ========== 商品详情相关 ==========
  399. /**
  400. * 获取产品详情
  401. */
  402. getGoodsDetails: function() {
  403. ProductSpuApi.getSpuDetail(this.activity.spuId).then(res => {
  404. let spu = res.data;
  405. let skus = res.data.skus;
  406. this.$set(this, 'spu', spu);
  407. this.$set(this.attr, 'properties', ProductUtil.convertProductPropertyList(skus));
  408. this.$set(this, 'skuMap', ProductUtil.convertProductSkuMap(skus));
  409. // 将秒杀活动的信息,合并到 SKU 里面,实现秒杀价格的显示
  410. this.activity.products.forEach(product => {
  411. this.spu.price = Math.min(this.spu.price, product.seckillPrice); // 设置 SPU 的最低价格
  412. });
  413. skus.forEach(sku => {
  414. const product = this.activity.products.find(product => product.skuId === sku.id);
  415. if (product) {
  416. sku.price = product.seckillPrice;
  417. sku.stock = Math.min(sku.stock, product.stock);
  418. } else { // 找不到可能是没配置,则不能发起秒杀
  419. sku.stock = 0;
  420. }
  421. // 设置限购数量
  422. if (this.activity.totalLimitCount > 0 && this.activity.singleLimitCount > 0) {
  423. sku.limitCount = Math.min(this.activity.totalLimitCount, this.activity.singleLimitCount);
  424. } else if (this.activity.totalLimitCount > 0) {
  425. sku.limitCount = this.activity.totalLimitCount;
  426. } else if (this.activity.singleLimitCount > 0) {
  427. sku.limitCount = this.activity.singleLimitCount;
  428. }
  429. });
  430. // 处理滚动条
  431. setTimeout(() => {
  432. this.infoScroll();
  433. }, 1000);
  434. // 设置或下载分销需要的图片
  435. // #ifdef H5
  436. this.storeImage = spu.picUrl;
  437. this.make();
  438. this.ShareInfo();
  439. // #endif
  440. // #ifdef MP
  441. this.getQrcode();
  442. this.imgTop = spu.picUrl;
  443. // #endif
  444. // #ifndef H5
  445. this.downloadFilestoreImage();
  446. // #endif
  447. // 选中默认 sku
  448. this.selectDefaultSku();
  449. }).catch(err => {
  450. return this.$util.Tips({
  451. title: err.toString()
  452. }, {
  453. tab: 3,
  454. url: 1
  455. });
  456. })
  457. },
  458. /**
  459. * 默认选中属性
  460. */
  461. selectDefaultSku: function() {
  462. const properties = this.attr.properties;
  463. // 获得选中的属性值的名字,例如说 "黑色,大",则 skuKey = ["黑色", "大"]
  464. let skuKey = undefined;
  465. for (let key in this.skuMap) {
  466. if (this.skuMap[key].stock > 0) {
  467. skuKey = key.split(",");
  468. break;
  469. }
  470. }
  471. if (!skuKey) { // 如果找不到,则选中第一个
  472. skuKey = Object.keys(this.skuMap)[0].split(",");
  473. }
  474. // 使用 index 属性表示当前选中的,值为属性值的名字
  475. for (let i = 0; i < properties.length; i++) {
  476. this.$set(properties[i], "index", skuKey[i]);
  477. }
  478. let sku = this.skuMap[skuKey.join(",")];
  479. if (!sku) {
  480. return
  481. }
  482. this.$set(this.attr.productSelect, "spuName", this.spu.name);
  483. this.$set(this.attr.productSelect, "id", sku.id);
  484. this.$set(this.attr.productSelect, "picUrl", sku.picUrl);
  485. this.$set(this.attr.productSelect, "price", sku.price);
  486. this.$set(this.attr.productSelect, "stock", sku.stock);
  487. this.$set(this.attr.productSelect, "cart_num", 1);
  488. this.$set(this.attr.productSelect, "limitCount", sku.limitCount); // 秒杀活动特有字段
  489. this.$set(this, "attrValue", skuKey.join(","));
  490. },
  491. /**
  492. * 打开 SKU 属性的选择
  493. */
  494. openAttr: function() {
  495. this.$set(this.attr, 'cartAttr', true);
  496. },
  497. /**
  498. * 关闭 productWindow 弹窗
  499. */
  500. closeAttr: function() {
  501. this.$set(this.attr, 'cartAttr', false);
  502. },
  503. /**
  504. * 属性变动赋值
  505. *
  506. * @param newSkuKey 新的 skuKey
  507. * @param propertyIndex properties 的下标
  508. * @param valueIndex values 的下标
  509. */
  510. ChangeAttr: function(newSkuKey, propertyIndex, valueIndex) {
  511. // SKU
  512. let sku = this.skuMap[newSkuKey];
  513. if (!sku) {
  514. return;
  515. }
  516. this.$set(this.attr.productSelect, "id", sku.id);
  517. this.$set(this.attr.productSelect, "picUrl", sku.picUrl);
  518. this.$set(this.attr.productSelect, "price", sku.price);
  519. this.$set(this.attr.productSelect, "stock", sku.stock);
  520. this.$set(this.attr.productSelect, "limitCount", sku.limitCount);
  521. this.$set(this.attr.productSelect, "cart_num", 1);
  522. // SKU 关联属性
  523. this.$set(this.attr.properties[propertyIndex], 'index',
  524. this.attr.properties[propertyIndex].values[valueIndex].name);
  525. this.$set(this, "attrValue", newSkuKey);
  526. },
  527. /**
  528. * 购物车数量加和数量减
  529. *
  530. * @param changeValue true 增加;false 减少
  531. */
  532. ChangeCartNum: function(changeValue) {
  533. // 获取当前 sku
  534. let sku = this.attr.productSelect;
  535. if (!sku) {
  536. return;
  537. }
  538. // 设置数量
  539. let stock = sku.stock || 0;
  540. let limitCount = sku.limitCount;
  541. if (changeValue) {
  542. sku.cart_num++;
  543. if (limitCount > 0 && sku.cart_num > limitCount) {
  544. this.$set(this.attr.productSelect, "cart_num", limitCount);
  545. this.$util.Tips({
  546. title: `该商品每次限购 ${sku.limitCount} ${this.spu.unitName}`
  547. });
  548. } else if (sku.cart_num > stock) {
  549. this.$set(this.attr.productSelect, "cart_num", stock);
  550. }
  551. } else {
  552. sku.cart_num--;
  553. if (sku.cart_num < 1) {
  554. this.$set(this.attr.productSelect, "cart_num", 1);
  555. }
  556. }
  557. },
  558. /**
  559. * 购物车手动填写
  560. *
  561. * @param number 数量
  562. */
  563. iptCartNum: function (number) {
  564. this.$set(this.attr.productSelect, 'cart_num', number ? number : 1);
  565. // 判断是否超限购
  566. let sku = this.attr.productSelect;
  567. let limitCount = sku.limitCount;
  568. if (limitCount !== undefined && number > limitCount) {
  569. this.$set(this.attr.productSelect, "cart_num", limitCount);
  570. this.$util.Tips({
  571. title: `该商品每次限购 ${sku.limitCount} ${this.spu.unitName}`
  572. });
  573. }
  574. },
  575. /**
  576. * 单独购买
  577. */
  578. openAlone: function() {
  579. uni.navigateTo({
  580. url: `/pages/goods_details/index?id=${this.activity.spuId}`
  581. })
  582. },
  583. /**
  584. * 下订单
  585. */
  586. goBuy: function() {
  587. // 未登录,需要跳转
  588. if (!this.isLogin) {
  589. toLogin();
  590. return;
  591. }
  592. // 【重要】如果 attr 组件未打开,此时需要先打开。等到选择完后,再立即购买
  593. if (!this.attr.cartAttr) {
  594. this.openAttr();
  595. return;
  596. }
  597. // 发起下单
  598. let sku = this.attr.productSelect;
  599. uni.navigateTo({
  600. url: '/pages/users/order_confirm/index?skuId=' + sku.id + '&count=' + sku.cart_num
  601. + '&seckillActivityId=' + this.id
  602. });
  603. },
  604. /**
  605. * 跳转到客服
  606. */
  607. kefuClick(){
  608. location.href = this.chatUrl;
  609. },
  610. // ========== 评价相关的方法 ==========
  611. /**
  612. * 获得商品评价列表
  613. */
  614. getProductReplyList: function() {
  615. ProductCommentApi.getCommentList(this.activity.spuId, 3).then(res => {
  616. this.reply = res.data;
  617. })
  618. },
  619. /**
  620. * 获得商品评价统计
  621. */
  622. getProductReplyCount: function() {
  623. ProductCommentApi.getCommentStatistics(this.activity.spuId).then(res => {
  624. const count = res.data.goodCount + res.data.mediocreCount + res.data.negativeCount;
  625. this.$set(this, 'replyChance', (100.0 * res.data.goodCount / count).toFixed(0));
  626. this.$set(this, 'replyCount', count);
  627. });
  628. },
  629. // ========== 收藏相关方法 ==========
  630. /**
  631. * 获得是否收藏
  632. */
  633. isFavoriteExists: function() {
  634. if (!this.isLogin) {
  635. return;
  636. }
  637. ProductFavoriteApi.isFavoriteExists(this.activity.spuId).then(res => {
  638. this.userCollect = res.data;
  639. });
  640. },
  641. /**
  642. * 收藏 / 取消商品
  643. */
  644. setCollect: function() {
  645. if (!this.isLogin) {
  646. toLogin();
  647. return;
  648. }
  649. // 情况一:取消收藏
  650. if (this.userCollect) {
  651. ProductFavoriteApi.deleteFavorite(this.activity.spuId).then(res => {
  652. this.$set(this, 'userCollect', false);
  653. })
  654. // 情况二:添加收藏
  655. } else {
  656. ProductFavoriteApi.createFavorite(this.activity.spuId).then(res => {
  657. this.$set(this, 'userCollect', true);
  658. })
  659. }
  660. },
  661. // ========== 分销相关的方法 ==========
  662. /**
  663. * 生成二维码,设置到 promotionCode 变量
  664. */
  665. make() {
  666. let href = location.href.split('?')[0] + "?id="+ this.id + "&spread=" + this.uid;
  667. uQRCode.make({
  668. canvasId: 'qrcode',
  669. text: href,
  670. size: this.qrcodeSize,
  671. margin: 10,
  672. success: res => {
  673. this.promotionCode = res;
  674. },
  675. complete: () => {},
  676. fail:res => {
  677. this.$util.Tips({
  678. title: '海报二维码生成失败!'
  679. });
  680. }
  681. })
  682. },
  683. /**
  684. * 设置微信公众号的分享标题、内容等信息
  685. */
  686. ShareInfo: function() {
  687. // 只处理微信环境
  688. if (!this.$wechat.isWeixin()) {
  689. return
  690. }
  691. const spu = this.spu;
  692. let href = location.href;
  693. href = href.indexOf("?") === -1 ?
  694. href + "?spread=" + this.uid :
  695. href + "&spread=" + this.uid;
  696. const configAppMessage = {
  697. title: spu.name,
  698. imgUrl: spu.picUrl,
  699. desc: spu.description,
  700. link: href
  701. };
  702. this.$wechat.wechatEvevt([
  703. "updateAppMessageShareData",
  704. "updateTimelineShareData",
  705. "onMenuShareAppMessage",
  706. "onMenuShareTimeline"
  707. ], configAppMessage);
  708. },
  709. /**
  710. * 获得商品的封面 base64
  711. */
  712. getImageBase64:function(images) {
  713. imageBase64({
  714. url: images
  715. }).then(res=>{
  716. this.imgTop = res.data.code
  717. })
  718. },
  719. /**
  720. * 获得小程序的二维码
  721. */
  722. getQrcode() {
  723. let data = {
  724. pid: this.uid,
  725. id: this.id,
  726. path: 'pages/activity/goods_seckill_details/index'
  727. }
  728. getQrcode(data).then(res=>{
  729. base64src(res.data.code, res => {
  730. this.promotionCode = res;
  731. });
  732. }).catch(err => {
  733. this.errT = err;
  734. });
  735. },
  736. /**
  737. * 生成海报
  738. */
  739. goPoster: function() {
  740. // 提示正在生成中
  741. uni.showLoading({
  742. title: '海报生成中',
  743. mask: true
  744. });
  745. this.posters = false;
  746. // 如果没有二维码图片,则说明加载失败,进行错误提示
  747. if(!this.promotionCode){
  748. uni.hideLoading();
  749. this.$util.Tips({
  750. title: this.errT
  751. });
  752. return
  753. }
  754. // 校验海报是否已经生成;如果失败,则进行错误提示
  755. setTimeout(() => {
  756. if (!this.imgTop) {
  757. uni.hideLoading();
  758. this.$util.Tips({
  759. title: '无法生成商品海报!'
  760. });
  761. }
  762. }, 1000);
  763. // 展示海报
  764. const that = this;
  765. let arrImagesUrlTop = '';
  766. uni.downloadFile({
  767. url: this.imgTop, //仅为示例,并非真实的资源
  768. success: (res) => {
  769. arrImagesUrlTop = res.tempFilePath;
  770. let arrImages = [that.posterbackgd, arrImagesUrlTop, that.promotionCode];
  771. const name = that.spu.name;
  772. const price = that.fen2yuan(that.spu.price);
  773. const marketPrice = that.fen2yuan(that.spu.marketPrice);
  774. setTimeout(() => {
  775. that.$util.PosterCanvas(arrImages, name, price, marketPrice,
  776. function(tempFilePath) {
  777. that.posterImage = tempFilePath;
  778. that.canvasStatus = true;
  779. uni.hideLoading();
  780. });
  781. }, 500);
  782. }
  783. });
  784. },
  785. /**
  786. * 关闭分享弹窗
  787. */
  788. closePosters: function() {
  789. this.posters = false;
  790. },
  791. /**
  792. * 隐藏海报
  793. */
  794. posterImageClose: function() {
  795. this.canvasStatus = false
  796. },
  797. /**
  798. * 获取海报产品图(解决跨域问题,只适用于小程序)
  799. */
  800. downloadFilestoreImage: function() {
  801. let that = this;
  802. uni.downloadFile({
  803. url: that.setDomain(that.spu.picUrl),
  804. success: function(res) {
  805. that.storeImage = res.tempFilePath;
  806. },
  807. fail: function() {
  808. return that.$util.Tips({
  809. title: ''
  810. });
  811. },
  812. });
  813. },
  814. /**
  815. * 替换安全域名
  816. */
  817. setDomain: function(url) {
  818. url = url ? url.toString() : '';
  819. // 本地调试打开,生产请注销
  820. if (url.indexOf("https://") > -1) {
  821. return url;
  822. }
  823. return url.replace('http://', 'https://');
  824. },
  825. /**
  826. * 分享打开
  827. */
  828. listenerActionSheet: function() {
  829. if (!this.isLogin) {
  830. toLogin();
  831. return
  832. }
  833. // #ifdef H5
  834. if (this.$wechat.isWeixin() === true) {
  835. this.weixinStatus = true;
  836. }
  837. // #endif
  838. this.posters = true;
  839. },
  840. /**
  841. * 微信小程序的保存图片到本机
  842. */
  843. // #ifdef MP
  844. savePosterPath: function() {
  845. let that = this;
  846. uni.getSetting({
  847. success(res) {
  848. if (!res.authSetting['scope.writePhotosAlbum']) {
  849. uni.authorize({
  850. scope: 'scope.writePhotosAlbum',
  851. success() {
  852. uni.saveImageToPhotosAlbum({
  853. filePath: that.posterImage,
  854. success: function(res) {
  855. that.posterImageClose();
  856. that.$util.Tips({
  857. title: '保存成功',
  858. icon: 'success'
  859. });
  860. },
  861. fail: function(res) {
  862. that.$util.Tips({
  863. title: '保存失败'
  864. });
  865. }
  866. })
  867. }
  868. })
  869. } else {
  870. uni.saveImageToPhotosAlbum({
  871. filePath: that.posterImage,
  872. success: function(res) {
  873. that.posterImageClose();
  874. that.$util.Tips({
  875. title: '保存成功',
  876. icon: 'success'
  877. });
  878. },
  879. fail: function(res) {
  880. that.$util.Tips({
  881. title: '保存失败'
  882. });
  883. },
  884. })
  885. }
  886. }
  887. })
  888. },
  889. // #endif
  890. // ========== 顶部 nav 相关的方法 ==========
  891. /**
  892. * 后退
  893. */
  894. returns: function() {
  895. uni.navigateBack()
  896. },
  897. /**
  898. * 点击指定 nav bar
  899. *
  900. * @param index 新的 navList 位置
  901. */
  902. tap: function(index) {
  903. this.$set(this, 'navActive', index);
  904. this.$set(this, 'lock', true);
  905. this.$set(this, 'scrollTop', index > 0 ? this.topArr[index] - (app.globalData.navHeight / 2)
  906. : this.topArr[index]);
  907. },
  908. /**
  909. * 滚动
  910. *
  911. * @param e 滚动事件
  912. */
  913. scroll: function(e) {
  914. const scrollY = e.detail.scrollTop;
  915. let opacity = scrollY / 200;
  916. opacity = opacity > 1 ? 1 : opacity;
  917. this.$set(this, 'opacity', opacity);
  918. this.$set(this, 'scrollY', scrollY);
  919. if (this.lock) {
  920. this.$set(this, 'lock', false)
  921. return;
  922. }
  923. // 设置选中的 nav
  924. for (let i = 0; i < this.topArr.length; i++) {
  925. if (scrollY < this.topArr[i] - (app.globalData.navHeight / 2) + this.heightArr[i]) {
  926. this.$set(this, 'navActive', i)
  927. break
  928. }
  929. }
  930. },
  931. /**
  932. * 处理器滚动条
  933. */
  934. infoScroll: function() {
  935. const topArr = [];
  936. const heightArr = [];
  937. for (let i = 0; i < this.navList.length; i++) {
  938. // 获取元素所在位置
  939. const query = wx.createSelectorQuery().in(this);
  940. const idView = "#past" + i;
  941. query.select(idView).boundingClientRect();
  942. query.exec(res => {
  943. const top = res[0].top;
  944. const height = res[0].height;
  945. topArr.push(top);
  946. heightArr.push(height);
  947. this.$set(this, 'topArr', topArr);
  948. this.$set(this, 'heightArr', heightArr);
  949. });
  950. }
  951. },
  952. fen2yuan(price) {
  953. return Util.fen2yuan(price)
  954. }
  955. },
  956. }
  957. </script>
  958. <style scoped lang="scss">
  959. .userEvaluation{
  960. i{
  961. display: inline-block;
  962. }
  963. }
  964. .product-con{
  965. .line1{
  966. width: 600rpx;
  967. }
  968. }
  969. .share-box {
  970. z-index: 1000;
  971. position: fixed;
  972. left: 0;
  973. top: 0;
  974. width: 100%;
  975. height: 100%;
  976. image {
  977. width: 100%;
  978. height: 100%;
  979. }
  980. }
  981. .generate-posters {
  982. width: 100%;
  983. height: 170rpx;
  984. background-color: #fff;
  985. position: fixed;
  986. left: 0;
  987. bottom: 0;
  988. z-index: 99;
  989. transform: translate3d(0, 100%, 0);
  990. transition: all 0.3s cubic-bezier(0.25, 0.5, 0.5, 0.9);
  991. border-top: 1rpx solid #eee;
  992. }
  993. .generate-posters.on {
  994. transform: translate3d(0, 0, 0);
  995. }
  996. .generate-posters .item {
  997. flex: 50%;
  998. text-align: center;
  999. font-size: 30rpx;
  1000. }
  1001. .generate-posters .item .iconfont {
  1002. font-size: 80rpx;
  1003. color: #5eae72;
  1004. }
  1005. .generate-posters .item .iconfont.icon-haibao {
  1006. color: #5391f1;
  1007. }
  1008. .navbar .header {
  1009. height: 96rpx;
  1010. font-size: 30rpx;
  1011. color: #050505;
  1012. background-color: #fff;
  1013. /* #ifdef MP */
  1014. padding-right: 95rpx;
  1015. /* #endif */
  1016. }
  1017. .icon-xiangzuo {
  1018. /* #ifdef H5 */
  1019. top: 30rpx !important;
  1020. /* #endif */
  1021. }
  1022. .navbar .header .item {
  1023. position: relative;
  1024. margin: 0 25rpx;
  1025. }
  1026. .navbar .header .item.on:before {
  1027. position: absolute;
  1028. width: 60rpx;
  1029. height: 5rpx;
  1030. background-repeat: no-repeat;
  1031. content: "";
  1032. background-image: linear-gradient(to right, #ff3366 0%, #ff6533 100%);
  1033. bottom: -10rpx;
  1034. left: 50%;
  1035. margin-left: -28rpx;
  1036. }
  1037. .navbar {
  1038. position: fixed;
  1039. background-color: #fff;
  1040. top: 0;
  1041. left: 0;
  1042. z-index: 999;
  1043. width: 100%;
  1044. }
  1045. .navbar .navbarH {
  1046. position: relative;
  1047. }
  1048. .navbar .navbarH .navbarCon {
  1049. position: absolute;
  1050. bottom: 0;
  1051. height: 100rpx;
  1052. width: 100%;
  1053. }
  1054. .icon-xiangzuo {
  1055. color: #000;
  1056. position: fixed;
  1057. font-size: 40rpx;
  1058. width: 100rpx;
  1059. height: 56rpx;
  1060. line-height: 54rpx;
  1061. z-index: 1000;
  1062. left: 33rpx;
  1063. }
  1064. .product-con .nav {
  1065. background-image: url('');
  1066. background-repeat: no-repeat;
  1067. background-size: 100% 100%;
  1068. width: 100%;
  1069. height: 110rpx;
  1070. padding: 0 30rpx;
  1071. box-sizing: border-box;
  1072. background-color: #E93323;
  1073. }
  1074. .product-con .nav .money {
  1075. font-size: 28rpx;
  1076. color: #fff;
  1077. }
  1078. .product-con .nav .money .num {
  1079. font-size: 48rpx;
  1080. }
  1081. .product-con .nav .money .y-money {
  1082. font-size: 26rpx;
  1083. margin-left: 10rpx;
  1084. text-decoration: line-through;
  1085. }
  1086. .product-con .nav .time {
  1087. font-size: 20rpx;
  1088. color: #fff;
  1089. text-align: center;
  1090. }
  1091. .product-con .nav .time .timeCon {
  1092. margin-top: 10rpx;
  1093. }
  1094. .product-con .nav .time .timeCon .num {
  1095. padding: 0 7rpx;
  1096. font-size: 22rpx;
  1097. color: #ff3d3d;
  1098. background-color: #fff;
  1099. border-radius: 2rpx;
  1100. }
  1101. .product-con .nav .timeState {
  1102. font-size: 28RPX;
  1103. color: #FFF;
  1104. }
  1105. .product-con .nav .iconfont {
  1106. color: #fff;
  1107. font-size: 30rpx;
  1108. margin-left: 20rpx;
  1109. }
  1110. .product-con .wrapper .introduce {
  1111. margin: 0;
  1112. }
  1113. .product-con .wrapper .introduce .infor {
  1114. width: 570rpx;
  1115. }
  1116. .product-con .wrapper .introduce .iconfont {
  1117. font-size: 36rpx;
  1118. color: #999999;
  1119. }
  1120. .product-con .wrapper .label {
  1121. margin: 18rpx 0 0 0;
  1122. font-size: 24rpx;
  1123. color: #82848f;
  1124. }
  1125. .product-con .wrapper .label .stock {
  1126. width: 255rpx;
  1127. margin-right: 28rpx;
  1128. }
  1129. .product-con .footer {
  1130. padding: 0 20rpx 0 30rpx;
  1131. position: fixed;
  1132. bottom: 0;
  1133. width: 100%;
  1134. box-sizing: border-box;
  1135. height: 100rpx;
  1136. background-color: #fff;
  1137. z-index: 99;
  1138. border-top: 1rpx solid #f0f0f0;
  1139. text-align: center;
  1140. }
  1141. .product-con .footer .item {
  1142. font-size: 18rpx;
  1143. color: #666;
  1144. }
  1145. .product-con .footer .item .iconfont {
  1146. text-align: center;
  1147. font-size: 40rpx;
  1148. }
  1149. .product-con .footer .item .iconfont.icon-shoucang1 {
  1150. color: #f00;
  1151. }
  1152. .product-con .footer .item .iconfont.icon-gouwuche1 {
  1153. font-size: 40rpx;
  1154. position: relative;
  1155. }
  1156. .product-con .footer .item .iconfont.icon-gouwuche1 .num {
  1157. color: #fff;
  1158. position: absolute;
  1159. font-size: 18rpx;
  1160. padding: 2rpx 8rpx 3rpx;
  1161. border-radius: 200rpx;
  1162. top: -10rpx;
  1163. right: -10rpx;
  1164. }
  1165. .product-con .footer .bnt {
  1166. width: 540rpx;
  1167. height: 76rpx;
  1168. }
  1169. .product-con .footer .bnt .bnts {
  1170. width: 270rpx;
  1171. text-align: center;
  1172. line-height: 76rpx;
  1173. color: #fff;
  1174. font-size: 28rpx;
  1175. }
  1176. .product-con .footer .bnt .joinCart {
  1177. border-radius: 50rpx 0 0 50rpx;
  1178. background-image: linear-gradient(to right, #fea10f 0%, #fa8013 100%);
  1179. }
  1180. .product-con .footer .bnt .buy {
  1181. border-radius: 0 50rpx 50rpx 0;
  1182. background-image: linear-gradient(to right, #fa6514 0%, #e93323 100%);
  1183. }
  1184. .setCollectBox {
  1185. font-size: 18rpx;
  1186. color: #666;
  1187. }
  1188. .bg-color-hui {
  1189. background: #bbbbbb !important;
  1190. }
  1191. .canvas {
  1192. position:fixed;
  1193. z-index: -5;
  1194. opacity: 0;
  1195. }
  1196. .poster-pop {
  1197. width: 450rpx;
  1198. height: 714rpx;
  1199. position: fixed;
  1200. left: 50%;
  1201. transform: translateX(-50%);
  1202. z-index: 99;
  1203. top: 50%;
  1204. margin-top: -357rpx;
  1205. }
  1206. .poster-pop image {
  1207. width: 100%;
  1208. height: 100%;
  1209. display: block;
  1210. }
  1211. .poster-pop .close {
  1212. width: 46rpx;
  1213. height: 75rpx;
  1214. position: fixed;
  1215. right: 0;
  1216. top: -73rpx;
  1217. display: block;
  1218. }
  1219. .poster-pop .save-poster {
  1220. background-color: #df2d0a;
  1221. font-size: :22rpx;
  1222. color: #fff;
  1223. text-align: center;
  1224. height: 76rpx;
  1225. line-height: 76rpx;
  1226. width: 100%;
  1227. }
  1228. .poster-pop .keep {
  1229. color: #fff;
  1230. text-align: center;
  1231. font-size: 25rpx;
  1232. margin-top: 10rpx;
  1233. }
  1234. .mask {
  1235. position: fixed;
  1236. top: 0;
  1237. left: 0;
  1238. right: 0;
  1239. bottom: 0;
  1240. background-color: rgba(0, 0, 0, 0.6);
  1241. z-index: 9;
  1242. }
  1243. .home-nav {
  1244. /* #ifdef H5 */
  1245. top: 20rpx !important;
  1246. /* #endif */
  1247. }
  1248. .home-nav {
  1249. color: #fff;
  1250. position: fixed;
  1251. font-size: 33rpx;
  1252. width: 56rpx;
  1253. height: 56rpx;
  1254. z-index: 999;
  1255. left: 33rpx;
  1256. background: rgba(190, 190, 190, 0.5);
  1257. border-radius: 50%;
  1258. &.on{
  1259. background: unset;
  1260. color: #333;
  1261. }
  1262. }
  1263. .home-nav .line {
  1264. width: 1rpx;
  1265. height: 24rpx;
  1266. background: rgba(255, 255, 255, 0.25);
  1267. }
  1268. .home-nav .icon-xiangzuo {
  1269. width: auto;
  1270. font-size: 28rpx;
  1271. }
  1272. </style>