详解使用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弹性盒模型开发笔记(三)
Apr 26 HTML / CSS
使用Filters滤镜弥补CSS3的跨浏览器问题以及兼容低版本IE
Jan 23 HTML / CSS
利用CSS3的transition属性实现滑动效果
Aug 05 HTML / CSS
CSS3绘制圆角矩形的简单示例
Sep 28 HTML / CSS
CSS实现聊天气泡效果
Apr 26 HTML / CSS
html标签之Object和EMBED标签详解
Jul 04 HTML / CSS
HTML5注册页面示例代码
Mar 27 HTML / CSS
html5 postMessage解决跨域、跨窗口消息传递方案
Dec 20 HTML / CSS
HTML5 canvas 瀑布流文字效果的示例代码
Jan 31 HTML / CSS
Html5页面播放M4a音频文件
Mar 30 HTML / CSS
HTTP中的Content-type详解
Jan 18 HTML / CSS
使用CSS实现音波加载效果
May 07 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
PHP的范围解析操作符(::)的含义分析说明
2011/07/03 PHP
解析php获取字符串的编码格式的方法(函数)
2013/06/21 PHP
php关闭warning问题的解决方法
2016/05/17 PHP
Javascript Math ceil()、floor()、round()三个函数的区别
2010/03/09 Javascript
Jquery replace 字符替换实现代码
2010/12/02 Javascript
javascript插入样式实现代码
2012/02/22 Javascript
关于extjs4如何获取grid修改后的数据的问题
2013/08/07 Javascript
javascript进行四舍五入方法汇总
2014/12/16 Javascript
jQuery.prop() 使用详解
2015/07/19 Javascript
jQuery多级手风琴菜单实例讲解
2015/10/22 Javascript
JavaScript实现经典排序算法之插入排序
2016/12/28 Javascript
微信小程序 简单教程实例详解
2017/01/13 Javascript
JavaScript html5利用FileReader实现上传功能
2020/03/27 Javascript
Dropify.js图片宽高自适应的方法
2017/11/27 Javascript
jQuery实现的简单手风琴效果示例
2018/08/29 jQuery
基于Vue-cli快速搭建项目的完整步骤
2018/11/03 Javascript
6行代码实现微信小程序页面返回顶部效果
2018/12/28 Javascript
详解在微信小程序的JS脚本中使用Promise来优化函数处理
2019/03/06 Javascript
[50:29]2014 DOTA2华西杯精英邀请赛 5 24 DK VS iG
2014/05/26 DOTA
[03:42]2014DOTA2西雅图国际邀请赛 Navi战队巡礼
2014/07/07 DOTA
使用PYTHON创建XML文档
2012/03/01 Python
Python中random模块生成随机数详解
2016/03/10 Python
Python实现向服务器请求压缩数据及解压缩数据的方法示例
2017/06/09 Python
Python3 操作符重载方法示例
2017/11/23 Python
Python定义函数功能与用法实例详解
2019/04/08 Python
python递归调用中的坑:打印有值, 返回却None
2020/03/16 Python
Grid 宫格常用布局的实现
2020/01/10 HTML / CSS
使用html2canvas将页面转成图并使用用canvas2image下载
2019/04/04 HTML / CSS
澳大利亚家具商店:Freedom
2020/12/17 全球购物
会计岗位职责范本
2014/03/07 职场文书
企业党员一句话承诺
2014/05/30 职场文书
党的群众路线教育实践活动制度建设计划
2014/11/03 职场文书
2015年全民国防教育日活动总结
2015/03/23 职场文书
使用GO语言实现Mysql数据库CURD的简单示例
2021/08/07 Golang
教你nginx跳转配置的四种方式
2022/07/07 Servers
Linux在两个服务器直接传文件的操作方法
2022/08/05 Servers