JavaScript實(shí)戰(zhàn)之帶收放動畫效果的導(dǎo)航菜單
雖然有很多插件可用,但為了共同提高,我做了一系列JavaScript實(shí)戰(zhàn)系列的實(shí)例,分享給大家,前輩們?nèi)粲泻玫慕ㄗh,請務(wù)必指出,免得誤人子弟??!
今天是第一戰(zhàn):帶收放動畫效果的菜單,效果如下圖:(樣式有點(diǎn)丑(-^-))
( 由于在寫本文時(shí),用的編輯器不同,暫時(shí)添加不了演示效果,這里有:最終完整代碼和演示 )
動畫效果:鼠標(biāo)hover改變所有目標(biāo)的背景和字體顏色,鼠標(biāo)移動到‘首頁導(dǎo)航',顯示下面的分組菜單,分組菜單有子菜單,點(diǎn)擊可縮放,帶動畫過度效果!而且,可以隨便添加和刪除導(dǎo)航菜單和子菜單,不影響效果!
如何實(shí)現(xiàn)呢?
第一步:用什么來實(shí)現(xiàn)菜單?HTML代碼設(shè)計(jì)如下,遵循JS代碼和HTML代碼分離的原則!這里你看不到一句JS代碼
未應(yīng)用樣式之前是這個(gè)樣子的:很古老吧?。?!
第二步:CSS樣式。鼠標(biāo)hover改變所有目標(biāo)的背景和字體顏色,直接用CSS的transition和:hover,而其他的CSS樣式布局就不全部列舉了,大家自己動手吧,主要注意以下幾點(diǎn):
#ul{ .... z-index: 100; } #ul li{ display: inline-block; position: relative; top: 0; left: -25px; width: 10%; min-width: 70px; height: 30px; text-align: center; line-height: 30px; border: 1px solid gray; border-radius:10px; background-color: aliceblue; cursor: pointer; -webkit-transition: all ease-in-out 0.3s; -moz-transition: all ease-in-out 0.3s; -ms-transition: all ease-in-out 0.3s; -o-transition: all ease-in-out 0.3s; transition: all ease-in-out 0.3s; } #ul li:hover{background-color: aquamarine;color: red;} ... .show-hide:hover{background-color: beige} .a-div{ background-color: aquamarine; border-radius:10px; color: black; display: none; opacity: 0 } .a{ z-index: -1; display: block; ... }
第三步:這一步是重點(diǎn)。如果給每個(gè)菜單選項(xiàng)和分組都添加事件監(jiān)聽,個(gè)人覺得好麻煩,且代碼量肯定多不少,有沒有什么辦法就在一個(gè)元素上加監(jiān)聽就能實(shí)現(xiàn)呢?
答案肯定是有的,利用事件的冒泡機(jī)制!在父元素ul標(biāo)簽上添加事件監(jiān)聽,而在監(jiān)聽函數(shù)里直接改變觸發(fā)事件的元素樣式就可以了,就這么簡單!
代碼如下:
var ul = document.getElementById('ul'); ul.addEventListener('mouseover',listener1,false); ul.addEventListener('mouseout',listener2,false); ul.addEventListener('click',listener3,false);
因?yàn)镮E8及以下版本沒有addEventListener,如果要兼容,還得加attachEvent對應(yīng)的代碼。
第四步:主角登場!實(shí)現(xiàn)listener1、listener2、listener3監(jiān)聽函數(shù)。
首先來最簡單的listener1函數(shù),代碼如下:
function listener1(event){ //event = event||window.event; //兼容IE8及以前版本 var target = event.target||event.srcElement; //兼容IE8及以前版本 if(target.tagName.toLowerCase() === 'li'){ var div1 = target.getElementsByTagName('div')[0]; div1.style.display = 'block'; var i = 0; var id; (function foo(){ if(i>=1){clearTimeout(id);id=null;return;} i+=0.2; div1.style.opacity = i; id = setTimeout(function(){clearTimeout(id);foo()},30); })(); } }
同樣,一切為了IE8及更舊版本?! ?/p>
1.因?yàn)樗膃vent沒有target屬性,只有相對應(yīng)得srcElement屬性
2.而這一句event = event||window.event;這里其實(shí)是可以省略的,只有當(dāng)用屬性來設(shè)置注冊事件監(jiān)聽時(shí),如ul.onmouseover = function(){},或<ul onmouseover='func'>,IE8及更舊版本只能通過window.event來取得當(dāng)前的Event對象
好了,現(xiàn)在獲得了當(dāng)前觸發(fā)事件的target,事情就簡單很多了,通過他就可以改變它自己和它的親戚!
下面是listener2函數(shù),用在mouseout時(shí)觸發(fā),主要是操控target的子元素DIV,代碼如下:
function listener2(event){ //event = event||window.event; var target = event.target||event.srcElement; if(target.tagName.toLowerCase() === 'li'){ var div1 = target.getElementsByTagName('div')[0]; div1.onmouseover = function(){ div1.style.display = 'block'; div1.style.opacity = 1; }; div1.onmouseout = function(){ div1.style.display = 'none'; div1.style.opacity = 0; }; div1.style.display = 'none'; //這一組是為了實(shí)現(xiàn)當(dāng)鼠標(biāo)從上方出去時(shí)隱藏div1 div1.style.opacity = 0; } }
好了,到這里,已經(jīng)實(shí)現(xiàn)了大部分效果了,還有最后一步,那就是1號主角了:listener3函數(shù),它主要負(fù)責(zé)鼠標(biāo)點(diǎn)擊時(shí)的縮放效果!
實(shí)現(xiàn)原理:
1.函數(shù)外面定義一個(gè)bool變量當(dāng)做開關(guān),鼠標(biāo)點(diǎn)一下開,再點(diǎn)一下關(guān);
2.通過setTimeout來實(shí)現(xiàn)動畫效果,動態(tài)的改變子菜單的height和opacity屬性,還有display屬性;
完整代碼如下:
var bool = true; function listener3(event) { var event = event || window.event; var target = event.target || event.srcElement; if (target.className === 'show-hide') { var parent = target.parentElement; var adiv = parent.getElementsByClassName('a-div')[0]; if (window.getComputedStyle(adiv,null).opacity>0.5){bool=false}else{bool=true} var height = 90, changeH, opacity, id; if (bool) { changeH = 0; opacity = 0; target.innerHTML = '財(cái)經(jīng) -'; (function show() { if (changeH > height) {clearTimeout(id);return} changeH += 5; opacity += 0.06; //console.log('opacity:'+adiv.style.opacity+',height :'+adiv.style.height); adiv.style.height = changeH + 'px'; adiv.style.opacity = opacity; adiv.style.display = 'block'; id = setTimeout(function () { clearTimeout(id); show(); }, 16.7); })(); bool = false; } else { changeH = height; opacity = 1; target.innerHTML = '財(cái)經(jīng) +'; (function hidden() { if (changeH < 0) {clearTimeout(id);adiv.style.display = 'none';return} changeH -= 10; opacity -= 0.11; //console.log('opacity:'+adiv.style.opacity+',height :'+adiv.style.height); adiv.style.height = changeH + 'px'; adiv.style.opacity = opacity; id = setTimeout(function () { clearTimeout(id); hidden(); }, 16.7); })(); bool = true; } } }
注意幾點(diǎn):
1.記得清除setTimeout的ID,然后退出,否則死循環(huán),如if (changeH < 0) {clearTimeout(id);adiv.style.display = 'none';return}
2.setTimeout的延遲時(shí)間設(shè)置為16.7是因?yàn)榉掀聊坏乃⑿侣?0FPS,看著舒服
3.調(diào)試過程中,設(shè)置changeH和opacity的遞增遞減值時(shí),記得打印出來,方便調(diào)試:
console.log('opacity:'+adiv.style.opacity+',height :'+adiv.style.height);
4.最后,整個(gè)菜單的實(shí)現(xiàn)中,最關(guān)鍵的是下面這一句,如果沒有這一句,你無法完美實(shí)現(xiàn)所有功能,比如:你點(diǎn)開一組子菜單,然后移動到其它組點(diǎn)擊的時(shí)候,情況將有很大不同;而window.getComputedStyle用這個(gè)的原因是,首次打開時(shí),點(diǎn)任意組的第一下都沒反應(yīng),因?yàn)橹苯油ㄟ^event.target在點(diǎn)第一下時(shí)是取不到opacity值的。
if (window.getComputedStyle(adiv,null).opacity>0.5){bool=false}else{bool=true};
不過,IE9以下不支持getComputedStyle方法,IE的Element對象有currentStyle屬性;
如果你對CSS的處理不是很熟悉,看看我的總結(jié):用原生JS讀寫CSS樣式的方法總結(jié)
如果你想多了解setTimeout()方法的應(yīng)用,看看這個(gè):你真的知道setTimeout是如何運(yùn)行的嗎
對于事件的處理機(jī)制,可以看看這個(gè):DOM中的事件處理概覽與原理的全面剖析
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- JS無限級導(dǎo)航菜單實(shí)現(xiàn)方法
- AngularJS+Bootstrap3多級導(dǎo)航菜單的實(shí)現(xiàn)代碼
- 基于JavaScript實(shí)現(xiàn)熔巖燈效果導(dǎo)航菜單
- js+div+css下拉導(dǎo)航菜單完整代碼分享
- JS中用三種方式實(shí)現(xiàn)導(dǎo)航菜單中的二級下拉菜單
- JS 實(shí)現(xiàn)導(dǎo)航菜單中的二級下拉菜單的幾種方式
- ABP框架中導(dǎo)航菜單的使用及JavaScript API獲取菜單的方法
- 原生js和jquery分別實(shí)現(xiàn)橫向?qū)Ш讲藛涡Ч?/a>
- javascript HTML+CSS實(shí)現(xiàn)經(jīng)典橙色導(dǎo)航菜單
- 以JavaScript來實(shí)現(xiàn)WordPress中的二級導(dǎo)航菜單的方法
- JS+CSS實(shí)現(xiàn)另類帶提示效果的豎向?qū)Ш讲藛?/a>
- JS+CSS實(shí)現(xiàn)簡單的二級下拉導(dǎo)航菜單效果
- JS實(shí)現(xiàn)移動端可折疊導(dǎo)航菜單(現(xiàn)代都市風(fēng))
相關(guān)文章
JavaScript實(shí)現(xiàn)經(jīng)典貪吃蛇游戲
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)經(jīng)典貪吃蛇游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-09-09uniapp微信小程序?qū)崿F(xiàn)一個(gè)頁面多個(gè)倒計(jì)時(shí)
這篇文章主要為大家詳細(xì)介紹了uniapp微信小程序?qū)崿F(xiàn)一個(gè)頁面多個(gè)倒計(jì)時(shí),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-11-11JavaScript樹形數(shù)據(jù)結(jié)構(gòu)處理
這篇文章主要介紹了JavaScript樹形數(shù)據(jù)結(jié)構(gòu)處理,文章圍繞主題展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下2022-07-07Bootstrap treeview實(shí)現(xiàn)動態(tài)加載數(shù)據(jù)并添加快捷搜索功能
本文實(shí)現(xiàn)了運(yùn)用bootstrap treeview實(shí)現(xiàn)動態(tài)加載數(shù)據(jù),并且添加快捷搜索功能,需要的朋友參考下2018-01-01