详解使用postMessage解决iframe跨域通信问题


Posted in HTML / CSS onNovember 01, 2019

这周碰到一个让人头疼的需求:要在我的web项目中嵌入另一个第三方web项目。第一时间想到的就是用iframe了,但问题来了,我和第三方web项目是有交互的,这就违反同源策略了,处理跨域问题是最让人头疼的事之一。

需求是这样的,在我的页面点击一些按钮,要实时反馈到iframe子页面,子页面再进行响应。

当时脑子里第一时间想到的解决方案是:用NGINX把两个项目代理到同一域名下。但这样似乎有点小题大做了,有没有更方便快捷的方法呢?

在window对象下有个postMessage方法,是专门用来解决跨域通信问题的。

关于postMessage的详细介绍请戳这里,不过MDN的文档太详细了,导致有些同学看完还是一脸懵逼,下面我们就来看看怎么用postMessage实现iframe跨域通信,当你会用了之后再回去看文档,感觉是完全不同的。

首先我们模拟场景,假设有两个不同源的页面,iframePage.html是index.html的子页面:

<!-- index.html -->

<body style="border:5px solid #333;">

  <h1>this is index</h1>

  <iframe src="./iframePage.html" id='myframe'></iframe>

</body>
<!-- iframePage -->

<body style="border:5px solid #333;">

  <h1>this is iframePage</h1>

</body>

现在这两个iframe是无法通信,因为它们是不同源的(假设存在跨域问题),这时候就要用到postMessage了。

我们先试着从父页面向子页面发送一条消息:

// idnex.html

//获取iframe元素
iFrame = document.getElementById('myframe')

//iframe加载完毕后再发送消息,否则子页面接收不到message
iFrame.onload = function(){

  //iframe加载完立即发送一条消息
  iFrame.contentWindow.postMessage('MessageFromIndex1','*');

}

我们知道postMessage是挂载在window对象上的,所以等iframe加载完毕后,用iFrame.contentWindow获取到iframe的window对象,然后调用postMessage方法,相当于给子页面发送了一条消息。

postMessage方法第一个参数是要发送的数据,可以是任何原始类型的数据。

Gecko 6.0 (Firefox 6.0 / Thunderbird 6.0 / SeaMonkey 2.3)之前,第一个参数必须是一个字符串。

postMessage方法第二个参数可以设置要发送到哪个url,如果当前子页面的url和设置的不一致,则会发送失败,我们设置为*,代表所有url都允许发送。

postMessage方法还有第三个参数,属于高级用法,这里不做讨论,可以稍后去MDN了解。

消息发送到iframePage.html,我们来接收message:

// iframePage.html

//回调函数
function receiveMessageFromIndex ( event ) {
  console.log( 'receiveMessageFromIndex', event )
}

//监听message事件
window.addEventListener("message", receiveMessageFromIndex, false);

我们只需要在子页面监听message事件,并且设置好回调函数即可,来看看打印出来的event:

详解使用postMessage解决iframe跨域通信问题

event对象中的data属性存放着我们从父页面传过来的数据,就这么简单!

让我们再试试从子页面发送数据给父页面:

// iframePage.html

//给父页面发送消息,data为对象
parent.postMessage( {msg: 'MessageFromIframePage'}, '*');

父页面接收数据:

//index.html

//回调函数
function receiveMessageFromIframePage (event) {
    console.log('receiveMessageFromIframePage', event)
}

//监听message事件
window.addEventListener("message", receiveMessageFromIframePage, false);

我看看到,的确可以传输不同的数据,此时data为一个对象:

详解使用postMessage解决iframe跨域通信问题

大家可以到postMessage-demo把代码clone下来运行试试看。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

HTML / CSS 相关文章推荐
CSS3制作日历实现代码
Jan 21 HTML / CSS
CSS3旋转——彩色扇子兼容firefox浏览器
Jun 04 HTML / CSS
CSS3正方体旋转示例代码
Aug 08 HTML / CSS
CSS的pointer-events属性详细介绍(作用和注意事项)
Apr 23 HTML / CSS
CSS3 分类菜单效果
May 27 HTML / CSS
CSS3关于z-index不生效问题的解决
Feb 19 HTML / CSS
html5中localStorage本地存储的简单使用
Jun 16 HTML / CSS
详解html5页面 rem 布局适配方法
Jan 12 HTML / CSS
html svg生成环形进度条的实现方法
Sep 23 HTML / CSS
web字体加载方案优化小结
Nov 29 HTML / CSS
AmazeUI 评论列表的实现示例
Aug 13 HTML / CSS
CSS使用Flex和Grid布局实现3D骰子
Aug 05 HTML / CSS
跨域修改iframe页面内容详解
Oct 31 #HTML / CSS
html如何对span设置宽度
Oct 30 #HTML / CSS
关于iframe跨域使用postMessage的实现
Oct 29 #HTML / CSS
使用canvas生成含有微信头像的邀请海报没有微信头像问题
Oct 29 #HTML / CSS
Html5与App的通讯方式详解
Oct 24 #HTML / CSS
html+js 实现markdown编辑器效果
Oct 23 #HTML / CSS
高清屏下canvas重置尺寸引发的问题的解决
Oct 14 #HTML / CSS
You might like
PHPMailer 中文使用说明小结
2010/01/22 PHP
几个比较实用的JavaScript 测试及效验工具
2010/04/18 Javascript
Jquery.TreeView结合ASP.Net和数据库生成菜单导航条
2010/08/27 Javascript
IE6/7 and IE8/9/10(IE7模式)依次隐藏具有absolute或relative的父元素和子元素后再显示父元素
2011/07/31 Javascript
js弹出框轻量级插件jquery.boxy使用介绍
2013/01/15 Javascript
JS数组(Array)处理函数整理
2014/12/07 Javascript
Javascript中的方法链(Method Chaining)介绍
2015/03/15 Javascript
JS中setTimeout的巧妙用法前端函数节流
2016/03/24 Javascript
JavaScript实现点击文本自动定位到下拉框选中操作
2016/06/15 Javascript
JS动态给对象添加事件的简单方法
2016/07/19 Javascript
javaScript如何跳出多重循环break、continue
2016/09/01 Javascript
JQuery实现图片轮播效果
2017/05/08 jQuery
深入理解vue.js中$watch的oldvalue与newValue
2017/08/07 Javascript
element-ui的回调函数Events的用法详解
2018/10/16 Javascript
如何HttpServletRequest文件对象并储存
2020/08/14 Javascript
JS性能优化实现方法及优点进行
2020/08/30 Javascript
Vue自定义组件双向绑定实现原理及方法详解
2020/09/03 Javascript
安装Python的web.py框架并从hello world开始编程
2015/04/25 Python
详解Django缓存处理中Vary头部的使用
2015/07/24 Python
Python正则抓取新闻标题和链接的方法示例
2017/04/24 Python
python实现word 2007文档转换为pdf文件
2018/03/15 Python
Django ORM 常用字段与不常用字段汇总
2019/08/09 Python
Python 函数用法简单示例【定义、参数、返回值、函数嵌套】
2019/09/20 Python
numpy ndarray 按条件筛选数组,关联筛选的例子
2019/11/26 Python
html5组织文档结构_动力节点Java学院整理
2017/07/11 HTML / CSS
优秀学生干部个人的自我评价
2013/10/04 职场文书
《有趣的发现》教学反思
2014/04/15 职场文书
岗位说明书范文
2014/05/07 职场文书
花坛标语大全
2014/06/30 职场文书
学习保证书
2015/01/17 职场文书
2015年护士工作总结范文
2015/03/31 职场文书
师德师风主题教育活动总结
2015/05/07 职场文书
2016应届毕业生实习评语
2015/12/01 职场文书
大学生各类奖学金申请书
2019/06/24 职场文书
Python可视化学习之seaborn调色盘
2022/02/24 Python
根德5570型九灯四波段立体声收音机是电子管收音机的楷模 ? 再论5570
2022/04/05 无线电