日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網(wǎng)站首頁 編程語言 正文

React常見跨窗口通信方式實例詳解_React

作者:jie19100 ? 更新時間: 2022-11-26 編程語言

iframe

跨窗口通信就是在嵌套了iframe的時候,實現(xiàn)iframe與父窗口的通信。

什么是iframe

  • 它是一個html標簽,它可以將一個網(wǎng)站作為一個dom元素,嵌入到另一個網(wǎng)站中。
  • iframe具有自己的windowdocument對象。

使用場景

  • 比如公司開發(fā)了一個完整的網(wǎng)站,需要在另一個項目中去使用。比如直播功能,一些插件,這時候就可以使用iframe嵌入的方式。減少了重復開發(fā)的時間,需要修改界面的時候,也只需要修改一份代碼即可。
  • 微應用,微應用也有很多是使用iframe來實現(xiàn)。

同源策略

當兩個網(wǎng)站同時滿足:同協(xié)議+同域名+同端口的時候。

當iframe與父窗口同源時

  • 父窗口可以對iframe進行完全訪問,如windowdocumentlocation等對象的訪問。
  • 父窗口可以調(diào)用iframe的全局函數(shù)。
  • 父窗口可以修改iframe的元素內(nèi)容

效果圖

index1.html嵌套同源的index2.html

html1

<body>
  <h1>html-1</h1>
  <iframe src="http://127.0.0.1:3000/index2.html" frameborder="0"></iframe>
  <script>
    const iframe = document.querySelector("iframe");
    iframe.onload = function () {
      console.log(iframe)
      // 獲取iframe的window對象
      const iWindow = iframe.contentWindow;
      // 獲取iframe的document對象
      const iDocument = iframe.contentDocument;
      console.log(iWindow)
      console.log(iWindow.location)
      iWindow.say()
      iDocument.body.innerHTML = "<h1>this is html-1</h1>"
    }
  </script>
</body>

html2

<body>
  <h1>html-2</h1>
  <script>
    function say() {
      console.log("saying!!!")
    }
  </script>
</body>

效果圖

index1.html嵌套同源的index2.html

發(fā)現(xiàn)子iframewindowdocumentlocation對象,以及子iframe的全局方法都可以訪問。

當iframe與父窗口不同源時

  • 父窗口無法訪問iframewindow的所有屬性與方法。
  • 父窗口無法訪問iframedocument
  • 無法調(diào)用iframe的全局方法。

效果圖

跨窗口通信

一:通過window.parent、frames、top

window.frames:獲取子iframe的列表,與document.querySelector("iframe")一樣

window.parent:獲取父window的引用

window.top:獲取最頂層窗口的window引用

上一節(jié)我們講到,當iframe同源時,不同窗口可以拿到對方的window對象,以及全局方法,那么我們可以利用全局方法來實現(xiàn)不同window窗口的通信。

html1

<body>
  <h1>html-1</h1>
  <div>
    <button onclick="send(Math.random(1))">發(fā)送數(shù)據(jù)給html2</button>
  </div>
  <iframe src="http://127.0.0.1:3000/index2.html" frameborder="0"></iframe>
  <script>
    const iframe = document.querySelector("iframe");
    let send;
    iframe.onload = function () {
      // 獲取iframe的window對象
      const iWindow = iframe.contentWindow;
      // 獲取iframe的document對象
      const iDocument = iframe.contentDocument;
      function receive(value) {
        console.log("這是html1,來了一條數(shù)據(jù):", value)
      }
      send = function (value) {
        iWindow.receive(value)
      }
    }
  </script>
</body>

html2

<body>
  <h1>html-2</h1>
  <div>
    <button onclick="send(Math.random(1))">發(fā)送數(shù)據(jù)給html1</button>
  </div>
  <script>
    function receive(value) {
      console.log("當前是html2,收到一條數(shù)據(jù):", value)
    }
    function send(value) {
      window.parent.receive(value)
    }
  </script>
</body>

效果圖

同理,window.top也可以這樣通信

二:window.postMessage

postMessage支持不同窗口之間的通信,即使是非同源的情況。

發(fā)送數(shù)據(jù)

當需要使用給其他窗口(window)發(fā)送數(shù)據(jù)時,需要調(diào)用對方windowpostMessage方法。

該方法接收兩個參數(shù)

  • 參數(shù)一:需要發(fā)送的數(shù)據(jù),數(shù)據(jù)最后為字符串形式,因為IE只支持字符串數(shù)據(jù)。
  • 參數(shù)二:接收方的地址(協(xié)議+域名+端口)

接收數(shù)據(jù)

監(jiān)聽message事件

該事件對象包含接收的數(shù)據(jù),以及發(fā)送方的地址等信息。

html1

<body>
  <h1>html-1</h1>
  <div>
    <button onclick="send(Math.random(1))">發(fā)送數(shù)據(jù)給html2</button>
  </div>
  <iframe src="http://127.0.0.1:3001/index2.html" frameborder="0"></iframe>
  <script>
    const iframe = document.querySelector("iframe");
    let send;
    iframe.onload = function () {
      // 獲取iframe的window對象
      const iWindow = iframe.contentWindow;
      send = function (value) {
        iWindow.postMessage("wuwuwuw", "http://127.0.0.1:3001")
      }
    }
    window.addEventListener("message", function (event) {
      if (event.origin != 'http://127.0.0.1:3001') {
        // 過濾指定地址的信息
        return;
      }
      if (window == event.source) {
        // 頁面初始化的時候會被瀏覽器觸發(fā)一次message,在這里根據(jù)發(fā)送方地址進行過濾
        return
      }
      console.log("html1收到的數(shù)據(jù) " + event.data);
    })
  </script>
</body>

html2

<body>
  <h1>html-2</h1>
  <div>
    <button onclick="send(Math.random(1))">發(fā)送數(shù)據(jù)給html1</button>
  </div>
  <script>
    function receive(value) {
      console.log("當前是html2,收到一條數(shù)據(jù):", value)
    }
    function send(value) {
      window.parent.postMessage(value, "http://127.0.0.1:3000")
    }
    window.addEventListener("message", function (event) {
      if (event.origin != 'http://127.0.0.1:3000') {
        // 過濾指定地址的信息
        return;
      }
      if (window == event.source) {
        // 頁面初始化的時候會被瀏覽器觸發(fā)一次message,在這里根據(jù)發(fā)送方地址進行過濾
        return;
      }
      console.log("html2收到的數(shù)據(jù) " + event.data);
    })
  </script>
</body>

效果圖

其他通信方法

  • 使用soket,需要后端支持
  • 使用本地存儲,監(jiān)聽本地存儲的數(shù)據(jù)變化

原文鏈接:https://juejin.cn/post/7155037804102156296

欄目分類
最近更新