jasonyeo avatar jasonyeo committed 8eeedd2 Draft

now working with add to cart

Comments (0)

Files changed (9)

static/js/collections/cart.js

+
+var app = app || {};
+
+(function() {
+    'use strict';
+
+    var ShoppingCart = Backbone.Collection.extend({
+        model: app.Product,
+        //url: '/shoppingcart'
+    });
+
+    app.Cart = new ShoppingCart();
+}());

static/js/lib/jquery/noty/default.js

+;(function($) {
+
+	$.noty.themes.defaultTheme = {
+		name: 'defaultTheme',
+		helpers: {
+			borderFix: function() {
+				if (this.options.dismissQueue) {
+					var selector = this.options.layout.container.selector + ' ' + this.options.layout.parent.selector;
+					switch (this.options.layout.name) {
+						case 'top':
+							$(selector).css({borderRadius: '0px 0px 0px 0px'});
+							$(selector).last().css({borderRadius: '0px 0px 5px 5px'}); break;
+						case 'topCenter': case 'topLeft': case 'topRight':
+						case 'bottomCenter': case 'bottomLeft': case 'bottomRight':
+						case 'center': case 'centerLeft': case 'centerRight': case 'inline':
+							$(selector).css({borderRadius: '0px 0px 0px 0px'});
+							$(selector).first().css({'border-top-left-radius': '5px', 'border-top-right-radius': '5px'});
+							$(selector).last().css({'border-bottom-left-radius': '5px', 'border-bottom-right-radius': '5px'}); break;
+						case 'bottom':
+							$(selector).css({borderRadius: '0px 0px 0px 0px'});
+							$(selector).first().css({borderRadius: '5px 5px 0px 0px'}); break;
+						default: break;
+					}
+				}
+			}
+		},
+		modal: {
+			css: {
+				position: 'fixed',
+				width: '100%',
+				height: '100%',
+				backgroundColor: '#000',
+				zIndex: 10000,
+				opacity: 0.6,
+				display: 'none',
+				left: 0,
+				top: 0
+			}
+		},
+		style: function() {
+
+			this.$bar.css({
+				overflow: 'hidden',
+				background: "url('') repeat-x scroll left top #fff"
+			});
+
+			this.$message.css({
+				fontSize: '13px',
+				lineHeight: '16px',
+				textAlign: 'center',
+				padding: '8px 10px 9px',
+				width: 'auto',
+				position: 'relative'
+			});
+
+			this.$closeButton.css({
+				position: 'absolute',
+				top: 4, right: 4,
+				width: 10, height: 10,
+				background: "url()",
+				display: 'none',
+				cursor: 'pointer'
+			});
+
+			this.$buttons.css({
+				padding: 5,
+				textAlign: 'right',
+				borderTop: '1px solid #ccc',
+				backgroundColor: '#fff'
+			});
+
+			this.$buttons.find('button').css({
+				marginLeft: 5
+			});
+
+			this.$buttons.find('button:first').css({
+				marginLeft: 0
+			});
+
+			this.$bar.bind({
+				mouseenter: function() { $(this).find('.noty_close').fadeIn(); },
+				mouseleave: function() { $(this).find('.noty_close').fadeOut(); }
+			});
+
+			switch (this.options.layout.name) {
+				case 'top':
+					this.$bar.css({
+						borderRadius: '0px 0px 5px 5px',
+						borderBottom: '2px solid #eee',
+						borderLeft: '2px solid #eee',
+						borderRight: '2px solid #eee',
+						boxShadow: "0 2px 4px rgba(0, 0, 0, 0.1)"
+					});
+				break;
+				case 'topCenter': case 'center': case 'bottomCenter': case 'inline':
+					this.$bar.css({
+						borderRadius: '5px',
+						border: '1px solid #eee',
+						boxShadow: "0 2px 4px rgba(0, 0, 0, 0.1)"
+					});
+					this.$message.css({fontSize: '13px', textAlign: 'center'});
+				break;
+				case 'topLeft': case 'topRight':
+				case 'bottomLeft': case 'bottomRight':
+				case 'centerLeft': case 'centerRight':
+					this.$bar.css({
+						borderRadius: '5px',
+						border: '1px solid #eee',
+						boxShadow: "0 2px 4px rgba(0, 0, 0, 0.1)"
+					});
+					this.$message.css({fontSize: '13px', textAlign: 'left'});
+				break;
+				case 'bottom':
+					this.$bar.css({
+						borderRadius: '5px 5px 0px 0px',
+						borderTop: '2px solid #eee',
+						borderLeft: '2px solid #eee',
+						borderRight: '2px solid #eee',
+						boxShadow: "0 -2px 4px rgba(0, 0, 0, 0.1)"
+					});
+				break;
+				default:
+					this.$bar.css({
+						border: '2px solid #eee',
+						boxShadow: "0 2px 4px rgba(0, 0, 0, 0.1)"
+					});
+				break;
+			}
+
+			switch (this.options.type) {
+				case 'alert': case 'notification':
+					this.$bar.css({backgroundColor: '#FFF', borderColor: '#CCC', color: '#444'}); break;
+				case 'warning':
+					this.$bar.css({backgroundColor: '#FFEAA8', borderColor: '#FFC237', color: '#826200'});
+					this.$buttons.css({borderTop: '1px solid #FFC237'}); break;
+				case 'error':
+					this.$bar.css({backgroundColor: 'red', borderColor: 'darkred', color: '#FFF'});
+					this.$message.css({fontWeight: 'bold'});
+					this.$buttons.css({borderTop: '1px solid darkred'}); break;
+				case 'information':
+					this.$bar.css({backgroundColor: '#57B7E2', borderColor: '#0B90C4', color: '#FFF'});
+					this.$buttons.css({borderTop: '1px solid #0B90C4'}); break;
+				case 'success':
+					this.$bar.css({backgroundColor: 'lightgreen', borderColor: '#50C24E', color: 'darkgreen'});
+					this.$buttons.css({borderTop: '1px solid #50C24E'});break;
+				default:
+					this.$bar.css({backgroundColor: '#FFF', borderColor: '#CCC', color: '#444'}); break;
+			}
+		},
+		callback: {
+			onShow: function() { $.noty.themes.defaultTheme.helpers.borderFix.apply(this); },
+			onClose: function() { $.noty.themes.defaultTheme.helpers.borderFix.apply(this); }
+		}
+	};
+
+})(jQuery);

static/js/lib/jquery/noty/jquery.noty.js

+/**
+ * noty - jQuery Notification Plugin v2.0.3
+ * Contributors: https://github.com/needim/noty/graphs/contributors
+ *
+ * Examples and Documentation - http://needim.github.com/noty/
+ *
+ * Licensed under the MIT licenses:
+ * http://www.opensource.org/licenses/mit-license.php
+ *
+ **/
+
+if (typeof Object.create !== 'function') {
+    Object.create = function (o) {
+        function F() {
+        }
+
+        F.prototype = o;
+        return new F();
+    };
+}
+
+(function ($) {
+
+    var NotyObject = {
+
+        init:function (options) {
+
+            // Mix in the passed in options with the default options
+            this.options = $.extend({}, $.noty.defaults, options);
+
+            this.options.layout = (this.options.custom) ? $.noty.layouts['inline'] : $.noty.layouts[this.options.layout];
+            this.options.theme = $.noty.themes[this.options.theme];
+
+            delete options.layout;
+            delete options.theme;
+
+            this.options = $.extend({}, this.options, this.options.layout.options);
+            this.options.id = 'noty_' + (new Date().getTime() * Math.floor(Math.random() * 1000000));
+
+            this.options = $.extend({}, this.options, options);
+
+            // Build the noty dom initial structure
+            this._build();
+
+            // return this so we can chain/use the bridge with less code.
+            return this;
+        }, // end init
+
+        _build:function () {
+
+            // Generating noty bar
+            var $bar = $('<div class="noty_bar"></div>').attr('id', this.options.id);
+            $bar.append(this.options.template).find('.noty_text').html(this.options.text);
+
+            this.$bar = (this.options.layout.parent.object !== null) ? $(this.options.layout.parent.object).css(this.options.layout.parent.css).append($bar) : $bar;
+
+            // Set buttons if available
+            if (this.options.buttons) {
+
+                // If we have button disable closeWith & timeout options
+                this.options.closeWith = [];
+                this.options.timeout = false;
+
+                var $buttons = $('<div/>').addClass('noty_buttons');
+
+                (this.options.layout.parent.object !== null) ? this.$bar.find('.noty_bar').append($buttons) : this.$bar.append($buttons);
+
+                var self = this;
+
+                $.each(this.options.buttons, function (i, button) {
+                    var $button = $('<button/>').addClass((button.addClass) ? button.addClass : 'gray').html(button.text)
+                        .appendTo(self.$bar.find('.noty_buttons'))
+                        .bind('click', function () {
+                            if ($.isFunction(button.onClick)) {
+                                button.onClick.call($button, self);
+                            }
+                        });
+                });
+            }
+
+            // For easy access
+            this.$message = this.$bar.find('.noty_message');
+            this.$closeButton = this.$bar.find('.noty_close');
+            this.$buttons = this.$bar.find('.noty_buttons');
+
+            $.noty.store[this.options.id] = this; // store noty for api
+
+        }, // end _build
+
+        show:function () {
+
+            var self = this;
+
+            $(self.options.layout.container.selector).append(self.$bar);
+
+            self.options.theme.style.apply(self);
+
+            ($.type(self.options.layout.css) === 'function') ? this.options.layout.css.apply(self.$bar) : self.$bar.css(this.options.layout.css || {});
+
+            self.$bar.addClass(self.options.layout.addClass);
+
+            self.options.layout.container.style.apply($(self.options.layout.container.selector));
+
+            self.options.theme.callback.onShow.apply(this);
+
+            if ($.inArray('click', self.options.closeWith) > -1)
+                self.$bar.css('cursor', 'pointer').one('click', function () {
+                    self.close();
+                });
+
+            if ($.inArray('hover', self.options.closeWith) > -1)
+                self.$bar.one('mouseenter', function () {
+                    self.close();
+                });
+
+            if ($.inArray('button', self.options.closeWith) > -1)
+                self.$closeButton.one('click', function () {
+                    self.close();
+                });
+
+            if ($.inArray('button', self.options.closeWith) == -1)
+                self.$closeButton.remove();
+
+            if (self.options.callback.onShow)
+                self.options.callback.onShow.apply(self);
+
+            self.$bar.animate(
+                self.options.animation.open,
+                self.options.animation.speed,
+                self.options.animation.easing,
+                function () {
+                    if (self.options.callback.afterShow) self.options.callback.afterShow.apply(self);
+                    self.shown = true;
+                });
+
+            // If noty is have a timeout option
+            if (self.options.timeout)
+                self.$bar.delay(self.options.timeout).promise().done(function () {
+                    self.close();
+                });
+
+            return this;
+
+        }, // end show
+
+        close:function () {
+
+            if (this.closed) return;
+
+            var self = this;
+
+            if (!this.shown) { // If we are still waiting in the queue just delete from queue
+                var queue = [];
+                $.each($.noty.queue, function (i, n) {
+                    if (n.options.id != self.options.id) {
+                        queue.push(n);
+                    }
+                });
+                $.noty.queue = queue;
+                return;
+            }
+
+            self.$bar.addClass('i-am-closing-now');
+
+            if (self.options.callback.onClose) {
+                self.options.callback.onClose.apply(self);
+            }
+
+            self.$bar.clearQueue().stop().animate(
+                self.options.animation.close,
+                self.options.animation.speed,
+                self.options.animation.easing,
+                function () {
+                    if (self.options.callback.afterClose) self.options.callback.afterClose.apply(self);
+                })
+                .promise().done(function () {
+
+                    // Modal Cleaning
+                    if (self.options.modal) {
+                        $.notyRenderer.setModalCount(-1);
+                        if ($.notyRenderer.getModalCount() == 0) $('.noty_modal').fadeOut('fast', function () {
+                            $(this).remove();
+                        });
+                    }
+
+                    // Layout Cleaning
+                    $.notyRenderer.setLayoutCountFor(self, -1);
+                    if ($.notyRenderer.getLayoutCountFor(self) == 0) $(self.options.layout.container.selector).remove();
+
+                    // Make sure self.$bar has not been removed before attempting to remove it
+                    if (typeof self.$bar !== 'undefined' && self.$bar !== null ) {
+                        self.$bar.remove();
+                        self.$bar = null;
+                        self.closed = true;
+                    }
+
+                    delete $.noty.store[self.options.id]; // deleting noty from store
+
+                    self.options.theme.callback.onClose.apply(self);
+
+                    if (!self.options.dismissQueue) {
+                        // Queue render
+                        $.noty.ontap = true;
+
+                        $.notyRenderer.render();
+                    }
+
+                });
+
+        }, // end close
+
+        setText:function (text) {
+            if (!this.closed) {
+                this.options.text = text;
+                this.$bar.find('.noty_text').html(text);
+            }
+            return this;
+        },
+
+        setType:function (type) {
+            if (!this.closed) {
+                this.options.type = type;
+                this.options.theme.style.apply(this);
+                this.options.theme.callback.onShow.apply(this);
+            }
+            return this;
+        },
+
+        setTimeout:function (time) {
+            if (!this.closed) {
+                var self = this;
+                this.options.timeout = time;
+                self.$bar.delay(self.options.timeout).promise().done(function () {
+                    self.close();
+                });
+            }
+            return this;
+        },
+
+        closed:false,
+        shown:false
+
+    }; // end NotyObject
+
+    $.notyRenderer = {};
+
+    $.notyRenderer.init = function (options) {
+
+        // Renderer creates a new noty
+        var notification = Object.create(NotyObject).init(options);
+
+        (notification.options.force) ? $.noty.queue.unshift(notification) : $.noty.queue.push(notification);
+
+        $.notyRenderer.render();
+
+        return ($.noty.returns == 'object') ? notification : notification.options.id;
+    };
+
+    $.notyRenderer.render = function () {
+
+        var instance = $.noty.queue[0];
+
+        if ($.type(instance) === 'object') {
+            if (instance.options.dismissQueue) {
+                $.notyRenderer.show($.noty.queue.shift());
+            } else {
+                if ($.noty.ontap) {
+                    $.notyRenderer.show($.noty.queue.shift());
+                    $.noty.ontap = false;
+                }
+            }
+        } else {
+            $.noty.ontap = true; // Queue is over
+        }
+
+    };
+
+    $.notyRenderer.show = function (notification) {
+
+        if (notification.options.modal) {
+            $.notyRenderer.createModalFor(notification);
+            $.notyRenderer.setModalCount(+1);
+        }
+
+        // Where is the container?
+        if ($(notification.options.layout.container.selector).length == 0) {
+            if (notification.options.custom) {
+                notification.options.custom.append($(notification.options.layout.container.object).addClass('i-am-new'));
+            } else {
+                $('body').append($(notification.options.layout.container.object).addClass('i-am-new'));
+            }
+        } else {
+            $(notification.options.layout.container.selector).removeClass('i-am-new');
+        }
+
+        $.notyRenderer.setLayoutCountFor(notification, +1);
+
+        notification.show();
+    };
+
+    $.notyRenderer.createModalFor = function (notification) {
+        if ($('.noty_modal').length == 0)
+            $('<div/>').addClass('noty_modal').data('noty_modal_count', 0).css(notification.options.theme.modal.css).prependTo($('body')).fadeIn('fast');
+    };
+
+    $.notyRenderer.getLayoutCountFor = function (notification) {
+        return $(notification.options.layout.container.selector).data('noty_layout_count') || 0;
+    };
+
+    $.notyRenderer.setLayoutCountFor = function (notification, arg) {
+        return $(notification.options.layout.container.selector).data('noty_layout_count', $.notyRenderer.getLayoutCountFor(notification) + arg);
+    };
+
+    $.notyRenderer.getModalCount = function () {
+        return $('.noty_modal').data('noty_modal_count') || 0;
+    };
+
+    $.notyRenderer.setModalCount = function (arg) {
+        return $('.noty_modal').data('noty_modal_count', $.notyRenderer.getModalCount() + arg);
+    };
+
+    // This is for custom container
+    $.fn.noty = function (options) {
+        options.custom = $(this);
+        return $.notyRenderer.init(options);
+    };
+
+    $.noty = {};
+    $.noty.queue = [];
+    $.noty.ontap = true;
+    $.noty.layouts = {};
+    $.noty.themes = {};
+    $.noty.returns = 'object';
+    $.noty.store = {};
+
+    $.noty.get = function (id) {
+        return $.noty.store.hasOwnProperty(id) ? $.noty.store[id] : false;
+    };
+
+    $.noty.close = function (id) {
+        return $.noty.get(id) ? $.noty.get(id).close() : false;
+    };
+
+    $.noty.setText = function (id, text) {
+        return $.noty.get(id) ? $.noty.get(id).setText(text) : false;
+    };
+
+    $.noty.setType = function (id, type) {
+        return $.noty.get(id) ? $.noty.get(id).setType(type) : false;
+    };
+
+    $.noty.clearQueue = function () {
+        $.noty.queue = [];
+    };
+
+    $.noty.closeAll = function () {
+        $.noty.clearQueue();
+        $.each($.noty.store, function (id, noty) {
+            noty.close();
+        });
+    };
+
+    var windowAlert = window.alert;
+
+    $.noty.consumeAlert = function (options) {
+        window.alert = function (text) {
+            if (options)
+                options.text = text;
+            else
+                options = {text:text};
+
+            $.notyRenderer.init(options);
+        };
+    };
+
+    $.noty.stopConsumeAlert = function () {
+        window.alert = windowAlert;
+    };
+
+    $.noty.defaults = {
+        layout:'top',
+        theme:'defaultTheme',
+        type:'alert',
+        text:'',
+        dismissQueue:true,
+        template:'<div class="noty_message"><span class="noty_text"></span><div class="noty_close"></div></div>',
+        animation:{
+            open:{height:'toggle'},
+            close:{height:'toggle'},
+            easing:'swing',
+            speed:500
+        },
+        timeout:false,
+        force:false,
+        modal:false,
+        closeWith:['click'],
+        callback:{
+            onShow:function () {
+            },
+            afterShow:function () {
+            },
+            onClose:function () {
+            },
+            afterClose:function () {
+            }
+        },
+        buttons:false
+    };
+
+    $(window).resize(function () {
+        $.each($.noty.layouts, function (index, layout) {
+            layout.container.style.apply($(layout.container.selector));
+        });
+    });
+
+})(jQuery);
+
+// Helpers
+function noty(options) {
+
+    // This is for BC  -  Will be deleted on v2.2.0
+    var using_old = 0
+        , old_to_new = {
+            'animateOpen':'animation.open',
+            'animateClose':'animation.close',
+            'easing':'animation.easing',
+            'speed':'animation.speed',
+            'onShow':'callback.onShow',
+            'onShown':'callback.afterShow',
+            'onClose':'callback.onClose',
+            'onClosed':'callback.afterClose'
+        };
+
+    jQuery.each(options, function (key, value) {
+        if (old_to_new[key]) {
+            using_old++;
+            var _new = old_to_new[key].split('.');
+
+            if (!options[_new[0]]) options[_new[0]] = {};
+
+            options[_new[0]][_new[1]] = (value) ? value : function () {
+            };
+            delete options[key];
+        }
+    });
+
+    if (!options.closeWith) {
+        options.closeWith = jQuery.noty.defaults.closeWith;
+    }
+
+    if (options.hasOwnProperty('closeButton')) {
+        using_old++;
+        if (options.closeButton) options.closeWith.push('button');
+        delete options.closeButton;
+    }
+
+    if (options.hasOwnProperty('closeOnSelfClick')) {
+        using_old++;
+        if (options.closeOnSelfClick) options.closeWith.push('click');
+        delete options.closeOnSelfClick;
+    }
+
+    if (options.hasOwnProperty('closeOnSelfOver')) {
+        using_old++;
+        if (options.closeOnSelfOver) options.closeWith.push('hover');
+        delete options.closeOnSelfOver;
+    }
+
+    if (options.hasOwnProperty('custom')) {
+        using_old++;
+        if (options.custom.container != 'null') options.custom = options.custom.container;
+    }
+
+    if (options.hasOwnProperty('cssPrefix')) {
+        using_old++;
+        delete options.cssPrefix;
+    }
+
+    if (options.theme == 'noty_theme_default') {
+        using_old++;
+        options.theme = 'defaultTheme';
+    }
+
+    if (!options.hasOwnProperty('dismissQueue')) {
+        if (options.layout == 'topLeft'
+            || options.layout == 'topRight'
+            || options.layout == 'bottomLeft'
+            || options.layout == 'bottomRight') {
+            options.dismissQueue = true;
+        } else {
+            options.dismissQueue = false;
+        }
+    }
+
+    if (options.buttons) {
+        jQuery.each(options.buttons, function (i, button) {
+            if (button.click) {
+                using_old++;
+                button.onClick = button.click;
+                delete button.click;
+            }
+            if (button.type) {
+                using_old++;
+                button.addClass = button.type;
+                delete button.type;
+            }
+        });
+    }
+
+    if (using_old) {
+        if (typeof console !== "undefined" && console.warn) {
+            console.warn('You are using noty v2 with v1.x.x options. @deprecated until v2.2.0 - Please update your options.');
+        }
+    }
+
+    // console.log(options);
+    // End of the BC
+
+    return jQuery.notyRenderer.init(options);
+}

static/js/lib/jquery/noty/topCenter.js

+;(function($) {
+
+	$.noty.layouts.topCenter = {
+		name: 'topCenter',
+		options: { // overrides options
+
+		},
+		container: {
+			object: '<ul id="noty_topCenter_layout_container" />',
+			selector: 'ul#noty_topCenter_layout_container',
+			style: function() {
+				$(this).css({
+					top: 20,
+					left: 0,
+					position: 'fixed',
+					width: '310px',
+					height: 'auto',
+					margin: 0,
+					padding: 0,
+					listStyleType: 'none',
+					zIndex: 10000000
+				});
+
+				$(this).css({
+					left: ($(window).width() - $(this).outerWidth(false)) / 2 + 'px'
+				});
+			}
+		},
+		parent: {
+			object: '<li />',
+			selector: 'li',
+			css: {}
+		},
+		css: {
+			display: 'none',
+			width: '310px'
+		},
+		addClass: ''
+	};
+
+})(jQuery);

static/js/views/cart.js

+var app = app || {};
+
+$(function( $ ) {
+    'use strict';
+
+    // Purchase View
+    // ----------
+    app.CartView = Backbone.View.extend({
+
+        template: _.template( $('#cart-template').html() ),
+
+        initialize: function() {
+            console.log('Init cart view');
+            window.app.Cart.on( 'reset', this.render, this );
+            window.app.Cart.on( 'add', this.render, this );
+            this.renderCartPopover();
+        },
+
+        render: function() {
+            console.log('rendering cart');
+            var thisCart = this;
+            $('#cart-popover-content').html('');
+            app.Cart.each(function(product) {
+                $('#cart-popover-content').append( thisCart.template( product.toJSON() ) );
+            });
+            console.log("html = " + this.$el.html());
+            console.log(this.$el);
+            this.renderCartPopover();
+            return this;
+        },
+
+        renderCartPopover: function() {
+            $('#shopping-cart-btn').popover({
+                offset: -0,
+                placement:'bottom',
+                html:'true',
+                content: function() {
+                    return $('#cart-popover-content').html();
+                },
+                title: function() {
+                    return $('#cart-popover-title').html();
+                }
+            });
+        },
+
+
+        events: {
+        }
+    });
+});

static/js/views/product.js

             return this;
         },
 
+        addToCart: function() {
+            app.Cart.add( this.model );
+            console.log('added '+this.model.toJSON()+' to cart');
+            var notify = window.noty({
+                text: 'Added ' + this.model.get('Name') + ' ' + this.model.get('Name') + ' to cart',
+                type: 'information',
+                layout: 'topCenter',
+                timeout: true
+            });
+        },
+
         events: {
+            'click #add-to-cart-btn': 'addToCart'
         }
     });
 });

static/js/views/purchase.js

 
         initialize: function() {
             console.log('Init purchase view');
-            this.$el.load('purchase');
+            this.$el.load('purchase', this.initCartPopover);
             window.app.Products.on( 'reset', this.addAll, this );
             //window.app.Products.on( 'add', this.addOne, this );
             window.app.Products.fetch();
             console.log('fetched products');
         },
 
+        initCartPopover: function() {
+            new app.CartView();
+        },
+
         render: function() {
             return this;
         },
         },
 
         addAll: function() {
+
             console.log('adding all');
             this.$('#products-list').html('');
             app.Products.each(this.addOne, this);
         },
 
         events: {
-            'click a[href="#tables"]': 'showTables'
+            'click a[href="#tables"]': 'showTables',
         }
     });
 });

templates/index.html

 
         </div>
 
+        <script type="text/template" id="cart-template">
+            <div class="container">
+                <div class="row">
+                    <div class="col span2">
+                        <%= Brand %> <%= Name %> <%= Pricebuy %>
+                    </div>
+                    <div class="col offset2">
+                        1
+                    </div>
+                </div>
+            </div>
+        </script>
+ 
 
         <script type="text/template" id="product-thumbnail-template">
             <div class="span3">
                 </div>
                 <div class="modal-footer">
                     <button class="btn" data-dismiss="modal" aria-hidden="true">Close</button>
-                    <button class="btn btn-primary">Add item to cart</button>
+                    <button id="add-to-cart-btn" class="btn btn-primary">Add item to cart</button>
                 </div>
             </div>
         </script>
         <script src="//ajax.googleapis.com/ajax/libs/jquery/1.8.2/jquery.min.js"></script>
+        <script src="../static/js/lib/jquery/noty/jquery.noty.js"></script>
+        <script src="../static/js/lib/jquery/noty/topCenter.js"></script>
+        <script src="../static/js/lib/jquery/noty/default.js"></script>
         <script src="//netdna.bootstrapcdn.com/twitter-bootstrap/2.1.1/js/bootstrap.min.js"></script>
         <script src="//cdnjs.cloudflare.com/ajax/libs/underscore.js/1.4.2/underscore-min.js"></script>
         <script src="//cdnjs.cloudflare.com/ajax/libs/backbone.js/0.9.2/backbone-min.js"></script>
         <script src="../static/js/models/product.js"></script>
         <script src="../static/js/collections/products.js"></script>
         <script src="../static/js/collections/cart.js"></script>
+        <script src="../static/js/views/cart.js"></script>
         <script src="../static/js/views/product.js"></script>
         <script src="../static/js/views/productthumbnail.js"></script>
         <script src="../static/js/views/purchase.js"></script>

templates/purchase.html

     <div class="container">
         <div class="navbar ">
             <!-- HOME BTN -->
-            <a class="btn btn-info pull-left" href="index.html" style="display: inline-block">
+            <a class="btn btn-info pull-left" href="/" style="display: inline-block">
                 <i class="icon-arrow-left"></i>Home </a>
-            <!--<a "id="shoppingCart" class="btn btn-info pull-right" style="display:inline-block"><i class="icon-shopping-cart"></i> Shopping Cart </a> -->
             <a class="brand" style="margin-left: 250px">&nbsp;&nbsp; Browse Product
             </a>
             <form class="navbar-search">
                 &nbsp;&nbsp; <input class="search-query" placeholder="Search Product" style="width: 180px" type="text"> &nbsp;&nbsp
             </form>
             <!-- Shopping cart popover -->
-            <a id="shoppingCart" class="btn btn-info pull-right" style="display: inline-block; margin-right: 20px">
+            <a id="shopping-cart-btn" class="btn btn-info pull-right" style="display: inline-block; margin-right: 20px" rel="popover">
                 <i class="icon-shopping-cart"></i>Shopping Cart </a>
-            <!--<a id ="shoppingCart" class="btn pull-right" rel="popover" ><<img src="http://www.placehold.it/30x30"> HEHE </a> -->
-            <div id="popover_content_title" class="container row" style="display: none">
+            <div id="cart-popover-title" class="container row" style="display: none">
                 <div class="col span2">
-                    Product Detail </div>
-                <div class="col">
-                    Qty </div>
-            </div>
-            <div id="popover_content_wrapper" style="display: none">
-                <div class="container">
-                    <div class="row">
-                        <div class="col span2">
-                            Product Name $100 </div>
-                        <div class="col offset2">
-                            1 </div>
-                    </div>
-                    <br />
-                    <div class="row">
-                        hehe </div>
+                    Product Detail
+                </div>
+                <div class="col offset2">
+                    Qty
                 </div>
             </div>
+            <div id="cart-popover-content" style="display: none">
+           </div>
         </div>
     </div>
 </div>
Tip: Filter by directory path e.g. /media app.js to search for public/media/app.js.
Tip: Use camelCasing e.g. ProjME to search for ProjectModifiedEvent.java.
Tip: Filter by extension type e.g. /repo .js to search for all .js files in the /repo directory.
Tip: Separate your search with spaces e.g. /ssh pom.xml to search for src/ssh/pom.xml.
Tip: Use ↑ and ↓ arrow keys to navigate and return to view the file.
Tip: You can also navigate files with Ctrl+j (next) and Ctrl+k (previous) and view the file with Ctrl+o.
Tip: You can also navigate files with Alt+j (next) and Alt+k (previous) and view the file with Alt+o.