Vue.js實現(xiàn)大轉(zhuǎn)盤抽獎總結(jié)及實現(xiàn)思路
大家好!先上圖看看本次案例的整體效果。

實現(xiàn)思路:
Vue component實現(xiàn)大轉(zhuǎn)盤組件,可以嵌套到任意要使用的頁面。
css3 transform控制大轉(zhuǎn)盤抽獎過程的動畫效果。
抽獎組件內(nèi)使用鉤子函數(shù)watch監(jiān)聽抽獎結(jié)果的返回情況播放大轉(zhuǎn)盤動畫并給用戶彈出中獎提示。
中獎結(jié)果彈窗,為抽獎組件服務(wù)。
實現(xiàn)步驟如下:
構(gòu)建api獎品配置信息和抽獎接口,vuex全局存放獎品配置和中獎結(jié)果數(shù)據(jù)信息。
api:
export default {
getPrizeList () {
let prizeList = [
{
id: 1,
name: '小米8',
img: 'https://i1.mifile.cn/f/i/g/2015/cn-index/m8-140.png'
},
{
id: 2,
name: '小米電視',
img: 'https://i1.mifile.cn/f/i/g/2015/TV4A-43QC.png'
}, {
id: 3,
name: '小米平衡車',
img: 'https://i1.mifile.cn/f/i/g/2015/cn-index/scooter-140!140x140.jpg'
}, {
id: 4,
name: '小米耳機',
img: 'https://c1.mifile.cn/f/i/g/2015/video/pinpai140!140x140.jpg'
}
]
return prizeList
},
lottery () {
return {
id: 4,
name: '小米耳機',
img: 'https://c1.mifile.cn/f/i/g/2015/video/pinpai140!140x140.jpg'
}
}
}
store:
import lotteryApi from '../../api/lottery.api.js'
const state = {
prizeList: [],
lotteryResult: {}
}
const getters = {
prizeList: state => state.prizeList,
lotteryResult: state => state.lotteryResult
}
const mutations = {
SetPrizeList (state, { prizeList }) {
state.prizeList = prizeList
},
SetLotteryResult (state, { lotteryResult }) {
state.lotteryResult = lotteryResult
}
}
const actions = {
getPrizeList ({ commit }) {
let result = lotteryApi.getPrizeList()
commit('SetPrizeList', { prizeList: result })
},
lottery ({ commit }) {
let result = lotteryApi.lottery()
commit('SetLotteryResult', { lotteryResult: result })
}
}
export default {
state,
getters,
mutations,
actions,
namespaced: true
}
編寫抽獎組件,為保證通用性,組件只負(fù)責(zé)播放抽獎結(jié)果。接收兩個數(shù)據(jù)和一個方法,如下:
數(shù)據(jù)一:預(yù)置的獎品列表數(shù)據(jù)(輪播獎品需要)
數(shù)據(jù)二:抽獎結(jié)果,播放抽獎動畫和彈出中獎結(jié)果需要
方法:抽獎動作,返回的抽獎結(jié)果數(shù)據(jù)即為數(shù)據(jù)二,響應(yīng)式傳遞給組件
大概代碼思路如下(僅供參考,不可運行)
<template>
<section>
<div class="lucky-item">
<img src="http://www.cnblogs.com/images/cnblogs_com/codeon/878827/o_backImage.jpg"
alt>
<div class="lucky-box">
<img src="http://www.cnblogs.com/images/cnblogs_com/codeon/878827/o_circle.jpg"
alt>
<ul id="wheel"
class="wheel-list"
:style="wheelStyle"
:class="transition">
<li v-for="(prize,index) in slotPrizes"
:style="{transform: 'rotate('+index*45+'deg)'}"
v-bind:key="index">
<div class="fan-item"
style="transform: rotate(15deg) skewY(45deg);"></div>
<div class="lucky-prize">
<h3>{{prize.name}}</h3>
</div>
</li>
</ul>
<div class="wheel-btn"
@click="$emit('lottery')">
<a>
<img src="http://images.cnblogs.com/cnblogs_com/codeon/878827/o_go.jpg"
alt>
</a>
</div>
</div>
<prize-pop :prize="lotteryResult"
v-if="showPrize"
@closeLotteryPop="showPrize=false" />
</div>
</section>
</template>
<script>
import PrizePop from './common/prize-pop.vue'
export default {
name: 'BigTurntable',
data () {
return {
isStart: false,
showPrize: false,
wheelStyle: { 'transform': 'rotate(0deg)' },
transition: 'transitionclear',
playTurns: 5 // 默認(rèn)先旋轉(zhuǎn)5圈
}
},
components: {
PrizePop
},
props: {
prizes: {
type: Array,
required: false
},
lotteryResult: {
type: Object,
default: () => { }
}
},
computed: {
slotPrizes () {
var self = this
console.log(self.prizes)
let prizeList = []
prizeList.push({ ...self.prizes[0], slotIndex: 1 })
prizeList.push({ name: '謝謝參與', slotIndex: 2 })
prizeList.push({ ...self.prizes[1], slotIndex: 3 })
prizeList.push({ name: '謝謝參與', slotIndex: 4 })
prizeList.push({ ...self.prizes[2], slotIndex: 5 })
prizeList.push({ name: '謝謝參與', slotIndex: 6 })
prizeList.push({ ...self.prizes[3], slotIndex: 7 })
prizeList.push({ name: '謝謝參與', slotIndex: 8 })
console.log(prizeList)
return prizeList
}
},
methods: {
/**
* 執(zhí)行抽獎動畫
*/
playWheel (index) {
},
/**
* 獲取中獎結(jié)果所在獎品列表中的索引,以確定抽獎動畫最終落在哪個獎品
*/
getPrizeIndex (prizeId) {
}
},
watch: {
/**
* 監(jiān)聽抽獎結(jié)果,一旦有中獎信息就開始執(zhí)行抽獎動畫
*/
lotteryResult (newVal, oldVal) {
var self = this
if (newVal.id && newVal.id > 0) {
let index = self.getPrizeIndex(newVal.id)
self.playWheel(index)
}
}
}
}
</script>
彈出中獎結(jié)果組件,依附于抽獎組件,在上一步的執(zhí)行抽獎結(jié)果動畫結(jié)束后執(zhí)行。
<template>
<div class="subject-pop" style="z-index: 10;" v-if="prize.id>0">
<div class="subject-pop-mask"></div>
<div class="subject-pop-box">
<h3>恭喜您</h3>
<p>
<img :src="prize.img" alt>
</p>
<h4>獲得
<span></span>
<span>{{prize.name}}</span>
</h4>
<div class="subject-pop-footer">
<a href="javascript:;" rel="external nofollow" class="november-btn1" @click="closeLotteryEmit">知道了</a>
</div>
</div>
</div>
</template>
<script>
export default {
props: {
prize: {
type: Object,
default: () => {
return {
id: 0
}
}
}
},
methods: {
closeLotteryEmit () {
this.$emit('closeLotteryPop')
}
}
}
</script>
抽獎組件運用在需要使用的頁面中,此頁面需要為抽獎組件提前準(zhǔn)備好預(yù)置獎品列表和中獎結(jié)果信息,并提供好抽獎方法供子組件(抽獎組件)觸發(fā),觸發(fā)完更改抽獎結(jié)果響應(yīng)式傳入到抽獎組件中。
<template>
<section>
<div style="width:100%;text-align:center;margin:2rem 0;">您有一次抽獎機會,祝君好運~~~</div>
<BigTurntable :prizes="prizeList"
:lotteryResult="lotteryResult"
@lottery="lottery" />
</section>
</template>
<script>
import { mapGetters, mapActions } from 'vuex'
import BigTurntable from '@/components/bigTurntable.vue'
export default {
name: 'BigTurntableRun',
created () {
var self = this
self.getPrizeList()
},
components: {
BigTurntable
},
computed: {
...mapGetters({
prizeList: 'lottery/prizeList',
lotteryResult: 'lottery/lotteryResult'
})
},
methods: {
...mapActions({
getPrizeList: 'lottery/getPrizeList',
lottery: 'lottery/lottery'
})
}
}
</script>
總結(jié)
以上所述是小編給大家介紹的Vue.js實現(xiàn)大轉(zhuǎn)盤抽獎總結(jié)及實現(xiàn)思路,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
如果你覺得本文對你有幫助,歡迎轉(zhuǎn)載,煩請注明出處,謝謝!
相關(guān)文章
windows下vue-cli及webpack搭建安裝環(huán)境
這篇文章主要介紹了windows下vue-cli及webpack搭建安裝環(huán)境,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-04-04
Vue中$emit調(diào)用父組件異步方法模擬.then實現(xiàn)方式
這篇文章主要介紹了Vue中$emit調(diào)用父組件異步方法模擬.then實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-09-09
VUE3中h()函數(shù)和createVNode()函數(shù)的使用解讀
這篇文章主要介紹了VUE3中h()函數(shù)和createVNode()函數(shù)的使用解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-08-08
如何用vue2+element-ui實現(xiàn)多行行內(nèi)表格編輯
最近開發(fā)項目,關(guān)于表格的數(shù)據(jù)操作比較多,這個地方個人覺得比較難搞,特此記錄一下,這篇文章主要給大家介紹了關(guān)于如何用vue2+element-ui實現(xiàn)多行行內(nèi)表格編輯的相關(guān)資料,需要的朋友可以參考下2024-08-08

