原生js實現下拉框選擇元件
本文例項為大家分享了js實現下拉框選擇元件的具體程式碼,供大家參考,具體內容如下
功能需求:
1、點選div後,div顯示聚焦狀態,同時顯示下拉框內容;
2、選擇兒童人數後,如果兒童人數大於0,在下方出現對應的兒童年齡選擇框數量;
3、成人人數的選擇範圍是1-7,兒童人數的選擇範圍是0-4,兒童年齡的選擇範圍是<1、1-17;
4、點選確認按鈕後,將選擇好的成人人數和兒童人數顯示在最上方的div內;
5、可以控制選擇框是否可點選;
6、當顯示一個ul列表時,點選另一個ul列表,將上一個ul列表隱藏;
7、點選隱藏框內除繫結事件元素外,將正在顯示的ul列表隱藏;
8、點選頁面中任意空白位置,將顯示的下拉框內容整體隱藏;
下拉框不可操作時的顯示狀態:
下拉框可操作時:
選擇兒童人數後,下方自動出現對應數量的兒童年齡選擇框:
點選確認按鈕後,將結果顯示在是上方的div內:
剛開始的想法是對select、ul下拉列表、btn按鈕分別進行事件監聽,此外還要有當點選下拉框內其它位置時,ul下拉列表隱藏、當點選body時整個下拉框內容隱藏。監聽事件過多,而且事件冒泡也會影響事件的執行,導致某些事件會出現執行多次的情況。
兒童年齡的選擇框是根據兒童的人數來生成的,有幾個兒童,就有幾個年齡選擇框。這種情況下,年齡的選擇框肯定是動態建立的,無法針對年齡的select進行事件監聽,只能採用事件委託的形式,所以最後把對select、ul下拉列表、btn按鈕的點選事件,還有當點選container內其它位置時,ul下拉列表隱藏。全部委託給了dropDownContainer元素。
下面附上程式碼
html結構程式碼:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width,initial-scale=1.0"> <title>select</title> </head> <body> <script type="module"> import Main from './js/Main.js'; //引數為false時,選擇框不可點選;為true時,選擇框可使用 let main=new Main(true); main.appendTo("body"); </script> </body> </html>
Main.js檔案:
import Utils from './Utils.js'; export default class Main{ static styles=false; listPrep; constructor(state){ //state控制下拉框是否可點選 this.state=state; this.elem=this.createE(); } createE(){ if(this.elem) return this.elem; let div=Utils.createE("div"); div.className="guestsNum"; div.innerHTML=`<span>人數未定</span><i></i> <div class="dropDownContainer none" id="dropDownContainer"> <div class="dropDownItem clearfix"> <span>每間</span> <div class="dropDownSelect"> <div class="dropDownCont"><span id="adultNum">2 成人</span><i></i></div> <ul class="dropDownList" tag="adult">${this.setDropDownList("adult")}</ul> </div> <div class="dropDownSelect"> <div class="dropDownCont"><span id="childrenNum">0 兒童</span><i></i></div> <ul class="dropDownList" tag="children"><li>0</li>${this.setDropDownList("children")}</ul> </div> </div> <div class="dropDownItem clearfix none" id="ItemAge"></div> <div class="dropDownBottom clearfix"> ${this.state?'':'<em class="dropDownTips">請優先選擇日期,以便查詢實時價格。</em>'} ${this.state?'<a class="dropDownBtn" id="dropDownBtn" href="javascript:void(0)" rel="external nofollow" rel="external nofollow" >確認</a>':'<a class="dropDownBtn disabled" href="javascript:void(0)" rel="external nofollow" rel="external nofollow" >確認</a>'} </div> </div>`; //設定樣式,因為樣式只設置一次就好,不需要例項化,所以使用靜態方法 Main.setStyles(); //獲取元素 Utils.getIdElem(div,this); //監聽div的點選事件 div.addEventListener("click",(e)=>this.guestsNumClickHandler(e)); //如果state為true,下拉框監聽點選事件 if(this.state) this.dropDownContainer.addEventListener("click",e=>this.dropDownContainerClick(e)); //document監聽點選事件,隱藏下拉框 document.addEventListener("click",e=>this.documentClick(e)); return div; } appendTo(parent){ Utils.appendTo(this.elem,parent); } guestsNumClickHandler(e){ //如果下拉框是顯示狀態,則直接跳出,避免重複操作 if(!Utils.hasClass(this.dropDownContainer,"none")) return; //如果點選的不是guestsNum,直接跳出,避免事件冒泡 if(e.target.nodeName!=="SPAN"&&e.target.nodeName!=="I"&&!Utils.hasClass(e.target,"guestsNum")) return; //給div新增聚集樣式 Utils.addClass(this.elem,"focus"); //將dropDownContainer顯示 Utils.removeClass(this.dropDownContainer,"none"); } dropDownContainerClick(e){ if(e.target.nodeName==="LI"){ //點選ul選擇列表 this.dropDownListClick(e); } else if(e.target.id==="dropDownBtn"){ //點選確認按鈕 this.dropDownBtnClick(); } else if(e.target.nodeName==="SPAN" || e.target.nodeName==="I") { //點選span或者i標籤時,將它們的父元素div作為引數 this.dropDownSelectClick(e.target.parentElement); } else if(Utils.hasClass(e.target,"dropDownCont")){ //點選div選擇框時,將div作為引數 this.dropDownSelectClick(e.target); } else { //點選下拉框內其它位置時,讓當前的ul列表隱藏 if(this.listPrep) this.listPrep.style.display="none"; } } dropDownSelectClick(div){ //隱藏掉上一個顯示的ul列表 if(this.listPrep) this.listPrep.style.display="none"; //當前點選的ul列表賦值給this.listPrep this.listPrep=div.nextElementSibling; //將當前點選的ul列表顯示 this.listPrep.style.display="block"; } dropDownListClick(e){ //獲取當前點選的ul的tag屬性值 let tag=this.listPrep.getAttribute("tag"); let unit=""; switch (tag){ case "adult": unit="成人";break; case "children": unit="兒童"; let txt=Number(e.target.innerText); //根據li的數值,自動建立下面的年齡選擇框 this.setDropDownItemAge(txt); break; case "age": unit="歲";break; } //將選擇的li的值,顯示出來 this.listPrep.previousElementSibling.firstElementChild.textContent=e.target.innerText+" "+unit; //顯示完成後,將當前顯示的ul隱藏 this.listPrep.style.display="none"; } setDropDownItemAge(txt){ let str="<span>兒童年齡</span>"; if(txt===0){ //如果是0,則年齡選擇框不顯示 this.ItemAge.style.display="none"; }else{ this.ItemAge.style.display="block"; //迴圈選擇的數值,建立年齡選擇框 for(let i=0;i<txt;i++){ str+=`<div class="dropDownSelect"> <div class="dropDownCont"><span><1歲</span><i></i></div> <ul class="dropDownList" tag="age"><li><1</li>${this.setDropDownList("age")}</ul> </div>`; } this.ItemAge.innerHTML=str; } } dropDownBtnClick(){ //將選擇的內容顯示在最上方的select框內 let resultStr=this.adultNum.innerText.replace(/\s/g,"")+" "+this.childrenNum.innerText.replace(/\s/g,""); this.elem.firstElementChild.textContent=resultStr; //隱藏dropDownContainer this.dropDownContainerHide(); } documentClick(e){ //避免事件冒泡 if(e.target!==document.documentElement && e.target!==document.body) return; //隱藏dropDownContainer this.dropDownContainerHide(); } dropDownContainerHide(){ //div去掉聚集狀態 Utils.removeClass(this.elem,"focus"); //dropDownContainer隱藏 Utils.addClass(this.dropDownContainer,"none"); //隱藏當前顯示的ul列表 if(this.listPrep) this.listPrep.style.display="none"; } setDropDownList(type){ //建立ul下拉列表內容 let li=""; let max=0; switch (type){ case "adult": max=8;break; case "children": max=5;break; case "age": max=18;break; } for(let i=1;i<max;i++){ li+="<li>"+i+"</li>"; } return li; } static setStyles(){ if(Main.styles) return; Main.style=true; Utils.insertCss(".guestsNum",{ width:"108px",height:"34px",padding:"0px 12px",border:"1px solid #ccc",borderRadius:"3px",position:"relative",fontSize:"14px",color:"#666",userSelect:"none",}) Utils.insertCss(".guestsNum.focus",{ borderColor:"#ffa800",boxShadow:"0 0 4px #ffa800" }) Utils.insertCss(".guestsNum>span",{ lineHeight:"34px" }) Utils.insertCss(".guestsNum>i",{ display:"inline-block",width:"16px",height:"16px",backgroundImage:"url(./image/user.jpg)",float:"right",margin:"8px 0px 0px 10px" }) Utils.insertCss(".dropDownContainer",{ border: "1px solid #ffa800",borderRadius: "4px",boxShadow: "0 0 4px #ffa800",backgroundColor: "#fff",padding: "20px 15px",width: "480px",fontSize:"12px",position:"absolute",left:"0px",top:"35px",}) Utils.insertCss(".dropDownItem",{ marginBottom:"12px" }) Utils.insertCss(".dropDownItem>span",{ display:"block",width:"60px",lineHeight:"28px",float:"left",}) Utils.insertCss(".dropDownSelect",{ width:"90px",height:"30px",marginRight:"10px",position:"relative" }) Utils.insertCss(".dropDownCont",{ border:"1px solid #ccc",height:"12px",padding:"6px 8px 10px",}) Utils.insertCss(".dropDownCont>span",width:"53px",height:"14px",lineHeight:"14px",borderRight:"1px solid #ccc" }) Utils.insertCss(".dropDownCont>i",width:"0px",height:"0px",border:"5px solid #c6c6c6",borderColor:"#c6c6c6 transparent transparent",margin: "6px 0px 0px 4px",float: "right" }) Utils.insertCss(".dropDownList",{ listStyle:"none",padding:"0px",margin:"0px",width:"88px",maxHeight:"200px",overflow:"auto",cursor:"pointer",backgroundColor:"#fff",borderRadius:"4px",top:"30px",zIndex:"2",boxShadow: "1px 1px 3px rgba(0,.1)",display:"none" }) Utils.insertCss(".dropDownList>li",{ lineHeight:"28px",paddingLeft:"8px",}) Utils.insertCss(".dropDownList>li:hover",{ background:"#f4f4f4" }) Utils.insertCss(".dropDownBottom",{ borderTop:"1px solid #ccc",marginTop:"20px",paddingTop:"20px" }) Utils.insertCss(".dropDownTips",{ fontStyle:"normal",fontSize: "12px",color: "#ef523d",lineHeight:"28px" }) Utils.insertCss(".dropDownBtn",{ textDecoration:"none",float: "right",display: "inline-block",padding: "2px 22px",backgroundColor: "#ffb200",fontSize: "14px",lineHeight: "24px",color: "#fff",}) Utils.insertCss(".dropDownBtn.disabled",{ backgroundColor: "#efefef",color: "#999" }) Utils.insertCss(".clearfix:after",{ content:"\".\"",display:"block",overflow:"hidden",visibility:"hidden",clear:"both",height:"0px" }) Utils.insertCss(".none",{ display:"none" }) } }
Utils.js檔案:
export default class Utils{ static createE(elem,style,prep){ elem=document.createElement(elem); if(style) for(let prop in style) elem.style[prop]=style[prop]; if(prep) for(let prop in prep) elem[prop]=prep[prop]; return elem; } static appendTo(elem,parent){ if (parent.constructor === String) parent = document.querySelector(parent); parent.appendChild(elem); } static randomNum(min,max){ return Math.floor(Math.random*(max-min)+min); } static randomColor(alpha){ alpha=alpha||Math.random().toFixed(1); if(isNaN(alpha)) alpha=1; if(alpha>1) alpha=1; if(alpha<0) alpha=0; let col="rgba("; for(let i=0;i<3;i++){ col+=Utils.randomNum(0,256)+","; } col+=alpha+")"; return col; } static insertCss(select,styles){ if(document.styleSheets.length===0){ let styleS=Utils.createE("style"); Utils.appendTo(styleS,document.head); } let styleSheet=document.styleSheets[document.styleSheets.length-1]; let str=select+"{"; for(var prop in styles){ str+=prop.replace(/[A-Z]/g,function(item){ return "-"+item.toLocaleLowerCase(); })+":"+styles[prop]+";"; } str+="}" styleSheet.insertRule(str,styleSheet.cssRules.length); } static getIdElem(elem,obj){ if(elem.id) obj[elem.id]=elem; if(elem.children.length===0) return obj; for(let i=0;i<elem.children.length;i++){ Utils.getIdElem(elem.children[i],obj); } } static addClass(elem,className){ let arr=(elem.className+" "+className).match(/\S+/g); arr=arr.filter((item,index)=>arr.indexOf(item,index+1)<0) elem.className=arr.join(" "); } static removeClass(elem,className){ if(!elem.className) return; let arr=elem.className.match(/\S+/g); let arr1=className.match(/\S+/g); arr1.forEach(item=>{ arr=arr.filter(t=>t!==item) }) elem.className=arr.join(" "); } static hasClass(elem,className){ if(!elem.className) return false; let arr=elem.className.match(/\S+/g); let arr1=className.match(/\S+/g); let res; arr1.forEach(item=>{ res= arr.some(it=>it===item) }) return res; } }
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。