-
Notifications
You must be signed in to change notification settings - Fork 4.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Enter editing mode via Enter or Spacebar #58795
Changes from 6 commits
030b6ea
75a8d55
4316b4a
8cc9253
08c81eb
d8e988e
ad788fc
39b0a10
0ef75fa
e9a2861
7441ad2
fea790d
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,128 @@ | ||
/** | ||
* WordPress dependencies | ||
*/ | ||
const { test, expect } = require( '@wordpress/e2e-test-utils-playwright' ); | ||
|
||
test.use( { | ||
editorNavigationUtils: async ( { page, pageUtils }, use ) => { | ||
await use( new EditorNavigationUtils( { page, pageUtils } ) ); | ||
}, | ||
} ); | ||
|
||
test.describe( 'Site editor navigation', () => { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. This test feels quite big and I worry that it's going to need a lot of work to maintain it as the editor continues to evolve. I think it would be better if we could find a way to just test the spacebar press on its own. |
||
test.beforeAll( async ( { requestUtils } ) => { | ||
await requestUtils.activateTheme( 'emptytheme' ); | ||
} ); | ||
|
||
test.afterAll( async ( { requestUtils } ) => { | ||
await requestUtils.activateTheme( 'twentytwentyone' ); | ||
} ); | ||
|
||
test( 'Can use keyboard to navigate the site editor', async ( { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. ' |
||
admin, | ||
editorNavigationUtils, | ||
page, | ||
pageUtils, | ||
} ) => { | ||
await admin.visitSiteEditor(); | ||
|
||
// Test: Can navigate to a sidebar item and into its subnavigation frame without losing focus | ||
// Go to the Pages button | ||
|
||
await editorNavigationUtils.tabToLabel( 'Pages', { times: 10 } ); | ||
jeryj marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
await expect( | ||
page.getByRole( 'button', { name: 'Pages' } ) | ||
).toBeFocused(); | ||
await pageUtils.pressKeys( 'Enter' ); | ||
// We should be in the Pages sidebar | ||
await expect( | ||
page.getByRole( 'button', { name: 'Back', exact: true } ) | ||
).toBeFocused(); | ||
await pageUtils.pressKeys( 'Enter' ); | ||
// Go back to the main navigation | ||
await expect( | ||
page.getByRole( 'button', { name: 'Pages' } ) | ||
).toBeFocused(); | ||
|
||
// Test: Can navigate into the iframe using the keyboard | ||
await editorNavigationUtils.tabToNode( 'IFRAME', { times: 10 } ); | ||
// Getting the actual iframe as a cleaner locator was suprisingly tricky, | ||
// so we're using a css selector with .is-focused which should be present when the iframe has focus. | ||
await expect( | ||
page.locator( 'iframe[name="editor-canvas"].is-focused' ) | ||
).toBeFocused(); | ||
jeryj marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// Enter into the site editor frame | ||
await pageUtils.pressKeys( 'Enter' ); | ||
// Focus should still be on the iframe. | ||
await expect( | ||
page.locator( 'iframe[name="editor-canvas"]' ) | ||
).toBeFocused(); | ||
// But did it do anything? | ||
// Test to make sure a Tab keypress works as expected. | ||
// As of this writing, we are in select mode and a tab | ||
/// keypress will reveal the header template select mode | ||
jeryj marked this conversation as resolved.
Show resolved
Hide resolved
|
||
// button. This test is not documenting that we _want_ | ||
// that action, but checking that we are within the site | ||
// editor and keypresses work as intened. | ||
await pageUtils.pressKeys( 'Tab' ); | ||
await expect( | ||
page.getByRole( 'button', { | ||
name: 'Template Part Block. Row 1. header', | ||
} ) | ||
).toBeFocused(); | ||
|
||
// Test: We can go back to the main navigation from the editor frame | ||
// Move to the document toolbar | ||
await pageUtils.pressKeys( 'alt+F10' ); | ||
// Go to the open navigation button | ||
await pageUtils.pressKeys( 'shift+Tab' ); | ||
|
||
// Open the sidebar again | ||
await expect( | ||
page.getByRole( 'button', { | ||
name: 'Open Navigation', | ||
exact: true, | ||
} ) | ||
).toBeFocused(); | ||
await pageUtils.pressKeys( 'Enter' ); | ||
|
||
await expect( | ||
page.getByLabel( 'Go to the Dashboard' ).first() | ||
).toBeFocused(); | ||
} ); | ||
} ); | ||
|
||
class EditorNavigationUtils { | ||
constructor( { page, pageUtils } ) { | ||
this.page = page; | ||
this.pageUtils = pageUtils; | ||
} | ||
|
||
async tabToLabel( label, { times = 10 } ) { | ||
jeryj marked this conversation as resolved.
Show resolved
Hide resolved
|
||
for ( let i = 0; i < times; i++ ) { | ||
await this.pageUtils.pressKeys( 'Tab' ); | ||
const activeLabel = await this.page.evaluate( () => { | ||
return ( | ||
document.activeElement.getAttribute( 'aria-label' ) || | ||
document.activeElement.textContent | ||
); | ||
} ); | ||
if ( activeLabel === label ) { | ||
return; | ||
} | ||
} | ||
} | ||
|
||
async tabToNode( nodeName, { times = 10 } ) { | ||
for ( let i = 0; i < times; i++ ) { | ||
await this.pageUtils.pressKeys( 'Tab' ); | ||
const activeNode = await this.page.evaluate( () => { | ||
return document.activeElement.nodeName; | ||
} ); | ||
if ( activeNode === nodeName ) { | ||
return; | ||
} | ||
} | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This doesn't make any sense. When you have multiple keydown handlers, all of them should execute
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The only time the
if
statement above it runs is when a keydown listener is passed to the iframe, which is only in theview
mode of the site editor. The shortcuts shouldn't be enabled at that point since we're not within the editor.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok, but that's not clear from reading this. Also there might be other iframe instances that pass onKeydown and do want shortcuts disabled. Using onKeydown to disable/enable shortcuts seems a bit strange, imo that should be done through a separate prop. Maybe reuse
isPreviewMode
?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure how all of that should be structured. The only thing important to me in this PR was fixing a bug where you couldn't use the keyboard to enter into Edit mode from View mode. This onKeydown was not allowed to run.
Very open to fixing this another way. I wasn't sold on this solution.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, this is fine, all I'm saying is that the
else
shouldn't have been added?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Oh - That's fine with me: #59474