/* Notify.js - http://notifyjs.com/ Copyright (c) 2015 MIT */ (function (factory) { // UMD start // https://github.com/umdjs/umd/blob/master/jqueryPluginCommonjs.js if (typeof define === 'function' && define.amd) { // AMD. Register as an anonymous module. define(['jquery'], factory); } else if (typeof module === 'object' && module.exports) { // Node/CommonJS module.exports = function( root, jQuery ) { if ( jQuery === undefined ) { // require('jQuery') returns a factory that requires window to // build a jQuery instance, we normalize how we use modules // that require this pattern but the window provided is a noop // if it's defined (how jquery works) if ( typeof window !== 'undefined' ) { jQuery = require('jquery'); } else { jQuery = require('jquery')(root); } } factory(jQuery); return jQuery; }; } else { // Browser globals factory(jQuery); } }(function ($) { //IE8 indexOf polyfill var indexOf = [].indexOf || function(item) { for (var i = 0, l = this.length; i < l; i++) { if (i in this && this[i] === item) { return i; } } return -1; }; var pluginName = "notify"; var pluginClassName = pluginName + "js"; var blankFieldName = pluginName + "!blank"; var positions = { t: "top", m: "middle", b: "bottom", l: "left", c: "center", r: "right" }; var hAligns = ["l", "c", "r"]; var vAligns = ["t", "m", "b"]; var mainPositions = ["t", "b", "l", "r"]; var opposites = { t: "b", m: null, b: "t", l: "r", c: null, r: "l" }; var parsePosition = function(str) { var pos; pos = []; $.each(str.split(/\W+/), function(i, word) { var w; w = word.toLowerCase().charAt(0); if (positions[w]) { return pos.push(w); } }); return pos; }; var styles = {}; var coreStyle = { name: "core", html: "
\n
\n
\n
", css: "." + pluginClassName + "-corner {\n position: fixed;\n margin: 5px;\n z-index: 1050;\n}\n\n." + pluginClassName + "-corner ." + pluginClassName + "-wrapper,\n." + pluginClassName + "-corner ." + pluginClassName + "-container {\n position: relative;\n display: block;\n height: inherit;\n width: inherit;\n margin: 3px;\n}\n\n." + pluginClassName + "-wrapper {\n z-index: 1;\n position: absolute;\n display: inline-block;\n height: 0;\n width: 0;\n}\n\n." + pluginClassName + "-container {\n display: none;\n z-index: 1;\n position: absolute;\n}\n\n." + pluginClassName + "-hidable {\n cursor: pointer;\n}\n\n[data-notify-text],[data-notify-html] {\n position: relative;\n}\n\n." + pluginClassName + "-arrow {\n position: absolute;\n z-index: 2;\n width: 0;\n height: 0;\n}" }; var stylePrefixes = { "border-radius": ["-webkit-", "-moz-"] }; var getStyle = function(name) { return styles[name]; }; var removeStyle = function(name) { if (!name) { throw "Missing Style name"; } if (styles[name]) { delete styles[name]; } }; var addStyle = function(name, def) { if (!name) { throw "Missing Style name"; } if (!def) { throw "Missing Style definition"; } if (!def.html) { throw "Missing Style HTML"; } //remove existing style var existing = styles[name]; if (existing && existing.cssElem) { if (window.console) { console.warn(pluginName + ": overwriting style '" + name + "'"); } styles[name].cssElem.remove(); } def.name = name; styles[name] = def; var cssText = ""; if (def.classes) { $.each(def.classes, function(className, props) { cssText += "." + pluginClassName + "-" + def.name + "-" + className + " {\n"; $.each(props, function(name, val) { if (stylePrefixes[name]) { $.each(stylePrefixes[name], function(i, prefix) { return cssText += " " + prefix + name + ": " + val + ";\n"; }); } return cssText += " " + name + ": " + val + ";\n"; }); return cssText += "}\n"; }); } if (def.css) { cssText += "/* styles for " + def.name + " */\n" + def.css; } if (cssText) { def.cssElem = insertCSS(cssText); def.cssElem.attr("id", "notify-" + def.name); } var fields = {}; var elem = $(def.html); findFields("html", elem, fields); findFields("text", elem, fields); def.fields = fields; }; var insertCSS = function(cssText) { var e, elem, error; elem = createElem("style"); elem.attr("type", 'text/css'); $("head").append(elem); try { elem.html(cssText); } catch (_) { elem[0].styleSheet.cssText = cssText; } return elem; }; var findFields = function(type, elem, fields) { var attr; if (type !== "html") { type = "text"; } attr = "data-notify-" + type; return find(elem, "[" + attr + "]").each(function() { var name; name = $(this).attr(attr); if (!name) { name = blankFieldName; } fields[name] = type; }); }; var find = function(elem, selector) { if (elem.is(selector)) { return elem; } else { return elem.find(selector); } }; var pluginOptions = { clickToHide: true, autoHide: true, autoHideDelay: 5000, arrowShow: true, arrowSize: 5, breakNewLines: true, elementPosition: "bottom", globalPosition: "top right", style: "bootstrap", className: "error", showAnimation: "slideDown", showDuration: 400, hideAnimation: "slideUp", hideDuration: 200, gap: 5 }; var inherit = function(a, b) { var F; F = function() {}; F.prototype = a; return $.extend(true, new F(), b); }; var defaults = function(opts) { return $.extend(pluginOptions, opts); }; var createElem = function(tag) { return $("<" + tag + ">"); }; var globalAnchors = {}; var getAnchorElement = function(element) { var radios; if (element.is('[type=radio]')) { radios = element.parents('form:first').find('[type=radio]').filter(function(i, e) { return $(e).attr("name") === element.attr("name"); }); element = radios.first(); } return element; }; var incr = function(obj, pos, val) { var opp, temp; if (typeof val === "string") { val = parseInt(val, 10); } else if (typeof val !== "number") { return; } if (isNaN(val)) { return; } opp = positions[opposites[pos.charAt(0)]]; temp = pos; if (obj[opp] !== undefined) { pos = positions[opp.charAt(0)]; val = -val; } if (obj[pos] === undefined) { obj[pos] = val; } else { obj[pos] += val; } return null; }; var realign = function(alignment, inner, outer) { if (alignment === "l" || alignment === "t") { return 0; } else if (alignment === "c" || alignment === "m") { return outer / 2 - inner / 2; } else if (alignment === "r" || alignment === "b") { return outer - inner; } throw "Invalid alignment"; }; var encode = function(text) { encode.e = encode.e || createElem("div"); return encode.e.text(text).html(); }; function Notification(elem, data, options) { if (typeof options === "string") { options = { className: options }; } this.options = inherit(pluginOptions, $.isPlainObject(options) ? options : {}); this.loadHTML(); this.wrapper = $(coreStyle.html); if (this.options.clickToHide) { this.wrapper.addClass(pluginClassName + "-hidable"); } this.wrapper.data(pluginClassName, this); this.arrow = this.wrapper.find("." + pluginClassName + "-arrow"); this.container = this.wrapper.find("." + pluginClassName + "-container"); this.container.append(this.userContainer); if (elem && elem.length) { this.elementType = elem.attr("type"); this.originalElement = elem; this.elem = getAnchorElement(elem); this.elem.data(pluginClassName, this); this.elem.before(this.wrapper); } this.container.hide(); this.run(data); } Notification.prototype.loadHTML = function() { var style; style = this.getStyle(); this.userContainer = $(style.html); this.userFields = style.fields; }; Notification.prototype.show = function(show, userCallback) { var args, callback, elems, fn, hidden; callback = (function(_this) { return function() { if (!show && !_this.elem) { _this.destroy(); } if (userCallback) { return userCallback(); } }; })(this); hidden = this.container.parent().parents(':hidden').length > 0; elems = this.container.add(this.arrow); args = []; if (hidden && show) { fn = "show"; } else if (hidden && !show) { fn = "hide"; } else if (!hidden && show) { fn = this.options.showAnimation; args.push(this.options.showDuration); } else if (!hidden && !show) { fn = this.options.hideAnimation; args.push(this.options.hideDuration); } else { return callback(); } args.push(callback); return elems[fn].apply(elems, args); }; Notification.prototype.setGlobalPosition = function() { var p = this.getPosition(); var pMain = p[0]; var pAlign = p[1]; var main = positions[pMain]; var align = positions[pAlign]; var key = pMain + "|" + pAlign; var anchor = globalAnchors[key]; if (!anchor || !document.body.contains(anchor[0])) { anchor = globalAnchors[key] = createElem("div"); var css = {}; css[main] = 0; if (align === "middle") { css.top = '45%'; } else if (align === "center") { css.left = '45%'; } else { css[align] = 0; } anchor.css(css).addClass(pluginClassName + "-corner"); $("body").append(anchor); } return anchor.prepend(this.wrapper); }; Notification.prototype.setElementPosition = function() { var arrowColor, arrowCss, arrowSize, color, contH, contW, css, elemH, elemIH, elemIW, elemPos, elemW, gap, j, k, len, len1, mainFull, margin, opp, oppFull, pAlign, pArrow, pMain, pos, posFull, position, ref, wrapPos; position = this.getPosition(); pMain = position[0]; pAlign = position[1]; pArrow = position[2]; elemPos = this.elem.position(); elemH = this.elem.outerHeight(); elemW = this.elem.outerWidth(); elemIH = this.elem.innerHeight(); elemIW = this.elem.innerWidth(); wrapPos = this.wrapper.position(); contH = this.container.height(); contW = this.container.width(); mainFull = positions[pMain]; opp = opposites[pMain]; oppFull = positions[opp]; css = {}; css[oppFull] = pMain === "b" ? elemH : pMain === "r" ? elemW : 0; incr(css, "top", elemPos.top - wrapPos.top); incr(css, "left", elemPos.left - wrapPos.left); ref = ["top", "left"]; for (j = 0, len = ref.length; j < len; j++) { pos = ref[j]; margin = parseInt(this.elem.css("margin-" + pos), 10); if (margin) { incr(css, pos, margin); } } gap = Math.max(0, this.options.gap - (this.options.arrowShow ? arrowSize : 0)); incr(css, oppFull, gap); if (!this.options.arrowShow) { this.arrow.hide(); } else { arrowSize = this.options.arrowSize; arrowCss = $.extend({}, css); arrowColor = this.userContainer.css("border-color") || this.userContainer.css("border-top-color") || this.userContainer.css("background-color") || "white"; for (k = 0, len1 = mainPositions.length; k < len1; k++) { pos = mainPositions[k]; posFull = positions[pos]; if (pos === opp) { continue; } color = posFull === mainFull ? arrowColor : "transparent"; arrowCss["border-" + posFull] = arrowSize + "px solid " + color; } incr(css, positions[opp], arrowSize); if (indexOf.call(mainPositions, pAlign) >= 0) { incr(arrowCss, positions[pAlign], arrowSize * 2); } } if (indexOf.call(vAligns, pMain) >= 0) { incr(css, "left", realign(pAlign, contW, elemW)); if (arrowCss) { incr(arrowCss, "left", realign(pAlign, arrowSize, elemIW)); } } else if (indexOf.call(hAligns, pMain) >= 0) { incr(css, "top", realign(pAlign, contH, elemH)); if (arrowCss) { incr(arrowCss, "top", realign(pAlign, arrowSize, elemIH)); } } if (this.container.is(":visible")) { css.display = "block"; } this.container.removeAttr("style").css(css); if (arrowCss) { return this.arrow.removeAttr("style").css(arrowCss); } }; Notification.prototype.getPosition = function() { var pos, ref, ref1, ref2, ref3, ref4, ref5, text; text = this.options.position || (this.elem ? this.options.elementPosition : this.options.globalPosition); pos = parsePosition(text); if (pos.length === 0) { pos[0] = "b"; } if (ref = pos[0], indexOf.call(mainPositions, ref) < 0) { throw "Must be one of [" + mainPositions + "]"; } if (pos.length === 1 || ((ref1 = pos[0], indexOf.call(vAligns, ref1) >= 0) && (ref2 = pos[1], indexOf.call(hAligns, ref2) < 0)) || ((ref3 = pos[0], indexOf.call(hAligns, ref3) >= 0) && (ref4 = pos[1], indexOf.call(vAligns, ref4) < 0))) { pos[1] = (ref5 = pos[0], indexOf.call(hAligns, ref5) >= 0) ? "m" : "l"; } if (pos.length === 2) { pos[2] = pos[1]; } return pos; }; Notification.prototype.getStyle = function(name) { var style; if (!name) { name = this.options.style; } if (!name) { name = "default"; } style = styles[name]; if (!style) { throw "Missing style: " + name; } return style; }; Notification.prototype.updateClasses = function() { var classes, style; classes = ["base"]; if ($.isArray(this.options.className)) { classes = classes.concat(this.options.className); } else if (this.options.className) { classes.push(this.options.className); } style = this.getStyle(); classes = $.map(classes, function(n) { return pluginClassName + "-" + style.name + "-" + n; }).join(" "); return this.userContainer.attr("class", classes); }; Notification.prototype.run = function(data, options) { var d, datas, name, type, value; if ($.isPlainObject(options)) { $.extend(this.options, options); } else if ($.type(options) === "string") { this.options.className = options; } if (this.container && !data) { this.show(false); return; } else if (!this.container && !data) { return; } datas = {}; if ($.isPlainObject(data)) { datas = data; } else { datas[blankFieldName] = data; } for (name in datas) { d = datas[name]; type = this.userFields[name]; if (!type) { continue; } if (type === "text") { d = encode(d); if (this.options.breakNewLines) { d = d.replace(/\n/g, '
'); } } value = name === blankFieldName ? '' : '=' + name; find(this.userContainer, "[data-notify-" + type + value + "]").html(d); } this.updateClasses(); if (this.elem) { this.setElementPosition(); } else { this.setGlobalPosition(); } this.show(true); if (this.options.autoHide) { clearTimeout(this.autohideTimer); this.autohideTimer = setTimeout(this.show.bind(this, false), this.options.autoHideDelay); } }; Notification.prototype.destroy = function() { this.wrapper.data(pluginClassName, null); this.wrapper.remove(); }; $[pluginName] = function(elem, data, options) { if ((elem && elem.nodeName) || elem.jquery) { $(elem)[pluginName](data, options); } else { options = data; data = elem; new Notification(null, data, options); } return elem; }; $.fn[pluginName] = function(data, options) { $(this).each(function() { var prev = getAnchorElement($(this)).data(pluginClassName); if (prev) { prev.destroy(); } var curr = new Notification($(this), data, options); }); return this; }; $.extend($[pluginName], { defaults: defaults, addStyle: addStyle, removeStyle: removeStyle, pluginOptions: pluginOptions, getStyle: getStyle, insertCSS: insertCSS }); //always include the default bootstrap style addStyle("bootstrap", { html: "
\n\n
", classes: { base: { "font-weight": "bold", "padding": "8px 15px 8px 14px", "text-shadow": "0 1px 0 rgba(255, 255, 255, 0.5)", "background-color": "#fcf8e3", "border": "1px solid #fbeed5", "border-radius": "4px", "white-space": "nowrap", "padding-left": "25px", "background-repeat": "no-repeat", "background-position": "3px 7px" }, error: { "color": "#B94A48", "background-color": "#F2DEDE", "border-color": "#EED3D7", "background-image": "url()" }, success: { "color": "#468847", "background-color": "#DFF0D8", "border-color": "#D6E9C6", "background-image": "url()" }, info: { "color": "#3A87AD", "background-color": "#D9EDF7", "border-color": "#BCE8F1", "background-image": "url()" }, warn: { "color": "#C09853", "background-color": "#FCF8E3", "border-color": "#FBEED5", "background-image": "url()" } } }); $(function() { insertCSS(coreStyle.css).attr("id", "core-notify"); $(document).on("click", "." + pluginClassName + "-hidable", function(e) { $(this).trigger("notify-hide"); }); $(document).on("notify-hide", "." + pluginClassName + "-wrapper", function(e) { var elem = $(this).data(pluginClassName); if(elem) { elem.show(false); } }); }); }));