一文教你用JavaScript制作個簡單的大轉(zhuǎn)盤游戲
背景
日常生活中,我們經(jīng)常會見到形形色色的抽獎活動,例如九宮格、大轉(zhuǎn)盤等等……以前都沒去深入考慮過,如果讓我去做這些小游戲,有哪些需要注意的事項(xiàng),不試不知道,一試全是坑。正好最近有需求讓我做一個大轉(zhuǎn)盤游戲,那我也總結(jié)一下我的一些感想和經(jīng)驗(yàn)。
一、開始前的準(zhǔn)備
首先就是確定產(chǎn)品需求,仔細(xì)一看,emmm,就是正常的一個大轉(zhuǎn)盤該有的東西,也沒啥特殊要求,唯一需要注意的是大轉(zhuǎn)盤的轉(zhuǎn)盤個數(shù)需要動態(tài)變化,即用戶設(shè)置多少獎品我就需要顯示多少塊區(qū)域。
既然要做大轉(zhuǎn)盤,不外乎三個步驟:
- 畫出大轉(zhuǎn)盤
- 把獎勵放上去
- 讓大轉(zhuǎn)盤滾起來
這有啥難的,讓UI給我N個大轉(zhuǎn)盤背景圖,用戶設(shè)了多少獎品我就顯示對應(yīng)的背景圖... 哦,那樣圖片太多了,太麻煩了,也行吧... 那給我每個角度的扇形切圖一張,我轉(zhuǎn)一下不就出來了?... ???不給啊,那沒事了...(當(dāng)時的表情:冷靜分析...仔細(xì)思索...眉頭稍皺...好耶,我涼了呀)
二、畫出大轉(zhuǎn)盤
畫出大轉(zhuǎn)盤底圖部分就不必多說了,最難的部分在于把一個圓動態(tài)平均分成N份,并讓其在底圖上正常顯示。
我采用的方法是:用戶選擇了多少獎品(除2個獎品以外)我就在底圖上生成多少個寬高為底圖50%的div,目的是使每個div的右下角正好與底圖的中心點(diǎn)重合(這里有個坑,看下去就知道了),接著計(jì)算出對應(yīng)的圓心角 angle = 360 / n ,然后將其形變后以右下角為圓心旋轉(zhuǎn)對應(yīng)的角度拼成一個圓形。其中,形變和旋轉(zhuǎn)分別采用skew與rotate進(jìn)行實(shí)現(xiàn)。
注:
- skew形變的角度為
(90 - angle)deg; - 如果要給每一塊加上漸變色的話,由于使用了skew形變,所以要顯示從左到右的漸變的效果,就需要從原先正方形的底邊變到右邊,即:
background: linear-gradient('45deg', color1, color2)
如想看到更為細(xì)致的旋轉(zhuǎn)原理與demo,請點(diǎn)擊下方鏈接:
原理:https://www.w3cplus.com/css3/building-a-circular-navigation-with-css-transforms.html
demo:https://tympanus.net/Tutorials/CircularNavigation/interactivedemo/index.html
// 以下為N為8時,用純JS寫的一個測試demo
const n = 8; // 獎品數(shù)量
const circle = document.getElementById('circle');
for(let i = 0; i < n ; i++) {
let item = document.createElement('div');
item.className = `circle-item circle-item-${n}`;
// n為2時,旋轉(zhuǎn)原點(diǎn)不在右下角改為底部中點(diǎn),不需要skew形變,并且寬度不是50%而是100%
if(n !== 2) {
item.style=
`transform: rotate(${i * angle}deg) skew(${90 - angle}deg);
transform-origin: bottom right;
background-color: ${colorList[i % colorList.length]}; //設(shè)置了每個格子的背景色,可以不設(shè)置
`
} else {
item.style=
`transform: rotate(${i * angle}deg);
transform-origin: bottom;
background-color: ${colorList[i % colorList.length]};
`
}
circle.appendChild(item);
}
本以為算解決了一個難題,直到我開始測試時,發(fā)現(xiàn) n == 3時出現(xiàn)了奇怪的現(xiàn)象:

好嘛,當(dāng) n == 3時,只設(shè)置50%的寬高顯然并不能填充滿整個背景圖。于是更改了當(dāng) n != 2時的樣式,改為寬高的60%,并設(shè)置其初始位置往左和上各平移10%。
.circle-item {
border: 1px solid;
height: 60%;
width: 60%;
position: absolute;
left: -10%;
top: -10%;
}
這樣乍一看是實(shí)現(xiàn)了平均分的問題,但仔細(xì)一想還有一個問題,那便是如果加上指針,指針永遠(yuǎn)是向上的,這樣看起來就怪怪的了,所以還需要將整個圖像進(jìn)行一個旋轉(zhuǎn),使初始指針默認(rèn)指向第一塊的中心位置。
旋轉(zhuǎn)角度為: (180 - angle) / 2。

三、把獎勵放上去
把獎勵放上去我想到兩種方法:
1、給每個獎勵生成一個與底部背景圖寬高一樣的正方形,然后將其對著中心點(diǎn)旋轉(zhuǎn)對應(yīng)的角度,將其一層層的放在底部背景圖上,即可完成。如下圖1所示,每一個正方形的大小都是與底部的背景圖的寬高是一樣的,只是將其進(jìn)行一個旋轉(zhuǎn)即可。(當(dāng)時以為不好顯示用戶抽中的扇形的部分,就沒采用這種方法,現(xiàn)在總結(jié)了才發(fā)現(xiàn)只需要將顏色填充在每個扇形中即可,不用填充在獎勵背景上,如下圖2所示,應(yīng)該也是能實(shí)現(xiàn)的)

圖1 每一層獎勵的范圍

圖2 總結(jié)時想到的實(shí)現(xiàn)方法
2、將獎勵直接放在每一個扇形區(qū)域里面,這樣我就直接修改每個扇形的背景色即可??瓷先ズ芎唵?,但實(shí)際開始操作了就發(fā)現(xiàn)了兩個問題:
- 由于每一塊方格是skew形變出來的,獎勵直接放上去也會產(chǎn)生形變,需要將形變消除;
- 獎勵居中顯示很麻煩,到現(xiàn)在我也沒有找到可以套用的公式;
第一點(diǎn)解決起來較為簡單,只需要將獎勵進(jìn)行一個反向的skew變形,旋轉(zhuǎn)角度為:-(90 – (angle / 2))deg。 第二點(diǎn)我到現(xiàn)在也沒有找到可以套用的公式,是每一種給他寫了一個樣式居中的。(還好產(chǎn)品只要求2-6的獎勵數(shù)量,不然可能就沒有這篇文章了,如果大家有好的方法也可以教教我)
兩種方法各有優(yōu)點(diǎn): 方法一的優(yōu)點(diǎn)是不需要將獎勵進(jìn)行反向形變,怎么放上去就是怎么顯示; 方法二的優(yōu)點(diǎn)是獎勵與扇形不分離,我不需要額外的添加太多的div來實(shí)現(xiàn)獎勵正確顯示。
四、讓大轉(zhuǎn)盤滾起來
大轉(zhuǎn)盤到現(xiàn)在已經(jīng)完成了七七八八了,現(xiàn)在就差最后一步,讓其滾起來,點(diǎn)擊一次后滾動到對應(yīng)的位置然后停下。
采用的方法是給大轉(zhuǎn)盤添加一個旋轉(zhuǎn)的動畫,突然想起來 transform 中有 ease-in-out 這個過渡方法,即慢-快-慢的過渡效果,正好滿足我抽獎所需,于是直接采用了這種方法。
這里有個小細(xì)節(jié),由于大轉(zhuǎn)盤上本來就設(shè)置了transform的動畫效果,如果不想現(xiàn)在的旋轉(zhuǎn)動畫覆蓋掉之前的動畫,就需要單獨(dú)給動畫加上一層div,放在大轉(zhuǎn)盤底圖的外邊一層。
const circle = 8; // 旋轉(zhuǎn)圈數(shù) let circleAdd = 0; // 抽獎次數(shù),每次抽完自增 let rotateAngle = 0; // 旋轉(zhuǎn)角度 let getPrizeIndex = 0; // 抽到的獎品的index
在定義完上面的內(nèi)容之后計(jì)算需要旋轉(zhuǎn)的角度: rotateAngle = circle * 360 * circleAdd - angle * getPrizeIndex,將這個rotateAngle放入需要旋轉(zhuǎn)的style中即可完成旋轉(zhuǎn)。
這里走了兩次彎路,一次是沒有加上circleAdd,然后就發(fā)現(xiàn)第一次大轉(zhuǎn)盤會正常旋轉(zhuǎn),第二次開始就不動或者只往前滾不到360°甚至往回滾!原因是因?yàn)槊恳淮谓ostyle賦新值的時候,由于已經(jīng)有旋轉(zhuǎn)的角度了,他會在你設(shè)置新的旋轉(zhuǎn)角度的時候從上一個旋轉(zhuǎn)角度開始執(zhí)行動畫。
第二次是我在寫計(jì)算公式的時候,想當(dāng)然的把上面的“-”寫成了“+”,想的是我先旋轉(zhuǎn)了circle * circleAdd圈,然后還要滾 getPrizeIndex 個獎勵就是我需要顯示的獎勵,然后發(fā)現(xiàn)指針指向的獎勵與我需要選擇的 getPrizeIndex 并不相符。仔細(xì)檢查了才發(fā)現(xiàn)雖然大轉(zhuǎn)盤旋轉(zhuǎn)是順時針方向的,但是獎勵的讀取方向應(yīng)該是逆向的,所以應(yīng)該是“-”而不是“+”。
個人總結(jié)
在寫這個大轉(zhuǎn)盤之前我一直沒覺得大轉(zhuǎn)盤這種小游戲有多難,無非就是轉(zhuǎn)一下,然后獎勵顯示一下,這有啥難的?寫了之后才發(fā)現(xiàn)自己真的too young too simple,一個簡單的大轉(zhuǎn)盤里面也有不少的數(shù)學(xué)的公式需要學(xué)習(xí)計(jì)算,真的是走一步卡半天,一步一個坑。還是需要多多學(xué)習(xí)!
到此這篇關(guān)于一文教你用JavaScript制作個簡單的大轉(zhuǎn)盤游戲的文章就介紹到這了,更多相關(guān)JavaScript大轉(zhuǎn)盤游戲內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript四種調(diào)用模式和this示例介紹
JavaScript調(diào)用時除了聲明時定義的形參外,每個函數(shù)接受兩個附加參數(shù):this 和arguments,下面為大家介紹下JavaScript四種調(diào)用模式和this2014-01-01
按下Enter焦點(diǎn)移至下一個控件的實(shí)現(xiàn)js代碼
正如標(biāo)題所言使用js操作按下Enter焦點(diǎn)移至下一個控件,具體的實(shí)現(xiàn)示例如下,需要的朋友可以參考下2013-12-12
javaScript window.event.keyCode 集合與測試方法
javaScript window.event.keyCode 集合,對于事件的代碼獲取可以用腳本監(jiān)聽來實(shí)現(xiàn)。2010-05-05
js以對象為索引的關(guān)聯(lián)數(shù)組
在代碼邏輯中更多的是用關(guān)聯(lián)數(shù)組的方式。但即使是這樣我們也很少使用對象類型作為鍵值對的鍵名。2010-07-07
JavaScript中call,apply,bind的區(qū)別與實(shí)現(xiàn)
這篇文章主要介紹了JavaScript中call,apply,bind的區(qū)別與實(shí)現(xiàn),文章通過圍繞主題思想展開詳細(xì)的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-09-09

