- 注册时间
- 2009-12-25
- 最后登录
- 2021-7-10
- 在线时间
- 3302 小时
- 阅读权限
- 200
- 积分
- 10
- 帖子
- 13416
- 精华
- 1
- UID
- 2036
|
通过Google搜索iframe 自适应高度,结果5W多条,搜索iframe 高度自适应,结果2W多条。 4 K( {3 p( Z, \/ o- U
我翻了前面的几十条,刨去大量的转载,有那么三五篇是原创的。而这几篇原创里面,基本上只谈到如何自适应静的东西,就是没有考虑到JS操作DOM之后,如何做动态同步的问题。另外,在兼容性方面,也研究的不彻底。 # k/ e# v# B5 {5 }8 w
这篇文章,希望在这两个方面再做一些深入。 ) a5 O) w" N* h6 {% v7 P: {, a
可能有人还没接触到这个问题过,先说明一下,什么是高度自适应吧。所谓iframe高度自适应,就是,基于界面美观和交互的考虑,隐藏了iframe的border和scrollbar,让人看不出它是个iframe。如果iframe始终调用同一个固定高度的页面,我们直接写死iframe高度就可以了。而如果iframe要切换页面,或者被包含页面要做DOM动态操作,这时候,就需要程序去同步iframe高度和被包含页的实际高度了。 9 J2 B2 z( z# Z: U. ?" o
顺便说下,iframe在迫不得已的时候才去用,它会给前端开发带来太多的麻烦。 7 q; t1 W2 t3 h6 O. `$ W1 `
传统做法大致有两个:
* v* y/ X* g' H5 S6 S0 {方法一,在每个被包含页在本身内容加载完毕之后,执行JS取得本页面的高度,然后去同步父页面的iframe高度。
% S3 C# M: @2 @4 H% ~, X v4 v; q方法二,在主页面iframe的onload事件中执行JS,去取得被包含页的高度内容,然后去同步高度。
* U0 A5 z$ N. @# h. [( m在代码维护角度考虑,方法二是优于方法一的,因为方法一,每个被包含页都要去引入一段相同的代码来做这个事情,创建了好多副本。 + c# f9 \ d! ^
两个方法都只处理了静的东西,就是只在内容加载的时候执行,如果JS去操作DOM引起的高度变化,都不太方便。
8 F9 h# l: J# |' g如果在主窗口做一个Interval,不停的来获取被包含页的高度,然后做同步,是不是即方便,又解决了JS操作DOM的问题了呢?答案是肯定的。 2 m. S- R8 O& i+ \( M- z
Demo页面:主页面 iframe_a.html ,被包含页面 iframe_b.htm 和 iframe_c.html _) h/ k; W4 R; k5 |' d# m
主页面代码示例:
0 @4 x7 U; y, {; V8 X. i( r. M<iframe id="frame_content" src="iframe_b.html" scrolling="no" frameborder="0"></iframe>
# [+ ^# h' M( z' U: Q9 F<script type="text/javascript">
y) W& M! q3 l5 D. K& m' ufunction reinitIframe(){
( o9 G. }6 t: [9 p$ |0 P( I; i! Mvar iframe = document.getElementById("frame_content");
, S& r4 j6 G z6 c( u5 Jtry{
, i L8 _3 K. `5 W1 ]: Biframe.height = iframe.contentWindow.document.documentElement.scrollHeight; a: ~$ y+ g1 |1 ^
}catch (ex){}
+ d( ^0 J8 ?* c4 d+ D} # d% y* y2 w, m* v; N
window.setInterval("reinitIframe()", 200); - }, A% L) j" O5 z) G
</script>一直执行,效率会不会有问题? ' C4 j( H9 e3 A) F1 Y
我做了测试,同时开5个窗口(IE6、IE7、FF、Opera、Safari)执行这个代码,不会对CPU有什么影响,甚至调整到2ms,也没影响(基本维持在0%占用率)。
1 c( R f8 z2 W. e6 Q下面谈谈各浏览器的兼容性问题,如何获取到正确的高度,主要是对body.scrollHeight和documentElement.scrollHeight两个值得比较。注意本文用的是这个doctype,不同的doctype应该不会影响结果,但是假如你的页面没有申明doctype,那还是先去加一个吧。
: z3 f: @& X# Z( r" \2 a: R<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/strict.dtd">在主页面追加以下测试代码,以输出这两个值,代码示例:
e) Y% H3 o5 n9 a<div><button >Check Height</button></div> & V4 [% P0 F* R# I2 ]
<script type="text/javascript"> ' s' _% R: |( q+ H5 F. d
function checkHeight() {
4 l5 H6 P( G9 Z% s/ @3 nvar iframe = document.getElementById("frame_content"); 2 J: |" H; J, b* K( F4 k
var bHeight = iframe.contentWindow.document.body.scrollHeight;
% W9 C6 t7 T/ K5 C' qvar dHeight = iframe.contentWindow.document.documentElement.scrollHeight;
* g/ B. Y8 B; [% c8 Kalert("bHeight:" + bHeight + ", dHeight:" + dHeight); 2 k- N5 x. M* Y% e4 n2 I9 }/ y
} ' D2 |; u# [. @* q5 {5 M4 ?1 d7 X
</script>被加载页面,可以切换一个绝对定位的层,来使页面高度动态改变。如果层展开,则会撑高页面高度。代码示例:
4 A( Q) V# E8 |( T/ A' S<div>
5 k( L5 r4 U% t<button >Toggle Overlay</button> " _2 Q1 p. P4 q+ F
</div>
8 I' A0 o' ^- S8 _! J<div style="height:160px;position:relative"> & B! y9 F* ?# V: R% E- x9 c2 j, X5 R
<div id="overlay" style="position:absolute;width:280px;height:280px;display:none;"></div>
) j4 c5 g g, D</div> 0 p- V- }& X+ y, R% ^
<script type="text/javascript"> : c% U; Q" j8 g- j2 D6 P5 j- b/ Z
function toggleOverlay() { ( ?, k- J( j k1 j; M; s+ ]0 g
var overlay = document.getElementById('overlay'); ! b# E7 v- _$ H' e) D
overlay.style.display = (overlay.style.display == 'none') ? 'block' : 'none';
* C; [ c( d" i0 a( `% i T} . P6 j8 f* Q# u, @7 j
</script>下面列出以上代码在各浏览器的测试值:
4 o4 L- ~! U P$ Q(bHeight = body.scrollHeight, dHeight = documentElement.scrollHeight, 红色 = 错误值, 绿色 = 正确值) " _' G% a Q7 K3 a$ _2 L
/ 层隐藏时 层展开时
4 x: [" Z9 x5 c1 S+ r1 ~" O% ibHeight dHeight bHeight dHeight * x; a% K2 |6 j* p; R% m' i# u
IE6 184 184 184 303 ! _7 D2 q$ a: {( X5 S
IE7 184 184 184 303 4 k$ O3 s, j1 p
FF 184 184 184 303
9 [. T5 \0 _+ q; @Opera 181 181 300 300
U8 j# W/ X( Q, wSafari 184 184 303 184
/ ?6 N* T9 W5 Q" [暂且无视Opera比别人少3像素的问题…可以看出,如果没有绝对定位的东西,两个值是相等的,取哪个都无所谓。 ) k) p& V# l5 s( b
但是如果有,那么各个浏览器的表现不太相同,单取哪个值都不对。但可以找到了一条规律,那就是取两个值得最大值可以兼容各浏览器。所以我们的主页面代码就要改造成这样了:
) w, B' s1 w, X# h& M4 u& Hfunction reinitIframe(){ ( l! n _) P; c* G$ t6 H
var iframe = document.getElementById("frame_content"); * C" h" b9 R1 p& K$ U9 U
try{ " \& Q$ k) \. ]( y$ ]8 ]0 `' x
var bHeight = iframe.contentWindow.document.body.scrollHeight;
, s, v6 N9 o5 d! Z4 Q1 Cvar dHeight = iframe.contentWindow.document.documentElement.scrollHeight;
; X2 j a) Q! B! c3 y7 hvar height = Math.max(bHeight, dHeight); 1 B0 a- G4 E# h6 Z
iframe.height = height;
z6 m2 N8 m5 A* l. u}catch (ex){} 2 Y/ h ^- }& {: Y
} 1 U' T# K# A8 ?' t8 R7 w
window.setInterval("reinitIframe()", 200);这样子,基本解决了兼容性问题。顺便说下,不光绝对定位的层会影响到值,float也会导致两个值的差异。 6 j: L e- S" y! \' k
如果你演示Demo后,会发现,除了IE,其他浏览器中,当层展开后再隐藏,取到的高度值还是维持在展开的高度303,而非隐藏回去的真正值184,就是说长高了之后缩不回去了。这个现象在不同被包含页面之间做切换也会发生,当从高的页面切换到矮页面的时候,取到的高度还是那个高的值。 4 O9 ~# s- T2 X1 m
可以归纳为,当iframe窗体高度高于文档实际高度的时候,高度取的是窗体高度,而当窗体高度低于实际文档高度时,取的是文档实际高度。因此,要想办法在同步高度之前把高度设置到一个比实际文档低的值。所以,在iframe的添加 onload=”this.height=100″,让页面加载的时候先缩到足够矮,然后再同步到一样的高度。 ' _) _! X" E @
这个值,在实际应用中决定,足够矮但又不能太矮,否则在FF等浏览器里会有很明显的闪烁。DOM操作的时候主页面无法监听到,只能DOM操作完了之后把高度变小了。 7 S. R+ o. u- `( \: Q
在我的一个实际项目中,在成本和收益之间权衡,我并没有做这个事情,因为每个DOM函数中都要插入这个代码,代价太高,其实层缩回去不缩掉也不是那么致命。包括Demo里,也没有去做这个事情。如果读者有更好的方法,请告诉我。
3 @5 Q* L* b1 H, L这是最终的主页面的代码:
2 j* c3 h6 N l% ]: C<iframe id="frame_content" src="iframe_b.html" scrolling="no" frameborder="0"
/ Q; ]; a% y4 O6 n></iframe>
$ h$ `6 E6 D1 C<script type="text/javascript">
# n, u& Q! t8 y2 R* I( xfunction reinitIframe(){
! s; m! W, R) s( R; ]: B& L! gvar iframe = document.getElementById("frame_content");
: r( I: O' T btry{ : t) G. i+ [# c5 J2 Y4 z
var bHeight = iframe.contentWindow.document.body.scrollHeight; 4 O* a$ J6 `4 T- E6 x+ l, R, v
var dHeight = iframe.contentWindow.document.documentElement.scrollHeight;
( M7 Z2 C) ?( C$ r5 ?: s5 N8 Q3 ~: Lvar height = Math.max(bHeight, dHeight); 9 u& w! `4 N+ l( P( _8 }
iframe.height = height; * Q' s8 i7 I0 U4 b. ]) f6 l$ k
}catch (ex){} 7 Q/ Z& Q4 o& z+ m
} ( T, [8 x8 O2 W% f0 j6 d
window.setInterval("reinitIframe()", 200); + r' U* X8 p) j9 |: x
</script>附Demo页面: 主页面 iframe_a.html ,被包含页面 iframe_b.htm 和 iframe_c.html |
|