UNPKG

@wordpress/block-library

Version:
149 lines (133 loc) 3.41 kB
/** * WordPress dependencies */ import { store, getContext, withSyncEvent } from '@wordpress/interactivity'; // Whether the hash has been handled for the current page load. // This is used to prevent the hash from being handled multiple times. let hashHandled = false; const { actions } = store( 'core/accordion', { state: { get isOpen() { const { id, accordionItems } = getContext(); const accordionItem = accordionItems.find( ( item ) => item.id === id ); return accordionItem ? accordionItem.isOpen : false; }, }, actions: { toggle: () => { const context = getContext(); const { id, autoclose, accordionItems } = context; const accordionItem = accordionItems.find( ( item ) => item.id === id ); if ( autoclose ) { accordionItems.forEach( ( item ) => { item.isOpen = item.id === id ? ! accordionItem.isOpen : false; } ); } else { accordionItem.isOpen = ! accordionItem.isOpen; } }, handleKeyDown: withSyncEvent( ( event ) => { if ( event.key !== 'ArrowUp' && event.key !== 'ArrowDown' && event.key !== 'Home' && event.key !== 'End' ) { return; } event.preventDefault(); const context = getContext(); const { id, accordionItems } = context; const currentIndex = accordionItems.findIndex( ( item ) => item.id === id ); let nextIndex; switch ( event.key ) { case 'ArrowUp': nextIndex = Math.max( 0, currentIndex - 1 ); break; case 'ArrowDown': nextIndex = Math.min( currentIndex + 1, accordionItems.length - 1 ); break; case 'Home': nextIndex = 0; break; case 'End': nextIndex = accordionItems.length - 1; break; } const nextId = accordionItems[ nextIndex ].id; const nextButton = document.getElementById( nextId ); if ( nextButton ) { nextButton.focus(); } } ), openPanelByHash: () => { if ( hashHandled || ! window.location?.hash?.length ) { return; } const context = getContext(); const { id, accordionItems, autoclose } = context; const hash = decodeURIComponent( window.location.hash.slice( 1 ) ); const targetElement = window.document.getElementById( hash ); if ( ! targetElement ) { return; } const panelElement = window.document.querySelector( '.wp-block-accordion-panel[aria-labelledby="' + id + '"]' ); if ( ! panelElement || ! panelElement.contains( targetElement ) ) { return; } hashHandled = true; if ( autoclose ) { accordionItems.forEach( ( item ) => { item.isOpen = item.id === id; } ); } else { const targetItem = accordionItems.find( ( item ) => item.id === id ); if ( targetItem ) { targetItem.isOpen = true; } } // Wait for the panel to be opened before scrolling to it. window.setTimeout( () => { targetElement.scrollIntoView(); }, 0 ); }, }, callbacks: { initAccordionItems: () => { const context = getContext(); const { id, openByDefault, accordionItems } = context; accordionItems.push( { id, isOpen: openByDefault, } ); actions.openPanelByHash(); }, hashChange: () => { hashHandled = false; actions.openPanelByHash(); }, }, }, { lock: true } );