可输入也可选择的下拉框(多选或单选或输入)

将第一版改造成允许单选和多选(依然可以输入新项)

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>自定义下拉选择框</title>
    <style type="text/css">
        *{margin:0; padding:0;}
        ul,li{list-style:none;}
        :focus{outline: none;}
        input[type=text]{border:1px solid #ccc;}
        input[type=text]:hover,input[type=text]:focus{border-color: #57bc80; box-shadow: none;}
        
        body{padding:50px; font-size: 12px;}
        .my-select-box{position: relative;}
        .my-select-box .my-select-input{ height: 24px; line-height: 24px; padding:0 5px;
            background: url() right center no-repeat;
            padding-right:20px !important;
            width:100%;
            box-sizing: border-box;
        }
        .my-select-box .my-select-list{position:absolute; left:0; z-index: 101; border:1px solid #ccc; border-top:none; max-height: 120px; overflow-y: auto; display: none; background: #fff; width:100%; box-sizing: border-box;}
        .my-select-box .my-select-list li{height: 22px; line-height: 22px; padding:0 3px; cursor:pointer; margin-bottom: 1px}
        .my-select-box .my-select-list li.choosed{background: #ccc; color:#fff;}
        .my-select-box .my-select-list li:hover{background: #ccc; color:#fff;}
    </style>
</head>
<body>
    <div id="multiple" class="my-select-box" style="width:500px;">
        <input type="text" class="my-select-input" placeholder="可输入也可选择,输入多个时以|分隔" />
        <ul class="my-select-list">
            <li data-value="1">第一项</li>
            <li data-value="2">第二项</li>
            <li data-value="3">第三项</li>
            <li data-value="4">第四项</li>
            <li data-value="5">第5项</li>
            <li data-value="6">第6项</li>
            <li data-value="7">第7项</li>
            <li data-value="8">第8项</li>
        </ul>
    </div>
    <div style="height: 30px"></div>
    <div id="single" class="my-select-box">
        <input type="text" class="my-select-input" placeholder="可输入也可选择" maxlength="20" />
        <ul class="my-select-list">
            <li data-value="1">第一项</li>
            <li data-value="2">第二项</li>
            <li data-value="3">第三项</li>
            <li data-value="4">第四项</li>
            <li data-value="5">第5项</li>
            <li data-value="6">第6项</li>
            <li data-value="7">第7项</li>
            <li data-value="8">第8项</li>
        </ul>
    </div>
    <script type="text/javascript" src="jquery-3.2.1.min.js"></script>
    <script>    
        ;(function($){
            $.fn.MySelect=function(options){
                var defaults={
                    splitChar:'|',   //默认的分隔符
                    multiple:false   //是否多选
                };
                var opt=$.extend({},defaults,options);
                this.each(function(){
                    var $box=$(this);
                    var $input=$box.find("input.my-select-input");  //输入框
                    var $list=$input.next();                          //ul装扮成下拉框
                    var inputHeight=$input.outerHeight();   //计算input输入框的高度和宽度,方便定位ul和设置ul及包裹元素的宽度
                    //var inputWidth=$input.innerWidth();
                    $list.css({"top":(inputHeight)});
                    //$box.width($input.outerWidth());

                    $input.focus(function(){   //输入框获得焦点后,显示下拉选择ul
                        var $nextUl=$(this).next();
                        if($nextUl.children().length>0){
                            $(this).next().show();
                        }           
                    }).bind('input propertychange',function(){  //绑定监测输入框的输入值更改
                        var $this=$(this);
                        var curText=$this.val();
                        var $nextUl=$this.next();
                        var $liList=$nextUl.find("li").removeClass("choosed");
                        if(!opt.multiple){  //如果是单选
                            $this.attr("data-id","");   
                            if($liList.length>0){
                                $liList.each(function(i,item){
                                    var txt=$(item).text();                 
                                    if(txt===curText){
                                        var v=$(item).attr("data-value");
                                        $this.attr("data-id",v);
                                        $(item).addClass("choosed");
                                    }
                                });
                            }
                        }else{  //如果是多选
                            var inputValArr=curText?curText.split(opt.splitChar):[];                            
                            if(inputValArr.length>0){
                                for(var i=0; i<inputValArr.length; i++){
                                    var txtItem=inputValArr[i];
                                    if($liList.length>0){
                                        $liList.each(function(i,item){
                                            var txt=$(item).text();                 
                                            if(txt===txtItem){
                                                //var v=$(item).attr("data-value");
                                                //$this.attr("data-id",v);
                                                $(item).addClass("choosed");
                                            }
                                        });
                                    }
                                }
                            }
                        }
                    });                 
                    //修改成如下事件绑定,为了给动态添加的li也可以产生点击效果
                    $list.off('click', 'li').on('click', 'li', function (e) {
                        var $this = $(this);
                        var value = $this.attr("data-value") || '';                     
                        if(!opt.multiple){  //如果是单选
                            $input.val($this.text()).attr("data-id", value);
                            $this.addClass("choosed").siblings().removeClass("choosed");
                            $this.parent().hide();  //隐藏ul
                        }else{ //如果是多选的情况下,单击li项时,情形一:li项的text已经在输入框中,情形二:li项的text不在输入框中
                            //正则表达式去判断li的text是否在输入框中
                            var curInputVal=$input.val();
                            var inputValArr=curInputVal?curInputVal.split(opt.splitChar):[];
                            var liText=$this.text();
                            //^abc$|^abc\||\|abc\||\|abc$
                            var regStr='';
                            if(opt.splitChar==="|" ||opt.splitChar==="$"){  //如果分隔符是特殊字符要进行转义
                                regStr="^"+liText+"$"+"|^"+liText+"\\"+opt.splitChar+"|"+"\\"+opt.splitChar+liText+"\\"+opt.splitChar+"|"+"\\"+opt.splitChar+liText+"$";        
                            }else{
                                regStr="^"+liText+"$"+"|^"+liText+opt.splitChar+"|"+opt.splitChar+liText+opt.splitChar+"|"+opt.splitChar+liText+"$";
                            }                           
                            //console.log("正则串",regStr);
                            var reg=new RegExp(regStr);
                            if(reg.test(curInputVal)){  //说明当前li的值已经在input中存在,此时把li项移除
                                if(inputValArr.length>0){
                                    for(var m=0; m<inputValArr.length; m++){
                                        if(inputValArr[m]==liText){
                                            inputValArr.splice(m,1);
                                            m--;
                                            //break;
                                        }
                                    }                                   
                                }                               
                                $this.removeClass("choosed");
                            }else{  //如果不在输入框中,则把该项
                                inputValArr.push($this.text());                             
                                $this.addClass("choosed");
                            }   
                            $input.val(inputValArr.join(opt.splitChar));  //将input的值用分隔符组装起来赋值给输入框                  
                        }                       
                    });         
                });

                $(document).click(function (e) {  //点击.my-select-box范围外时隐藏ul下拉框
                    var target=e.target;
                    var $target=$(target);
                    var $parent=$target.closest('.my-select-box');
                    if($parent.length<1){  //说明不是.my-select-box范围内点击,将ul隐藏
                        $(".my-select-list").hide();
                    }else if($parent.length==1){  //如果存在多个my-select-box的情况,将其余的非这项以外的都隐藏
                        var $ul=$parent.find(".my-select-list");
                        var flag=$ul.is(":hidden");
                        $(".my-select-list").hide();
                        if(!flag) $ul.show();
                    }
                });
                return this;
            }
        })(jQuery);

        $("#multiple").MySelect({multiple:true});
        $("#single").MySelect();
    </script>
</body>
</html>
Paste_Image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,456评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,370评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,337评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,583评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,596评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,572评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,936评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,595评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,850评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,601评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,685评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,371评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,951评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,934评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,167评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,636评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,411评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,398评论 25 707
  • 1、【单选题】当向Excel工作表单元格输入公式时,使用单元格地址D$2引用D列2行单元格,该单元格的引用称为__...
    雾熏阅读 3,472评论 1 37
  • 1 风若没有手脚 怎么能稳稳抓住树枝 2 一片树林 雨水最期待的归乡 3 被树林锁住的昆虫与野兽 鸟群与河流 ——...
    朔邪阅读 251评论 1 3
  • 杭州的冬天是越来越冷了,再过一个多月马上就要到农历丁酉年了,好吧,其实就是2017年。 在中国,生肖文化尤其繁荣,...
    i图纹系阅读 837评论 4 20
  • 转眼已经出来实习整整两个月了,在很多同学才刚刚期末考试完出来实习或者还没找到实习单位的时候,我已经提前出来尝到了工...
    小龙虾米阅读 276评论 0 2