IE6 DIV 覆盖 Select 元素 Form 控件

本为讲述了如何解决在IE6下动态创建的DIV等Element会被Select等Form控件覆盖的问题。

下拉框,即html的SELECT元素,.net设计时的DropDownList,是html中的windowed element,尤其ie6之后,几乎是唯一的windowed element(还有popup等少量极少用的的)。

普通的元素,textbox, div, table……这些,属于windowless element,它们之间互相遮盖的情况由z-index决定,在它们之上,是SELECT这些windowed element。所以一般情况下div、table等不能遮盖select。

这个问题广泛存在于各种弹出式控件的使用之中,比如日历控件等。

如果要显示div,以前的做法是,动态的,在显示的时候,让div区域的select不可见,div消失的时候,再恢复这些select元素。这种做法比较奇怪,因为它严格上并不是“遮盖”了select,而是,让她整个消失了,如果calendar弹出元素只是应该遮盖select元素的一部分,但select却整个不见,用户也许会觉得奇怪;做起来也麻烦,要用js逐一判断各select的位置。

ie5.5之后,有一个新的小技巧,称之为“iframe shim”(iframe加塞:p),可以真正的“遮盖”select元素。

它利用了一种特殊的元素:iframe。在ie5.5之前,iframe也是windowed element,但从5.5开始,iframe就是普通的windowless element了,可是,虽然是windowless element,iframe却可以盖住select。这种做法的原理就是:放一个iframe与你要显示的东西(比如说一个div)同样大小、位置,并设置z-index使得iframe在此DIV之下;这样,iframe遮盖了select,同时,iframe又在要显示的div的下面,div就露出来了。

限制:仅适用于ie5.5及以后版本。

参考文章链接:
http://dotnetjunkies.com/WebLog/jking/archive/2003/07/21/488.aspx

示例程序代码:
//html.select.hack.iframe shim.htm
<html>
<head>
 <script>
  function DivSetVisible(state)
  {
   var DivRef = document.getElementById(‘PopupDiv’);
   var IfrRef = document.getElementById(‘DivShim’);
   if(state)
   {
    DivRef.style.display = “block”;
    IfrRef.style.width = DivRef.offsetWidth;
    IfrRef.style.height = DivRef.offsetHeight;
    IfrRef.style.top = DivRef.style.top;
    IfrRef.style.left = DivRef.style.left;
    IfrRef.style.zIndex = DivRef.style.zIndex – 1;
    IfrRef.style.display = “block”;
   }
   else
   {
    DivRef.style.display = “none”;
    IfrRef.style.display = “none”;
   }
  }
 </script>
</head>
<body background=”/A-A-A/2005/07/17/20050717095153122477_1.gif”>
 <form>
  <select>
   <option>A Select Box is Born ….</option>
  </select>
 </form>
 <div
  id=”PopupDiv”
  style=”position:absolute;font:italic normal bolder 12pt Arial; top:25px; left:50px; padding:4px; display:none; color:#ffff00; z-index:100″>
  …. and a DIV can cover it up<br>through the help of an IFRAME.
 </div>
 <iframe
  id=”DivShim”
  src=”javascript:false;”
  scrolling=”no”
  frameborder=”0″
  style=”position:absolute; top:0px; left:0px; display:none;filter=progid:DXImageTransform.Microsoft.Alpha(style=0,opacity=0);”>
 </iframe>
 <br>
 <br>
  <a href=”#” onclick=”DivSetVisible(true)”>Click to show DIV.</a>
 <br>
 <br>
  <a href=”#” onclick=”DivSetVisible(false)”>Click to hide DIV.</a>
</body>
</html>

转自:http://www.dedecms.com/plus/view.php?aid=28759

Tags: , , , ,

5 comments

  1. 解决DIV遮盖select的方法

    由于IE存在一个bug,会使得DIV层无法遮盖select控件。
    我今天说的解决方法是用iframe找到的。其实这个问题的解决方法我今天在网上搜也会很多,不过大部分都是直接用html来实现的。但在实际应用中,我们会更倾向与用js动态生成对象来进行封装。

    首先这个实现方式是从google-suggest那得到的启发,起初看suggest的js代码的时候,发现它里面用到了iframe,但是不清楚和div究竟是怎么关联上而又不互相影响的。昨天突然想到可以用selenium的viewdom模块看到其生成的dom对象是如何的,后来发现,suggest是通过两个div来实现的,一个专门用于包含iframe,一个专门用于表现数据,然后再将这两个div重叠即可。很巧妙,但是我怎么没想到。这就应了那句话:简单的才是美的。

    现在开始说代码,hoho。
    首先是用于遮盖select的层:s

    // 创建层
    var s=document.createElement(“DIV”);

    // 设置层的相关属性
    s.style.visibility=””;

    // 定义层的样式
    s.style.position=”absolute”;
    s.style.left=”200″;
    s.style.top=”100″;
    s.style.width=”500″;
    s.style.height=”90″;
    s.style.border=”black 1px solid”;
    s.style.backgroundColor=”white”;

    // 生成iframe
    var L=document.createElement(“IFRAME”);
    L.name=”completionFrame”;

    // 定义iframe的样式,宽高与s相同
    L.width=s.style.width;
    L.height=s.style.height;

    // 附加L到s
    s.appendChild(L);

    // 显示s
    document.body.appendChild(s);

    其次是你要表现数据的DIV层:t
    js如下:

    // 创建层
    var t=document.createElement(“DIV”);

    // 设置相关属性
    t.style.visibility=””;
    t.zIndex=1;

    // 定义样式表,长宽高和定位等
    t.style.position=”absolute”;
    t.style.left=”200″;
    t.style.top=”100″;
    t.style.width=”500″;
    t.style.height=”90″;
    t.style.border=”blue 1px solid”;
    t.style.backgroundColor=”white”;

    // 定义你要表现的内容
    t.innerHTML = “Hello World”;

    // 显示
    document.body.appendChild(t);

    这样就大功告成了,经过测试,在ie和firefox下都是可以的。

    这里有几个问题需要说明一下:

    生成的L(IFRAME)我并没有指定其src,但是不影响效果。google-suggest里的iframe对象指定了src:http://www.google.com/webhp?complete=1&hl=en,其实就是一个空白页,我想这么写起来可能更严谨一些吧
    按照我上面写的代码,在实际显示的时候,包含iframe的div层可能会稍大一些,也就是会露出“白边”,大家在实际使用的时候可以进行调整
    注意两个层显示的顺序,也就是append到body的顺序,我是先让包含iframe的层显示,再让显示数据的层显示,如果调换,包含iframe的层就会遮住表现数据的层,即使是改两个层的zindex也不能解决,希望有人能关注一下哈
    如果你拷贝我的代码到你的script中,发现不能运行,有可能是你网页编码的问题,因为我的注释全都是用的中文,如果你用en或utf8输出的话,可能就会有问题
    即使是用iframe,在ie下仍然会有一个bug,就是如果其遮盖的地方有获得焦点的输入框时,它无法遮盖那个闪烁的光标。这个问题在gmail里面也没有解决。不过可以通过一些focus操作来解决这个小小的bug。

  2. div遮盖select问题

    select,object,iframe标签属于系统级别,所以如果div的z-index设为多少都无法遮盖select

    解决办法:
    可以将select的visibility样式改为hidden
    .selectClass{visibility:”hidden”;}

  3. <iframe id=backiframe style=”position:absolute; z-index:9; width:expression(this.nextSibling.offsetWidth); height:expression(this.nextSibling.offsetHeight); top:expression(this.nextSibling.offsetTop); left:expression(this.nextSibling.offsetLeft);” frameborder=”0″ ></iframe>
    <div id=”popupArea” style=”z-index:10;position:absolute;width:0;height:0; left:100px; top:50px;”></div>
    先用iframe 遮盖住select iframe大小位置自动跟随层变化

Leave a Reply

Your email address will not be published.

*