index.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  1. <template>
  2. <page-meta :page-style="lock ? 'overflow: hidden' : 'overflow: auto'" />
  3. <page :nav="false" ref="pageRef">
  4. <view
  5. class="detailWrapper"
  6. :style="{ backgroundImage: 'url(' + ossurl.box.headDetailbj + ')' }"
  7. >
  8. <view
  9. class="pagae-nav"
  10. :style="{ paddingTop: statusBarHeight + 'px', background: headerBg }"
  11. >
  12. <view class="flex-align-center relative" style="height: 80rpx">
  13. <text class="cuIcon-back back" @click.stop="back"></text>
  14. <view v-if="data" class="title line-ellipsis">
  15. {{ data && data.name }}
  16. </view>
  17. </view>
  18. </view>
  19. <view :style="{ height: 72 + statusBarHeight * 2 + 'rpx' }"></view>
  20. <top
  21. v-if="data"
  22. :data="data"
  23. :horse="horse"
  24. @onShowRecord="showRecord"
  25. @onShowUpdate="showUpdate"
  26. :type="1"
  27. />
  28. <view class="SwiperBox">
  29. <swiper
  30. class="productSwiper"
  31. :indicator-dots="false"
  32. circular="true"
  33. :autoplay="false"
  34. interval="5000"
  35. duration="500"
  36. :current="current"
  37. @change="handleChange"
  38. >
  39. <swiper-item
  40. v-for="(item, index) in products"
  41. :key="index"
  42. @click="showSwipe(index)"
  43. >
  44. <view class="imageBox">
  45. <image class="image" :src="item.cover" mode="aspectFit" />
  46. </view>
  47. <view
  48. class="level"
  49. :style="{ backgroundImage: 'url(' + ossurl.mine.butbj + ')' }"
  50. >
  51. {{ getLevel(item.level) }}({{ item.probability }}%)
  52. </view>
  53. </swiper-item>
  54. </swiper>
  55. <view class="prev" @click="handlePrev" v-if="current !== 0">
  56. <image :src="ossurl.box.detail.left" mode="aspectFit" />
  57. </view>
  58. <view class="next" @click="handleNext" v-if="products&&current !== products.length - 1">
  59. <image :src="ossurl.box.detail.right" mode="aspectFit" />
  60. </view>
  61. </view>
  62. <view class="swiperBottom">
  63. <image :src="ossurl.box.detail.bottom" style="width: 100%;" mode="widthFix" />
  64. </view>
  65. </view>
  66. <product-a
  67. v-if="products"
  68. :products="products"
  69. :levelist="data.luckGoodsLevelProbabilityList"
  70. :poolId="poolId"
  71. />
  72. <!-- <product-b v-if="products" :products="products" /> -->
  73. <explain v-if="products" />
  74. <view class="fill-height"></view>
  75. <view class="bottom paddingX10" v-if="data">
  76. <view class="bottom-btns flex-align-around">
  77. <view
  78. class="relative btnItem"
  79. @click="pay(1)"
  80. :style="{ backgroundImage: 'url(' + ossurl.box.detail.yfBtn + '); color: #000;' }"
  81. >
  82. 一发入魂
  83. <view v-if="$valid.isNumber2(data.specialPrice)" class="text">
  84. (¥{{ data.specialPrice }})
  85. </view>
  86. <view v-else class="text">(¥{{ data.price }})</view>
  87. </view>
  88. <view
  89. class="relative btnItem"
  90. @click="pay(5)"
  91. :style="{ backgroundImage: 'url(' + ossurl.box.detail.wlBtn + ')' }"
  92. >
  93. <view>
  94. 五连绝世
  95. <view v-if="data.specialPriceFive" class="text">
  96. (¥{{ data.specialPriceFive }})
  97. </view>
  98. <view v-else class="text">(¥{{ data.price * 5 }})</view>
  99. </view>
  100. </view>
  101. <view
  102. class="relative btnItem"
  103. @click="showFreedom"
  104. :style="{ backgroundImage: 'url(' + ossurl.box.detail.csBtn + ')' }"
  105. >
  106. 自由超神
  107. </view>
  108. </view>
  109. </view>
  110. <image class="ck" :src="ossurl.box.detail.warehouse" webp @click="showCK" mode="widthFix" />
  111. <image class="mj" :src="ossurl.box.detail.stone" webp @click="$router.push('magic')" v-if="user" mode="widthFix"/>
  112. <image class="record" :src="ossurl.box.detail.record" webp @click="showRecord" mode="widthFix" />
  113. <image class="rule" :src="ossurl.box.detail.rule" webp @click="showRule" mode="widthFix" />
  114. <image
  115. v-if="data && data.demoFlag === 1"
  116. class="sw"
  117. :src="ossurl.box.detail.demoBtn"
  118. webp
  119. @click="tryDemo"
  120. mode="widthFix"
  121. />
  122. <record ref="recordRef" @close="onClose" :poolId="poolId" />
  123. <checkout
  124. ref="checkoutRef"
  125. v-if="data"
  126. :data="data"
  127. @close="onClose"
  128. :poolId="poolId"
  129. @success="onSuccess"
  130. />
  131. <pay-freedom
  132. ref="freeRef"
  133. v-if="data"
  134. :data="data"
  135. @close="onClose"
  136. :poolId="poolId"
  137. @pay="pay"
  138. />
  139. <!-- <update-record ref="updateRef" :poolId="poolId" /> -->
  140. <image
  141. v-if="topVisible"
  142. :src="ossurl.common.topBtn"
  143. webp
  144. class="fixed"
  145. style="right: 30rpx; bottom: 300rpx; width: 68rpx; height: 98rpx"
  146. @click="$common.scrollTop()"
  147. />
  148. <ruleDialog ref="ruleDialog"></ruleDialog>
  149. </page>
  150. </template>
  151. <script>
  152. import top from './top'
  153. import levelRatio from './level_ratio'
  154. import productA from './product_a'
  155. import productB from './product_b'
  156. import explain from './explain'
  157. import record from './record'
  158. import official from './official'
  159. import checkout from './checkout'
  160. import updateRecord from './update_record'
  161. import loginMixin from '@/mixin/login'
  162. import prince from './prince'
  163. import payFreedom from './pay_freedom'
  164. import { throttle } from '@/utils'
  165. import ossurl from '@/utils/ossurl'
  166. import { LEVEL_MAP } from '@/utils/config'
  167. import ruleDialog from './ruleDialog'
  168. export default {
  169. mixins: [loginMixin],
  170. components: {
  171. ruleDialog,
  172. top,
  173. levelRatio,
  174. productA,
  175. productB,
  176. explain,
  177. record,
  178. official,
  179. checkout,
  180. updateRecord,
  181. prince,
  182. payFreedom
  183. },
  184. data() {
  185. return {
  186. current: 0,
  187. ossurl,
  188. poolId: null,
  189. data: null,
  190. products: null,
  191. horse: null,
  192. show: false,
  193. lock: false,
  194. topVisible: false,
  195. smartPayFlag: true,
  196. scrollTop: 0
  197. }
  198. },
  199. computed: {
  200. statusBarHeight() {
  201. if (this.$store.state.systemInfo) {
  202. return this.$store.state.systemInfo.statusBarHeight
  203. }
  204. return 20
  205. },
  206. headerBg() {
  207. return this.scrollTop > 0 ? '#fff' : 'transparent'
  208. },
  209. user() {
  210. return this.$store.state.userInfo
  211. },
  212. },
  213. onPageScroll(e) {
  214. this.scrollTop = e.scrollTop
  215. this.topVisible = e.scrollTop > 800
  216. },
  217. onLoad(options) {
  218. console.log(options)
  219. this.poolId = options.poolId
  220. this.poolIn()
  221. if(this.$store.state.hide){
  222. this.showRule()
  223. }
  224. },
  225. onShow() {},
  226. mounted() {
  227. setTimeout(() => {
  228. this.init(true)
  229. }, 100)
  230. },
  231. onUnload() {
  232. this.poolOut()
  233. },
  234. onShareAppMessage(res) {
  235. return {
  236. title: this.data.name,
  237. path: '/pages/award/index',
  238. imageUrl: this.data.cover
  239. }
  240. },
  241. onShareTimeline() {
  242. return {
  243. title: '燚火漫域 次元聚集地,潮玩新势力',
  244. query: 'poolId=' + this.poolId,
  245. imageUrl: this.data.cover
  246. }
  247. },
  248. methods: {
  249. showSwipe(index) {
  250. this.$router.push('award_detail_swipe', { poolId: this.poolId, index })
  251. },
  252. getLevel(level) {
  253. let item = LEVEL_MAP[level]
  254. if (item) {
  255. return item.title
  256. } else {
  257. return ''
  258. }
  259. },
  260. poolIn() {
  261. if (this.$common.isLogin()) {
  262. this.$service.award.poolIn(this.poolId)
  263. }
  264. },
  265. poolOut() {
  266. if (this.$common.isLogin()) {
  267. this.$service.award.poolOut(this.poolId)
  268. }
  269. },
  270. async init(loading = false) {
  271. this.getData(loading)
  272. this.getHorse()
  273. },
  274. async getData(loading = false) {
  275. const res = await this.$service.award.detail(this.poolId, loading)
  276. this.data = res
  277. res && (this.products = res.luckGoodsList)
  278. this.$refs.princeRef && this.$refs.princeRef.refresh()
  279. },
  280. async getHorse() {
  281. const res = await this.$service.award.detailHorse(this.poolId)
  282. this.horse = res
  283. },
  284. showRecord() {
  285. this.$refs.recordRef.show()
  286. this.lock = true
  287. },
  288. onClose() {
  289. this.lock = false
  290. },
  291. showUpdate() {
  292. // this.$refs.updateRef.show()
  293. },
  294. async pay(num) {
  295. throttle.call(() => {
  296. this.previewPay(num)
  297. })
  298. },
  299. async previewPay(num) {
  300. if (!this.$common.isLogin()) {
  301. this.$router.push('login')
  302. return
  303. }
  304. const res = await this.$service.award.preview(this.poolId, num)
  305. if (res) {
  306. this.$refs.checkoutRef.show(num, res)
  307. this.lock = true
  308. }
  309. },
  310. showFreedom() {
  311. if (!this.$common.isLogin()) {
  312. this.$router.push('login')
  313. return
  314. }
  315. this.$refs.freeRef.show()
  316. this.lock = true
  317. },
  318. back() {
  319. this.$router.back()
  320. },
  321. onSuccess(param) {
  322. const closeAnimal = this.$store.getters.closeAnimal
  323. if (param.num >= 10 || closeAnimal) {
  324. this.$router.push('lottery_more', param)
  325. } else {
  326. this.$cache.set(this.$cache.key.AWARD_GOODS, this.products)
  327. this.$router.push('lottery', param)
  328. }
  329. setTimeout(() => {
  330. this.init(false)
  331. }, 500)
  332. if (this.$store.state.hide == 1) {
  333. this.$service.user.getHide()
  334. this.pageConfig('magic_luck')
  335. }
  336. },
  337. async pageConfig(key) {
  338. const res = await this.$service.base.pageConfig(key)
  339. let pageConfig = this.$store.state.pageConfig || {}
  340. pageConfig[key] = res
  341. this.$store.commit(SET_PAGE_CONFIG, { ...pageConfig })
  342. },
  343. async tryDemo() {
  344. const res = await this.$service.award.tryDemo(this.poolId, 1)
  345. if (res) {
  346. this.$cache.set(this.$cache.key.AWARD_GOODS, this.products)
  347. this.$cache.set(this.$cache.key.AWARD_DEMO, res)
  348. this.$router.push('lottery', { num: 1 })
  349. this.getData()
  350. this.$event.emit(this.$event.key.AWARD_REFRESH)
  351. }
  352. },
  353. showCK() {
  354. this.$router.push('store')
  355. },
  356. closeSmartTitle() {
  357. this.smartPayFlag = false
  358. },
  359. showRule() {
  360. // this.$router.push('playing_method', { type: 1 })
  361. this.$refs.ruleDialog.visible = true
  362. },
  363. handleChange(e) {
  364. this.current = e.detail.current
  365. },
  366. handlePrev() {
  367. this.current -= 1
  368. },
  369. handleNext() {
  370. this.current += 1
  371. }
  372. }
  373. }
  374. </script>
  375. <style></style>
  376. <style lang="scss" scoped>
  377. .award-detail {
  378. padding-bottom: calc(136rpx + env(safe-area-inset-bottom));
  379. }
  380. .detailWrapper {
  381. background-size: 100% auto;
  382. background-repeat: no-repeat;
  383. height: 1240rpx;
  384. .SwiperBox {
  385. position: relative;
  386. .prev,
  387. .next {
  388. position: absolute;
  389. top: 20%;
  390. z-index: 90;
  391. image {
  392. width: 108rpx;
  393. height: 108rpx;
  394. }
  395. }
  396. .prev {
  397. left: 0;
  398. }
  399. .next {
  400. right: 0;
  401. }
  402. }
  403. .productSwiper {
  404. height: 750rpx;
  405. .imageBox {
  406. text-align: center;
  407. height: 40.5vh;
  408. font-size: 0;
  409. line-height: 0;
  410. .image {
  411. width: 520rpx;
  412. height: 40.5vh;
  413. animation: animateImage 3s;
  414. animation-play-state: running;
  415. animation-iteration-count: infinite;
  416. animation-timing-function: linear;
  417. }
  418. }
  419. .level {
  420. text-align: center;
  421. font-size: 24rpx;
  422. font-family: Alimama ShuHeiTi, Alimama ShuHeiTi;
  423. font-weight: 700;
  424. color: #333;
  425. background-size: 272rpx 62rpx;
  426. background-position: center;
  427. background-repeat: no-repeat;
  428. line-height: 62rpx;
  429. }
  430. @keyframes animateImage {
  431. from,
  432. to {
  433. transform: translate3d(0, 0, 0);
  434. }
  435. 25% {
  436. transform: translate3d(0, 10px, 0);
  437. }
  438. 50% {
  439. transform: translate3d(0, 0, 0);
  440. }
  441. 75% {
  442. transform: translate3d(0, -10px, 0);
  443. }
  444. 100% {
  445. transform: translate3d(0, 0, 0);
  446. }
  447. }
  448. }
  449. }
  450. .pagae-nav {
  451. position: fixed;
  452. left: 0;
  453. top: 0;
  454. width: 100%;
  455. z-index: 100;
  456. align-items: center;
  457. .title {
  458. font-size: 30rpx;
  459. font-weight: bold;
  460. text-align: center;
  461. line-height: 72rpx;
  462. width: 500rpx;
  463. }
  464. .back {
  465. z-index: 1800;
  466. font-size: 32rpx;
  467. line-height: 32rpx;
  468. position: absolute;
  469. left: 20rpx;
  470. top: 24rpx;
  471. }
  472. }
  473. .bg {
  474. position: fixed;
  475. z-index: -1;
  476. left: 0;
  477. right: 0;
  478. top: 0;
  479. width: 100%;
  480. height: 100%;
  481. }
  482. .smart_pay {
  483. height: 50rpx;
  484. background: linear-gradient(90deg, #3800b3 0%, #655b82 100%);
  485. margin: 20rpx 28rpx;
  486. border-radius: 20rpx;
  487. padding: 0 20rpx;
  488. color: #fff;
  489. line-height: 50rpx;
  490. .smart_rule {
  491. display: inline-block;
  492. width: 36rpx;
  493. height: 36rpx;
  494. line-height: 30rpx;
  495. margin: 0 0 0 10rpx;
  496. vertical-align: text-top;
  497. }
  498. .smart_X {
  499. position: absolute;
  500. right: 20rpx;
  501. top: 5rpx;
  502. width: 32rpx;
  503. height: 32rpx;
  504. }
  505. }
  506. .fill-height {
  507. height: 168rpx;
  508. }
  509. .bottom {
  510. position: fixed;
  511. left: 0;
  512. right: 0;
  513. bottom: 0;
  514. background: #373737;
  515. height: 168rpx;
  516. padding-bottom: 40rpx;
  517. .bottom-btns {
  518. height: 128rpx;
  519. .btnItem {
  520. font-size: 28rpx;
  521. font-family: Source Han Sans CN, Source Han Sans CN;
  522. font-weight: 700;
  523. color: #ffffff;
  524. background-size: 100% 100%;
  525. padding: 20rpx 12rpx;
  526. text-align: center;
  527. flex: 1;
  528. margin: 0 13rpx;
  529. display: flex;
  530. align-items: center;
  531. justify-content: center;
  532. .text {
  533. display: inline-block;
  534. font-size: 18rpx;
  535. line-height: 18rpx;
  536. }
  537. &:first-child {
  538. margin-left: 0;
  539. }
  540. &:last-child {
  541. margin-right: 0;
  542. }
  543. }
  544. }
  545. .image {
  546. height: 108rpx;
  547. width: 0;
  548. }
  549. }
  550. .sw,
  551. .mj,
  552. .ck {
  553. position: fixed;
  554. right: 0rpx;
  555. width: 84rpx;
  556. }
  557. .mj {
  558. bottom: 800rpx;
  559. }
  560. .sw {
  561. bottom: 540rpx;
  562. width: 108rpx;
  563. }
  564. .ck {
  565. bottom: 690rpx;
  566. }
  567. .record {
  568. position: fixed;
  569. left: 0rpx;
  570. width: 96rpx;
  571. bottom: 670rpx;
  572. }
  573. .rule {
  574. position: fixed;
  575. left: 0rpx;
  576. bottom: 800rpx;
  577. width: 84rpx;
  578. }
  579. </style>