UNPKG

react-toast-notifications

Version:

A configurable, composable, toast notification system for react.

295 lines (236 loc) 11.4 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.useToasts = exports.withToastManager = exports.ToastConsumer = exports.ToastProvider = undefined; var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }(); var _react = require('react'); var _react2 = _interopRequireDefault(_react); var _reactDom = require('react-dom'); var _reactTransitionGroup = require('react-transition-group'); var _ToastController = require('./ToastController'); var _ToastContainer = require('./ToastContainer'); var _ToastElement = require('./ToastElement'); var _utils = require('./utils'); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _objectWithoutProperties(obj, keys) { var target = {}; for (var i in obj) { if (keys.indexOf(i) >= 0) continue; if (!Object.prototype.hasOwnProperty.call(obj, i)) continue; target[i] = obj[i]; } return target; } function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; } function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; } var defaultComponents = { Toast: _ToastElement.DefaultToast, ToastContainer: _ToastContainer.ToastContainer }; // $FlowFixMe `createContext` var ToastContext = _react2.default.createContext(); var Consumer = ToastContext.Consumer, Provider = ToastContext.Provider; var canUseDOM = !!(typeof window !== 'undefined' && window.document && window.document.createElement); // Provider // ============================== var ToastProvider = exports.ToastProvider = function (_Component) { _inherits(ToastProvider, _Component); function ToastProvider() { var _ref; var _temp, _this, _ret; _classCallCheck(this, ToastProvider); for (var _len = arguments.length, args = Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } return _ret = (_temp = (_this = _possibleConstructorReturn(this, (_ref = ToastProvider.__proto__ || Object.getPrototypeOf(ToastProvider)).call.apply(_ref, [this].concat(args))), _this), _this.state = { toasts: [] }, _this.has = function (id) { if (!_this.state.toasts.length) { return false; } return Boolean(_this.state.toasts.filter(function (t) { return t.id === id; }).length); }, _this.onDismiss = function (id) { var cb = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : _utils.NOOP; return function () { cb(id); _this.remove(id); }; }, _this.add = function (content) { var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var cb = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : _utils.NOOP; var id = options.id ? options.id : (0, _utils.generateUEID)(); var callback = function callback() { return cb(id); }; // bail if a toast exists with this ID if (_this.has(id)) { return; } // update the toast stack _this.setState(function (state) { var newToast = _extends({ content: content, id: id }, options); var toasts = _this.props.newestOnTop ? [newToast].concat(_toConsumableArray(state.toasts)) : [].concat(_toConsumableArray(state.toasts), [newToast]); return { toasts: toasts }; }, callback); // consumer may want to do something with the generated ID (and not use the callback) return id; }, _this.remove = function (id) { var cb = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : _utils.NOOP; var callback = function callback() { return cb(id); }; // bail if NO toasts exists with this ID if (!_this.has(id)) { return; } _this.setState(function (state) { var toasts = state.toasts.filter(function (t) { return t.id !== id; }); return { toasts: toasts }; }, callback); }, _this.removeAll = function () { if (!_this.state.toasts.length) { return; } _this.state.toasts.forEach(function (t) { return _this.remove(t.id); }); }, _this.update = function (id) { var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var cb = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : _utils.NOOP; var callback = function callback() { return cb(id); }; // bail if NO toasts exists with this ID if (!_this.has(id)) { return; } // update the toast stack _this.setState(function (state) { var old = state.toasts; var i = old.findIndex(function (t) { return t.id === id; }); var updatedToast = _extends({}, old[i], options); var toasts = [].concat(_toConsumableArray(old.slice(0, i)), [updatedToast], _toConsumableArray(old.slice(i + 1))); return { toasts: toasts }; }, callback); }, _temp), _possibleConstructorReturn(_this, _ret); } // Internal Helpers // ------------------------------ // Public API // ------------------------------ _createClass(ToastProvider, [{ key: 'render', value: function render() { var _this2 = this; var _props = this.props, inheritedAutoDismiss = _props.autoDismiss, autoDismissTimeout = _props.autoDismissTimeout, children = _props.children, components = _props.components, placement = _props.placement, portalTargetSelector = _props.portalTargetSelector, transitionDuration = _props.transitionDuration; var _defaultComponents$co = _extends({}, defaultComponents, components), Toast = _defaultComponents$co.Toast, ToastContainer = _defaultComponents$co.ToastContainer; var add = this.add, remove = this.remove, removeAll = this.removeAll, update = this.update; var toasts = Object.freeze(this.state.toasts); var hasToasts = Boolean(toasts.length); var portalTarget = canUseDOM ? portalTargetSelector ? document.querySelector(portalTargetSelector) : document.body : null; // appease flow return _react2.default.createElement( Provider, { value: { add: add, remove: remove, removeAll: removeAll, update: update, toasts: toasts } }, children, portalTarget ? (0, _reactDom.createPortal)(_react2.default.createElement( ToastContainer, { placement: placement, hasToasts: hasToasts }, _react2.default.createElement( _reactTransitionGroup.TransitionGroup, { component: null }, toasts.map(function (_ref2) { var appearance = _ref2.appearance, autoDismiss = _ref2.autoDismiss, content = _ref2.content, id = _ref2.id, onDismiss = _ref2.onDismiss, unknownConsumerProps = _objectWithoutProperties(_ref2, ['appearance', 'autoDismiss', 'content', 'id', 'onDismiss']); return _react2.default.createElement( _reactTransitionGroup.Transition, { appear: true, key: id, mountOnEnter: true, timeout: transitionDuration, unmountOnExit: true }, function (transitionState) { return _react2.default.createElement( _ToastController.ToastController, _extends({ appearance: appearance, autoDismiss: autoDismiss !== undefined ? autoDismiss : inheritedAutoDismiss, autoDismissTimeout: autoDismissTimeout, component: Toast, key: id, onDismiss: _this2.onDismiss(id, onDismiss), placement: placement, transitionDuration: transitionDuration, transitionState: transitionState }, unknownConsumerProps), content ); } ); }) ) ), portalTarget) : _react2.default.createElement(ToastContainer, { placement: placement, hasToasts: hasToasts }) // keep ReactDOM.hydrate happy ); } }]); return ToastProvider; }(_react.Component); ToastProvider.defaultProps = { autoDismiss: false, autoDismissTimeout: 5000, components: defaultComponents, newestOnTop: false, placement: 'top-right', transitionDuration: 220 }; var ToastConsumer = exports.ToastConsumer = function ToastConsumer(_ref3) { var children = _ref3.children; return _react2.default.createElement( Consumer, null, function (context) { return children(context); } ); }; var withToastManager = exports.withToastManager = function withToastManager(Comp // $FlowFixMe `forwardRef` ) { return _react2.default.forwardRef(function (props, ref) { return _react2.default.createElement( ToastConsumer, null, function (context) { return _react2.default.createElement(Comp, _extends({ toastManager: context }, props, { ref: ref })); } ); }); }; var useToasts = exports.useToasts = function useToasts() { var ctx = (0, _react.useContext)(ToastContext); if (!ctx) { throw Error('The `useToasts` hook must be called from a descendent of the `ToastProvider`.'); } return { addToast: ctx.add, removeToast: ctx.remove, removeAllToasts: ctx.removeAll, updateToast: ctx.update, toastStack: ctx.toasts }; };