github源碼:https://github.com/boycy815/topProxy
為了偷懶所以依賴了Kissy:http://docs.kissyui.com/
用法舉例:
需求是在http://www.demo.org/top.html中通過iframe方式嵌入http://www.iframe.com/iframe.html,而在iframe頁面中希望通過點擊一個按鈕,調用top頁面的一個js方法。
1. 在top頁面中建立方法供內部頁面使用
function testFun (text) { alert(text); }
2. 在http://www.demo.org/top.html中嵌入iframe
<iframe width="320" height="240" src="http://www.iframe.com/iframe.html"></iframe>
3. 建立www.demo.org域下建立一個代理頁面http://www.demo.org/proxy.html用於跨域通信使用
4. 在http://www.demo.org/proxy.html加入用於代理頁面的邏輯
5. 在http://www.iframe.com/iframe.html頁面中通過js配置代理頁面地址
TopProxyConfig = {url:'http://www.demo.org/proxy.html'};
6. 通過kissy加載通信模塊
KISSY.use('topproxy', function(S, TopProxy){ //執行代碼 });
7. 在http://www.iframe.com/iframe.html通過TopProxy.call直接訪問http://www.demo.org/top.html中的方法,如
KISSY.use('topproxy', function(S, TopProxy){ TopProxy.call('testFun', '這是一個真實的故事'); });
其中call方法的第一個參數是外部網頁的全局方法名,支持“.”,后面參數無限個,均傳到目標方法里去。
注:
1. 調用后可能不會立即執行,會等到iframe加載完畢后才觸發,如果想預加載可以提前執行一個沒用的方法。
2. 如果不設置TopProxyConfig,那么會認為引用iframe和父iframe同域(大域)並直接調用top對象。
3. 在IE678下可能直接調用top中的系統方法會報錯,由於系統方法不支持apply。
原理:
A頁面嵌日的iframe頁面B,其中B想調用A的方法並傳遞數據,那么可在B中嵌入A頁面同域下的一個iframe頁面C,C可以通過window.top訪問到A的window。那么在B中可以改變C的href的hash向C傳遞一些信息,然后C再把這些信息傳遞給A,從而間接達到B給A傳遞信息的目的。