react-toast-notifications
Version:
A configurable, composable, toast notification system for react.
295 lines (236 loc) • 11.4 kB
JavaScript
;
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
};
};