父子 iframe 传值
2021-11-03
最近项目中有个页面中使用到 iframe,涉及到不同 iframe 的传值等操作,踩坑不少,简单记录一下。
外部的页面称为父页面,内部嵌入的 iframe 称为子页面。一个父页面中可以嵌入多个子页面。不同子页面之间称为兄弟页面。
任务需要:父获取子的属性、子获取父的属性。
父获取子的属性
首先界面中的 iframe 需要设置不同的 ID,这样父组件中,可以通过 document.getElementById 等获取对应的 iframe。
<iframe id="test1" title="detail1" src='http://www.baidu.com'></iframe>
<iframe id="test2" title="detail2" src='http://www.seafile.com'></iframe>
<iframe id="test1" title="detail1" src='http://www.baidu.com'></iframe>
<iframe id="test2" title="detail2" src='http://www.seafile.com'></iframe>
子获取父的属性
子 iframe 中获取父元素的变量,使用 window.parent 可以获取父页面的 DOM 节点,然后获取对应的属性。
例如,这里获取父窗体的属性是否是移动端
const parentWin = window.parent;
return parentWin.window.isMobile
const parentWin = window.parent;
return parentWin.window.isMobile
或者获取父页面中其他的DOM节点
const parentWin = window.parent;
parentWin.getElementsByTagName("div")[0]
const parentWin = window.parent;
parentWin.getElementsByTagName("div")[0]
子获取当前 iframe 的属性
需求:界面中有 N 个 iframe,某个 iframe 需要获取自己是第几个 iframe,然后改变内部的属性。
那么我们可以使用 iframe.contentWindow == window 来判断。
首先获取父页面,然后找到全部的 iframe 并循环,如果某个 iframe.contentWindow == window ,那么就可以获取对应的标识。
var a = parent.document.getElementsByTagName("IFRAME");
for(var i = 0; i < a.length; i++) {
if(a[i].contentWindow == window){
// a[i]是当前的页面
// 那么可以获取对应的属性
}
}
var a = parent.document.getElementsByTagName("IFRAME");
for(var i = 0; i < a.length; i++) {
if(a[i].contentWindow == window){
// a[i]是当前的页面
// 那么可以获取对应的属性
}
}
具体我们可以操作:界面中两个 iframe 如下
<iframe id="test1" title="detail1" src='http://www.baidu.com'></iframe>
<iframe id="test2" title="detail2" src='http://www.seafile.com'></iframe>
<iframe id="test1" title="detail1" src='http://www.baidu.com'></iframe>
<iframe id="test2" title="detail2" src='http://www.seafile.com'></iframe>
我们就可以根据下面的语句判断当前的 iframe 是哪一个(如果是 test1,那么显示XXX,如果是 test2,那么显示XXX)
(typeof (window) !== 'undefined' && (window.parent.document.getElementById('test1').contentWindow == window))
(typeof (window) !== 'undefined' && (window.parent.document.getElementById('test1').contentWindow == window))
父子之间通信
一个 window postMessage,另一个监听 message 事件,这样就可以通信
window.postMessage(message, targetOrigin, [transfer]);
// window 是发送消息的iframe
// message 是发送的消息(可以是字符串)
// targetOrgin 是目标的 iframe,或者是通配符 *
// transfer 可选,是一串和 message 同时传递的 Transferable 对象。
window.postMessage(message, targetOrigin, [transfer]);
// window 是发送消息的iframe
// message 是发送的消息(可以是字符串)
// targetOrgin 是目标的 iframe,或者是通配符 *
// transfer 可选,是一串和 message 同时传递的 Transferable 对象。
例子:一个窗口(https://www.baidu.com)发出事件
window.postMessage('hello', 'www.sina.com');
window.postMessage('hello', 'www.sina.com');
另一个窗口(www.sina.com)监听事件
window.addEventListener('message', function (e) {
if (e.origin !== "https://www.baidu.com") {
return;
}
console.log(e.data);
});
window.addEventListener('message', function (e) {
if (e.origin !== "https://www.baidu.com") {
return;
}
console.log(e.data);
});
具体参考:https://www.runoob.com/js/met-win-postmessage.html
注意
1、iframe 在加载状态发生变化时会发生 onreadystatechange 事件,可以通过这个事件改变界面的其他属性。
2、外部的样式和内部的样式完全不起作用,如果更改样式,需要改变单页面的样式,不能通过父子页面的样式继承。