详解使用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对图片照片进行边缘模糊处理的实现
Aug 08 HTML / CSS
CSS3实战第一波 让我们尽情的圆角吧
Aug 27 HTML / CSS
css3 旋转按钮 使用CSS3创建一个旋转可变色按钮
Dec 31 HTML / CSS
基于css3实现漂亮便签样式
Mar 18 HTML / CSS
使用css3制作动感导航条示例
Jan 26 HTML / CSS
浅谈CSS3动画的回调处理
Jul 21 HTML / CSS
CSS3实现缺角矩形,折角矩形以及缺角边框
Dec 20 HTML / CSS
HTML5 canvas 基本语法
Aug 26 HTML / CSS
Html5定位终极解决方案
Feb 05 HTML / CSS
html5 datalist 选中option选项后的触发事件
Mar 05 HTML / CSS
CSS3实现三角形不断放大效果
Apr 13 HTML / CSS
html form表单基础入门案例讲解
Jul 15 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
1 Tube Radio
2021/03/02 无线电
深入解析php中的foreach问题
2013/06/30 PHP
php 获取SWF动画截图示例代码
2014/02/10 PHP
php导出word文档与excel电子表格的简单示例代码
2014/03/08 PHP
php简单分页类实现方法
2015/02/26 PHP
php文件操作相关类实例
2015/06/18 PHP
php实现Session存储到Redis
2015/11/11 PHP
PHP实现超简单的SSL加密解密、验证及签名的方法示例
2017/08/28 PHP
zeroclipboard复制到剪切板的flash
2010/08/04 Javascript
jQuery使用slideUp方法实现控制元素缓慢收起
2015/03/27 Javascript
JS实现点击上移下移LI行数据的方法
2015/08/05 Javascript
jquery实现点击向下展开菜单项(伸缩导航)效果
2015/08/22 Javascript
Bootstrap所支持的表单控件实例详解
2016/05/16 Javascript
通过button将form表单的数据提交到action层的实例
2017/09/08 Javascript
Angular实现搜索框及价格上下限功能
2018/01/19 Javascript
JavaScript Math对象和调试程序的方法分析
2019/05/13 Javascript
jQuery删除/清空指定元素的所有子节点实例代码
2019/07/04 jQuery
vue实现在线预览pdf文件和下载(pdf.js)
2019/11/26 Javascript
el-form 多层级表单的实现示例
2020/09/10 Javascript
[39:21]LGD vs OG 2019国际邀请赛淘汰赛 胜者组 BO3 第二场 8.24
2019/09/10 DOTA
在Python dataframe中出生日期转化为年龄的实现方法
2018/10/20 Python
对python读写文件去重、RE、set的使用详解
2018/12/11 Python
python读写csv文件实例代码
2019/07/05 Python
Python 实现自动获取种子磁力链接方式
2020/01/16 Python
python 高阶函数简单介绍
2021/02/19 Python
Crocs美国官方网站:卡骆驰洞洞鞋
2017/08/04 全球购物
用Python写一个for循环的例子
2016/07/19 面试题
工程师求职简历的自我评价分享
2013/10/10 职场文书
搞笑获奖感言
2014/01/30 职场文书
行政内勤岗位职责
2014/04/07 职场文书
小学生九一八纪念日83周年演讲稿500字
2014/09/17 职场文书
2016全国“质量月”活动标语口号
2015/12/26 职场文书
详解CSS开发过程中的20个快速提升技巧
2021/05/21 HTML / CSS
Mybatis-plus在项目中的简单应用
2021/07/01 Java/Android
部分武汉产收音机展览
2022/04/07 无线电
nginx lua 操作 mysql
2022/05/15 Servers