woocommerce/plugins/woocommerce-admin/client/layout/transient-notices/snackbar/index.js

111 lines
2.6 KiB
JavaScript

/**
* External dependencies
*/
import { noop } from 'lodash';
import classnames from 'classnames';
import { speak } from '@wordpress/a11y';
import { useEffect, forwardRef, renderToString } from '@wordpress/element';
import { __ } from '@wordpress/i18n';
import warning from '@wordpress/warning';
import { Button } from '@wordpress/components';
const NOTICE_TIMEOUT = 10000;
/** @typedef {import('@wordpress/element').WPElement} WPElement */
/**
* Custom hook which announces the message with the given politeness, if a
* valid message is provided.
*
* @param {string|WPElement} [message] Message to announce.
* @param {'polite'|'assertive'} politeness Politeness to announce.
*/
function useSpokenMessage( message, politeness ) {
const spokenMessage =
typeof message === 'string' ? message : renderToString( message );
useEffect( () => {
if ( spokenMessage ) {
speak( spokenMessage, politeness );
}
}, [ spokenMessage, politeness ] );
}
function Snackbar(
{
className,
children,
spokenMessage = children,
politeness = 'polite',
actions = [],
onRemove = noop,
icon = null,
},
ref
) {
useSpokenMessage( spokenMessage, politeness );
useEffect( () => {
const timeoutHandle = setTimeout( () => {
onRemove();
}, NOTICE_TIMEOUT );
return () => clearTimeout( timeoutHandle );
}, [] );
const classes = classnames( className, 'components-snackbar' );
if ( actions && actions.length > 1 ) {
// we need to inform developers that snackbar only accepts 1 action
warning(
'Snackbar can only have 1 action, use Notice if your message require many messages'
);
// return first element only while keeping it inside an array
actions = [ actions[ 0 ] ];
}
const snackbarContentClassnames = classnames(
'components-snackbar__content',
{
'components-snackbar__content-with-icon': !! icon,
}
);
return (
<div
ref={ ref }
className={ classes }
onClick={ onRemove }
tabIndex="0"
role="button"
onKeyPress={ onRemove }
aria-label={ __( 'Dismiss this notice' ) }
>
<div className={ snackbarContentClassnames }>
{ icon && (
<div className="components-snackbar__icon">{ icon }</div>
) }
{ children }
{ actions.map( ( { label, onClick, url }, index ) => {
return (
<Button
key={ index }
href={ url }
isTertiary
onClick={ ( event ) => {
event.stopPropagation();
if ( onClick ) {
onClick( event );
}
} }
className="components-snackbar__action"
>
{ label }
</Button>
);
} ) }
</div>
</div>
);
}
export default forwardRef( Snackbar );