- 注册时间
- 2009-12-25
- 最后登录
- 2021-7-10
- 在线时间
- 3302 小时
- 阅读权限
- 200
- 积分
- 10
- 帖子
- 13416
- 精华
- 1
- UID
- 2036
  
|
通过Google搜索iframe 自适应高度,结果5W多条,搜索iframe 高度自适应,结果2W多条。
1 z0 X( p! u( k6 z) d我翻了前面的几十条,刨去大量的转载,有那么三五篇是原创的。而这几篇原创里面,基本上只谈到如何自适应静的东西,就是没有考虑到JS操作DOM之后,如何做动态同步的问题。另外,在兼容性方面,也研究的不彻底。
( x# D1 E2 ^$ m; `% B# J: I这篇文章,希望在这两个方面再做一些深入。 2 Q: `6 z5 w, C
可能有人还没接触到这个问题过,先说明一下,什么是高度自适应吧。所谓iframe高度自适应,就是,基于界面美观和交互的考虑,隐藏了iframe的border和scrollbar,让人看不出它是个iframe。如果iframe始终调用同一个固定高度的页面,我们直接写死iframe高度就可以了。而如果iframe要切换页面,或者被包含页面要做DOM动态操作,这时候,就需要程序去同步iframe高度和被包含页的实际高度了。
, c# t7 u, j" y! F顺便说下,iframe在迫不得已的时候才去用,它会给前端开发带来太多的麻烦。 4 s/ r6 C7 L8 R( Z
传统做法大致有两个:
V" j, T5 @9 O4 s; V4 w方法一,在每个被包含页在本身内容加载完毕之后,执行JS取得本页面的高度,然后去同步父页面的iframe高度。
* _$ S6 P5 M0 X9 W方法二,在主页面iframe的onload事件中执行JS,去取得被包含页的高度内容,然后去同步高度。 2 o9 g9 @% B6 n# G2 i
在代码维护角度考虑,方法二是优于方法一的,因为方法一,每个被包含页都要去引入一段相同的代码来做这个事情,创建了好多副本。
; M' O" Z! C t% [5 ?两个方法都只处理了静的东西,就是只在内容加载的时候执行,如果JS去操作DOM引起的高度变化,都不太方便。
, \" P3 m0 Y& O. k6 d3 d. V如果在主窗口做一个Interval,不停的来获取被包含页的高度,然后做同步,是不是即方便,又解决了JS操作DOM的问题了呢?答案是肯定的。
' M {7 s: ^, I h) I1 N: d9 q/ zDemo页面:主页面 iframe_a.html ,被包含页面 iframe_b.htm 和 iframe_c.html
5 e- a% E# @ H7 q主页面代码示例:
% I+ P( V; s4 C0 S& }. {: S- E<iframe id="frame_content" src="iframe_b.html" scrolling="no" frameborder="0"></iframe>
9 q7 p( X- U7 ^3 O9 C0 Q<script type="text/javascript"> ' A3 z: Q$ ^7 m. g9 ^6 ~8 p
function reinitIframe(){
" h* U% m- X# ~# I; U4 h! P) a6 ^var iframe = document.getElementById("frame_content");
+ m& y; d, _* `( t) K! r+ |try{ ( M1 [5 B9 p* N3 [1 v8 q
iframe.height = iframe.contentWindow.document.documentElement.scrollHeight; 5 |4 t! I3 v9 I8 B X
}catch (ex){} , t% D/ z! F# S' f/ v
} 9 Y3 [1 c$ z6 @# i3 ~+ E
window.setInterval("reinitIframe()", 200);
6 c/ q1 Y, q+ C2 V</script>一直执行,效率会不会有问题? 2 l1 F; j$ C8 E; C3 _( P' m
我做了测试,同时开5个窗口(IE6、IE7、FF、Opera、Safari)执行这个代码,不会对CPU有什么影响,甚至调整到2ms,也没影响(基本维持在0%占用率)。
, ~! l2 Z" \; m3 ?5 P, t下面谈谈各浏览器的兼容性问题,如何获取到正确的高度,主要是对body.scrollHeight和documentElement.scrollHeight两个值得比较。注意本文用的是这个doctype,不同的doctype应该不会影响结果,但是假如你的页面没有申明doctype,那还是先去加一个吧。
' h. t9 Q6 i" j<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">在主页面追加以下测试代码,以输出这两个值,代码示例:
4 n- M5 `& b9 f. d<div><button >Check Height</button></div> 1 a: E# n2 r# A8 c
<script type="text/javascript">
3 b% x0 T5 r7 l: C( ]* W& N/ Bfunction checkHeight() { 4 C: t0 x5 j1 \4 e c# Q! R- |
var iframe = document.getElementById("frame_content");
' w* K% e/ t- h2 `+ K. Cvar bHeight = iframe.contentWindow.document.body.scrollHeight;
% f& T1 R4 S% L* Q9 N, R( u }var dHeight = iframe.contentWindow.document.documentElement.scrollHeight; 1 I7 }3 v" v8 L! x2 K5 Y
alert("bHeight:" + bHeight + ", dHeight:" + dHeight); , E! j$ e; q& \: y. [
} " d0 S+ P2 ]! \ }* _5 i! r
</script>被加载页面,可以切换一个绝对定位的层,来使页面高度动态改变。如果层展开,则会撑高页面高度。代码示例: 9 ]4 i- c2 P2 {( g9 ?
<div> 9 w' a; N1 o, T7 s; I% |8 J. g" Y
<button >Toggle Overlay</button>
6 T5 _6 l8 p L5 a' A; m8 ~/ F</div> 6 U; s8 H# e+ b v
<div style="height:160px;position:relative"> 8 x- b) I3 [2 o7 S+ ^ T& u
<div id="overlay" style="position:absolute;width:280px;height:280px;display:none;"></div>
* n" _6 o% i4 o S; G3 e</div>
* P7 x' Z! m3 t9 R<script type="text/javascript"> / O3 O+ T) [; @3 _ a5 O
function toggleOverlay() { 5 @" S+ r9 X l [* N* |
var overlay = document.getElementById('overlay'); 2 J- n; F+ V# @
overlay.style.display = (overlay.style.display == 'none') ? 'block' : 'none'; F: d' ]3 l: ^$ W$ ~( H" A
}
2 Y6 `6 w7 c8 b8 |0 V, X! G</script>下面列出以上代码在各浏览器的测试值: " p/ B" k* Z* V# c
(bHeight = body.scrollHeight, dHeight = documentElement.scrollHeight, 红色 = 错误值, 绿色 = 正确值) , C4 E; S0 B) Y8 F3 q
/ 层隐藏时 层展开时
2 J s5 [9 x0 w* rbHeight dHeight bHeight dHeight 5 _8 F! n: l& a" t5 T+ @
IE6 184 184 184 303 ( O, z2 ^2 W8 T9 `! A, E
IE7 184 184 184 303
6 ~, ]9 x3 A! J* i# lFF 184 184 184 303 4 {0 q) W+ B4 v5 O8 B, u
Opera 181 181 300 300
9 C R, M: e( c R& tSafari 184 184 303 184
; g+ }" u* L/ F# x# s, ]5 u8 _暂且无视Opera比别人少3像素的问题…可以看出,如果没有绝对定位的东西,两个值是相等的,取哪个都无所谓。
) J- q: ~0 ^# m7 Z! |4 ~但是如果有,那么各个浏览器的表现不太相同,单取哪个值都不对。但可以找到了一条规律,那就是取两个值得最大值可以兼容各浏览器。所以我们的主页面代码就要改造成这样了: ( w5 a) {) |3 G7 D9 {+ V
function reinitIframe(){ % t/ }- n& k8 k0 T3 ^- \" H1 T! g
var iframe = document.getElementById("frame_content");
! N4 e o2 b& M" l$ F* Qtry{ , _3 N% ]' v) X
var bHeight = iframe.contentWindow.document.body.scrollHeight;
$ {# i- H. s0 |) xvar dHeight = iframe.contentWindow.document.documentElement.scrollHeight; & q" Q( C6 c2 o. F; o
var height = Math.max(bHeight, dHeight);
3 P- r b) x, N3 D9 E- oiframe.height = height; / N5 W% M% y* \, O- w" C
}catch (ex){} , [ R' Z4 x9 C/ `* V
}
/ w8 D x5 w( k1 }: b+ Fwindow.setInterval("reinitIframe()", 200);这样子,基本解决了兼容性问题。顺便说下,不光绝对定位的层会影响到值,float也会导致两个值的差异。
2 Y* H* H9 `: Q如果你演示Demo后,会发现,除了IE,其他浏览器中,当层展开后再隐藏,取到的高度值还是维持在展开的高度303,而非隐藏回去的真正值184,就是说长高了之后缩不回去了。这个现象在不同被包含页面之间做切换也会发生,当从高的页面切换到矮页面的时候,取到的高度还是那个高的值。 1 H1 m6 B1 s& a+ f
可以归纳为,当iframe窗体高度高于文档实际高度的时候,高度取的是窗体高度,而当窗体高度低于实际文档高度时,取的是文档实际高度。因此,要想办法在同步高度之前把高度设置到一个比实际文档低的值。所以,在iframe的添加 onload=”this.height=100″,让页面加载的时候先缩到足够矮,然后再同步到一样的高度。 ( _5 z. z6 U9 u, A: }' p
这个值,在实际应用中决定,足够矮但又不能太矮,否则在FF等浏览器里会有很明显的闪烁。DOM操作的时候主页面无法监听到,只能DOM操作完了之后把高度变小了。 4 x( v2 M4 w! e( E3 M7 i3 p
在我的一个实际项目中,在成本和收益之间权衡,我并没有做这个事情,因为每个DOM函数中都要插入这个代码,代价太高,其实层缩回去不缩掉也不是那么致命。包括Demo里,也没有去做这个事情。如果读者有更好的方法,请告诉我。
1 R& ]8 E- Y6 i1 E$ f6 {这是最终的主页面的代码: ( z/ h7 q# o* A" W
<iframe id="frame_content" src="iframe_b.html" scrolling="no" frameborder="0"
1 |9 y5 R: T# {, b1 ?( }9 t! J></iframe> , C( R$ w7 V/ p9 H' H3 }0 ?
<script type="text/javascript"> 0 t- _/ C4 B" ]
function reinitIframe(){ 2 K P! l! Y* L, c% `
var iframe = document.getElementById("frame_content"); K) F! \6 m' p5 I) ^
try{
4 c8 a8 n" D4 ^! q" m Y: Pvar bHeight = iframe.contentWindow.document.body.scrollHeight;
- T$ V3 B! H/ N8 }5 ?% g6 _5 W$ H% i! nvar dHeight = iframe.contentWindow.document.documentElement.scrollHeight; 8 Q/ n0 |# U* i% |; G
var height = Math.max(bHeight, dHeight); / n8 `' e. {$ X1 N7 C
iframe.height = height;
& l+ o& r: a( c J) x+ ]}catch (ex){}
2 j. e7 N: |& G8 M/ a( J9 ?}
( @3 y* J- r+ x" W" E( _window.setInterval("reinitIframe()", 200);
. b: ]+ i* [ }; ^* [" ?0 R1 B( \</script>附Demo页面: 主页面 iframe_a.html ,被包含页面 iframe_b.htm 和 iframe_c.html |
|