php iframe 自适应高度,JS解决iframe之间通信和自适应高度的问题

首先说明下,iframe通信 分为:同域通信 和 跨域通信。

一、 同域通信

所谓同域通信是指 http://localhost/demo/iframe/iframeA.html 下的a.html页面嵌套 iframe

比如: 的B.html页面,这两个页面数据进行通信,比如我想在父页面A.html 调用子页面当中的函数 我们很容易想到或者google下 document.getElementById('iframeA').contentWindow.b();这种方法,其中b 是子页面B.html中的一个函数。但是这样调用下有个问题我纠结了很久,就是既然在火狐下报这样的错误, 如下:

e2917bbc026d61c741474e41ca5e9ae5.png

b不是个函数 但是我在子页面明明定义了这么一个函数,那么为什么会报这样的错误呢?经过仔细分析及google,发现有这么一个问题需要理解,当iframe没有加载完成后 我就去执行这个js会报这样的错误,所以就试着在火狐下 用iframe.onload这个函数 进行测试,果然没有报错,是正确的 所以就确定是这个问题。所以就想写个兼容IE和火狐 google写个函数 来确定iframe已经加载完成!,其实给个回调函数来调用我们上面的方法。

综合上面的思路 就可以写个这样的代码:

topNddddddddddddddddame

function A(){

alert("A");

}

var iframe = document.getElementById('iframeA');

iframeIsLoad(iframe,function(){

var obj = document.getElementById('iframeA').contentWindow;

obj.b();

});

function iframeIsLoad(iframe,callback){

if(iframe.attachEvent) {

iframe.attachEvent('onload',function(){

callback && callback();

});

}else {

iframe.onload = function(){

callback && callback();

}

}

}

B.html 代码如下:

var b = function(){

alert("B");

}

子页面调用父页面的函数很简单,只要这样搞下就ok了,window.parent.A();

子页面取父页面元素的值: window.parent.document.getElementById("topName").innerHTML等方法。

二: iframe跨域通信。

iframe跨域访问一般分为2种情况,第一种是同主域,不同子域的跨域。 第二种是:不同主域跨域。

1、 是同主域下面,不同子域之间的跨域;可以通过document.domain 来设置相同的主域来解决。

假如现在我有个域 abc.example.com 下有个页面叫abc.html, 页面上嵌套了一个iframe 如下:,我想在abc域下的页面abc.html 访问 def域下的def.html  我们都知道由于安全性 游览器的同源策略的限制,js不能操作页面不同域下 不同协议下 不同端口的页面,所以就要解决跨域访问了,假如父页面abc.html 页面有个js函数:function test(){console.log(1);};我想在子页面调用这个函数 还是按照上面的同域方式调用 parent.test();这样,通过在火狐下看 已经跨域了 解决的办法是 在各个js函数顶部 加一句 document.domain = 'example.com',就可以解决了。

abc.html代码如下:

// 跨域 子页调用父页的 函数 (假设是下面test函数)

document.domain = 'example.com';

function test(){console.log(1);};

def.html代码如下:

/*

* 子页调用父页的方法

*/

document.domain = 'example.com';

//window.top.test();

window.parent.test();

还是这两个页面 我想父页调用子页 如下方法:

a.html代码如下:

/*

* 跨域 父页想调用子页的的函数

*/

document.domain = 'example.com';

var iframe = document.getElementById('iframe2');

iframeIsLoad(iframe,function(){

var obj = iframe.contentWindow;

obj.child();

});

function iframeIsLoad(iframe,callback){

if(iframe.attachEvent) {

iframe.attachEvent('onload',function(){

callback && callback();

});

}else {

iframe.onload = function(){

callback && callback();

}

}

}

假如现在def.html页面有个child函数 代码如下:

document.domain = 'example.com';

function child(){console.log('我是子页');}

就可以跨域调用了 不管是子页面调用父页面 还是父页面调用子页面。一切ok!

2、 是不同主域跨域;

虽然google有几种方法关于不同主域上的跨域问题 有通过location.hash方法或者window.name方法或者html5及flash等等,但是我觉得下面iframe这种方法值得学习下,

如下图所示:域a.com的页面request.html(即http://a.com/demo/ajax/ajaxproxy/request.html)里面嵌套了一个iframe指向域b.com(http://b.com/demo/ajax/ajaxproxy/response.html)的response.html,而response.html里又嵌套了域a.com的proxy.html。

126ca0bdd1be969a8d7edeadcdb2fbf1.png

思路:要实现a.com域下的request.html页面请求域b.com下的process.php,可以将请求参数通过url传给response.html,由response.html向process.php发起真正的ajax请求(response.html与process.php都属于域b.com),然后将返回的结果通过url传给proxy.html,最后由于proxy.html和request.html是在同个域下,所以可以在proxy.html利用window.top 将结果返回在request.html完成真正的跨域。

77dc6321b9d2815f5c9f7c9e7b3b73b4.png

ok, 先看看页面结构

a.com域下有:

request.html

proxy.html

b.com域下有:

response.html

process.php

先来看看request.html页面如下:

New Document

这里将会填上响应的结果

点击,发送跨域请求

document.getElementById('sendBtn').onclick = function() {

var url = 'http://b.com/demo/ajax/ajaxproxy/reponse.html',

fn = 'GetPerson', //这是定义在response.html的方法

reqdata = '{"id" : 24}', //这是请求的参数

callback = "CallBack"; //这是请求全过程完成后执行的回调函数,执行最后的动作

CrossRequest(url, fn, reqdata, callback); //发送请求

}

function CrossRequest(url,fn,reqdata,callback) {

var server = document.getElementById('serverIf');

server.src = url + '?fn=' +encodeURIComponent(fn) + "&data=" +encodeURIComponent(reqdata) + "&callback="+encodeURIComponent(callback);

}

//回调函数

function CallBack(data) {

var str = "My name is " + data.name + ". I am a " + data.sex + ". I am " + data.age + " years old.";

document.getElementById("result").innerHTML = str;

}

这个页面其实就是要告诉response.html:我要让你执行你定义好的方法GetPerson,并且要用我给你的参数'{"id" : 24}'。response.html纯粹是负责将CallBack这个方法名传递给下一位仁兄proxy.html,而proxy.html拿到了CallBack这个方法名就可以执行了,因为proxy.html和request.html是同域的。

response.html代码如下:

New Document

// 通用方法 ajax请求

function _request (reqdata,url,callback) {

var xmlhttp;

if(window.XMLHttpRequest) {

xmlhttp = new XMLHttpRequest();

}else {

xmlhttp = new ActiveXObject("Microsoft.XMLHTTP");

}

xmlhttp.onreadystatechange = function(){

if(xmlhttp.readyState == 4 && xmlhttp.status == 200) {

var data = xmlhttp.responseText;

callback(data);

}

}

xmlhttp.open('POST',url);

xmlhttp.setRequestHeader("Content-Type", "application/json; charset=utf-8");

xmlhttp.send(reqdata);

}

// 通用方法 获取url参数

function _getQuery(key) {

var query = location.href.split('?')[1],

value = decodeURIComponent(query.split(key + "=")[1].split("&")[0]);

return value;

}

//向process.php发送ajax请求

function GetPerson(reqdata,callback) {

var url = 'http://b.com/demo/ajax/ajaxproxy/process.php';

var fn = function(data) {

var proxy = document.getElementById('proxy');

proxy.src = "http://a.com/demo/ajax/ajaxproxy/Proxy.html?data=" + encodeURIComponent(data) + "&callback=" + encodeURIComponent(callback);

};

_request(reqdata, url, fn);

}

(function(){

var fn = _getQuery('fn'),

reqdata = _getQuery("data"),

callback = _getQuery("callback");

eval(fn + "('" + reqdata +"', '" + callback + "')");

})();

这里其实就是接收来自request.html的请求得到请求参数和方法后向服务器process.php发出真正的ajax请求,然后将从服务器返回的数据以及从request.html传过来的回调函数名传递给proxy.html。

接下来看看php代码如下,其实就是想返回一个json数据:

$data = json_decode(file_get_contents("php://input"));

header("Content-Type: application/json; charset=utf-8");

echo ('{"id" : ' . $data->id . ', "age" : 24, "sex" : "boy", "name" : "huangxueming"}');

?>

最后就是proxy.html代码:

New Document

function _getUrl(key) {//通用方法,获取URL参数

var query = location.href.split("?")[1],

value = decodeURIComponent(query.split(key + "=")[1].split("&")[0]);

return value;

}

(function() {

var callback = _getUrl("callback"),

data = _getUrl("data");

eval("window.top." + decodeURIComponent(callback) + "(" + decodeURIComponent(data) + ")");

})();

这里也是最后一步了,proxy终于拿到了request.html透过response.html传过来的回调函数名以及从response.html直接传过来的响应数据,利用window.top执行request.html里定义的回调函数。

三、iframe高度自适应的问题。

iframe高度自适应分为2种,一种是同域下自适应  另外一种是跨域下自适应,下面我们来看看同域下iframe高度自适应的问题。

1.同域下iframe高度自适应的问题:

思路:获取被嵌套iframe元素,通过JavaScript取得被嵌套页面最终高度,然后在主页面进行设置来实现。

假如我们demo有iframe1.html和iframe2.html

下面贴上iframe1.html代码如下:

New Document

*{margin:0;padding:0;}

window.onload = function() {

var iframeid = document.getElementById('iframe');

if(iframeid && !window.opera) {

if(iframeid.contentDocument && iframeid.contentDocument.body.offsetHeight) {

iframeid.height = iframeid.contentDocument.body.offsetHeight;

}else if(iframeid.Document && iframeid.Document.body.scrollHeight){

iframeid.height = iframeid.Document.body.scrollHeight;

}

}

}

iframe2.html

New Document

*{margin:0;padding:0;}

就可以动态设置iframe1页面的高度为iframe2的高度了。

2. 跨域下iframe高度自适应的问题。

首先我们知道iframe跨域我们是不能用上面js方式来控制了,所以我们只能用个中间键 我们可以在a.com域下iframe1.html页面嵌套一个b.com域下的iframe2.html页面,然后我在iframe2.html页面嵌套个和iframe1.html相同域的iframe3.html页面了,这样的话 iframe1.html和iframe3.html就可以无障碍的进行通信了,因为页面iframe2.html嵌套iframe3.html,所以iframe2.html可以改写iframe3.html的href值。

iframe1中的内容:

iframe1.html内容主要接受iframe3.html页面传过来的内容并且去完成相应的操作。iframe1.html代码如下:

var ifr_el = document.getElementById("iframe");

function getIfrData(data){

ifr_el.style.height = data+"px";

}

iframe2.html中的内容:

iframe2.html内容是怎么把值传给iframe3.html页面,刚才说了是将值传递到iframe3.html页面的href中,所以只要修改iframe的src就可以,因为不用刷新C页面,所以可以用过hash的方式传递给iframe3.html页面.iframe2.html代码如下:

New Document

*{margin:0;padding:0;}

var oldHeight = 0,

ifr_el = document.getElementById("iframe");

t && clearInterval(t);

var t = setInterval(function(){

var height = document.body.scrollHeight;

if(oldHeight != height) {

oldHeight = height;

ifr_el.src += '#' +oldHeight;

}

},200);

可以看到 默认情况下 iframe1.html 页面我给iframe2.html的高度是200像素 但是在iframe2.html我给iframe3.html高度是230像素,那么正常情况下是有滚动条的,那么现在我是想在iframe2.html获取滚动条的高度,把高度传给通过iframe3.html的src里面去,然后在iframe3.html页面里获取这个高度值 传给iframe1.html(因为iframe1.html和iframe3.html是同域的),所以iframe1.html能取到这个高度值,再设置下本身的高度就是这个值就ok了。

iframe3.html页面的唯一功能就是接收iframe2.html页面通过href传进来的值并且传递给iframe1.html页面,可到iframe2.html页面传来的值可以通过一个定时器不停去查看location.href是 否被改变,但是这样感觉效率很低,还有个方式就是在新的浏览器中通过onhashchange事件 (IE8+,Chrome5.0+,Firefox3.6+,Safari5.0+,Opera10.6+)来监听href的改变。

iframe3.html代码如下:

var oldHeight = 0;

t && clearInterval(t);

var t = setInterval(function(){

var height = location.href.split('#')[1];

if(height && height != oldHeight) {

oldHeight = height;

if(window.parent.parent.getIfrData) {

window.parent.parent.getIfrData(oldHeight);

}

}

},200);

这样就可以解决通过跨域实现iframe自适应高度的问题了。

四、总结

以上就是本文的全部内容了,希望对大家的学习工作能有所帮助。如果有疑问可以留言讨论。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Vue中使用iframe实现内容嵌套时,有时会遇到跨域和自适应高度问题。这是因为浏览器出于安全原因,不允许从一个源(域名)加载另一个源的内容,除非服务器提供了适当的CORS(跨源资源共享)策略。对于跨域问题,你需要在服务器端设置允许跨域请求,或者在前端使用JSONP或者代理服务器来间接访问。 至于iframe高度适应,Vue本身并不能直接处理iframe高度变化,因为它是在DOM渲染后处理数据的。但你可以通过以下方法解决: 1. 利用JavaScript事件监听:给iframe添加`load`或`message`事件,在事件触发时获取iframe的实际高度并更新其CSS样式。 ```javascript new Vue({ el: '#myIframe', methods: { updateIframeHeight() { this.$refs.myIframe.addEventListener('load', () => { this.$refs.myIframe.style.height = `${this.myIframeContentHeight}px`; }); } }, mounted() { this.updateIframeHeight(); } }) ``` 2. 使用Intersection Observer API:这是一个现代浏览器提供的API,可以检测元素是否进入或离开视口,适用于动态内容。你可以创建一个观察者,当iframe进入视口时计算其高度。 ```javascript const observer = new IntersectionObserver((entries) => { entries.forEach((entry) => { if (entry.isIntersecting) { this.myIframeContentHeight = entry.target.contentDocument.body.scrollHeight; } }); }); observer.observe(this.$refs.myIframe); ``` 3. 使用第三方库如`vue-observe-visibility-events`:这个库可以帮助你在元素可见时执行回调,可以配合上述方法使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值