@automattic/simple-components
Version:
React components, as used on WordPress.com
130 lines (114 loc) • 2.95 kB
JSX
/** @format */
/**
* External dependencies
*/
import PropTypes from 'prop-types';
import React, { Component } from 'react';
import classnames from 'classnames';
import { noop } from 'lodash';
import { localize } from 'i18n-calypso';
import Gridicon from 'gridicons';
export class Notice extends Component {
static defaultProps = {
className: '',
duration: 0,
icon: null,
isCompact: false,
isLoading: false,
onDismissClick: noop,
status: null,
text: null,
};
static propTypes = {
className: PropTypes.string,
duration: PropTypes.number,
icon: PropTypes.string,
isCompact: PropTypes.bool,
isLoading: PropTypes.bool,
onDismissClick: PropTypes.func,
showDismiss: PropTypes.bool,
status: PropTypes.oneOf( [ 'is-error', 'is-info', 'is-success', 'is-warning', 'is-plain' ] ),
text: PropTypes.oneOfType( [
PropTypes.arrayOf( PropTypes.oneOfType( [ PropTypes.string, PropTypes.node ] ) ),
PropTypes.oneOfType( [ PropTypes.string, PropTypes.node ] ),
] ),
translate: PropTypes.func.isRequired,
};
dismissTimeout = null;
componentDidMount() {
if ( this.props.duration > 0 ) {
this.dismissTimeout = setTimeout( this.props.onDismissClick, this.props.duration );
}
}
componentWillUnmount() {
if ( this.dismissTimeout ) {
clearTimeout( this.dismissTimeout );
}
}
componentDidUpdate() {
clearTimeout( this.dismissTimeout );
if ( this.props.duration > 0 ) {
this.dismissTimeout = setTimeout( this.props.onDismissClick, this.props.duration );
}
}
getIcon() {
let icon;
switch ( this.props.status ) {
case 'is-info':
icon = 'info';
break;
case 'is-success':
icon = 'checkmark';
break;
case 'is-error':
icon = 'notice';
break;
case 'is-warning':
icon = 'notice';
break;
default:
icon = 'info';
break;
}
return icon;
}
render() {
const {
children,
className,
icon,
isCompact,
isLoading,
onDismissClick,
showDismiss = ! isCompact, // by default, show on normal notices, don't show on compact ones
status,
text,
translate,
} = this.props;
const classes = classnames( 'notice', status, className, {
'is-compact': isCompact,
'is-loading': isLoading,
'is-dismissable': showDismiss,
} );
return (
<div className={ classes }>
<span className="notice__icon-wrapper">
<Gridicon className="notice__icon" icon={ icon || this.getIcon() } size={ 24 } />
</span>
<span className="notice__content">
<span className="notice__text">{ text ? text : children }</span>
</span>
{ text ? children : null }
{ showDismiss && (
<span tabIndex="0" className="notice__dismiss" onClick={ onDismissClick }>
<Gridicon icon="cross" size={ 24 } />
<span className="notice__screen-reader-text screen-reader-text">
{ translate( 'Dismiss' ) }
</span>
</span>
) }
</div>
);
}
}
export default localize( Notice );