| <!DOCTYPE html> |
| <title>Test Script-Based Focus for Fenced Frames</title> |
| <script src="/resources/testharness.js"></script> |
| <script src="/resources/testharnessreport.js"></script> |
| <script src="/resources/testdriver.js"></script> |
| <script src="/resources/testdriver-actions.js"></script> |
| <script src="/resources/testdriver-vendor.js"></script> |
| <script src="/common/utils.js"></script> |
| <script src="resources/utils.js"></script> |
| <script src="/common/dispatcher/dispatcher.js"></script> |
| |
| <script src="/common/get-host-info.sub.js"></script> |
| |
| <body> |
| <script> |
| async function AttemptButtonFocus(frame, expecting_focus) { |
| await frame.execute(async (expecting_focus) => { |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_equals(document.activeElement == button, expecting_focus, |
| "Button's focus should match expected focus"); |
| }, [expecting_focus]); |
| } |
| |
| async function ClickOn(element, actions) { |
| // Wait until the window size is initialized. |
| while (window.innerWidth == 0) { |
| await new Promise(resolve => requestAnimationFrame(resolve)); |
| } |
| await actions.pointerMove(0, 0, {origin: element}) |
| .pointerDown() |
| .pointerUp() |
| .send(); |
| } |
| |
| async function SetupTest(click=true) { |
| // Clean up any leftover frames from prior tests. |
| document.querySelectorAll("fencedframe").forEach(e => { |
| e.remove(); |
| }) |
| |
| const actions = new test_driver.Actions(); |
| |
| const frame = attachFencedFrameContext(); |
| const fencedframe_element = frame.element; |
| |
| if (click) |
| await ClickOn(document.body, actions); |
| |
| return [actions, frame, fencedframe_element]; |
| } |
| |
| promise_test(async () => { |
| const [actions, ff1, ff1_element] = await SetupTest(false); |
| |
| await ClickOn(ff1_element, actions); |
| await AttemptButtonFocus(ff1, true); |
| |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_true(document.activeElement == button, |
| "The button should have focus"); |
| assert_false(navigator.userActivation.isActive, |
| "Window should not have user activation"); |
| }, "An embedder can focus out of a fenced frame"); |
| |
| promise_test(async () => { |
| const [actions, frame, fencedframe_element] = await SetupTest(); |
| |
| await AttemptButtonFocus(frame, false); |
| await ClickOn(fencedframe_element, actions); |
| await AttemptButtonFocus(frame, true); |
| }, "Fenced frames can't pull script focus until getting user activation"); |
| |
| promise_test(async t => { |
| const [actions, frame, fencedframe_element] = await SetupTest(); |
| |
| await ClickOn(fencedframe_element, actions); |
| await ClickOn(document.body, actions); |
| |
| await AttemptButtonFocus(frame, true); |
| |
| // Give the browser time to receive the focus event before attempting |
| // another focus. |
| await t.step_timeout(async () => {await AttemptButtonFocus(frame, true);}, |
| 500); |
| }, "Focused fenced frames can move programmatic focus within frame"); |
| |
| promise_test(async () => { |
| const [actions, frame, fencedframe_element] = await SetupTest(); |
| |
| await ClickOn(fencedframe_element, actions); |
| await ClickOn(document.body, actions); |
| |
| // This will pull focus across a frame boundary and consume user activation. |
| await AttemptButtonFocus(frame, true); |
| |
| await ClickOn(document.body, actions); |
| await AttemptButtonFocus(frame, false); |
| }, "Script focus into a fenced frame consumes user activation"); |
| |
| promise_test(async () => { |
| const [actions, ff1, ff1_element] = await SetupTest(); |
| |
| const ff2 = attachFencedFrameContext(); |
| const ff2_element = ff2.element; |
| |
| await ClickOn(ff1_element, actions); |
| |
| await AttemptButtonFocus(ff1, true); |
| await AttemptButtonFocus(ff2, false); |
| }, "Another fenced frame cannot pull focus out of a focused fenced frame"); |
| |
| promise_test(async () => { |
| const [actions, ff1, ff1_element] = await SetupTest(); |
| |
| await ClickOn(ff1_element, actions); |
| await AttemptButtonFocus(ff1, true); |
| |
| await ff1.execute(async () => { |
| const ff2 = attachFencedFrameContext(); |
| |
| await ff2.execute(async () => { |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_false(document.activeElement == button, |
| "The button should not have focus"); |
| assert_false(navigator.userActivation.isActive, |
| "The fenced frame should not have user activation"); |
| }); |
| }); |
| }, "A fenced frame nested in another fenced frame cannot pull focus"); |
| |
| promise_test(async () => { |
| const [actions, ff1, ff1_element] = await SetupTest(); |
| |
| await ClickOn(document.body, actions); |
| |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_equals(document.activeElement, button, |
| "The button in the main page should have focus."); |
| |
| await ff1.execute(async () => { |
| assert_false(navigator.userActivation.isActive, |
| "The fenced frame should not have user activation."); |
| window.focus(); |
| }); |
| |
| assert_equals(document.activeElement, button, |
| "The button in the main page should still have focus."); |
| }, "A fenced frame cannot pull window.focus() without user activation"); |
| |
| promise_test(async () => { |
| const [actions, ff1, ff1_element] = await SetupTest(); |
| |
| await ClickOn(ff1_element, actions); |
| await ClickOn(document.body, actions); |
| |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_equals(document.activeElement, button, |
| "The button should have focus."); |
| |
| await ff1.execute(async () => { |
| assert_true(navigator.userActivation.isActive, |
| "The fenced frame should have user activation."); |
| window.focus(); |
| assert_false(navigator.userActivation.isActive, |
| "The fenced frame's user activation should be consumed by the focus"); |
| }); |
| |
| assert_equals(document.activeElement, document.body, |
| "The main page's focus should be pulled away from the button."); |
| }, "A fenced frame can pull window.focus() after user activation"); |
| |
| promise_test(async () => { |
| var actions = new test_driver.Actions(); |
| |
| const frame = attachIFrameContext( |
| {origin: get_host_info().HTTPS_REMOTE_ORIGIN}); |
| const iframe_element = |
| document.body.getElementsByTagName('iframe')[0]; |
| |
| await frame.execute(async () => { |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_equals(document.activeElement, button, |
| "The button in the iframe should have focus."); |
| }, [true]); |
| |
| const button = document.createElement("button"); |
| document.body.append(button); |
| button.focus(); |
| assert_equals(document.activeElement, button, |
| "The button in the main page should have focus."); |
| }, "An cross-origin iframe can pull focus back and forth without activation"); |
| |
| </script> |
| </body> |