You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

351 lines
12 KiB

  1. /*
  2. * Toastr
  3. * Copyright 2012-2014 John Papa and Hans Fjällemark.
  4. * All Rights Reserved.
  5. * Use, reproduction, distribution, and modification of this code is subject to the terms and
  6. * conditions of the MIT license, available at http://www.opensource.org/licenses/mit-license.php
  7. *
  8. * Author: John Papa and Hans Fjällemark
  9. * ARIA Support: Greta Krafsig
  10. * Project: https://github.com/CodeSeven/toastr
  11. */
  12. ; (function (define) {
  13. define(['jquery'], function ($) {
  14. return (function () {
  15. var $container;
  16. var listener;
  17. var toastId = 0;
  18. var toastType = {
  19. error: 'error',
  20. info: 'info',
  21. success: 'success',
  22. warning: 'warning'
  23. };
  24. var toastr = {
  25. clear: clear,
  26. remove: remove,
  27. error: error,
  28. getContainer: getContainer,
  29. info: info,
  30. options: {},
  31. subscribe: subscribe,
  32. success: success,
  33. version: '2.1.0',
  34. warning: warning
  35. };
  36. var previousToast;
  37. return toastr;
  38. //#region Accessible Methods
  39. function error(message, title, optionsOverride) {
  40. return notify({
  41. type: toastType.error,
  42. iconClass: getOptions().iconClasses.error,
  43. message: message,
  44. optionsOverride: optionsOverride,
  45. title: title
  46. });
  47. }
  48. function getContainer(options, create) {
  49. if (!options) { options = getOptions(); }
  50. $container = $('#' + options.containerId);
  51. if ($container.length) {
  52. return $container;
  53. }
  54. if(create) {
  55. $container = createContainer(options);
  56. }
  57. return $container;
  58. }
  59. function info(message, title, optionsOverride) {
  60. return notify({
  61. type: toastType.info,
  62. iconClass: getOptions().iconClasses.info,
  63. message: message,
  64. optionsOverride: optionsOverride,
  65. title: title
  66. });
  67. }
  68. function subscribe(callback) {
  69. listener = callback;
  70. }
  71. function success(message, title, optionsOverride) {
  72. return notify({
  73. type: toastType.success,
  74. iconClass: getOptions().iconClasses.success,
  75. message: message,
  76. optionsOverride: optionsOverride,
  77. title: title
  78. });
  79. }
  80. function warning(message, title, optionsOverride) {
  81. return notify({
  82. type: toastType.warning,
  83. iconClass: getOptions().iconClasses.warning,
  84. message: message,
  85. optionsOverride: optionsOverride,
  86. title: title
  87. });
  88. }
  89. function clear($toastElement) {
  90. var options = getOptions();
  91. if (!$container) { getContainer(options); }
  92. if (!clearToast($toastElement, options)) {
  93. clearContainer(options);
  94. }
  95. }
  96. function remove($toastElement) {
  97. var options = getOptions();
  98. if (!$container) { getContainer(options); }
  99. if ($toastElement && $(':focus', $toastElement).length === 0) {
  100. removeToast($toastElement);
  101. return;
  102. }
  103. if ($container.children().length) {
  104. $container.remove();
  105. }
  106. }
  107. //#endregion
  108. //#region Internal Methods
  109. function clearContainer(options){
  110. var toastsToClear = $container.children();
  111. for (var i = toastsToClear.length - 1; i >= 0; i--) {
  112. clearToast($(toastsToClear[i]), options);
  113. };
  114. }
  115. function clearToast($toastElement, options){
  116. if ($toastElement && $(':focus', $toastElement).length === 0) {
  117. $toastElement[options.hideMethod]({
  118. duration: options.hideDuration,
  119. easing: options.hideEasing,
  120. complete: function () { removeToast($toastElement); }
  121. });
  122. return true;
  123. }
  124. return false;
  125. }
  126. function createContainer(options) {
  127. $container = $('<div/>')
  128. .attr('id', options.containerId)
  129. .addClass(options.positionClass)
  130. .attr('aria-live', 'polite')
  131. .attr('role', 'alert');
  132. $container.appendTo($(options.target));
  133. return $container;
  134. }
  135. function getDefaults() {
  136. return {
  137. tapToDismiss: true,
  138. toastClass: 'toast',
  139. containerId: 'toast-container',
  140. debug: false,
  141. showMethod: 'fadeIn', //fadeIn, slideDown, and show are built into jQuery
  142. showDuration: 300,
  143. showEasing: 'swing', //swing and linear are built into jQuery
  144. onShown: undefined,
  145. hideMethod: 'fadeOut',
  146. hideDuration: 1000,
  147. hideEasing: 'swing',
  148. onHidden: undefined,
  149. extendedTimeOut: 1000,
  150. iconClasses: {
  151. error: 'toast-error',
  152. info: 'toast-info',
  153. success: 'toast-success',
  154. warning: 'toast-warning'
  155. },
  156. iconClass: 'toast-info',
  157. positionClass: 'toast-top-right',
  158. timeOut: 5000, // Set timeOut and extendedTimeOut to 0 to make it sticky
  159. titleClass: 'toast-title',
  160. messageClass: 'toast-message',
  161. target: 'body',
  162. closeHtml: '<button>&times;</button>',
  163. newestOnTop: true,
  164. preventDuplicates: false
  165. };
  166. }
  167. function publish(args) {
  168. if (!listener) { return; }
  169. listener(args);
  170. }
  171. function notify(map) {
  172. var options = getOptions(),
  173. iconClass = map.iconClass || options.iconClass;
  174. if(options.preventDuplicates){
  175. if(map.message === previousToast){
  176. return;
  177. }
  178. else{
  179. previousToast = map.message;
  180. }
  181. }
  182. if (typeof (map.optionsOverride) !== 'undefined') {
  183. options = $.extend(options, map.optionsOverride);
  184. iconClass = map.optionsOverride.iconClass || iconClass;
  185. }
  186. toastId++;
  187. $container = getContainer(options, true);
  188. var intervalId = null,
  189. $toastElement = $('<div/>'),
  190. $titleElement = $('<div/>'),
  191. $messageElement = $('<div/>'),
  192. $closeElement = $(options.closeHtml),
  193. response = {
  194. toastId: toastId,
  195. state: 'visible',
  196. startTime: new Date(),
  197. options: options,
  198. map: map
  199. };
  200. if (map.iconClass) {
  201. $toastElement.addClass(options.toastClass).addClass(iconClass);
  202. }
  203. if (map.title) {
  204. $titleElement.append(map.title).addClass(options.titleClass);
  205. $toastElement.append($titleElement);
  206. }
  207. if (map.message) {
  208. $messageElement.append(map.message).addClass(options.messageClass);
  209. $toastElement.append($messageElement);
  210. }
  211. if (options.closeButton) {
  212. $closeElement.addClass('toast-close-button').attr("role", "button");
  213. $toastElement.prepend($closeElement);
  214. }
  215. $toastElement.hide();
  216. if (options.newestOnTop) {
  217. $container.prepend($toastElement);
  218. } else {
  219. $container.append($toastElement);
  220. }
  221. $toastElement[options.showMethod](
  222. { duration: options.showDuration, easing: options.showEasing, complete: options.onShown }
  223. );
  224. if (options.timeOut > 0) {
  225. intervalId = setTimeout(hideToast, options.timeOut);
  226. }
  227. $toastElement.hover(stickAround, delayedHideToast);
  228. if (!options.onclick && options.tapToDismiss) {
  229. $toastElement.click(hideToast);
  230. }
  231. if (options.closeButton && $closeElement) {
  232. $closeElement.click(function (event) {
  233. if( event.stopPropagation ) {
  234. event.stopPropagation();
  235. } else if( event.cancelBubble !== undefined && event.cancelBubble !== true ) {
  236. event.cancelBubble = true;
  237. }
  238. hideToast(true);
  239. });
  240. }
  241. if (options.onclick) {
  242. $toastElement.click(function () {
  243. options.onclick();
  244. hideToast();
  245. });
  246. }
  247. publish(response);
  248. if (options.debug && console) {
  249. console.log(response);
  250. }
  251. return $toastElement;
  252. function hideToast(override) {
  253. if ($(':focus', $toastElement).length && !override) {
  254. return;
  255. }
  256. return $toastElement[options.hideMethod]({
  257. duration: options.hideDuration,
  258. easing: options.hideEasing,
  259. complete: function () {
  260. removeToast($toastElement);
  261. if (options.onHidden && response.state !== 'hidden') {
  262. options.onHidden();
  263. }
  264. response.state = 'hidden';
  265. response.endTime = new Date();
  266. publish(response);
  267. }
  268. });
  269. }
  270. function delayedHideToast() {
  271. if (options.timeOut > 0 || options.extendedTimeOut > 0) {
  272. intervalId = setTimeout(hideToast, options.extendedTimeOut);
  273. }
  274. }
  275. function stickAround() {
  276. clearTimeout(intervalId);
  277. $toastElement.stop(true, true)[options.showMethod](
  278. { duration: options.showDuration, easing: options.showEasing }
  279. );
  280. }
  281. }
  282. function getOptions() {
  283. return $.extend({}, getDefaults(), toastr.options);
  284. }
  285. function removeToast($toastElement) {
  286. if (!$container) { $container = getContainer(); }
  287. if ($toastElement.is(':visible')) {
  288. return;
  289. }
  290. $toastElement.remove();
  291. $toastElement = null;
  292. if ($container.children().length === 0) {
  293. $container.remove();
  294. }
  295. }
  296. //#endregion
  297. })();
  298. });
  299. }(typeof define === 'function' && define.amd ? define : function (deps, factory) {
  300. if (typeof module !== 'undefined' && module.exports) { //Node
  301. module.exports = factory(require('jquery'));
  302. } else {
  303. window['toastr'] = factory(window['jQuery']);
  304. }
  305. }));