| // Opens |url| in an iframe, establish a message channel with it, and waits for |
| // a message from the frame content. Returns a promise that resolves with the |
| // data of the message, or rejects on 2000ms timeout. |
| function openSXGInIframeAndWaitForMessage(test_object, url) { |
| return new Promise(async (resolve, reject) => { |
| // We can't catch the network error on iframe. So we use the timer. |
| test_object.step_timeout(() => reject('timeout'), 2000); |
| |
| const frame = await withIframe(url, 'sxg_iframe'); |
| const channel = new MessageChannel(); |
| channel.port1.onmessage = (event) => resolve(event.data); |
| frame.contentWindow.postMessage( |
| {port: channel.port2}, '*', [channel.port2]); |
| }); |
| } |
| |
| function withIframe(url, name) { |
| return new Promise((resolve, reject) => { |
| const frame = document.createElement('iframe'); |
| frame.src = url; |
| frame.name = name; |
| frame.onload = () => resolve(frame); |
| frame.onerror = () => reject('failed to load ' + url); |
| document.body.appendChild(frame); |
| }); |
| } |
| |
| function loadScript(url) { |
| return new Promise((resolve, reject) => { |
| const scriptTag = document.createElement('script'); |
| scriptTag.src = url; |
| scriptTag.onload = () => resolve(); |
| scriptTag.onerror = () => reject('failed to load ' + url); |
| document.head.appendChild(scriptTag); |
| }); |
| } |
| |
| // Used to delay the iframe creation after "didFinishLoadForFrame" is printed. |
| // This is intended to avoid the flakiness of the result outputs. |
| const waitUntilDidFinishLoadForFrame = new Promise((resolve) => { |
| window.addEventListener('load', () => setTimeout(resolve, 0)); |
| }); |