(function () { 'use strict'; var website = openerp.website; website.templates.push('/website/static/src/xml/website.snippets.xml'); website.EditorBar.include({ start: function () { var self = this; $("body").on('click', function (event) { var tag = event.srcElement.tagName.toLowerCase(); var $this = $(event.srcElement); if (!(tag === 'a' || tag === "button") && !$this.parents("a, button").length) { self.$('[data-action="edit"]').parent().effect('bounce', {distance: 18, times: 5}, 250); } }); return this._super(); }, edit: function () { $("body").off('click'); var self = this; self.on('rte:ready', this, function () { window.snippets = self.snippets = new website.snippet.BuildingBlock(this); self.snippets.appendTo(self.$el); }); return this._super.apply(this, arguments); }, save: function () { remove_added_snippet_id(); this._super(); }, }); $(document).ready(function () { hack_to_add_snippet_id(); $("[data-snippet-id]").each(function() { var $snipped_id = $(this); if (typeof $snipped_id.data("snippet-view") === 'undefined' && website.snippet.animationRegistry[$snipped_id.data("snippet-id")]) { $snipped_id.data("snippet-view", new website.snippet.animationRegistry[$snipped_id.data("snippet-id")]($snipped_id)); } }); }); /* ----- SNIPPET SELECTOR ---- */ website.snippet = {}; // puts $el at the same absolute position as $target function hack_to_add_snippet_id () { _.each(website.snippet.selector, function (val) { $(val[0]).each(function() { if (!$(this).is("[data-snippet-id]") && $(this).parents("[data-oe-model]").length) { $(this).attr("data-snippet-id", val[1]); } }); }); } function remove_added_snippet_id () { _.each(website.snippet.selector, function (val) { $(val[0]).each(function() { if ($(this).data("snippet-id") === val[1]) { $(this).removeAttr("data-snippet-id"); } }); }); } website.snippet.selector = []; website.snippet.BuildingBlock = openerp.Widget.extend({ template: 'website.snippets', activeSnippets: [], init: function (parent) { this.parent = parent; this._super.apply(this, arguments); if(!$('#oe_manipulators').length){ $("
").appendTo('body'); } this.$active_snipped_id = false; hack_to_add_snippet_id(); $("body").on('DOMNodeInserted', hack_to_add_snippet_id); }, dom_filter: function (dom) { if (typeof dom === "string") { var exclude = ":not(#oe_manipulators):not(#website-top-navbar):not(.oe_snippet)"; var sdom = dom.split(','); dom = ""; _.each(sdom, function (val) { val = val.replace(/^\s+|\s+$/g, ''); dom += "body > " + exclude + " " + val + ", "; val = val.split(" "); dom += "body > "+ val.shift() + exclude + val.join(" ") + ", "; }); dom = dom.replace(/,\s*$/g, ''); return $(dom); } else { return $(dom).parents("#oe_manipulators, #website-top-navbar").length || $(dom).hasClass("oe_snippet") ? $("") : $(dom); } }, start: function() { var self = this; var $ul = this.parent.$("#website-top-edit ul"); var $button = $(openerp.qweb.render('website.snippets_button')) .click(function () {self.$el.toggleClass("hidden");}) .prependTo($ul); this.fetch_snippet_templates(); this.bind_selected_manipulator(); this.bind_snippet_click_editor(); this.activate_overlay_zones(); this.scrollspy(); this.$el.addClass("hidden"); this.$modal = $(openerp.qweb.render('website.snippets_modal')); this.$modal.appendTo("body"); }, fetch_snippet_templates: function () { var self = this; this.style_templates = {}; _.each(openerp.qweb.compiled_templates, function (val, key) { if (key.indexOf('website.snippets.') === 0) { var $snippet = $(openerp.qweb.render(key)).addClass("oe_snippet"); var snipped_id = $snippet.data("snippet-id"); if (snipped_id) { if ($snippet.data('category') === 'style') { self.style_templates[snipped_id] = { 'snipped-id' : snipped_id, 'selector': $snippet.data('selector'), 'class': $snippet.find(".oe_snippet_class").text(), 'label': $snippet.find(".oe_snippet_label").text(), '$el': $snippet }; } if ($snippet.find(".oe_snippet_thumbnail").length) { self.$el.find('#snippet_' + $snippet.data('category')).append($snippet); self.make_snippet_draggable($snippet); } } } }); }, scrollspy: function (){ var self = this; var $ul = this.$("ul"); var $pill = self.$(".pill-content"); var padding = parseInt($pill.css("padding-left")); var $scroll = this.$(".scroll"); $scroll.scroll(function () { $pill.find("> div").each(function () { if ($(this).position().left <= padding) { $ul.find("> li").removeClass('active'); $ul.find("a[href='#" + $(this).attr("id") + "']").parent("li").addClass('active'); } }); }); $ul.find("a").click(function (event) { event.preventDefault(); $scroll.scrollLeft( $scroll.scrollLeft() + $($(event.currentTarget).attr("href")).position().left - padding ); return false; }); $pill.css("padding-right", self.$el.outerWidth() - padding + 10 - $pill.find("> div:last").outerWidth()); $ul.find("> li:first").addClass('active'); }, cover_target: function ($el, $target){ var pos = $target.offset(); var mt = parseInt($target.css("margin-top") || 0); var mb = parseInt($target.css("margin-bottom") || 0); $el.css({ 'position': 'absolute', 'width': $target.outerWidth(), 'height': $target.outerHeight() + mt + mb, 'top': pos.top - mt, 'left': pos.left }); }, show: function () { this.$el.removeClass("hidden"); }, hide: function () { this.$el.addClass("hidden"); }, bind_snippet_click_editor: function () { var self = this; var snipped_event_flag = false; $("body").on('click', "[data-snippet-id]", function (event) { if (snipped_event_flag) { return; } snipped_event_flag = true; setTimeout(function () {snipped_event_flag = false;}, 0); var $target = $(event.currentTarget); if (self.$active_snipped_id && self.$active_snipped_id.is($target)) { return; } self.make_active($target); }); $("body > :not(:has(#website-top-view)):not(#oe_manipulators)").on('click', function (ev) { if (!snipped_event_flag && self.$active_snipped_id && !self.$active_snipped_id.parents("[data-snippet-id]:first")) { self.make_active(false); } }); }, snippet_blur: function ($snipped_id) { if ($snipped_id) { if ($snipped_id.data("snippet-editor")) { $snipped_id.data("snippet-editor").onBlur(); } if ($snipped_id.data("snippet-view")) { $snipped_id.data("snippet-view").onBlurEdit(); } } }, snippet_focus: function ($snipped_id) { if ($snipped_id) { if ($snipped_id.data("snippet-view")) { $snipped_id.data("snippet-view").onFocusEdit(); } if ($snipped_id.data("snippet-editor")) { $snipped_id.data("snippet-editor").onFocus(); } } }, make_active: function ($snipped_id) { if ($snipped_id && this.$active_snipped_id && this.$active_snipped_id.get(0) === $snipped_id.get(0)) { return; } if (this.$active_snipped_id) { this.snippet_blur(this.$active_snipped_id); } if ($snipped_id) { this.$active_snipped_id = $snipped_id; this.create_overlay(this.$active_snipped_id); this.snippet_focus($snipped_id); } else { self.$active_snipped_id = false; } }, create_overlay: function ($snipped_id) { if (typeof $snipped_id.data("snippet-editor") === 'undefined') { this.activate_overlay_zones($snipped_id); var editor = website.snippet.editorRegistry[$snipped_id.data("snippet-id")] || website.snippet.editorRegistry.resize; $snipped_id.data("snippet-editor", new editor(this, $snipped_id)); } this.cover_target($snipped_id.data('overlay'), $snipped_id); }, path_eval: function(path){ var obj = window; path = path.split('.'); do{ obj = obj[path.shift()]; }while(path.length && obj); return obj; }, bind_selected_manipulator: function () { var self = this; var $selected_target = null; $("body").mouseover(function (event){ var $target = $(event.srcElement).parents("[data-snippet-id]:first"); $target = self.dom_filter($target); if($target.length && !self.editor_busy) { if($selected_target != $target){ if($selected_target && $selected_target.data('overlay')){ $selected_target.data('overlay').removeClass('oe_selected'); } $selected_target = $target; self.create_overlay($target); $target.data('overlay').addClass('oe_selected'); } } else if($selected_target && $selected_target.data('overlay')) { $selected_target.data('overlay').removeClass('oe_selected'); } }); }, // activate drag and drop for the snippets in the snippet toolbar make_snippet_draggable: function($snippet){ var self = this; var $toInsert = false; var $tumb = $snippet.find(".oe_snippet_thumbnail"); var dropped = false; $snippet.draggable({ greedy: true, helper: 'clone', zIndex: '1000', appendTo: 'body', cursor: "move", handle: ".oe_snippet_thumbnail", cursorAt: { 'left': $tumb.outerWidth()/2, 'top': $tumb.outerHeight()/2 }, start: function(){ self.hide(); dropped = false; var snipped_id = $snippet.data('snippet-id'); var action = $snippet.find('.oe_snippet_body').size() ? 'insert' : 'mutate'; if( action === 'insert'){ if (!$snippet.data('selector-siblings') && !$snippet.data('selector-children') && !$snippet.data('selector-vertical-children')) { console.debug($snippet.data("snippet-id") + " have oe_snippet_body class and have not for insert action"+ "data-selector-siblings, data-selector-children or data-selector-vertical-children tag for mutate action"); return; } self.activate_insertion_zones({ siblings: $snippet.data('selector-siblings'), children: $snippet.data('selector-children'), vertical_children: $snippet.data('selector-vertical-children') }); $toInsert = $snippet.find('.oe_snippet_body').clone(); $toInsert.removeClass('oe_snippet_body'); $toInsert.attr('data-snippet-id', snipped_id); } else if( action === 'mutate' ){ if (!$snippet.data('selector')) { console.debug($snippet.data("snippet-id") + " have not oe_snippet_body class and have not data-selector tag"); return; } var $targets = self.activate_overlay_zones($snippet.data('selector')); $targets.each(function(){ var $clone = $(this).data('overlay').clone(); $clone.addClass("oe_drop_zone").data('target', $(this)); $(this).data('overlay').after($clone); }); } $('.oe_drop_zone').droppable({ over: function(){ if( action === 'insert'){ dropped = true; $(this).first().after($toInsert); } }, out: function(){ if( action === 'insert'){ dropped = false; $toInsert.detach(); } }, drop: function(){ dropped = true; var $target = false; if(action === 'insert'){ $target = $toInsert; if (website.snippet.animationRegistry[snipped_id]) { new website.snippet.animationRegistry[snipped_id]($target); } self.create_overlay($target); $target.data("snippet-editor").build_snippet($target); } else { $target = $(this).data('target'); self.create_overlay($target); if (website.snippet.editorRegistry[snipped_id]) { var snippet = new website.snippet.editorRegistry[snipped_id](self, $target); snippet.build_snippet($target); } } $('.oe_drop_zone').remove(); setTimeout(function () {self.make_active($target);},0); }, }); }, stop: function(){ $('.oe_drop_zone').droppable('destroy').remove(); if (!dropped) { self.$modal.modal('toggle'); } }, }); }, // return the original snippet in the editor bar from a snippet id (string) get_snippet_from_id: function(id){ return $('.oe_snippet').filter(function(){ return $(this).data('snippet-id') === id; }).first(); }, // Create element insertion drop zones. two css selectors can be provided // selector.children -> will insert drop zones as direct child of the selected elements // in case the selected elements have children themselves, dropzones will be interleaved // with them. // selector.siblings -> will insert drop zones after and before selected elements activate_insertion_zones: function(selector){ var self = this; var child_selector = selector.children; var sibling_selector = selector.siblings; var vertical_child_selector = selector.vertical_children; var zone_template = ""; if(child_selector){ self.dom_filter(child_selector).each(function (){ var $zone = $(this); $zone.find('> *:not(.oe_drop_zone):visible').after(zone_template); $zone.prepend(zone_template); }); } if(vertical_child_selector){ self.dom_filter(vertical_child_selector).each(function (){ var $zone = $(this); var $template = $(zone_template).addClass("oe_vertical"); var nb = 0; var $lastinsert = false; var left = 0; var temp_left = 0; $zone.find('> *:not(.oe_drop_zone):visible').each(function () { var $col = $(this); $template.css('height', ($col.outerHeight() + parseInt($col.css("margin-top")) + parseInt($col.css("margin-bottom")))+'px'); $lastinsert = $template.clone(); $(this).after($lastinsert); temp_left = $col.position().left; if (left === temp_left) { $col.prev(".oe_drop_zone.oe_vertical").remove(); $col.before($template.clone().css("clear", "left")); } else if (!nb) { $col.before($template.clone()); } left = temp_left; nb ++; }); if (!nb) { $zone.prepend($template.css('height', $zone.outerHeight()+'px')); } }); } if(sibling_selector){ self.dom_filter(sibling_selector).each(function (){ var $zone = $(this); if($zone.prev('.oe_drop_zone:visible').length === 0){ $zone.before(zone_template); } if($zone.next('.oe_drop_zone:visible').length === 0){ $zone.after(zone_template); } }); } var count; do { count = 0; // var $zones = $('.oe_drop_zone + .oe_drop_zone'); // no two consecutive zones // count += $zones.length; // $zones.remove(); $zones = $('.oe_drop_zone > .oe_drop_zone:not(.oe_vertical)').remove(); // no recusrive zones count += $zones.length; $zones.remove(); } while (count > 0); // Cleaning up zones placed between floating or inline elements. We do not like these kind of zones. var $zones = $('.oe_drop_zone:not(.oe_vertical)'); $zones.each(function (){ var zone = $(this); var prev = zone.prev(); var next = zone.next(); var float_prev = zone.prev().css('float') || 'none'; var float_next = zone.next().css('float') || 'none'; var disp_prev = zone.prev().css('display') || null; var disp_next = zone.next().css('display') || null; if( (float_prev === 'left' || float_prev === 'right') && (float_next === 'left' || float_next === 'right') ){ zone.remove(); return; }else if( !( disp_prev === null || disp_next === null || disp_prev === 'block' || disp_next === 'block' )){ zone.remove(); } }); }, // generate drop zones covering the elements selected by the selector // we generate overlay drop zones only to get an idea of where the snippet are, the drop activate_overlay_zones: function(selector){ var $targets = this.dom_filter(selector || '[data-snippet-id]'); var self = this; function is_visible($el){ return $el.css('display') != 'none' && $el.css('opacity') != '0' && $el.css('visibility') != 'hidden'; } // filter out invisible elements $targets = $targets.filter(function(){ return is_visible($(this)); }); // filter out elements with invisible parents $targets = $targets.filter(function(){ var parents = $(this).parents().filter(function(){ return !is_visible($(this)); }); return parents.length === 0; }); $targets.each(function () { var $target = $(this); if (!$target.data('overlay')) { var $zone = $(openerp.qweb.render('website.snippet_overlay')); $zone.appendTo('#oe_manipulators'); $zone.data('target',$target); $target.data('overlay',$zone); $target.on("DOMNodeInserted DOMNodeRemoved DOMSubtreeModified", function () { self.cover_target($zone, $target); }); $('body').on("resize", function () { self.cover_target($zone, $target); }); } self.cover_target($target.data('overlay'), $target); }); return $targets; } }); website.snippet.animationRegistry = {}; website.snippet.Animation = openerp.Class.extend({ $: function () { return this.$el.find.apply(this.$el, arguments); }, init: function (dom) { this.$el = this.$target = $(dom); this.start(); }, /* * start * This method is called after init */ start: function () { }, /* onFocusEdit * if they are an editor for this snippet-id * Called before onFocus of snippet editor */ onFocusEdit : function () {}, /* onBlurEdit * if they are an editor for this snippet-id * Called after onBlur of snippet editor */ onBlurEdit : function () {}, /* getOptions * Read data saved for your snippet animation. */ getOptions: function () { var options = this.$el.data("snippet-options"); return options ? JSON.parse(options) : undefined; }, }); website.snippet.editorRegistry = {}; website.snippet.Editor = openerp.Class.extend({ init: function (parent, dom) { this.parent = parent; this.$target = $(dom); this.$overlay = this.$target.data('overlay'); this.snippet_id = this.$target.data("snippet-id"); this._readXMLData(); this.load_style_options(); this.get_parent_block(); this.start(); }, /* * _readXMLData * Read data XML and set value into: * this.$el : * all xml data * this.$overlay : * Dom hover the $target who content options * this.$editor : * content of .oe_snippet_options * Displayed into the overlay options on focus * this.$thumbnail : * content of .oe_snippet_thumbnail * Displayed in bottom editor menu, when the user click on "Building Blocks" * this.$body : * content of .oe_snippet_body * Insert into the view when the thumbnail is drag and droped into a drop zone */ _readXMLData: function() { this.$el = $(openerp.qweb.render("website.snippets." + this.snippet_id, {widget: this}).trim()); this.$editor = this.$el.find(".oe_snippet_options"); this.$thumbnail = this.$el.find(".oe_snippet_thumbnail"); this.$body = this.$el.find(".oe_snippet_body"); var $options = this.$overlay.find(".oe_overlay_options"); this.$editor.prependTo($options.find(".oe_options ul")); if ($options.find(".oe_options ul li").length) { $options.find(".oe_options").removeClass("hidden"); } }, // activate drag and drop for the snippets in the snippet toolbar _drag_and_drop: function(){ var self = this; var dropped = false; var $move = this.$overlay.find(".oe_snippet_move"); this.$overlay.draggable({ greedy: true, appendTo: 'body', cursor: "move", handle: ".oe_snippet_move", cursorAt: { left: 18, top: 14 }, helper: function() { var $clone = $(this).clone().css({width: "24px", height: "24px", border: 0}); $clone.find(".oe_overlay_options >:not(.oe_snippet_move), .oe_handle").remove(); $clone.find(":not(.glyphicon)").css({position: 'absolute', top: 0, left: 0}); return $clone.appendTo("body").removeClass("hidden"); }, start: function(){ self.parent.hide(); self.parent.editor_busy = true; self.$target.after(""); self.$target.detach(); self.$overlay.addClass("hidden"); self.parent.activate_insertion_zones({ siblings: self.$el ? self.$el.data('selector-siblings') : false, children: self.$el ? self.$el.data('selector-children') : false, vertical_children: self.$el ? self.$el.data('selector-vertical-children') : false, }); $("body").addClass('move-important'); $('.oe_drop_zone').droppable({ over: function(){ $(".oe_drop_zone.hide").removeClass("hide"); $(this).addClass("hide").first().after(self.$target); dropped = true; }, out: function(){ $(this).removeClass("hide"); self.$target.detach(); dropped = false; }, }); }, stop: function(){ if (!dropped) { $(".oe_drop_clone").after(self.$target); } self.$overlay.removeClass("hidden"); $("body").removeClass('move-important'); $('.oe_drop_zone').droppable('destroy').remove(); $(".oe_drop_clone").remove(); self.parent.editor_busy = false; self.get_parent_block(); setTimeout(function () {self.parent.create_overlay(self.$target);},0); }, }); }, _clone: function () { var self = this; this.$overlay.on('click', '.oe_snippet_clone', function () { var $clone = self.$target.clone(false); self.$target.after($clone); setTimeout(function () { self.parent.create_overlay($clone); self.parent.make_active($clone); },0); }); }, load_style_options: function () { var self = this; var $styles = this.$overlay.find('.oe_options'); var $ul = $styles.find('ul'); _.each(this.parent.style_templates, function (val, key) { if (!self.parent.dom_filter(val.selector).is(self.$target)) { return; } var $li = $("").data(val); $li.append($('').text(val.label)); $ul.append($li); if (self.$target.hasClass( "oe_snippet_" + $li.data("snipped-id") )) { $li.addClass("active"); } $styles.removeClass("hidden"); }); $styles.on('click', 'li.oe_style a', _.bind(this.change_style, this)); }, change_style: function (event) { var $li = $(event.currentTarget).parent(); var snipped_id = $li.data("snipped-id"); var active = $li.hasClass("active"); if (website.snippet.editorRegistry[snipped_id]) { var snippet = new website.snippet.editorRegistry[snipped_id](this, this.$target); snippet.build_snippet(this.$target); } var _class = "oe_snippet_" + snipped_id + " " + ($li.data("class") || ""); if (active) { this.$target.removeClass(_class); } else { this.$target.addClass(_class); } $li.toggleClass("active"); }, get_parent_block: function () { var self = this; var $button = this.$overlay.find('.oe_snippet_parent'); var $parent = this.$target.parents("[data-snippet-id]:first"); if ($parent.length) { $button.removeClass("hidden"); $button.off("click").on('click', function (event) { event.preventDefault(); setTimeout(function () { self.parent.make_active($parent); }, 0); }); } else { $button.addClass("hidden"); } }, /* * start * This method is called after init and _readXMLData */ start: function () { var self = this; this.$overlay.on('click', '.oe_snippet_remove', function () { self.$target.detach(); self.onBlur(); self.$target.remove(); return false; }); this._drag_and_drop(); this._clone(); }, /* * build_snippet * This method is called just after that a thumbnail is drag and droped into a drop zone * (after the insertion of this.$body, if this.$body exists) */ build_snippet: function ($target) { }, /* onFocus * This method is called when the user click inside the snippet in the dom */ onFocus : function () { this.$overlay.addClass('oe_active'); }, /* onFocus * This method is called when the user click outide the snippet in the dom, after a focus */ onBlur : function () { this.$overlay.removeClass('oe_active'); }, /* setOptions * Use this method when you want to save some data for your snippet animation. */ setOptions: function (options) { $target.attr("data-snippet-options", JSON.stringify(options)); }, /* getOptions * Read data saved for your snippet animation. */ getOptions: function () { var options = this.$target.data("snippet-options"); return options ? JSON.parse(options) : undefined; }, }); website.snippet.editorRegistry.resize = website.snippet.Editor.extend({ start: function () { var self = this; this._super(); var $box = $(openerp.qweb.render("website.snippets.resize")); var resize_values = this.getSize(); if (!resize_values.n) $box.find(".oe_handle.n").remove(); if (!resize_values.s) $box.find(".oe_handle.s").remove(); if (!resize_values.e) $box.find(".oe_handle.e").remove(); if (!resize_values.w) $box.find(".oe_handle.w").remove(); this.$overlay.append($box.find(".oe_handles").html()); this.$overlay.find(".oe_handle").on('mousedown', function (event){ event.preventDefault(); var $handle = $(this); var resize_values = self.getSize(); var compass = false; var XY = false; if ($handle.hasClass('n')) { compass = 'n'; XY = 'Y'; } else if ($handle.hasClass('s')) { compass = 's'; XY = 'Y'; } else if ($handle.hasClass('e')) { compass = 'e'; XY = 'X'; } else if ($handle.hasClass('w')) { compass = 'w'; XY = 'X'; } var resize = resize_values[compass]; if (!resize) return; var current = resize[2] || 0; _.each(resize[0], function (val, key) { if (self.$target.hasClass(val)) { current = key; } }); self.parent.editor_busy = true; var xy = event['page'+XY]; var begin = current; var beginClass = self.$target.attr("class"); var regClass = new RegExp("\\s*" + resize[0][begin].replace(/[-]*[0-9]+/, '[-]*[0-9]+'), 'g'); var cursor = $handle.css("cursor")+'-important'; $("body").addClass(cursor); self.$overlay.addClass('oe_hover'); var body_mousemove = function (event){ event.preventDefault(); var dd = event['page'+XY] - xy + resize[1][begin]; var next = current+1 === resize[1].length ? current : (current+1); var prev = current ? (current-1) : 0; var change = false; if (dd > (2*resize[1][next] + resize[1][current])/3) { self.$target.attr("class",self.$target.attr("class").replace(regClass, '')); self.$target.addClass(resize[0][next]); current = next; change = true; } if (prev != current && dd < (2*resize[1][prev] + resize[1][current])/3) { self.$target.attr("class",self.$target.attr("class").replace(regClass, '')); self.$target.addClass(resize[0][prev]); current = prev; change = true; } if (change) { self.on_resize(compass, beginClass, current); self.parent.cover_target(self.$overlay, self.$target); } }; $('body').mousemove(body_mousemove); var body_mouseup = function(){ $('body').unbind('mousemove', body_mousemove); $('body').unbind('mouseup', body_mouseup); $("body").removeClass(cursor); self.parent.editor_busy = false; }; $('body').mouseup(body_mouseup); }); }, getSize: function () { var grid = [0,4,8,16,32,48,64,92,128]; this.grid = { n: [_.map(grid, function (v) {return 'mt'+v;}), grid], s: [_.map(grid, function (v) {return 'mb'+v;}), grid] }; return this.grid; }, /* on_resize * called when the box is resizing and the class change, before the cover_target * @compass: resize direction : 'n', 's', 'e', 'w' * @beginClass: attributes class at the begin * @current: curent increment in this.grid */ on_resize: function (compass, beginClass, current) { } }); website.snippet.editorRegistry.colmd = website.snippet.editorRegistry.resize.extend({ getSize: function () { this.grid = this._super(); var width = this.$target.parents(".row:first").first().outerWidth(); var grid = [1,2,3,4,5,6,7,8,9,10,11,12]; this.grid.e = [_.map(grid, function (v) {return 'col-md-'+v;}), _.map(grid, function (v) {return width/12*v;})]; var grid = [-12,-11,-10,-9,-8,-7,-6,-5,-4,-3,-2,-1,0,1,2,3,4,5,6,7,8,9,10,11]; this.grid.w = [_.map(grid, function (v) {return 'col-md-offset-'+v;}), _.map(grid, function (v) {return width/12*v;}), 12]; return this.grid; }, on_resize: function (compass, beginClass, current) { if (compass !== 'w') return; // don't change the rigth border position when we change the offset (replace col size) var beginCol = Number(beginClass.match(/col-md-([0-9]+)|$/)[1] || 0); var beginOffset = Number(beginClass.match(/col-md-offset-([0-9-]+)|$/)[1] || beginClass.match(/col-lg-offset-([0-9-]+)|$/)[1] || 0); var offset = Number(this.grid.w[0][current].match(/col-md-offset-([0-9-]+)|$/)[1] || 0); this.$target.attr("class",this.$target.attr("class").replace(/\s*(col-lg-offset-|col-md-offset-|col-md-)([0-9-]+)/g, '')); var colSize = beginCol - (offset - beginOffset); this.$target.addClass('col-md-' + (colSize > 12 ? 12 : colSize)); if (offset > 0) { this.$target.addClass('col-md-offset-' + offset); } }, }); website.snippet.editorRegistry.carousel = website.snippet.editorRegistry.resize.extend({ build_snippet: function($target) { var id = "myCarousel" + $("body .carousel").size(); $target.attr("id", id); $target.find(".carousel-control").attr("href", "#"+id); }, start : function () { this._super(); this.$editor.find(".js_add").on('click', this, this.on_add); this.$editor.find(".js_remove").on('click', this, this.on_remove); this.change_background(); this.change_style(); this.change_size(); }, on_add: function (e) { e.preventDefault(); var $inner = e.data.$target.find('.carousel-inner'); var cycle = $inner.find('.item').size(); $inner.find('.item.active').clone().removeClass('active').appendTo($inner); e.data.$target.carousel(cycle); }, on_remove: function (e) { e.preventDefault(); var $inner = e.data.$target.find('.carousel-inner'); if ($inner.find('.item').size() > 1) { $inner .find('.item.active').remove().end() .find('.item:first').addClass('active'); e.data.$target.carousel(0); } }, change_background: function () { var self = this; var bg = this.$target.find('.carousel-inner .item.active').css('background-image').replace(/url\((.*)\)/g, '$1'); var selected = this.$editor.find('select[name="carousel-background"] option[value="'+bg+'"], select[name="carousel-background"] option[value="'+bg.replace(window.location.protocol+'//'+window.location.host, '')+'"]') .prop('selected', true).length; if (!selected) { this.$editor.find('.carousel-background input').val(bg); } this.$editor.find('select[name="carousel-background"], input') .on('click', function (event) {event.preventDefault(); return false;}) .on('change', function () { self.$target.find('.carousel-inner .item.active').css('background-image', 'url(' + $(this).val() + ')'); $(this).next().val(""); }); }, change_style: function () { var self = this; var style = false; var el = this.$target.find('.carousel-inner .item.active'); if (el.hasClass('text_only')) style = 'text_only'; if (el.hasClass('image_text')) style = 'image_text'; if (el.hasClass('text_image')) style = 'text_image'; this.$editor.find('select[name="carousel-style"] option[value="'+style+'"]').prop('selected', true); this.$editor.find('select[name="carousel-style"]').on('change', function(e) { var $container = self.$target.find('.carousel-inner .item.active'); $container.removeClass('image_text text_image text_only'); $container.addClass($(e.currentTarget).val()); }); }, change_size: function () { var self = this; var size = 'big'; if (this.$target.hasClass('oe_small')) size = 'small'; else if (this.$target.hasClass('oe_medium')) size = 'medium'; this.$editor.find('select[name="carousel-size"] option[value="'+size+'"]').prop('selected', true); this.$editor.find('select[name="carousel-size"]').on('change', function(e) { self.$target .removeClass('oe_big oe_small oe_medium') .addClass("oe_" + $(e.currentTarget).val()); }); } }); website.snippet.animationRegistry.vomify = website.snippet.Animation.extend({ start: function() { this._super(); var hue=0; var beat = false; var self = this; self.$target.append(''); var a = setInterval(function(){ self.$target.next().css({'-webkit-filter':'hue-rotate('+hue+'deg)'}); self.$target.prev().css({'-webkit-filter':'hue-rotate('+(-hue)+'deg)'}); hue -= 5; }, 10); setTimeout(function(){ clearInterval(a); setInterval(function(){ var filter = 'hue-rotate('+hue+'deg)'+ (beat ? ' invert()' : ''); $(document.documentElement).css({'-webkit-filter': filter}); hue += 5; if(hue % 35 === 0){ beat = !beat; } }, 10); },5000); } }); website.snippet.selector.push([ _.map([1,2,3,4,5,6,7,8,9,10,11,12], function (v) {return '.row > .col-md-'+v;}).join(","), 'colmd']); website.snippet.selector.push(['hr', 'hr']); })();