Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 495|回复: 0

IE8下JQuery clone 出的select元素使用append添加option异常解决记录

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-7-13 00:50:55 | 显示全部楼层 |阅读模式

      遇到一个怪现象,由于配置参数是多实例的, 故采用JQuery对模板HTML代码进行clone,

    HTML代码中包括select标签, 在克隆之后需要对select进行添加option。

    在firefox和chrome浏览器上都没有问题,在IE10下也没有问题,

    但是在IE8下就出现问题,使用append添加option后,IE8上就显示不出来新添加option。

      示例代码如下,对于clone出的第二个select有问题,但是通过打印,发现添加后的option数目是正确的3个。

    这个就太令人费解了。

    <html>
    <head>
        <script src="./jquery.js"></script>
    </head>
    <body>
        <div name="template">
            <select>
            </select>
            <input type="button" name="testBtn" value="click me">
        </div>
     <script>
           
            $("[name='testBtn']").on("click",function(){
                alert("enter")
                var temp = $(this).parents("[name='template']");
                $("select", temp).empty();
               
                $("select", temp).append("<option value='auto'>auto</option>");
                $("select", temp).append("<option value='1'>1</option>");
                $("select", temp).append("<option value='2'>2</option>");
                alert("option len="+$("option", $("select", temp)).length)
            });
           
            $("[name='template']").clone(true).appendTo("body");
        </script>
    </body>
    </html>

     

      点击第二个select,下拉框内容页显示不出来,第一个select是原始select,是没有这个问题的。

    细细思讨怀疑可能是clone出来的副本样式渲染上没有更新,

    故在select在填充完option后, 主动做一次隐藏后再次显示的动作,select恢复正常。

    <html>
    <head>
        <script src="./jquery.js"></script>
    </head>
    <body>
        <div name="template">
            <select>
            </select>
            <input type="button" name="testBtn" value="click me">
        </div>
     <script>
           
            $("[name='testBtn']").on("click",function(){
                alert("enter")
                var temp = $(this).parents("[name='template']");
                $("select", temp).empty();
               
                $("select", temp).append("<option value='auto'>auto</option>");
                $("select", temp).append("<option value='1'>1</option>");
                $("select", temp).append("<option value='2'>2</option>");
                alert("option len="+$("option", $("select", temp)).length)
               $("select", temp).hide().show()
            });
           
            $("[name='template']").clone(true).appendTo("body");
        </script>
    </body>
    </html>

     

      但是这种规避方法,似乎也不好,每次给select替换option,都需要隐藏后再显示,给用户视觉带来冲击,控件闪烁,牺牲网页的可访问性(有违WCAG),故寻找其他保持select控件显示不变的方法。

      在http://bbs.csdn.net/topics/390559926找到相同问题讨论中的一则说明:

    IE 下的 option 不能当普通标签来看,appendChild,innerHTML...都不能用
    通过可以 select.options.app( new Option(text,value)   )

      真是高人,实验了appendChild确实不能添加option,于是借鉴此思路,为了保持JQuery append option string的写法, 即时不改变原有代码,通过新添加一个无用option,然后再删除它,来达到恢复select样式的目的。

      示例代码如下:

    <html>
    <head>
        <script src="./jquery.js"></script>
    </head>
    <body>
        <div name="template">
            <select>
                <option>jj</option>
            </select>
            <input type="text" value="heeh">
            <input type="button" name="testBtn" value="click me">
        </div>
        <script>
            
            $("[name='testBtn']").on("click",function(){
                //alert("enter")
                var temp = $(this).parents("[name='template']");
                $("select", temp).empty();
                
                $("select", temp).append("<option value='auto'>auto</option>");
                $("select", temp).append("<option value='1'>1</option>");
                $("select", temp).append("<option value='2'>2</option>");
                //alert("option len="+$("option", $("select", temp)).length);
                
                //$("select", temp).hide().show()
                
                var select = document.getElementsByTagName("select")[1]; var option = document.createElement("option"); select.add( option ); select.remove(select.length-1);
            });
            
            $("[name='template']").clone(true).appendTo("body");
            $("input[type='text']").eq(1).val("reset")
        </script>
    </body>
    </html>

     

      这种方法也是属于偏的方法,既然怀疑是样式问题,我想还是使用样式的方法来纠正,

    使用IE8调试器审查两个select看不出有啥异样,瞎试吧,select是行内元素,display:inline赋值试下果然OK:)

    但是第一次OK, 第二次之后还是有问题的,应该是每次给option添加后,需要出发样式的变化,才能解决这个问题,

    于是先赋值 inline-block 后改为inline,可以彻底解决这个问题。推荐这个方法。

    <html>
    <head>
        <script src="./jquery.js"></script>
    </head>
    <body>
        <div name="template">
            <select>
                <option>jj</option>
            </select>
            <input type="text" value="heeh">
            <input type="button" name="testBtn" value="click me">
        </div>
        <script>
            
            $("[name='testBtn']").on("click",function(){
                //alert("enter")
                var temp = $(this).parents("[name='template']");
                $("select", temp).empty();
                
                $("select", temp).append("<option value='auto'>auto</option>");
                $("select", temp).append("<option value='1'>1</option>");
                $("select", temp).append("<option value='2'>2</option>");
                //alert("option len="+$("option", $("select", temp)).length);
                
                //$("select", temp).hide().show()
                
                /*
                var select = document.getElementsByTagName("select")[1];
                var option = document.createElement("option");
                select.add( option );
                select.remove(select.length-1);*/ $("select", temp).css("display", "inline-block"); $("select", temp).css("display", "inline");
                
            });
            
            $("[name='template']").clone(true).appendTo("body");
            $("input[type='text']").eq(1).val("reset")
        </script>
    </body>
    </html>

     

      补充一种另外一种解决方法, 不使用向select中append option,

    而将select整体替换为 “<select><option></option></select>”, 上代码:

    <html>
    <head>
        <script src="./jquery.js"></script>
    </head>
    <body>
        <div name="template">
            <select>
            </select>
            <input type="button" name="testBtn" value="click me">
        </div>
     <script>
           
            $("[name='testBtn']").on("click",function(){
                alert("enter")
                var temp = $(this).parents("[name='template']");
               
                var selectStr = "<select>"
                    + "<option value='auto'>auto</option>"
                    + "<option value='1'>1</option>"
                    + "<option value='2'>2</option>"
                    + "</select>";
                //console.log(selectStr);
               
                $(selectStr).replaceAll($("select", temp));
                //$("select", temp).replaceWith(selectStr);
                alert("option len="+$("option", $("select", temp)).length)
            });
           
            $("[name='template']").clone(true).appendTo("body");
        </script>
    </body>
    </html>

     

     

      与大家分享下吧, 至于JQuery克隆为啥会把select样式弄乱,还请大侠赐教。

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-22 12:12 , Processed in 0.061972 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表