seckill.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458
  1. <template>
  2. <view class="app-container">
  3. <view class="bg-container">
  4. <image src="@/static/home/top_bg.png" mode="widthFix"></image>
  5. </view>
  6. <view class="top-container">
  7. <image src="@/static/home/top_bg2.png" mode="widthFix" class="front-bg" v-if="StatusBar <= 20"></image>
  8. <image src="@/static/home/top_bg3.png" mode="widthFix" class="front-bg" v-if="StatusBar > 20"></image>
  9. <view class="content" :style="'height:' + (StatusBar > 20 ? getpx(350) : getpx(300)) + 'px'">
  10. <view class="title" :style="cuStyle">
  11. <view class="left" @tap="toBack">
  12. <image src="@/static/icon/back.png"></image>
  13. </view>
  14. <view class="tit">限时秒杀</view>
  15. <view class="right"></view>
  16. </view>
  17. <view class="time-list">
  18. <block v-for="(item, index) in seckillTimeList" :key='index'>
  19. <view class="item" :class="seckillTimeCurrent == index ? 'current':''" @tap="changeSeckillTime(index)">
  20. <view class="time">{{item.startHour}}:00</view>
  21. <view class="tag" v-if="item.type == 'yjs'">已结束</view>
  22. <view class="tag" v-if="item.type == 'jxz'">{{countdownTime}}</view>
  23. <view class="tag" v-if="item.type == 'wks'">即将开始</view>
  24. </view>
  25. </block>
  26. </view>
  27. </view>
  28. </view>
  29. <view class="list-container">
  30. <scroll-view class="scroll-view" scroll-y :refresher-triggered="refresherTriggered" @scrolltolower="scrolltolower" @refresherrefresh="refresherrefresh" refresher-enabled :class="StatusBar > 20 ? 'higher':''">
  31. <block v-for="(item, index) in goodsList" :key='index'>
  32. <div class="item" @tap="toSeckillGoodsDetail(item.goodsId)">
  33. <image :src="item.imgUrl" mode="aspectFill"></image>
  34. <view class="right">
  35. <view class="title ellipsis-2">{{item.goodsName}}</view>
  36. <view class="des ellipsis-2">{{item.describeText ? item.describeText : ''}}</view>
  37. <view class="stock-sales">
  38. <view class="stock">
  39. <text>剩余{{item.secStockNum}}件</text>
  40. <view class="progress-box">
  41. <!-- 库存 / 总数 * 100 = 剩余百分比 -->
  42. <progress :percent="item.secStockNum / item.limitBuy * 100" activeColor="#FF3F42" active stroke-width="6" />
  43. </view>
  44. </view>
  45. <view class="sales">销量:{{item.salesVolume}}</view>
  46. </view>
  47. <view class="bottom">
  48. <view class="price">
  49. <view class="price-1">¥{{item.price | numToFixed}}</view>
  50. <view class="price-2">¥{{item.goodsPrice | numToFixed}}</view>
  51. </view>
  52. <view class="btn" v-if="seckillTimeList[seckillTimeCurrent].type == 'jxz'">马上抢</view>
  53. <view class="btn2" v-else>马上抢</view>
  54. </view>
  55. </view>
  56. </div>
  57. </block>
  58. <no-data v-if="!goodsList.length" :showText="'暂无商品'"></no-data>
  59. <loading-text v-if="goodsList.length" :loading="loading" :noMore="noMore" ></loading-text>
  60. </scroll-view>
  61. </view>
  62. <drag-button :isDock="true" :customBar="false" ref="dragButton"></drag-button>
  63. </view>
  64. </template>
  65. <script>
  66. import {mapState} from 'vuex';
  67. import dragButton from '@/components/drag-button.vue';
  68. export default {
  69. components:{
  70. dragButton
  71. },
  72. data() {
  73. return {
  74. scrollTop: 0, // 滚动高度(用于控制自定义导航)
  75. seckillTimeList: [], // 秒杀时间段列表
  76. seckillTimeCurrent: 0, // 秒杀时间段选择
  77. goodsList: [], // 秒杀商品列表
  78. pageNum: 1,
  79. pageSize: 8,
  80. noMore: false,
  81. loading: false,
  82. countdownTime: '', // 倒计时
  83. endDatetime: '', // 倒计时结束时间
  84. nowDate: null, // 当前时间
  85. dateInterval: null, // 时间定时器
  86. refresherTriggered: false,
  87. StatusBar: this.StatusBar
  88. }
  89. },
  90. computed:{
  91. cuStyle(){
  92. return `height:${this.CustomBar-this.StatusBar}px; padding-top:${this.StatusBar}px;`
  93. },
  94. ...mapState(['userInfo', 'isLogin', 'userId']),
  95. isLoaded() {
  96. uni.hideLoading();
  97. return this.isLoaded_banner && this.isLoaded_coupon && this.isLoaded_seckill && this.isLoaded_tab && this.isLoaded_myManager;
  98. }
  99. },
  100. watch: {
  101. nowDate() {
  102. let hh = this.nowDate.getHours(),
  103. mm = this.nowDate.getMinutes(),
  104. ss = this.nowDate.getSeconds();
  105. let hs = [10, 12, 15, 18, 20];
  106. if(mm == 0 && ss == 0 && hs.indexOf(hh) >= 0) {
  107. setTimeout(() => {
  108. this.getSeckillTimeList();
  109. }, 1000)
  110. }
  111. }
  112. },
  113. onShow() {
  114. this.$refs.dragButton.init();
  115. },
  116. onLoad() {
  117. this.getSeckillTimeList();
  118. },
  119. methods: {
  120. toBack(){
  121. uni.navigateBack({
  122. delta:1
  123. })
  124. },
  125. getpx(val) {
  126. return uni.upx2px(val);
  127. },
  128. // 下拉刷新
  129. refresherrefresh() {
  130. this.refresherTriggered = true;
  131. this.pageNum = 1;
  132. this.getSeckillGoodsList();
  133. },
  134. // 上拉加载
  135. scrolltolower() {
  136. this.getSeckillGoodsList(true);
  137. },
  138. findElem(array, attr, val) {
  139. for (var i = 0; i < array.length; i++) {
  140. if (array[i][attr] == val) {
  141. return i; //返回当前索引值
  142. }
  143. }
  144. return -1;
  145. },
  146. // 计算倒计时
  147. countTime() {
  148. let endDatetime = this.endDatetime.replace(/\-/g, '/');
  149. // console.log(endDatetime)
  150. var nowtime = new Date(), //获取当前时间
  151. endtime = new Date(endDatetime); //定义结束时间
  152. var lefttime = endtime.getTime() - nowtime.getTime(), //距离结束时间的毫秒数
  153. hh = Math.floor(lefttime/(1000*60*60)), //计算小时数
  154. mm = Math.floor(lefttime/(1000*60)%60), //计算分钟数
  155. ss = Math.floor(lefttime/1000%60); //计算秒数
  156. // console.log(new Date(endDatetime))
  157. function checkTime(i){
  158. if (i<10) {
  159. i = "0"+i;
  160. }
  161. return i;
  162. }
  163. setTimeout(() => {
  164. this.countTime();
  165. }, 1000);
  166. this.countdownTime = checkTime(hh) + ":" + checkTime(mm) + ":" + checkTime(ss);
  167. // console.log(this.countdownTime)
  168. },
  169. // 获取秒杀时间列表
  170. getSeckillTimeList() {
  171. this.$axios({
  172. url: '/goods/sec/time',
  173. method: 'get',
  174. params: {}
  175. }).then(res => {
  176. if(res.data.length < 1) {return false;}
  177. this.seckillTimeList = res.data;
  178. this.seckillTimeCurrent = this.findElem(this.seckillTimeList, 'type', 'jxz');
  179. if(this.seckillTimeCurrent == -1) {
  180. this.seckillTimeCurrent = 0;
  181. this.getSeckillGoodsList();
  182. }else {
  183. this.endDatetime = this.seckillTimeList[this.seckillTimeCurrent].endDatetime;
  184. this.countTime();
  185. this.getSeckillGoodsList();
  186. }
  187. }).finally(res => {
  188. this.isLoaded_seckill = true;
  189. })
  190. },
  191. // 切换秒杀时间
  192. changeSeckillTime(index) {
  193. this.seckillTimeCurrent = index;
  194. this.getSeckillGoodsList();
  195. },
  196. // 获取秒杀商品列表
  197. getSeckillGoodsList(loadMore) {
  198. if(this.noMore && loadMore)return;
  199. this.noMore = false;
  200. if(!loadMore){
  201. this.pageNum = 1;
  202. }else{
  203. this.loading = true;
  204. }
  205. let secKillId = this.seckillTimeList[this.seckillTimeCurrent].secKillId;
  206. this.$axios({
  207. url: '/goods/sec/goods/list',
  208. method: 'get',
  209. params: {
  210. pageNum: this.pageNum,
  211. pageSize: this.pageSize,
  212. secKillId: secKillId,
  213. }
  214. }).then(res => {
  215. let _list = res.data.records;
  216. let pageTotal = res.data.pages;
  217. if(this.pageNum >= pageTotal){
  218. this.noMore = true;
  219. }
  220. if (_list.length) {
  221. this.pageNum += 1
  222. }
  223. if (loadMore) {
  224. this.goodsList = this.goodsList.concat(_list);
  225. this.loading = false;
  226. } else {
  227. this.goodsList = _list;
  228. }
  229. }).finally(res => {
  230. this.refresherTriggered = false;
  231. })
  232. },
  233. // 进入秒杀商品详情
  234. toSeckillGoodsDetail(id) {
  235. if(!id) {
  236. return false;
  237. }
  238. if(this.seckillTimeList[this.seckillTimeCurrent].type == 'wks') {
  239. return this.$toast('活动未开始');
  240. }
  241. uni.navigateTo({
  242. url: '/packageGoods/pages/detail?id=' + id
  243. })
  244. },
  245. }
  246. }
  247. </script>
  248. <style lang="scss">
  249. .app-container {
  250. background: #F4F2F2;
  251. box-sizing: border-box;
  252. }
  253. .bg-container {
  254. position: fixed;
  255. top: 0;
  256. left: 0;
  257. z-index: 0;
  258. image {
  259. width: 750rpx;
  260. }
  261. }
  262. .top-container {
  263. position: relative;
  264. .front-bg {
  265. display: block;
  266. width: 750rpx;
  267. position: absolute;
  268. top: 0;
  269. z-index: 0;
  270. }
  271. .content {
  272. width: 750rpx;
  273. padding: 0 20rpx;
  274. position: relative;
  275. box-sizing: border-box;
  276. z-index: 1;
  277. display: flex;
  278. flex-direction: column;
  279. justify-content: space-between;
  280. }
  281. .title {
  282. display: flex;
  283. align-items: center;
  284. justify-content: space-between;
  285. color: #FFFFFF;
  286. font-size: 36rpx;
  287. .left {
  288. width: 50rpx;
  289. height: 50rpx;
  290. display: flex;
  291. align-items: center;
  292. justify-content: center;
  293. }
  294. image {
  295. width: 32rpx;
  296. height: 32rpx;
  297. display: block;
  298. }
  299. .right {
  300. width: 50rpx;
  301. height: 50rpx;
  302. }
  303. }
  304. .time-list {
  305. flex: 1;
  306. display: flex;
  307. overflow-x: scroll;
  308. align-items: center;
  309. .item {
  310. display: flex;
  311. flex-direction: column;
  312. align-items: center;
  313. flex-shrink: 0;
  314. width: 140rpx;
  315. margin-right: 48rpx;
  316. .time {
  317. font-size: 32rpx;
  318. color: #FFFFFF;
  319. }
  320. .tag {
  321. width: 140rpx;
  322. height: 44rpx;
  323. border-radius: 44rpx;
  324. border: 1px solid #FFFFFF;
  325. font-size: 24rpx;
  326. color: #FFFFFF;
  327. box-sizing: border-box;
  328. margin-top: 12rpx;
  329. display: flex;
  330. align-items: center;
  331. justify-content: center;
  332. }
  333. &.current {
  334. .tag {
  335. background: #FFFFFF;
  336. color: #FF3F42;
  337. }
  338. }
  339. }
  340. }
  341. }
  342. .list-container {
  343. padding: 0 20rpx;
  344. .scroll-view {
  345. height: calc(100vh - 300rpx);
  346. &.higher {
  347. height: calc(100vh - 350rpx);
  348. }
  349. }
  350. .item {
  351. background: #FFFFFF;
  352. border-radius: 10rpx;
  353. display: flex;
  354. padding: 20rpx;
  355. margin-bottom: 20rpx;
  356. image {
  357. display: block;
  358. width: 180rpx;
  359. height: 180rpx;
  360. flex-shrink: 0;
  361. }
  362. .right {
  363. width: 490rpx;
  364. box-sizing: border-box;
  365. padding-left: 20rpx;
  366. .title {
  367. font-size: 30rpx;
  368. color: #333333;
  369. line-height: 36rpx;
  370. font-weight: 600;
  371. }
  372. .des {
  373. font-size: 24rpx;
  374. line-height: 30rpx;
  375. color: #999999;
  376. margin-top: 6rpx;
  377. }
  378. .stock-sales {
  379. display: flex;
  380. justify-content: space-between;
  381. align-items: center;
  382. margin-top: 10rpx;
  383. font-size: 24rpx;
  384. color: #666666;
  385. .stock {
  386. display: flex;
  387. align-items: center;
  388. text {
  389. font-size: 24rpx;
  390. color: #666666;
  391. }
  392. .progress-box {
  393. width: 140rpx;
  394. border-radius: 6px;
  395. overflow: hidden;
  396. margin-left: 10rpx;
  397. }
  398. }
  399. }
  400. .bottom {
  401. display: flex;
  402. justify-content: space-between;
  403. align-items: center;
  404. margin-top: 10rpx;
  405. .price {
  406. display: flex;
  407. flex-direction: column;
  408. }
  409. .price-1 {
  410. font-size: 32rpx;
  411. color: #FF3F42;
  412. line-height: 36rpx;
  413. }
  414. .price-2 {
  415. font-size: 26rpx;
  416. color: #666666;
  417. line-height: 30rpx;
  418. text-decoration: line-through;
  419. }
  420. .btn {
  421. width: 110rpx;
  422. height: 44rpx;
  423. background: #FF3F42;
  424. border-radius: 5rpx;
  425. font-size: 28rpx;
  426. color: #FFFFFF;
  427. text-align: center;
  428. line-height: 44rpx;
  429. }
  430. .btn2 {
  431. width: 110rpx;
  432. height: 44rpx;
  433. background: #AAAAAA;
  434. border-radius: 5rpx;
  435. font-size: 28rpx;
  436. color: #FFFFFF;
  437. text-align: center;
  438. line-height: 44rpx;
  439. }
  440. }
  441. }
  442. }
  443. }
  444. </style>