canvas實(shí)現(xiàn)動(dòng)態(tài)小球重疊效果
前面的話(huà)
在javascript運(yùn)動(dòng)系列中,詳細(xì)介紹了各種運(yùn)動(dòng),其中就包括碰壁運(yùn)動(dòng)。但是,如果用canvas去實(shí)現(xiàn),卻是另一種思路。本文將詳細(xì)介紹canvas動(dòng)態(tài)小球重疊效果
靜態(tài)小球
首先,生成隨機(jī)半徑、隨機(jī)位置的50個(gè)靜態(tài)小球
<button id="btn">按鈕</button> <canvas id="canvas" width="500" height="300" style="border:1px solid black">當(dāng)前瀏覽器不支持canvas,請(qǐng)更換瀏覽器后再試</canvas> <script> var canvas = document.getElementById('canvas'); var H=300,W=500; btn.onclick = function(){ getBalls(); } getBalls(); function getBalls(){ canvas.height = H; if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < 50; i++){ var tempR = Math.floor(Math.random()*255); var tempG = Math.floor(Math.random()*255); var tempB = Math.floor(Math.random()*255); cxt.fillStyle = 'rgb(' + tempR + ',' + tempG + ',' + tempB + ')'; var tempW = Math.floor(Math.random()*W); var tempH = Math.floor(Math.random()*H); var tempR = Math.floor(Math.random()*50); cxt.beginPath(); cxt.arc(tempW,tempH,tempR,0,Math.PI*2); cxt.fill(); } } } </script>
隨機(jī)運(yùn)動(dòng)
接著,這50個(gè)小球做隨機(jī)運(yùn)動(dòng),需要配合定時(shí)器更新小球的運(yùn)動(dòng)狀態(tài)。這時(shí),需要對(duì)上面代碼進(jìn)行改寫(xiě)
<button id="btn">更新</button> <canvas id="canvas" width="500" height="300" style="border:1px solid black">當(dāng)前瀏覽器不支持canvas,請(qǐng)更換瀏覽器后再試</canvas> <script> btn.onclick = function(){history.go();} var canvas = document.getElementById('canvas'); //存儲(chǔ)畫(huà)布寬高 var H=300,W=500; //存儲(chǔ)小球個(gè)數(shù) var NUM = 50; //存儲(chǔ)小球 var balls = []; function getBalls(){ if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < NUM; i++){ var tempR = Math.floor(Math.random()*255); var tempG = Math.floor(Math.random()*255); var tempB = Math.floor(Math.random()*255); var tempColor = 'rgb(' + tempR + ',' + tempG + ',' + tempB + ')'; var tempX = Math.floor(Math.random()*W); var tempY = Math.floor(Math.random()*H); var tempR = Math.floor(Math.random()*30+20); var tempBall = { x:tempX, y:tempY, r:tempR, stepX:Math.floor(Math.random() * 4 -2), stepY:Math.floor(Math.random() * 4 -2), color:tempColor, disX:Math.floor(Math.random() * 3 -1), disY:Math.floor(Math.random() * 3 -1) }; balls.push(tempBall); } } } function updateBalls(){ for(var i = 0; i < balls.length; i++){ balls[i].stepY += balls[i].disY; balls[i].stepX += balls[i].disX; balls[i].x += balls[i].stepX; balls[i].y += balls[i].stepY; } } function renderBalls(){ //重置畫(huà)布高度,達(dá)到清空畫(huà)布的效果 canvas.height = H; if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < balls.length; i++){ cxt.beginPath(); cxt.arc(balls[i].x,balls[i].y,balls[i].r,0,2*Math.PI); cxt.fillStyle = balls[i].color; cxt.closePath(); cxt.fill(); } } } getBalls(); clearInterval(oTimer); var oTimer = setInterval(function(){ //更新小球運(yùn)動(dòng)狀態(tài) updateBalls(); //渲染小球 renderBalls(); },50); </script>
碰壁檢測(cè)
下面,增加小球的碰壁檢測(cè)功能,當(dāng)小球碰壁時(shí),變?yōu)橄喾捶较?/p>
function bumpTest(ele){ //左側(cè) if(ele.x <= ele.r){ ele.x = ele.r; ele.stepX = -ele.stepX; } //右側(cè) if(ele.x >= W - ele.r){ ele.x = W - ele.r; ele.stepX = -ele.stepX; } //上側(cè) if(ele.y <= ele.r){ ele.y = ele.r; ele.stepY = -ele.stepY; } //下側(cè) if(ele.y >= H - ele.r){ ele.y = H - ele.r; ele.stepY = -ele.stepY; } }
<button id="btn">更新</button> <canvas id="canvas" width="500" height="300" style="border:1px solid black">當(dāng)前瀏覽器不支持canvas,請(qǐng)更換瀏覽器后再試</canvas> <script> btn.onclick = function(){history.go();} var canvas = document.getElementById('canvas'); //存儲(chǔ)畫(huà)布寬高 var H=300,W=500; //存儲(chǔ)小球個(gè)數(shù) var NUM = 30; //存儲(chǔ)小球 var balls = []; function getBalls(){ if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < NUM; i++){ var tempR = Math.floor(Math.random()*255); var tempG = Math.floor(Math.random()*255); var tempB = Math.floor(Math.random()*255); var tempColor = 'rgb(' + tempR + ',' + tempG + ',' + tempB + ')'; var tempR = Math.floor(Math.random()*30+20); var tempX = Math.floor(Math.random()*(W-tempR) + tempR); var tempY = Math.floor(Math.random()*(H-tempR) + tempR); var tempBall = { x:tempX, y:tempY, r:tempR, stepX:Math.floor(Math.random() * 13 -6), stepY:Math.floor(Math.random() * 13 -6), color:tempColor }; balls.push(tempBall); } } } function updateBalls(){ for(var i = 0; i < balls.length; i++){ balls[i].x += balls[i].stepX; balls[i].y += balls[i].stepY; bumpTest(balls[i]); } } function bumpTest(ele){ //左側(cè) if(ele.x <= ele.r){ ele.x = ele.r; ele.stepX = -ele.stepX; } //右側(cè) if(ele.x >= W - ele.r){ ele.x = W - ele.r; ele.stepX = -ele.stepX; } //上側(cè) if(ele.y <= ele.r){ ele.y = ele.r; ele.stepY = -ele.stepY; } //下側(cè) if(ele.y >= H - ele.r){ ele.y = H - ele.r; ele.stepY = -ele.stepY; } } function renderBalls(){ //重置畫(huà)布高度,達(dá)到清空畫(huà)布的效果 canvas.height = H; if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < balls.length; i++){ cxt.beginPath(); cxt.arc(balls[i].x,balls[i].y,balls[i].r,0,2*Math.PI); cxt.fillStyle = balls[i].color; cxt.closePath(); cxt.fill(); } } } getBalls(); clearInterval(oTimer); var oTimer = setInterval(function(){ //更新小球運(yùn)動(dòng)狀態(tài) updateBalls(); //渲染小球 renderBalls(); },50); </script>
重疊效果
canvas的合成屬性globalCompositeOperation表示后繪制的圖形怎樣與先繪制的圖形結(jié)合,屬性值是字符串,可能值如下: source-over(默認(rèn)):后繪制的圖形位于先繪制的圖形上方 source-in:后繪制的圖形與先繪制的圖形重疊的部分可見(jiàn),兩者其他部分完全透明 source-out:后繪制的圖形與先繪制的圖形不重疊的部分可見(jiàn),先繪制的圖形完全透明 source-atop:后繪制的圖形與先繪制的圖形重疊的部分可見(jiàn),先繪制的圖形不受影響 destination-over:后繪制的圖形位于先繪制的圖形下方,只有之前透明像素下的部分才可見(jiàn) destination-in:后繪制的圖形位于先繪制的圖形下方,兩者不重疊的部分完全透明 destination-out:后繪制的圖形擦除與先繪制的圖形重疊的部分 destination-atop:后繪制的圖形位于先繪制的圖形下方,在兩者不重疊的地方,先繪制的圖形會(huì)變透明 lighter:后繪制的圖形與先繪制的圖形重疊部分的值相加,使該部分變亮 copy:后繪制的圖形完全替代與之重疊的先繪制圖形 xor:后繪制的圖形與先繪制的圖形重疊的部分執(zhí)行"異或"操作
增加小球的重疊效果為'xor',即為最終的效果展示
<button id="btn">變換</button> <canvas id="canvas" width="500" height="300" style="border:1px solid black">當(dāng)前瀏覽器不支持canvas,請(qǐng)更換瀏覽器后再試</canvas> <script> btn.onclick = function(){history.go();} var canvas = document.getElementById('canvas'); //存儲(chǔ)畫(huà)布寬高 var H=300,W=500; //存儲(chǔ)小球個(gè)數(shù) var NUM = 30; //存儲(chǔ)小球 var balls = []; function getBalls(){ if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < NUM; i++){ var tempR = Math.floor(Math.random()*255); var tempG = Math.floor(Math.random()*255); var tempB = Math.floor(Math.random()*255); var tempColor = 'rgb(' + tempR + ',' + tempG + ',' + tempB + ')'; var tempR = Math.floor(Math.random()*30+20); var tempX = Math.floor(Math.random()*(W-tempR) + tempR); var tempY = Math.floor(Math.random()*(H-tempR) + tempR); var tempBall = { x:tempX, y:tempY, r:tempR, stepX:Math.floor(Math.random() * 21 -10), stepY:Math.floor(Math.random() * 21 -10), color:tempColor }; balls.push(tempBall); } } } function updateBalls(){ for(var i = 0; i < balls.length; i++){ balls[i].x += balls[i].stepX; balls[i].y += balls[i].stepY; bumpTest(balls[i]); } } function bumpTest(ele){ //左側(cè) if(ele.x <= ele.r){ ele.x = ele.r; ele.stepX = -ele.stepX; } //右側(cè) if(ele.x >= W - ele.r){ ele.x = W - ele.r; ele.stepX = -ele.stepX; } //上側(cè) if(ele.y <= ele.r){ ele.y = ele.r; ele.stepY = -ele.stepY; } //下側(cè) if(ele.y >= H - ele.r){ ele.y = H - ele.r; ele.stepY = -ele.stepY; } } function renderBalls(){ //重置畫(huà)布高度,達(dá)到清空畫(huà)布的效果 canvas.height = H; if(canvas.getContext){ var cxt = canvas.getContext('2d'); for(var i = 0; i < balls.length; i++){ cxt.beginPath(); cxt.arc(balls[i].x,balls[i].y,balls[i].r,0,2*Math.PI); cxt.fillStyle = balls[i].color; cxt.globalCompositeOperation = 'xor'; cxt.closePath(); cxt.fill(); } } } getBalls(); clearInterval(oTimer); var oTimer = setInterval(function(){ //更新小球運(yùn)動(dòng)狀態(tài) updateBalls(); //渲染小球 renderBalls(); },50); </script>
以上就是本文的全部?jī)?nèi)容,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,同時(shí)也希望多多支持腳本之家!
相關(guān)文章
如何自定義刪除無(wú)依賴(lài)文件的webpack插件
通過(guò)自定義webpack插件,利用執(zhí)行完成編譯的封存階段后,產(chǎn)生的產(chǎn)物module.fileDependencies,生成依賴(lài)的文件組,通過(guò)讀文件的方式,將待掃描的文件組和有依賴(lài)關(guān)系的文件進(jìn)行對(duì)比,這篇文章主要介紹了自定義刪除無(wú)依賴(lài)文件的webpack插件,需要的朋友可以參考下2023-12-12一文詳解preact的高性能狀態(tài)管理Signals
這篇文章主要介紹了一文詳解preact的高性能狀態(tài)管理Signals,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,感興趣的朋友可以參考一下2022-09-09js使用removeChild方法動(dòng)態(tài)刪除div元素
本節(jié)為大家介紹了js使用removeChild方法動(dòng)態(tài)刪除div元素,需要的朋友可以參考下2014-08-08JS判斷不同分辨率調(diào)用不同的CSS樣式文件實(shí)現(xiàn)思路及測(cè)試代碼
最近看一個(gè)網(wǎng)站,發(fā)現(xiàn)顯示器不同的分辨率,樣式文件調(diào)用的也不一樣,于是很好奇研究并寫(xiě)了一個(gè),經(jīng)測(cè)試感覺(jué)還不錯(cuò),感興趣的你可以來(lái)看看哦2013-01-01JavaScript使用sort函數(shù)實(shí)現(xiàn)漢字排序
JavaScript中的sort函數(shù)是一個(gè)強(qiáng)大且多用途的工具,能夠?qū)?shù)組的元素進(jìn)行排序,而漢字按照拼音排序又是一個(gè)常見(jiàn)需求,下面我們就來(lái)看看如何使用JavaScript實(shí)現(xiàn)漢字排序吧2023-12-12JavaScript對(duì)象拷貝與Object.assign用法實(shí)例分析
這篇文章主要介紹了JavaScript對(duì)象拷貝與Object.assign用法,結(jié)合實(shí)例形式分析了javascript深拷貝與淺拷貝以及Object.assign的功能與相關(guān)使用技巧,需要的朋友可以參考下2018-06-06js 鍵盤(pán)記錄實(shí)現(xiàn)(兼容FireFox和IE)
用js實(shí)現(xiàn)鍵盤(pán)記錄,要關(guān)注瀏覽器的三種按鍵事件類(lèi)型,即keydown,keypress和keyup,它們分別對(duì)應(yīng)onkeydown、onkeypress和onkeyup這三個(gè)事件句柄。一個(gè)典型的按鍵會(huì)產(chǎn)生所有這三種事件,依次是keydown,keypress,然后是按鍵釋放時(shí)候的keyup。2010-02-02Bootstrap柵格系統(tǒng)使用方法及頁(yè)面調(diào)整變形的解決方法
這篇文章主要介紹了Bootstrap柵格系統(tǒng)使用方法及頁(yè)面調(diào)整變形的解決方法,需要的朋友可以參考下2017-03-03