diff --git a/public/js/notify-combined.js b/public/js/notify-combined.js
new file mode 100644
index 0000000..8d9405a
--- /dev/null
+++ b/public/js/notify-combined.js
@@ -0,0 +1,593 @@
+/** Notify.js - v0.3.1 - 2014/06/29
+ * http://notifyjs.com/
+ * Copyright (c) 2014 Jaime Pillora - MIT
+ */
+(function(window,document,$,undefined) {
+'use strict';
+
+var Notification, addStyle, blankFieldName, coreStyle, createElem, defaults, encode, find, findFields, getAnchorElement, getStyle, globalAnchors, hAligns, incr, inherit, insertCSS, mainPositions, opposites, parsePosition, pluginClassName, pluginName, pluginOptions, positions, realign, stylePrefixes, styles, vAligns,
+ __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; };
+
+pluginName = 'notify';
+
+pluginClassName = pluginName + 'js';
+
+blankFieldName = pluginName + "!blank";
+
+positions = {
+ t: 'top',
+ m: 'middle',
+ b: 'bottom',
+ l: 'left',
+ c: 'center',
+ r: 'right'
+};
+
+hAligns = ['l', 'c', 'r'];
+
+vAligns = ['t', 'm', 'b'];
+
+mainPositions = ['t', 'b', 'l', 'r'];
+
+opposites = {
+ t: 'b',
+ m: null,
+ b: 't',
+ l: 'r',
+ c: null,
+ r: 'l'
+};
+
+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;
+};
+
+styles = {};
+
+coreStyle = {
+ name: 'core',
+ html: "
",
+ 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}"
+};
+
+stylePrefixes = {
+ "border-radius": ["-webkit-", "-moz-"]
+};
+
+getStyle = function(name) {
+ return styles[name];
+};
+
+addStyle = function(name, def) {
+ var cssText, elem, fields, _ref;
+ if (!name) {
+ throw "Missing Style name";
+ }
+ if (!def) {
+ throw "Missing Style definition";
+ }
+ if (!def.html) {
+ throw "Missing Style HTML";
+ }
+ if ((_ref = styles[name]) != null ? _ref.cssElem : void 0) {
+ if (window.console) {
+ console.warn("" + pluginName + ": overwriting style '" + name + "'");
+ }
+ styles[name].cssElem.remove();
+ }
+ def.name = name;
+ styles[name] = def;
+ 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);
+ }
+ fields = {};
+ elem = $(def.html);
+ findFields('html', elem, fields);
+ findFields('text', elem, fields);
+ return def.fields = fields;
+};
+
+insertCSS = function(cssText) {
+ var elem;
+ elem = createElem("style");
+ elem.attr('type', 'text/css');
+ $("head").append(elem);
+ try {
+ elem.html(cssText);
+ } catch (e) {
+ elem[0].styleSheet.cssText = cssText;
+ }
+ return elem;
+};
+
+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;
+ }
+ return fields[name] = type;
+ });
+};
+
+find = function(elem, selector) {
+ if (elem.is(selector)) {
+ return elem;
+ } else {
+ return elem.find(selector);
+ }
+};
+
+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
+};
+
+inherit = function(a, b) {
+ var F;
+ F = function() {};
+ F.prototype = a;
+ return $.extend(true, new F(), b);
+};
+
+defaults = function(opts) {
+ return $.extend(pluginOptions, opts);
+};
+
+createElem = function(tag) {
+ return $("<" + tag + ">" + tag + ">");
+};
+
+globalAnchors = {};
+
+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;
+};
+
+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;
+};
+
+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";
+};
+
+encode = function(text) {
+ encode.e = encode.e || createElem("div");
+ return encode.e.text(text).html();
+};
+
+Notification = (function() {
+
+ 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);
+ return this.userFields = style.fields;
+ };
+
+ Notification.prototype.show = function(show, userCallback) {
+ var args, callback, elems, fn, hidden,
+ _this = this;
+ callback = function() {
+ if (!show && !_this.elem) {
+ _this.destroy();
+ }
+ if (userCallback) {
+ return userCallback();
+ }
+ };
+ 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 align, anchor, css, key, main, pAlign, pMain, _ref;
+ _ref = this.getPosition(), pMain = _ref[0], pAlign = _ref[1];
+ main = positions[pMain];
+ align = positions[pAlign];
+ key = pMain + "|" + pAlign;
+ anchor = globalAnchors[key];
+ if (!anchor) {
+ anchor = globalAnchors[key] = createElem("div");
+ 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, mainFull, margin, opp, oppFull, pAlign, pArrow, pMain, pos, posFull, position, wrapPos, _i, _j, _len, _len1, _ref;
+ 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 (_i = 0, _len = _ref.length; _i < _len; _i++) {
+ pos = _ref[_i];
+ 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("background-color") || 'white';
+ for (_j = 0, _len1 = mainPositions.length; _j < _len1; _j++) {
+ pos = mainPositions[_j];
+ 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, text, _ref, _ref1, _ref2, _ref3, _ref4, _ref5;
+ 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,
+ _this = this;
+ 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);
+ return this.autohideTimer = setTimeout(function() {
+ return _this.show(false);
+ }, this.options.autoHideDelay);
+ }
+ };
+
+ Notification.prototype.destroy = function() {
+ return this.wrapper.remove();
+ };
+
+ return Notification;
+
+})();
+
+$[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 inst;
+ inst = getAnchorElement($(this)).data(pluginClassName);
+ if (inst) {
+ return inst.run(data, options);
+ } else {
+ return new Notification($(this), data, options);
+ }
+ });
+ return this;
+};
+
+$.extend($[pluginName], {
+ defaults: defaults,
+ addStyle: addStyle,
+ pluginOptions: pluginOptions,
+ getStyle: getStyle,
+ insertCSS: insertCSS
+});
+
+$(function() {
+ insertCSS(coreStyle.css).attr('id', 'core-notify');
+ $(document).on('click', "." + pluginClassName + "-hidable", function(e) {
+ return $(this).trigger('notify-hide');
+ });
+ return $(document).on('notify-hide', "." + pluginClassName + "-wrapper", function(e) {
+ var _ref;
+ return (_ref = $(this).data(pluginClassName)) != null ? _ref.show(false) : void 0;
+ });
+});
+
+}(window,document,jQuery));
+
+$.notify.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()"
+ }
+ }
+});