JavaScript實現(xiàn)煙花特效(面向?qū)ο?
更新時間:2021年09月10日 11:44:10 作者:山河不識
這篇文章主要為大家詳細介紹了JavaScript使用面向?qū)ο缶幊虒崿F(xiàn)煙花特效,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
本文實例為大家分享了JavaScript實現(xiàn)煙花特效的具體代碼,供大家參考,具體內(nèi)容如下
本特效使用面向?qū)ο缶幊?/strong>
分析
OOA
- 點擊觸發(fā)事件
- 煙花運動分成兩個階段
向上飛
爆炸
OOD
class FireWork{ constructor(){ } bindEvent(){ let _this = this; ele.onclick = function(){ //fly結(jié)束再去調(diào)用boom函數(shù) _this.fly(_this.boom); } } fly(boom){ } boom(){ } }
CSS設計實現(xiàn)
CSS代碼
*{ margin: 0; padding: 0; } #box{ width:800px; height:600px; position: relative; margin: 100px auto; background:#000000; border:2px solid red; overflow: hidden; } .ball{ width: 40px; height: 40px; border-radius: 50%; position: absolute; bottom: 0; }
JS編程實現(xiàn)
javascript代碼
<script src="./utils.js"></script> <script> // class FireWork{ constructor(){ this.box = document.getElementById("box"); this.box_offset = { left : box.offsetLeft, top : box.offsetTop } } bindEvent(){ let _this = this; this.box.onclick = function( e ){ e = e || event; _this.x = e.clientX - _this.box_offset.left; _this.y = e.clientY - _this.box_offset.top; _this.fly(_this.boom); } } fly( boom ){ let ele = this.createEle(); let _this = this; // 放入box之中; ele.style.left = this.x + "px"; let _left = this.x ; let _top = this.y ; animate( ele , { top : this.y } , function(){ ele.remove(); _this.boom( _left , _top ); }); } boom( x , y ){ let r = 100; let count = 0 ; let _this = this; for(let i = 0 ; i < 20 ; i ++){ let ele = this.createEle(); ele.style.left = x +"px"; ele.style.top = y + "px"; let _left = parseInt(Math.cos( Math.PI / 10 * i ) * r ) + x ; let _top = parseInt(Math.sin( Math.PI / 10 * i ) * r) + y animate( ele , { left : _left, top : _top, opacity : 0 } , function(){ ele.remove(); }) } } createEle(){ let ele = document.createElement("div"); ele.className = "ball"; ele.style.backgroundColor = `rgb(${parseInt(Math.random() * 255)},${parseInt(Math.random() * 255)},${parseInt(Math.random() * 255)})`; this.box.appendChild( ele ); return ele ; } } new FireWork().bindEvent(); </script>
引用的utils.js文件
function on(dom, event_name, handler_selector, delegation_handler) { if( typeof handler_selector === "string" && typeof delegation_handler === "function"){ return delegation(dom, event_name, handler_selector, delegation_handler); } // 在dom對象里面建立一個事件名 : 事件處理函數(shù)對應的數(shù)組; // 判定當前事件處理函數(shù)是否在dom對象之中; var event_type = "_" + event_name; if (event_type in dom) { dom[event_type].push(handler_selector); } else { dom[event_type] = [handler_selector]; } // 如果直接用事件名當成對象之中的key值,那么會和原有的dom功能名稱沖突; // 因為特殊的事件名會導致事件無法觸發(fā),所以我們在這里要對事件名進行拆分處理; dom.addEventListener(event_name.split(".")[0], handler_selector) } function off(dom, event_name) { // 獲取到dom對象里面事件名對應的一組事件處理函數(shù); var callback_list = dom["_" + event_name]; // 根據(jù)列表里面的所有函數(shù)進行事件移除 ; callback_list.forEach(function (event_handler) { dom.removeEventListener(event_name.split(".")[0], event_handler); }) // 清空dom對象里面的事件處理函數(shù)組; dom["_" + event_name].length = 0; } function trigger(dom, event_type) { dom.dispatchEvent(new Event(event_type)); } function delegation(dom, event_name, selector, event_handler) { dom.addEventListener(event_name, function (e) { e = e || event; var target = e.target || e.srcElement; while (target !== dom) { switch (selector[0]) { case ".": if (selector.slice(1) === target.className) { event_handler.call(target , e ) return; } case "#": if (selector.slice(1) === target.id) { event_handler.call(target, e) return; } default: if (selector.toUpperCase() === target.nodeName) { event_handler.call(target, e) return; } } target = target.parentNode; } }) } function animate( dom , attrs , callback , transition = "buffer", speed = 10 ){ // transition : 有兩種動畫方式 "buffer" , "liner" var _style = getComputedStyle( dom ); // - 1. 數(shù)據(jù)變形 ; for(var attr in attrs ){ attrs[attr] = { target : attrs[attr], now : _style[attr] } // - 2. 速度 : 勻速運動速度正負 ; if( transition === "liner" ){ attrs[attr].speed = attrs[attr].target > attrs[attr].now ? speed : - speed; } if( attr === "opacity"){ attrs[attr].target *= 100 attrs[attr].now *= 100 }else{ attrs[attr].now = parseInt(attrs[attr].now) } } // - 關(guān)閉開啟定時器; clearInterval( dom.interval ); dom.interval = setInterval( function(){ for(var attr in attrs ){ // 取出當前值和屬性值 ; // attrs[attr].target : 目標值; // attrs[attr].now : 當前值; let { target , now } = attrs[attr]; // 緩沖運動時候的速度; if( transition === "buffer" ){ var speed = (target - now ) / 10 ; speed = speed > 0 ? Math.ceil( speed ) : Math.floor( speed ); }else if(transition === "liner"){ var speed = attrs[attr].speed; } if( Math.abs(target - now) <= Math.abs( speed ) ){ if( attr === "opacity"){ dom.style[attr] = target / 100; }else{ dom.style[attr] = target + "px"; } delete attrs[attr]; for(var attr in attrs ){ // 如果有數(shù)據(jù)循環(huán)會執(zhí)行至少一次; return false; } clearInterval(dom.interval); typeof callback === "function" ? callback() : ""; }else{ now += speed; if( attr === "opacity"){ dom.style[attr] = now / 100; }else{ dom.style[attr] = now + "px"; } // 給對象賦值; attrs[attr].now = now; } } } , 30) }
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
JavaScript中子函數(shù)訪問外部變量的3種解決方法
任何在函數(shù)中定義的變量,都可認為是私有變量,因為不能在函數(shù)外部訪問這些變量,這篇文章主要給大家介紹了關(guān)于JavaScript中子函數(shù)訪問外部變量的3種解決方法,需要的朋友可以參考下2021-06-06Bootstrap中的Dropdown下拉菜單更改為懸停(hover)觸發(fā)
在使用bootstrap制作響應式導航條時,dropdown組件用的比較多,dropdown默認鼠標左鍵單擊才展開,如果使用鼠標放上去(hover)就展開則會省去點擊時間,這樣能提高效率,下面小編給大家解答下實現(xiàn)思路2016-08-08Javascript hasOwnProperty 方法 & in 關(guān)鍵字
hasOwnProperty :如果 object 具有指定名稱的屬性,那么方法返回 true;反之則返回 false。2008-11-11JavaScript設計模式之構(gòu)造函數(shù)模式實例教程
這篇文章主要介紹了JavaScript設計模式之構(gòu)造函數(shù)模式,結(jié)合實例形式分析了構(gòu)造函數(shù)模式的概念、功能、定義及使用方法,需要的朋友可以參考下2018-07-07