2013-08-29 10:29:56 +00:00
|
|
|
(function () {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var website = openerp.website;
|
2013-10-29 16:11:28 +00:00
|
|
|
website.add_template_file('/website/static/src/xml/website.snippets.xml');
|
2013-09-06 15:04:08 +00:00
|
|
|
|
2013-08-29 10:29:56 +00:00
|
|
|
website.EditorBar.include({
|
2013-09-18 08:33:24 +00:00
|
|
|
start: function () {
|
|
|
|
var self = this;
|
2013-09-20 15:21:40 +00:00
|
|
|
$("[data-oe-model]").on('click', function (event) {
|
2013-09-18 08:33:24 +00:00
|
|
|
var $this = $(event.srcElement);
|
2013-10-01 14:38:17 +00:00
|
|
|
var tag = $this[0].tagName.toLowerCase();
|
2013-09-18 08:33:24 +00:00
|
|
|
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();
|
|
|
|
},
|
2013-09-06 15:04:08 +00:00
|
|
|
edit: function () {
|
2013-11-07 09:42:56 +00:00
|
|
|
var self = this;
|
2013-09-18 08:33:24 +00:00
|
|
|
$("body").off('click');
|
2013-10-04 10:41:33 +00:00
|
|
|
window.snippets = this.snippets = new website.snippet.BuildingBlock(this);
|
|
|
|
this.snippets.appendTo(this.$el);
|
2013-11-07 09:42:56 +00:00
|
|
|
|
2013-11-08 14:38:48 +00:00
|
|
|
this.on('rte:ready', this, function () {
|
2013-11-07 09:42:56 +00:00
|
|
|
self.snippets.$button.removeClass("hidden");
|
2013-11-12 16:20:36 +00:00
|
|
|
website.snippet.stop_animation();
|
|
|
|
website.snippet.start_animation();
|
2013-11-07 09:42:56 +00:00
|
|
|
});
|
|
|
|
|
2013-09-11 12:56:32 +00:00
|
|
|
return this._super.apply(this, arguments);
|
2013-09-09 13:28:00 +00:00
|
|
|
},
|
2013-09-12 11:51:30 +00:00
|
|
|
save: function () {
|
2013-09-23 09:51:22 +00:00
|
|
|
this.snippets.make_active(false);
|
2013-10-19 20:59:31 +00:00
|
|
|
|
2013-10-24 14:10:24 +00:00
|
|
|
// FIXME: call clean_for_save on all snippets of the page, not only modified ones
|
|
|
|
// important for banner of parallax that changes data automatically.
|
2013-10-15 15:43:51 +00:00
|
|
|
this.snippets.clean_for_save();
|
2013-09-16 11:28:34 +00:00
|
|
|
remove_added_snippet_id();
|
2013-09-12 11:51:30 +00:00
|
|
|
this._super();
|
|
|
|
},
|
2013-08-29 10:29:56 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
/* ----- SNIPPET SELECTOR ---- */
|
2013-11-08 14:38:48 +00:00
|
|
|
|
2013-10-16 12:37:08 +00:00
|
|
|
var observer = new website.Observer(function (mutations) {
|
|
|
|
if (!_(mutations).find(function (m) {
|
|
|
|
return m.type === 'childList' && m.addedNodes.length > 0;
|
|
|
|
})) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
hack_to_add_snippet_id()
|
|
|
|
});
|
2013-09-16 11:28:34 +00:00
|
|
|
|
2013-09-09 13:28:00 +00:00
|
|
|
// puts $el at the same absolute position as $target
|
2013-09-12 17:06:52 +00:00
|
|
|
function hack_to_add_snippet_id () {
|
|
|
|
_.each(website.snippet.selector, function (val) {
|
|
|
|
$(val[0]).each(function() {
|
2013-09-16 11:28:34 +00:00
|
|
|
if (!$(this).is("[data-snippet-id]") && $(this).parents("[data-oe-model]").length) {
|
2013-09-12 17:06:52 +00:00
|
|
|
$(this).attr("data-snippet-id", val[1]);
|
2013-09-16 11:28:34 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
function remove_added_snippet_id () {
|
|
|
|
_.each(website.snippet.selector, function (val) {
|
|
|
|
$(val[0]).each(function() {
|
2013-09-17 11:00:43 +00:00
|
|
|
if ($(this).data("snippet-id") === val[1]) {
|
2013-09-16 11:28:34 +00:00
|
|
|
$(this).removeAttr("data-snippet-id");
|
|
|
|
}
|
2013-09-12 17:06:52 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2013-11-12 16:20:36 +00:00
|
|
|
$(document).ready(function() {
|
|
|
|
hack_to_add_snippet_id();
|
|
|
|
});
|
|
|
|
|
2014-01-21 12:39:12 +00:00
|
|
|
// 'snippet-dropped' is triggered on '#oe_snippets' whith $target as attribute when a snippet is dropped
|
|
|
|
// 'snippet-activated' is triggered on '#oe_snippets' (and on snippet) when a snippet is activated
|
|
|
|
|
2013-11-21 12:34:39 +00:00
|
|
|
website.snippet.styles = {};
|
2013-09-12 17:06:52 +00:00
|
|
|
website.snippet.selector = [];
|
2013-09-06 15:04:08 +00:00
|
|
|
website.snippet.BuildingBlock = openerp.Widget.extend({
|
2013-08-29 10:29:56 +00:00
|
|
|
template: 'website.snippets',
|
2013-09-12 11:51:30 +00:00
|
|
|
activeSnippets: [],
|
2013-09-09 13:28:00 +00:00
|
|
|
init: function (parent) {
|
|
|
|
this.parent = parent;
|
2013-08-29 10:29:56 +00:00
|
|
|
this._super.apply(this, arguments);
|
2013-09-21 09:51:16 +00:00
|
|
|
if(!$('#oe_manipulators').length){
|
|
|
|
$("<div id='oe_manipulators'></div>").appendTo('body');
|
|
|
|
}
|
2013-09-11 12:56:32 +00:00
|
|
|
this.$active_snipped_id = false;
|
2013-09-12 17:06:52 +00:00
|
|
|
hack_to_add_snippet_id();
|
2013-10-15 15:43:51 +00:00
|
|
|
this.snippets = [];
|
2013-10-16 12:37:08 +00:00
|
|
|
|
|
|
|
observer.observe(document.body, {
|
|
|
|
childList: true,
|
|
|
|
subtree: true,
|
|
|
|
});
|
2013-08-29 10:29:56 +00:00
|
|
|
},
|
2013-10-04 10:41:33 +00:00
|
|
|
dom_filter: function (dom, sibling) {
|
2013-09-17 07:48:47 +00:00
|
|
|
if (typeof dom === "string") {
|
2013-09-20 15:21:40 +00:00
|
|
|
var include = "[data-oe-model]";
|
2013-09-17 07:48:47 +00:00
|
|
|
var sdom = dom.split(',');
|
|
|
|
dom = "";
|
|
|
|
_.each(sdom, function (val) {
|
|
|
|
val = val.replace(/^\s+|\s+$/g, '');
|
2013-10-04 10:41:33 +00:00
|
|
|
dom += include + " " + val + ", ";
|
|
|
|
if (!sibling) {
|
|
|
|
val = val.split(" ");
|
|
|
|
dom += val.shift() + include + val.join(" ") + ", ";
|
|
|
|
}
|
2013-09-17 07:48:47 +00:00
|
|
|
});
|
|
|
|
dom = dom.replace(/,\s*$/g, '');
|
|
|
|
return $(dom);
|
|
|
|
} else {
|
2013-10-04 10:41:33 +00:00
|
|
|
return (!sibling && $(dom).is("[data-oe-model]")) || $(dom).parents("[data-oe-model]").length ? $(dom) : $("");
|
2013-09-17 07:48:47 +00:00
|
|
|
}
|
2013-09-16 12:29:42 +00:00
|
|
|
},
|
2013-08-29 10:29:56 +00:00
|
|
|
start: function() {
|
|
|
|
var self = this;
|
|
|
|
|
2013-11-07 09:42:56 +00:00
|
|
|
this.$button = $(openerp.qweb.render('website.snippets_button'))
|
|
|
|
.prependTo(this.parent.$("#website-top-edit ul"))
|
|
|
|
.find("button");
|
2013-09-19 10:20:14 +00:00
|
|
|
|
2013-11-07 09:42:56 +00:00
|
|
|
this.$button.click(function () {
|
2013-09-23 09:51:22 +00:00
|
|
|
self.make_active(false);
|
|
|
|
self.$el.toggleClass("hidden");
|
|
|
|
});
|
2014-01-08 14:50:46 +00:00
|
|
|
$("#wrapwrap").click(function () {
|
|
|
|
self.$el.addClass("hidden");
|
|
|
|
});
|
2013-09-19 10:20:14 +00:00
|
|
|
|
|
|
|
this.fetch_snippet_templates();
|
|
|
|
|
|
|
|
this.bind_snippet_click_editor();
|
|
|
|
|
|
|
|
this.$el.addClass("hidden");
|
2013-09-19 13:50:01 +00:00
|
|
|
|
|
|
|
this.$modal = $(openerp.qweb.render('website.snippets_modal'));
|
|
|
|
this.$modal.appendTo("body");
|
2013-12-04 17:16:12 +00:00
|
|
|
|
|
|
|
$(document).on('click', '.dropdown-submenu a[tabindex]', function (e) {
|
|
|
|
e.preventDefault();
|
|
|
|
});
|
2013-12-18 17:39:15 +00:00
|
|
|
|
2013-12-18 20:16:19 +00:00
|
|
|
this.getParent().on('change:height', this, function (editor) {
|
|
|
|
self.$el.css('top', editor.get('height'));
|
2013-12-18 17:39:15 +00:00
|
|
|
});
|
2014-01-02 17:18:25 +00:00
|
|
|
self.$el.css('top', this.parent.get('height'));
|
2013-09-19 10:20:14 +00:00
|
|
|
},
|
2014-01-13 12:55:34 +00:00
|
|
|
_get_snippet_url: function () {
|
|
|
|
return '/website/snippets';
|
|
|
|
},
|
2013-09-19 10:20:14 +00:00
|
|
|
fetch_snippet_templates: function () {
|
|
|
|
var self = this;
|
|
|
|
|
2014-01-13 12:55:34 +00:00
|
|
|
openerp.jsonRpc(this._get_snippet_url(), 'call', {})
|
2013-10-04 10:41:33 +00:00
|
|
|
.then(function (html) {
|
|
|
|
var $html = $(html);
|
|
|
|
|
2013-11-21 12:34:39 +00:00
|
|
|
var $styles = $html.find("[data-snippet-style-id]");
|
|
|
|
$styles.each(function () {
|
2013-10-04 10:41:33 +00:00
|
|
|
var $style = $(this);
|
2013-11-21 12:34:39 +00:00
|
|
|
var style_id = $style.data('snippet-style-id');
|
|
|
|
website.snippet.styles[style_id] = {
|
|
|
|
'snippet-style-id' : style_id,
|
2013-10-04 10:41:33 +00:00
|
|
|
'selector': $style.data('selector'),
|
2013-11-21 12:34:39 +00:00
|
|
|
'$el': $style,
|
2013-10-04 10:41:33 +00:00
|
|
|
};
|
|
|
|
});
|
2013-11-20 11:52:50 +00:00
|
|
|
$styles.addClass("hidden");
|
2013-09-18 15:48:31 +00:00
|
|
|
|
2013-10-04 10:41:33 +00:00
|
|
|
self.$snippets = $html.find(".tab-content > div > div").addClass("oe_snippet");
|
|
|
|
self.$el.append($html);
|
2013-09-18 15:48:31 +00:00
|
|
|
|
2014-01-16 18:05:35 +00:00
|
|
|
|
|
|
|
var snippets = 0;
|
|
|
|
self.$snippets.each(function () {
|
|
|
|
if (self.snippet_have_dropzone($(this)))
|
|
|
|
snippets++;
|
|
|
|
});
|
|
|
|
if (!snippets) self.$button.css("display", "none");
|
|
|
|
|
2013-10-04 10:41:33 +00:00
|
|
|
self.make_snippet_draggable(self.$snippets);
|
|
|
|
});
|
2013-09-16 11:28:34 +00:00
|
|
|
},
|
|
|
|
cover_target: function ($el, $target){
|
2013-09-17 11:00:43 +00:00
|
|
|
var pos = $target.offset();
|
|
|
|
var mt = parseInt($target.css("margin-top") || 0);
|
|
|
|
var mb = parseInt($target.css("margin-bottom") || 0);
|
2013-09-16 11:28:34 +00:00
|
|
|
$el.css({
|
|
|
|
'position': 'absolute',
|
2013-09-17 13:31:44 +00:00
|
|
|
'width': $target.outerWidth(),
|
2014-01-15 12:59:23 +00:00
|
|
|
'height': $target.outerHeight() + mt + mb+1,
|
2013-09-17 11:00:43 +00:00
|
|
|
'top': pos.top - mt,
|
2013-09-17 13:31:44 +00:00
|
|
|
'left': pos.left
|
2013-09-16 11:28:34 +00:00
|
|
|
});
|
2014-01-15 12:59:23 +00:00
|
|
|
$el.find(".oe_handle.size").css("bottom", (mb-7)+'px');
|
2013-09-16 11:28:34 +00:00
|
|
|
},
|
|
|
|
show: function () {
|
2013-09-18 15:48:31 +00:00
|
|
|
this.$el.removeClass("hidden");
|
2013-09-16 11:28:34 +00:00
|
|
|
},
|
|
|
|
hide: function () {
|
2013-09-18 15:48:31 +00:00
|
|
|
this.$el.addClass("hidden");
|
2013-09-11 12:56:32 +00:00
|
|
|
},
|
2013-09-09 13:28:00 +00:00
|
|
|
|
2014-01-16 18:05:35 +00:00
|
|
|
snippet_have_dropzone: function ($snippet) {
|
|
|
|
return (($snippet.data('selector-siblings') && this.dom_filter($snippet.data('selector-siblings')).size() > 0) ||
|
|
|
|
($snippet.data('selector-children') && this.dom_filter($snippet.data('selector-children')).size() > 0) ||
|
|
|
|
($snippet.data('selector-vertical-children') && this.dom_filter($snippet.data('selector-vertical-children')).size() > 0));
|
|
|
|
},
|
|
|
|
|
2013-09-11 12:56:32 +00:00
|
|
|
bind_snippet_click_editor: function () {
|
|
|
|
var self = this;
|
|
|
|
var snipped_event_flag = false;
|
2013-12-03 10:04:49 +00:00
|
|
|
$("#wrapwrap").on('click', function (event) {
|
|
|
|
if (snipped_event_flag) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
snipped_event_flag = true;
|
|
|
|
setTimeout(function () {snipped_event_flag = false;}, 0);
|
|
|
|
var $target = $(event.srcElement);
|
|
|
|
if (!$target.attr("data-snippet-id")) {
|
|
|
|
$target = $target.parents("[data-snippet-id]:first");
|
|
|
|
}
|
|
|
|
if (!$target.attr("data-oe-model") && !$target.parents("[data-oe-model]:first").length) {
|
|
|
|
$target = false;
|
|
|
|
}
|
|
|
|
if (self.$active_snipped_id && self.$active_snipped_id.is($target)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
self.make_active($target);
|
|
|
|
});
|
2013-09-11 12:56:32 +00:00
|
|
|
},
|
2013-12-06 13:32:39 +00:00
|
|
|
snippet_blur: function ($snippet) {
|
|
|
|
if ($snippet) {
|
|
|
|
if ($snippet.data("snippet-editor")) {
|
|
|
|
$snippet.data("snippet-editor").onBlur();
|
2013-09-11 12:56:32 +00:00
|
|
|
}
|
|
|
|
}
|
2013-08-29 10:29:56 +00:00
|
|
|
},
|
2013-12-06 13:32:39 +00:00
|
|
|
snippet_focus: function ($snippet) {
|
|
|
|
if ($snippet) {
|
|
|
|
if ($snippet.data("snippet-editor")) {
|
|
|
|
$snippet.data("snippet-editor").onFocus();
|
2013-09-11 12:56:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2013-10-15 15:43:51 +00:00
|
|
|
clean_for_save: function () {
|
|
|
|
for (var k in this.snippets) {
|
2013-10-22 15:11:44 +00:00
|
|
|
if (!this.snippets.hasOwnProperty(k)) { continue; }
|
2013-10-16 07:36:30 +00:00
|
|
|
var editor = $(this.snippets[k]).data("snippet-editor");
|
|
|
|
if (editor) {
|
|
|
|
editor.clean_for_save();
|
|
|
|
}
|
2013-10-15 15:43:51 +00:00
|
|
|
}
|
|
|
|
},
|
2013-12-06 13:32:39 +00:00
|
|
|
make_active: function ($snippet) {
|
|
|
|
if ($snippet && this.$active_snipped_id && this.$active_snipped_id.get(0) === $snippet.get(0)) {
|
2013-09-11 12:56:32 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (this.$active_snipped_id) {
|
|
|
|
this.snippet_blur(this.$active_snipped_id);
|
2013-11-06 16:39:27 +00:00
|
|
|
this.$active_snipped_id = false;
|
2013-09-11 12:56:32 +00:00
|
|
|
}
|
2013-12-06 13:32:39 +00:00
|
|
|
if ($snippet && $snippet.length) {
|
|
|
|
if(_.indexOf(this.snippets, $snippet.get(0)) === -1) {
|
|
|
|
this.snippets.push($snippet.get(0));
|
2013-10-15 15:43:51 +00:00
|
|
|
}
|
2013-12-06 13:32:39 +00:00
|
|
|
this.$active_snipped_id = $snippet;
|
2013-09-11 12:56:32 +00:00
|
|
|
this.create_overlay(this.$active_snipped_id);
|
2013-12-06 13:32:39 +00:00
|
|
|
this.snippet_focus($snippet);
|
|
|
|
}
|
2013-12-06 15:47:59 +00:00
|
|
|
$("#oe_snippets").trigger('snippet-activated', $snippet);
|
2013-12-09 11:25:34 +00:00
|
|
|
if ($snippet) {
|
|
|
|
$snippet.trigger('snippet-activated', $snippet);
|
|
|
|
}
|
2013-09-11 12:56:32 +00:00
|
|
|
},
|
2013-12-06 13:32:39 +00:00
|
|
|
create_overlay: function ($snippet) {
|
|
|
|
if (typeof $snippet.data("snippet-editor") === 'undefined') {
|
|
|
|
var $targets = this.activate_overlay_zones($snippet);
|
2013-10-04 10:41:33 +00:00
|
|
|
if (!$targets.length) return;
|
2013-12-06 13:32:39 +00:00
|
|
|
var editor = website.snippet.editorRegistry[$snippet.data("snippet-id")] || website.snippet.editorRegistry.resize;
|
|
|
|
$snippet.data("snippet-editor", new editor(this, $snippet));
|
2013-09-11 12:56:32 +00:00
|
|
|
}
|
2013-12-06 13:32:39 +00:00
|
|
|
this.cover_target($snippet.data('overlay'), $snippet);
|
2013-09-11 12:56:32 +00:00
|
|
|
},
|
|
|
|
|
2013-08-29 10:29:56 +00:00
|
|
|
path_eval: function(path){
|
|
|
|
var obj = window;
|
|
|
|
path = path.split('.');
|
|
|
|
do{
|
|
|
|
obj = obj[path.shift()];
|
|
|
|
}while(path.length && obj);
|
|
|
|
return obj;
|
|
|
|
},
|
|
|
|
|
2013-09-06 15:04:08 +00:00
|
|
|
// activate drag and drop for the snippets in the snippet toolbar
|
2013-10-04 10:41:33 +00:00
|
|
|
make_snippet_draggable: function($snippets){
|
2013-08-29 10:29:56 +00:00
|
|
|
var self = this;
|
2013-10-04 10:41:33 +00:00
|
|
|
var $tumb = $snippets.find(".oe_snippet_thumbnail:first");
|
|
|
|
var left = $tumb.outerWidth()/2;
|
|
|
|
var top = $tumb.outerHeight()/2;
|
2013-10-07 11:36:20 +00:00
|
|
|
var $toInsert, dropped, $snippet, action, snipped_id;
|
2013-08-29 10:29:56 +00:00
|
|
|
|
2013-10-04 10:41:33 +00:00
|
|
|
$snippets.draggable({
|
2013-09-19 13:57:16 +00:00
|
|
|
greedy: true,
|
2013-08-29 10:29:56 +00:00
|
|
|
helper: 'clone',
|
|
|
|
zIndex: '1000',
|
|
|
|
appendTo: 'body',
|
2013-09-13 09:35:18 +00:00
|
|
|
cursor: "move",
|
2013-09-18 15:54:40 +00:00
|
|
|
handle: ".oe_snippet_thumbnail",
|
|
|
|
cursorAt: {
|
2013-10-04 10:41:33 +00:00
|
|
|
'left': left,
|
|
|
|
'top': top
|
2013-09-18 15:54:40 +00:00
|
|
|
},
|
2013-08-29 10:29:56 +00:00
|
|
|
start: function(){
|
2013-09-20 09:19:17 +00:00
|
|
|
self.hide();
|
2013-09-19 13:50:01 +00:00
|
|
|
dropped = false;
|
2013-10-04 10:41:33 +00:00
|
|
|
$snippet = $(this);
|
2013-10-07 11:36:20 +00:00
|
|
|
snipped_id = $snippet.data('snippet-id');
|
|
|
|
action = $snippet.find('.oe_snippet_body').size() ? 'insert' : 'mutate';
|
2013-08-29 10:29:56 +00:00
|
|
|
if( action === 'insert'){
|
2013-09-16 12:32:57 +00:00
|
|
|
if (!$snippet.data('selector-siblings') && !$snippet.data('selector-children') && !$snippet.data('selector-vertical-children')) {
|
2013-09-16 08:57:55 +00:00
|
|
|
console.debug($snippet.data("snippet-id") + " have oe_snippet_body class and have not for insert action"+
|
2013-09-16 12:32:57 +00:00
|
|
|
"data-selector-siblings, data-selector-children or data-selector-vertical-children tag for mutate action");
|
2013-09-16 08:57:55 +00:00
|
|
|
return;
|
|
|
|
}
|
2013-08-29 10:29:56 +00:00
|
|
|
self.activate_insertion_zones({
|
2013-09-06 15:04:08 +00:00
|
|
|
siblings: $snippet.data('selector-siblings'),
|
2013-09-16 12:32:57 +00:00
|
|
|
children: $snippet.data('selector-children'),
|
|
|
|
vertical_children: $snippet.data('selector-vertical-children')
|
2013-08-29 10:29:56 +00:00
|
|
|
});
|
2013-09-11 12:56:32 +00:00
|
|
|
|
2013-09-17 07:18:06 +00:00
|
|
|
$toInsert = $snippet.find('.oe_snippet_body').clone();
|
|
|
|
$toInsert.removeClass('oe_snippet_body');
|
2013-12-06 16:31:22 +00:00
|
|
|
$toInsert.data('src-snippet-id', snipped_id);
|
2013-12-04 11:54:24 +00:00
|
|
|
if (!$toInsert.data('snippet-id')) {
|
|
|
|
$toInsert.attr('data-snippet-id', snipped_id);
|
|
|
|
} else {
|
|
|
|
snipped_id = $toInsert.data('snippet-id');
|
|
|
|
}
|
2013-09-17 07:18:06 +00:00
|
|
|
|
2013-09-16 08:57:55 +00:00
|
|
|
} 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;
|
|
|
|
}
|
2013-09-11 12:56:32 +00:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2013-08-29 10:29:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$('.oe_drop_zone').droppable({
|
|
|
|
over: function(){
|
2013-09-17 07:18:06 +00:00
|
|
|
if( action === 'insert'){
|
2013-09-20 09:19:17 +00:00
|
|
|
dropped = true;
|
2013-10-17 14:19:59 +00:00
|
|
|
$(this).first().after($toInsert);
|
2013-09-17 07:18:06 +00:00
|
|
|
}
|
2013-08-29 10:29:56 +00:00
|
|
|
},
|
|
|
|
out: function(){
|
2013-11-27 14:31:19 +00:00
|
|
|
var prev = $toInsert.prev();
|
2013-11-27 14:42:52 +00:00
|
|
|
if( action === 'insert' && this === prev[0]){
|
2013-09-20 09:19:17 +00:00
|
|
|
dropped = false;
|
2013-09-17 07:18:06 +00:00
|
|
|
$toInsert.detach();
|
|
|
|
}
|
2013-11-27 14:31:19 +00:00
|
|
|
}
|
2013-08-29 10:29:56 +00:00
|
|
|
});
|
|
|
|
},
|
2013-10-12 08:44:52 +00:00
|
|
|
stop: function(ev, ui){
|
2014-01-24 09:34:21 +00:00
|
|
|
if (action === 'insert' && ! dropped && $('.oe_drop_zone') && ui.position.top > 50) {
|
2013-11-16 08:53:50 +00:00
|
|
|
var el = $('.oe_drop_zone').nearest({x: ui.position.left, y: ui.position.top}).first();
|
|
|
|
if (el.length) {
|
|
|
|
el.after($toInsert);
|
2013-10-21 15:01:24 +00:00
|
|
|
dropped = true;
|
|
|
|
}
|
|
|
|
}
|
2013-10-17 14:19:59 +00:00
|
|
|
|
2013-09-11 12:56:32 +00:00
|
|
|
$('.oe_drop_zone').droppable('destroy').remove();
|
2013-10-07 11:36:20 +00:00
|
|
|
if (dropped) {
|
|
|
|
var $target = false;
|
|
|
|
if(action === 'insert'){
|
|
|
|
$target = $toInsert;
|
|
|
|
|
2013-11-12 16:20:36 +00:00
|
|
|
website.snippet.start_animation();
|
2013-10-07 11:36:20 +00:00
|
|
|
|
|
|
|
self.create_overlay($target);
|
2013-11-29 16:00:45 +00:00
|
|
|
if ($snippet.data("snippet-editor")) {
|
|
|
|
$target.data("snippet-editor").drop_and_build_snippet($target);
|
|
|
|
}
|
|
|
|
|
|
|
|
$target.find("[data-snippet-id]").each(function () {
|
|
|
|
var $snippet = $(this);
|
|
|
|
var snippet_id = $snippet.data("data-snippet-id");
|
|
|
|
self.create_overlay($snippet);
|
|
|
|
if ($snippet.data("snippet-editor")) {
|
|
|
|
$snippet.data("snippet-editor").drop_and_build_snippet($snippet);
|
|
|
|
}
|
|
|
|
});
|
2013-10-07 11:36:20 +00:00
|
|
|
|
|
|
|
} 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);
|
2013-11-06 13:39:17 +00:00
|
|
|
snippet.drop_and_build_snippet($target);
|
2013-10-07 11:36:20 +00:00
|
|
|
}
|
|
|
|
}
|
2013-12-06 15:08:55 +00:00
|
|
|
setTimeout(function () {
|
2013-12-06 15:47:59 +00:00
|
|
|
$("#oe_snippets").trigger('snippet-dropped', $target);
|
2013-12-06 15:08:55 +00:00
|
|
|
self.make_active($target);
|
|
|
|
},0);
|
2013-10-07 11:36:20 +00:00
|
|
|
} else {
|
|
|
|
$toInsert.remove();
|
|
|
|
if (self.$modal.find('input:not(:checked)').length) {
|
|
|
|
self.$modal.modal('toggle');
|
|
|
|
}
|
2013-09-19 13:50:01 +00:00
|
|
|
}
|
2013-08-29 10:29:56 +00:00
|
|
|
},
|
|
|
|
});
|
2013-09-06 15:04:08 +00:00
|
|
|
},
|
2013-08-29 10:29:56 +00:00
|
|
|
|
2013-09-06 15:04:08 +00:00
|
|
|
// 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();
|
2013-08-29 10:29:56 +00:00
|
|
|
},
|
2013-09-06 15:04:08 +00:00
|
|
|
|
2013-08-29 10:29:56 +00:00
|
|
|
// Create element insertion drop zones. two css selectors can be provided
|
2013-09-16 12:32:57 +00:00
|
|
|
// selector.children -> will insert drop zones as direct child of the selected elements
|
2013-08-29 10:29:56 +00:00
|
|
|
// 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;
|
2013-09-16 12:32:57 +00:00
|
|
|
var child_selector = selector.children;
|
2013-09-16 12:29:42 +00:00
|
|
|
var sibling_selector = selector.siblings;
|
2013-09-16 12:32:57 +00:00
|
|
|
var vertical_child_selector = selector.vertical_children;
|
2013-09-13 10:19:11 +00:00
|
|
|
|
2013-10-17 14:19:59 +00:00
|
|
|
var zone_template = "<div class='oe_drop_zone oe_insert'></div>";
|
2013-08-29 10:29:56 +00:00
|
|
|
|
|
|
|
if(child_selector){
|
2013-09-16 12:29:42 +00:00
|
|
|
self.dom_filter(child_selector).each(function (){
|
2013-09-13 10:19:11 +00:00
|
|
|
var $zone = $(this);
|
|
|
|
$zone.find('> *:not(.oe_drop_zone):visible').after(zone_template);
|
|
|
|
$zone.prepend(zone_template);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if(vertical_child_selector){
|
2013-09-16 12:29:42 +00:00
|
|
|
self.dom_filter(vertical_child_selector).each(function (){
|
2013-09-13 10:19:11 +00:00
|
|
|
var $zone = $(this);
|
2013-09-20 10:30:35 +00:00
|
|
|
var $template = $(zone_template).addClass("oe_vertical");
|
2013-09-20 10:26:00 +00:00
|
|
|
var nb = 0;
|
2013-09-20 13:01:22 +00:00
|
|
|
var $lastinsert = false;
|
|
|
|
var left = 0;
|
|
|
|
var temp_left = 0;
|
2013-09-20 10:26:00 +00:00
|
|
|
$zone.find('> *:not(.oe_drop_zone):visible').each(function () {
|
2013-09-20 10:30:35 +00:00
|
|
|
var $col = $(this);
|
2013-10-17 14:19:59 +00:00
|
|
|
$template.css('height', ($col.outerHeight() + parseInt($col.css("margin-top")) + parseInt($col.css("margin-bottom")))+'px');
|
2013-09-20 13:01:22 +00:00
|
|
|
$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());
|
2013-09-20 10:26:00 +00:00
|
|
|
}
|
2013-09-20 13:01:22 +00:00
|
|
|
left = temp_left;
|
2013-09-20 10:26:00 +00:00
|
|
|
nb ++;
|
|
|
|
});
|
|
|
|
if (!nb) {
|
2013-09-20 10:30:35 +00:00
|
|
|
$zone.prepend($template.css('height', $zone.outerHeight()+'px'));
|
2013-09-20 10:26:00 +00:00
|
|
|
}
|
2013-09-13 10:19:11 +00:00
|
|
|
});
|
2013-08-29 10:29:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(sibling_selector){
|
2013-10-04 10:41:33 +00:00
|
|
|
self.dom_filter(sibling_selector, true).each(function (){
|
2013-09-13 10:19:11 +00:00
|
|
|
var $zone = $(this);
|
|
|
|
if($zone.prev('.oe_drop_zone:visible').length === 0){
|
|
|
|
$zone.before(zone_template);
|
2013-08-29 10:29:56 +00:00
|
|
|
}
|
2013-09-13 10:19:11 +00:00
|
|
|
if($zone.next('.oe_drop_zone:visible').length === 0){
|
|
|
|
$zone.after(zone_template);
|
2013-08-29 10:29:56 +00:00
|
|
|
}
|
2013-09-13 10:19:11 +00:00
|
|
|
});
|
2013-08-29 10:29:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var count;
|
|
|
|
do {
|
|
|
|
count = 0;
|
2013-09-20 13:01:22 +00:00
|
|
|
// var $zones = $('.oe_drop_zone + .oe_drop_zone'); // no two consecutive zones
|
|
|
|
// count += $zones.length;
|
|
|
|
// $zones.remove();
|
2013-08-29 10:29:56 +00:00
|
|
|
|
2013-10-22 15:11:44 +00:00
|
|
|
$zones = $('.oe_drop_zone > .oe_drop_zone:not(.oe_vertical)').remove(); // no recursive zones
|
2013-08-29 10:29:56 +00:00
|
|
|
count += $zones.length;
|
|
|
|
$zones.remove();
|
2013-09-13 10:19:11 +00:00
|
|
|
} while (count > 0);
|
2013-08-29 10:29:56 +00:00
|
|
|
|
|
|
|
// Cleaning up zones placed between floating or inline elements. We do not like these kind of zones.
|
2013-09-13 10:19:11 +00:00
|
|
|
var $zones = $('.oe_drop_zone:not(.oe_vertical)');
|
|
|
|
$zones.each(function (){
|
|
|
|
var zone = $(this);
|
2013-08-29 10:29:56 +00:00
|
|
|
var prev = zone.prev();
|
|
|
|
var next = zone.next();
|
2013-10-22 15:11:44 +00:00
|
|
|
var float_prev = prev.css('float') || 'none';
|
|
|
|
var float_next = next.css('float') || 'none';
|
|
|
|
var disp_prev = prev.css('display') || null;
|
|
|
|
var disp_next = next.css('display') || null;
|
2013-08-29 10:29:56 +00:00
|
|
|
if( (float_prev === 'left' || float_prev === 'right')
|
|
|
|
&& (float_next === 'left' || float_next === 'right') ){
|
|
|
|
zone.remove();
|
|
|
|
}else if( !( disp_prev === null
|
|
|
|
|| disp_next === null
|
|
|
|
|| disp_prev === 'block'
|
|
|
|
|| disp_next === 'block' )){
|
|
|
|
zone.remove();
|
|
|
|
}
|
2013-09-13 10:19:11 +00:00
|
|
|
});
|
2013-08-29 10:29:56 +00:00
|
|
|
},
|
|
|
|
|
2013-09-06 15:04:08 +00:00
|
|
|
// generate drop zones covering the elements selected by the selector
|
2013-09-11 12:56:32 +00:00
|
|
|
// we generate overlay drop zones only to get an idea of where the snippet are, the drop
|
2013-08-29 10:29:56 +00:00
|
|
|
activate_overlay_zones: function(selector){
|
2013-09-16 12:29:42 +00:00
|
|
|
var $targets = this.dom_filter(selector || '[data-snippet-id]');
|
2013-09-16 11:28:34 +00:00
|
|
|
var self = this;
|
2013-10-04 10:41:33 +00:00
|
|
|
|
|
|
|
if (typeof selector !== 'string' && !$targets.length) {
|
|
|
|
console.debug( "A good node must have a [data-oe-model] attribute or must have at least one parent with [data-oe-model] attribute.");
|
|
|
|
console.debug( "Wrong node(s): ", selector);
|
|
|
|
}
|
2013-11-08 14:38:48 +00:00
|
|
|
|
2013-08-29 10:29:56 +00:00
|
|
|
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;
|
|
|
|
});
|
2013-11-08 14:38:48 +00:00
|
|
|
|
2013-09-11 12:56:32 +00:00
|
|
|
$targets.each(function () {
|
|
|
|
var $target = $(this);
|
|
|
|
if (!$target.data('overlay')) {
|
2013-09-12 14:53:02 +00:00
|
|
|
var $zone = $(openerp.qweb.render('website.snippet_overlay'));
|
2013-09-11 12:56:32 +00:00
|
|
|
$zone.appendTo('#oe_manipulators');
|
|
|
|
$zone.data('target',$target);
|
|
|
|
$target.data('overlay',$zone);
|
|
|
|
|
|
|
|
$target.on("DOMNodeInserted DOMNodeRemoved DOMSubtreeModified", function () {
|
2013-09-16 11:28:34 +00:00
|
|
|
self.cover_target($zone, $target);
|
|
|
|
});
|
|
|
|
$('body').on("resize", function () {
|
|
|
|
self.cover_target($zone, $target);
|
2013-09-11 12:56:32 +00:00
|
|
|
});
|
|
|
|
}
|
2013-09-16 11:28:34 +00:00
|
|
|
self.cover_target($target.data('overlay'), $target);
|
2013-09-06 15:04:08 +00:00
|
|
|
});
|
2013-09-11 12:56:32 +00:00
|
|
|
return $targets;
|
2013-09-16 11:28:34 +00:00
|
|
|
}
|
2013-08-29 10:29:56 +00:00
|
|
|
});
|
2013-09-06 15:04:08 +00:00
|
|
|
|
|
|
|
|
2013-11-21 12:34:39 +00:00
|
|
|
website.snippet.styleRegistry = {};
|
|
|
|
website.snippet.StyleEditor = openerp.Class.extend({
|
|
|
|
// initialisation (don't overwrite)
|
|
|
|
init: function (parent, $target, snippet_id) {
|
|
|
|
this.parent = parent;
|
|
|
|
this.$target = $target;
|
|
|
|
var styles = this.$target.data("snippet-style-ids") || {};
|
|
|
|
styles[snippet_id] = this;
|
|
|
|
this.$target.data("snippet-style-ids", styles);
|
|
|
|
this.$overlay = this.$target.data('overlay');
|
|
|
|
this['snippet-style-id'] = snippet_id;
|
|
|
|
this.$el = website.snippet.styles[snippet_id].$el.find(">li").clone();
|
|
|
|
|
|
|
|
this.required = this.$el.data("required");
|
|
|
|
|
|
|
|
this.set_active();
|
|
|
|
this.$el.find('li[data-class] a').on('mouseover mouseout click', _.bind(this._mouse, this));
|
|
|
|
this.$target.on('snippet-style-reset', _.bind(this.set_active, this));
|
|
|
|
|
|
|
|
this.start();
|
|
|
|
},
|
|
|
|
_mouse: function (event) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
if (event.type === 'mouseout') {
|
|
|
|
if (!this.over) return;
|
|
|
|
this.over = false;
|
|
|
|
} else if (event.type === 'click') {
|
|
|
|
this.over = false;
|
|
|
|
}else {
|
|
|
|
this.over = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
var $prev, $next;
|
|
|
|
if (event.type === 'mouseout') {
|
|
|
|
$prev = $(event.currentTarget).parent();
|
|
|
|
$next = this.$el.find("li[data-class].active");
|
|
|
|
} else {
|
|
|
|
$prev = this.$el.find("li[data-class].active");
|
|
|
|
$next = $(event.currentTarget).parent();
|
|
|
|
}
|
|
|
|
if (!$prev.length) {
|
|
|
|
$prev = false;
|
|
|
|
}
|
|
|
|
if ($prev && $prev[0] === $next[0]) {
|
|
|
|
$next = false;
|
|
|
|
if (this.required) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var np = {'$next': $next, '$prev': $prev};
|
|
|
|
|
|
|
|
if (event.type === 'click') {
|
|
|
|
setTimeout(function () {
|
|
|
|
self.set_active();
|
|
|
|
self.$target.trigger("snippet-style-change", [self, np]);
|
|
|
|
},0);
|
|
|
|
this.select(event, {'$next': $next, '$prev': $prev});
|
|
|
|
} else {
|
|
|
|
setTimeout(function () {
|
|
|
|
self.$target.trigger("snippet-style-preview", [self, np]);
|
|
|
|
},0);
|
|
|
|
this.preview(event, np);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// start is call just after the init
|
|
|
|
start: function () {
|
|
|
|
},
|
|
|
|
/* select
|
|
|
|
* called when a user select an item
|
|
|
|
* variables: np = {$next, $prev}
|
|
|
|
* $next is false if they are no next item selected
|
|
|
|
* $prev is false if they are no previous item selected
|
|
|
|
*/
|
|
|
|
select: function (event, np) {
|
|
|
|
var self = this;
|
|
|
|
// add or remove html class
|
|
|
|
if (np.$prev) {
|
|
|
|
this.$target.removeClass(np.$prev.data('class' || ""));
|
|
|
|
}
|
|
|
|
if (np.$next) {
|
|
|
|
this.$target.addClass(np.$next.data('class') || "");
|
|
|
|
}
|
|
|
|
},
|
|
|
|
/* preview
|
|
|
|
* called when a user is on mouse over or mouse out of an item
|
|
|
|
* variables: np = {$next, $prev}
|
|
|
|
* $next is false if they are no next item selected
|
|
|
|
* $prev is false if they are no previous item selected
|
|
|
|
*/
|
|
|
|
preview: function (event, np) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
// add or remove html class
|
|
|
|
if (np.$prev) {
|
|
|
|
this.$target.removeClass(np.$prev.data('class') || "");
|
|
|
|
}
|
|
|
|
if (np.$next) {
|
|
|
|
this.$target.addClass(np.$next.data('class') || "");
|
|
|
|
}
|
|
|
|
},
|
|
|
|
/* set_active
|
|
|
|
* select and set item active or not (add highlight item and his parents)
|
|
|
|
* called before start
|
|
|
|
*/
|
|
|
|
set_active: function () {
|
|
|
|
var self = this;
|
|
|
|
this.$el.find('li').removeClass("active");
|
|
|
|
var $active = this.$el.find('li[data-class]')
|
|
|
|
.filter(function () {
|
|
|
|
var $li = $(this);
|
|
|
|
return ($li.data('class') && self.$target.hasClass($li.data('class')));
|
|
|
|
})
|
|
|
|
.first()
|
|
|
|
.addClass("active");
|
|
|
|
this.$el.find('li:has(li[data-class].active)').addClass("active");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
website.snippet.styleRegistry.background = website.snippet.StyleEditor.extend({
|
|
|
|
_get_bg: function () {
|
|
|
|
return this.$target.css("background-image").replace(/url\(['"]*|['"]*\)|^none$/g, "");
|
|
|
|
},
|
|
|
|
_set_bg: function (src) {
|
|
|
|
this.$target.css("background-image", src && src !== "" ? 'url(' + src + ')' : "");
|
|
|
|
},
|
|
|
|
start: function () {
|
|
|
|
this._super();
|
|
|
|
var src = this._get_bg();
|
|
|
|
this.$el.find("li[data-class].active.oe_custom_bg").data("src", src);
|
|
|
|
},
|
|
|
|
select: function(event, np) {
|
|
|
|
var self = this;
|
|
|
|
this._super(event, np);
|
|
|
|
if (np.$next) {
|
|
|
|
if (np.$next.hasClass("oe_custom_bg")) {
|
|
|
|
var editor = new website.editor.ImageDialog();
|
2013-11-27 16:59:14 +00:00
|
|
|
editor.on('start', self, function (o) {o.url = np.$prev && np.$prev.data("src") || np.$next && np.$next.data("src") || "";});
|
2013-11-21 12:34:39 +00:00
|
|
|
editor.on('save', self, function (o) {
|
|
|
|
self._set_bg(o.url);
|
|
|
|
np.$next.data("src", o.url);
|
|
|
|
self.$target.trigger("snippet-style-change", [self, np]);
|
|
|
|
});
|
|
|
|
editor.on('cancel', self, function () {
|
|
|
|
if (!np.$prev || np.$prev.data("src") === "") {
|
|
|
|
self.$target.removeClass(np.$next.data("class"));
|
|
|
|
self.$target.trigger("snippet-style-change", [self, np]);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
editor.appendTo($('body'));
|
|
|
|
} else {
|
|
|
|
this._set_bg(np.$next.data("src"));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this._set_bg(false);
|
2013-11-27 16:59:14 +00:00
|
|
|
this.$target.removeClass(np.$prev.data("class"));
|
2013-11-21 12:34:39 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
preview: function (event, np) {
|
|
|
|
this._super(event, np);
|
|
|
|
if (np.$next) {
|
|
|
|
this._set_bg(np.$next.data("src"));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
set_active: function () {
|
|
|
|
var self = this;
|
|
|
|
var bg = self.$target.css("background-image");
|
|
|
|
this.$el.find('li').removeClass("active");
|
|
|
|
var $active = this.$el.find('li[data-class]')
|
|
|
|
.filter(function () {
|
|
|
|
var $li = $(this);
|
|
|
|
return ($li.data('src') && bg.indexOf($li.data('src')) >= 0) ||
|
|
|
|
(!$li.data('src') && self.$target.hasClass($li.data('class')));
|
|
|
|
})
|
|
|
|
.first();
|
|
|
|
if (!$active.length) {
|
|
|
|
$active = this.$target.css("background-image") !== 'none' ?
|
|
|
|
this.$el.find('li[data-class].oe_custom_bg') :
|
|
|
|
this.$el.find('li[data-class=""]');
|
|
|
|
}
|
|
|
|
$active.addClass("active");
|
|
|
|
this.$el.find('li:has(li[data-class].active)').addClass("active");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2013-09-06 15:04:08 +00:00
|
|
|
website.snippet.editorRegistry = {};
|
2013-09-10 10:21:40 +00:00
|
|
|
website.snippet.Editor = openerp.Class.extend({
|
2013-11-21 12:34:39 +00:00
|
|
|
init: function (parent, dom) {
|
2013-09-06 15:04:08 +00:00
|
|
|
this.parent = parent;
|
2013-09-11 12:56:32 +00:00
|
|
|
this.$target = $(dom);
|
|
|
|
this.$overlay = this.$target.data('overlay');
|
2013-11-21 12:34:39 +00:00
|
|
|
this.snippet_id = this.$target.data("snippet-id");
|
2013-09-10 10:21:40 +00:00
|
|
|
this._readXMLData();
|
2013-09-19 12:32:06 +00:00
|
|
|
this.load_style_options();
|
2013-09-19 12:58:54 +00:00
|
|
|
this.get_parent_block();
|
2013-09-06 15:04:08 +00:00
|
|
|
this.start();
|
|
|
|
},
|
|
|
|
|
2013-09-10 10:21:40 +00:00
|
|
|
/*
|
|
|
|
* _readXMLData
|
|
|
|
* Read data XML and set value into:
|
|
|
|
* this.$el :
|
|
|
|
* all xml data
|
2013-09-11 12:56:32 +00:00
|
|
|
* this.$overlay :
|
|
|
|
* Dom hover the $target who content options
|
2013-09-10 10:21:40 +00:00
|
|
|
* this.$editor :
|
2013-09-11 12:56:32 +00:00
|
|
|
* content of .oe_snippet_options
|
|
|
|
* Displayed into the overlay options on focus
|
2013-09-10 10:21:40 +00:00
|
|
|
*/
|
|
|
|
_readXMLData: function() {
|
2013-11-04 10:29:14 +00:00
|
|
|
var self = this;
|
|
|
|
this.$el = this.parent.$snippets.filter(function () { return $(this).data("snippet-id") == self.snippet_id; }).clone();
|
2013-09-17 11:52:35 +00:00
|
|
|
this.$editor = this.$el.find(".oe_snippet_options");
|
|
|
|
var $options = this.$overlay.find(".oe_overlay_options");
|
|
|
|
this.$editor.prependTo($options.find(".oe_options ul"));
|
2013-09-20 08:49:52 +00:00
|
|
|
if ($options.find(".oe_options ul li").length) {
|
|
|
|
$options.find(".oe_options").removeClass("hidden");
|
2013-09-19 12:32:06 +00:00
|
|
|
}
|
2013-09-06 15:04:08 +00:00
|
|
|
},
|
|
|
|
|
2013-09-13 09:35:18 +00:00
|
|
|
|
|
|
|
// activate drag and drop for the snippets in the snippet toolbar
|
|
|
|
_drag_and_drop: function(){
|
|
|
|
var self = this;
|
2013-09-23 09:51:22 +00:00
|
|
|
this.dropped = false;
|
2013-09-13 09:35:18 +00:00
|
|
|
this.$overlay.draggable({
|
2013-09-13 11:44:56 +00:00
|
|
|
greedy: true,
|
2013-09-13 09:35:18 +00:00
|
|
|
appendTo: 'body',
|
|
|
|
cursor: "move",
|
2013-09-19 10:20:14 +00:00
|
|
|
handle: ".oe_snippet_move",
|
|
|
|
cursorAt: {
|
|
|
|
left: 18,
|
|
|
|
top: 14
|
|
|
|
},
|
2013-09-17 11:52:35 +00:00
|
|
|
helper: function() {
|
2013-09-20 10:08:21 +00:00
|
|
|
var $clone = $(this).clone().css({width: "24px", height: "24px", border: 0});
|
2013-09-23 09:51:22 +00:00
|
|
|
$clone.find(".oe_overlay_options >:not(:contains(.oe_snippet_move)), .oe_handle").remove();
|
2013-09-19 10:20:14 +00:00
|
|
|
$clone.find(":not(.glyphicon)").css({position: 'absolute', top: 0, left: 0});
|
2013-09-23 09:51:22 +00:00
|
|
|
$clone.appendTo("body").removeClass("hidden");
|
|
|
|
return $clone;
|
2013-09-17 11:52:35 +00:00
|
|
|
},
|
2013-09-23 09:51:22 +00:00
|
|
|
start: _.bind(self._drag_and_drop_start, self),
|
|
|
|
stop: _.bind(self._drag_and_drop_stop, self)
|
|
|
|
});
|
|
|
|
},
|
|
|
|
_drag_and_drop_after_insert_dropzone: function (){},
|
2013-10-17 14:19:59 +00:00
|
|
|
_drag_and_drop_active_drop_zone: function ($zones){
|
2013-09-23 09:51:22 +00:00
|
|
|
var self = this;
|
2013-10-17 14:19:59 +00:00
|
|
|
$zones.droppable({
|
2013-09-23 09:51:22 +00:00
|
|
|
over: function(){
|
2013-10-17 14:19:59 +00:00
|
|
|
$(".oe_drop_zone.hide").removeClass("hide");
|
|
|
|
$(this).addClass("hide").first().after(self.$target);
|
2013-09-23 09:51:22 +00:00
|
|
|
self.dropped = true;
|
2013-09-13 09:35:18 +00:00
|
|
|
},
|
2013-09-23 09:51:22 +00:00
|
|
|
out: function(){
|
2013-10-17 14:19:59 +00:00
|
|
|
$(this).removeClass("hide");
|
2013-09-23 09:51:22 +00:00
|
|
|
self.$target.detach();
|
|
|
|
self.dropped = false;
|
2013-09-13 09:35:18 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
2013-09-23 09:51:22 +00:00
|
|
|
_drag_and_drop_start: function (){
|
|
|
|
var self = this;
|
|
|
|
self.parent.hide();
|
|
|
|
self.parent.editor_busy = true;
|
|
|
|
self.size = {
|
|
|
|
width: self.$target.width(),
|
|
|
|
height: self.$target.height()
|
|
|
|
};
|
|
|
|
self.$target.after("<div class='oe_drop_clone' style='display: none;'/>");
|
|
|
|
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');
|
2013-11-08 14:38:48 +00:00
|
|
|
|
2013-09-23 09:51:22 +00:00
|
|
|
self._drag_and_drop_after_insert_dropzone();
|
2013-10-17 14:19:59 +00:00
|
|
|
self._drag_and_drop_active_drop_zone($('.oe_drop_zone'));
|
2013-09-23 09:51:22 +00:00
|
|
|
},
|
|
|
|
_drag_and_drop_stop: function (){
|
|
|
|
var self = this;
|
|
|
|
if (!self.dropped) {
|
|
|
|
$(".oe_drop_clone").after(self.$target);
|
|
|
|
}
|
|
|
|
self.$overlay.removeClass("hidden");
|
|
|
|
$("body").removeClass('move-important');
|
|
|
|
$('.oe_drop_zone').droppable('destroy').remove();
|
|
|
|
$(".oe_drop_clone, .oe_drop_to_remove").remove();
|
|
|
|
self.parent.editor_busy = false;
|
|
|
|
self.get_parent_block();
|
|
|
|
setTimeout(function () {self.parent.create_overlay(self.$target);},0);
|
|
|
|
},
|
2013-09-13 09:35:18 +00:00
|
|
|
|
2013-09-19 12:32:06 +00:00
|
|
|
load_style_options: function () {
|
|
|
|
var self = this;
|
2013-09-20 08:49:52 +00:00
|
|
|
var $styles = this.$overlay.find('.oe_options');
|
2013-09-27 12:41:16 +00:00
|
|
|
var $ul = $styles.find('ul:first');
|
2013-11-21 12:34:39 +00:00
|
|
|
_.each(website.snippet.styles, function (val) {
|
2013-09-19 12:32:06 +00:00
|
|
|
if (!self.parent.dom_filter(val.selector).is(self.$target)) {
|
|
|
|
return;
|
|
|
|
}
|
2013-11-21 12:34:39 +00:00
|
|
|
var Editor = website.snippet.styleRegistry[val['snippet-style-id']] || website.snippet.StyleEditor;
|
|
|
|
var editor = new Editor(self, self.$target, val['snippet-style-id']);
|
2014-01-08 12:21:13 +00:00
|
|
|
$ul.prepend(editor.$el.addClass("snippet-style-" + val['snippet-style-id']));
|
2013-09-19 12:32:06 +00:00
|
|
|
});
|
2013-12-06 13:32:39 +00:00
|
|
|
|
2013-11-20 11:52:50 +00:00
|
|
|
if ($ul.find("li").length) {
|
|
|
|
$styles.removeClass("hidden");
|
2013-09-19 12:32:06 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2013-09-19 12:58:54 +00:00
|
|
|
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) {
|
2013-09-20 08:49:52 +00:00
|
|
|
$button.removeClass("hidden");
|
2013-09-19 12:58:54 +00:00
|
|
|
$button.off("click").on('click', function (event) {
|
|
|
|
event.preventDefault();
|
|
|
|
setTimeout(function () {
|
|
|
|
self.parent.make_active($parent);
|
|
|
|
}, 0);
|
|
|
|
});
|
|
|
|
} else {
|
2013-09-20 08:49:52 +00:00
|
|
|
$button.addClass("hidden");
|
2013-09-19 12:58:54 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2013-09-10 10:21:40 +00:00
|
|
|
/*
|
2013-09-11 12:56:32 +00:00
|
|
|
* start
|
|
|
|
* This method is called after init and _readXMLData
|
2013-09-10 10:21:40 +00:00
|
|
|
*/
|
2013-09-11 12:56:32 +00:00
|
|
|
start: function () {
|
2013-09-12 17:06:52 +00:00
|
|
|
var self = this;
|
2013-11-29 16:00:45 +00:00
|
|
|
this.$overlay.on('click', '.oe_snippet_clone', _.bind(this.on_clone, this));
|
|
|
|
this.$overlay.on('click', '.oe_snippet_remove', _.bind(this.on_remove, this));
|
2013-09-13 09:35:18 +00:00
|
|
|
this._drag_and_drop();
|
2013-11-29 16:00:45 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
on_clone: function () {
|
|
|
|
var $clone = this.$target.clone(false);
|
|
|
|
this.$target.after($clone);
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
|
|
|
|
on_remove: function () {
|
|
|
|
this.onBlur();
|
|
|
|
var index = _.indexOf(this.parent.snippets, this.$target.get(0));
|
|
|
|
delete this.parent.snippets[index];
|
|
|
|
this.$target.remove();
|
|
|
|
this.$overlay.remove();
|
|
|
|
return false;
|
2013-09-10 10:21:40 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/*
|
2013-11-06 13:39:17 +00:00
|
|
|
* drop_and_build_snippet
|
2013-10-22 15:11:44 +00:00
|
|
|
* This method is called just after that a thumbnail is drag and dropped into a drop zone
|
2013-09-10 10:21:40 +00:00
|
|
|
* (after the insertion of this.$body, if this.$body exists)
|
|
|
|
*/
|
2013-11-06 13:39:17 +00:00
|
|
|
drop_and_build_snippet: function ($target) {
|
2013-09-09 15:32:51 +00:00
|
|
|
},
|
|
|
|
|
2013-09-06 15:04:08 +00:00
|
|
|
/* onFocus
|
2013-09-10 10:21:40 +00:00
|
|
|
* This method is called when the user click inside the snippet in the dom
|
2013-09-06 15:04:08 +00:00
|
|
|
*/
|
|
|
|
onFocus : function () {
|
2013-11-09 14:27:51 +00:00
|
|
|
this.$overlay.addClass('oe_active');
|
2013-09-06 15:04:08 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/* onFocus
|
2013-10-22 15:11:44 +00:00
|
|
|
* This method is called when the user click outside the snippet in the dom, after a focus
|
2013-09-06 15:04:08 +00:00
|
|
|
*/
|
|
|
|
onBlur : function () {
|
2013-09-11 12:56:32 +00:00
|
|
|
this.$overlay.removeClass('oe_active');
|
2013-09-30 15:32:50 +00:00
|
|
|
},
|
|
|
|
|
2013-10-15 15:43:51 +00:00
|
|
|
/* clean_for_save
|
|
|
|
* function called just before save vue
|
|
|
|
*/
|
|
|
|
clean_for_save: function () {
|
2013-12-04 11:54:24 +00:00
|
|
|
this.$target.removeAttr('contentEditable')
|
|
|
|
.find('*').removeAttr('contentEditable');
|
2014-01-15 15:25:26 +00:00
|
|
|
this.$target.removeAttr('attributeEditable')
|
|
|
|
.find('*').removeAttr('attributeEditable');
|
2013-10-15 15:43:51 +00:00
|
|
|
},
|
2013-09-06 15:04:08 +00:00
|
|
|
});
|
|
|
|
|
2013-09-06 16:00:06 +00:00
|
|
|
|
2013-09-12 11:51:30 +00:00
|
|
|
website.snippet.editorRegistry.resize = website.snippet.Editor.extend({
|
2013-09-11 12:56:32 +00:00
|
|
|
start: function () {
|
2013-09-09 13:28:00 +00:00
|
|
|
var self = this;
|
2013-09-11 12:56:32 +00:00
|
|
|
this._super();
|
2013-09-12 11:51:30 +00:00
|
|
|
var $box = $(openerp.qweb.render("website.snippets.resize"));
|
2013-09-11 12:56:32 +00:00
|
|
|
|
|
|
|
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();
|
2014-01-08 12:21:13 +00:00
|
|
|
if (!resize_values.size) $box.find(".oe_handle.size").remove();
|
2013-11-08 14:38:48 +00:00
|
|
|
|
2013-09-11 12:56:32 +00:00
|
|
|
this.$overlay.append($box.find(".oe_handles").html());
|
|
|
|
|
2014-01-15 12:59:23 +00:00
|
|
|
this.$overlay.find(".oe_handle:not(:has(.oe_handle_button)), .oe_handle .oe_handle_button").on('mousedown', function (event){
|
2014-01-08 12:21:13 +00:00
|
|
|
event.preventDefault();
|
2013-09-06 15:04:08 +00:00
|
|
|
|
2014-01-08 12:21:13 +00:00
|
|
|
var $handle = $(this);
|
2013-09-09 13:28:00 +00:00
|
|
|
|
2014-01-08 12:21:13 +00:00
|
|
|
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';
|
|
|
|
}
|
|
|
|
else if ($handle.hasClass('size')) {
|
|
|
|
compass = 'size';
|
|
|
|
XY = 'Y';
|
|
|
|
}
|
2013-09-09 13:28:00 +00:00
|
|
|
|
2014-01-08 12:21:13 +00:00
|
|
|
var resize = resize_values[compass];
|
|
|
|
if (!resize) return;
|
2013-09-09 13:28:00 +00:00
|
|
|
|
2014-01-08 12:21:13 +00:00
|
|
|
|
|
|
|
if (compass === 'size') {
|
|
|
|
var offset = self.$target.offset().top;
|
|
|
|
if (self.$target.css("background").match(/rgba\(0, 0, 0, 0\)/)) {
|
|
|
|
self.$target.addClass("resize_editor_busy");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
var xy = event['page'+XY];
|
2013-09-12 11:51:30 +00:00
|
|
|
var current = resize[2] || 0;
|
2013-09-11 12:56:32 +00:00
|
|
|
_.each(resize[0], function (val, key) {
|
|
|
|
if (self.$target.hasClass(val)) {
|
2013-09-12 15:17:18 +00:00
|
|
|
current = key;
|
2013-09-11 12:56:32 +00:00
|
|
|
}
|
|
|
|
});
|
2013-09-12 11:51:30 +00:00
|
|
|
var begin = current;
|
|
|
|
var beginClass = self.$target.attr("class");
|
2013-09-20 13:01:22 +00:00
|
|
|
var regClass = new RegExp("\\s*" + resize[0][begin].replace(/[-]*[0-9]+/, '[-]*[0-9]+'), 'g');
|
2014-01-08 12:21:13 +00:00
|
|
|
}
|
2013-09-11 12:56:32 +00:00
|
|
|
|
2014-01-08 12:21:13 +00:00
|
|
|
self.parent.editor_busy = true;
|
2013-09-11 12:56:32 +00:00
|
|
|
|
2014-01-08 12:21:13 +00:00
|
|
|
var cursor = $handle.css("cursor")+'-important';
|
|
|
|
var $body = $(document.body);
|
|
|
|
$body.addClass(cursor);
|
|
|
|
|
|
|
|
var body_mousemove = function (event){
|
|
|
|
event.preventDefault();
|
|
|
|
if (compass === 'size') {
|
|
|
|
var dy = event.pageY-offset;
|
2014-01-15 12:59:23 +00:00
|
|
|
dy = dy - dy%resize;
|
|
|
|
if (dy <= 0) dy = resize;
|
2014-01-08 12:21:13 +00:00
|
|
|
self.$target.css("height", dy+"px");
|
2014-01-15 12:59:23 +00:00
|
|
|
self.on_resize(compass, null, dy);
|
2014-01-08 12:21:13 +00:00
|
|
|
self.parent.cover_target(self.$overlay, self.$target);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var body_mouseup = function(){
|
|
|
|
$body.unbind('mousemove', body_mousemove);
|
|
|
|
$body.unbind('mouseup', body_mouseup);
|
|
|
|
$body.removeClass(cursor);
|
|
|
|
self.parent.editor_busy = false;
|
|
|
|
self.$target.removeClass("resize_editor_busy");
|
|
|
|
};
|
|
|
|
$body.mousemove(body_mousemove);
|
|
|
|
$body.mouseup(body_mouseup);
|
|
|
|
});
|
2013-09-09 13:28:00 +00:00
|
|
|
},
|
2013-09-11 12:56:32 +00:00
|
|
|
getSize: function () {
|
|
|
|
var grid = [0,4,8,16,32,48,64,92,128];
|
2013-09-12 11:51:30 +00:00
|
|
|
this.grid = {
|
2014-01-08 12:21:13 +00:00
|
|
|
// list of class (Array), grid (Array), default value (INT)
|
2013-09-17 11:00:43 +00:00
|
|
|
n: [_.map(grid, function (v) {return 'mt'+v;}), grid],
|
2014-01-08 12:21:13 +00:00
|
|
|
s: [_.map(grid, function (v) {return 'mb'+v;}), grid],
|
2014-01-15 12:59:23 +00:00
|
|
|
// INT if the user can resize the snippet (resizing per INT px)
|
|
|
|
size: null
|
2013-09-11 12:56:32 +00:00
|
|
|
};
|
2013-09-12 11:51:30 +00:00
|
|
|
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) {
|
|
|
|
|
2013-09-11 12:56:32 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2013-09-12 11:51:30 +00:00
|
|
|
website.snippet.editorRegistry.colmd = website.snippet.editorRegistry.resize.extend({
|
2013-09-11 12:56:32 +00:00
|
|
|
getSize: function () {
|
2013-09-12 11:51:30 +00:00
|
|
|
this.grid = this._super();
|
2013-09-11 12:56:32 +00:00
|
|
|
var width = this.$target.parents(".row:first").first().outerWidth();
|
2013-09-12 11:51:30 +00:00
|
|
|
|
2013-09-12 15:17:18 +00:00
|
|
|
var grid = [1,2,3,4,5,6,7,8,9,10,11,12];
|
2013-09-12 11:51:30 +00:00
|
|
|
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];
|
2013-09-17 13:31:44 +00:00
|
|
|
this.grid.w = [_.map(grid, function (v) {return 'col-md-offset-'+v;}), _.map(grid, function (v) {return width/12*v;}), 12];
|
2013-09-12 11:51:30 +00:00
|
|
|
|
|
|
|
return this.grid;
|
|
|
|
},
|
2013-09-23 09:51:22 +00:00
|
|
|
_drag_and_drop_after_insert_dropzone: function(){
|
|
|
|
var self = this;
|
2013-10-21 15:39:18 +00:00
|
|
|
var $zones = $(".row:has(> .oe_drop_zone)").each(function () {
|
|
|
|
var $row = $(this);
|
|
|
|
var width = $row.innerWidth();
|
|
|
|
var pos = 0;
|
|
|
|
while (width > pos + self.size.width) {
|
|
|
|
var $last = $row.find("> .oe_drop_zone:last");
|
|
|
|
$last.each(function () {
|
|
|
|
pos = $(this).position().left;
|
|
|
|
});
|
|
|
|
if (width > pos + self.size.width) {
|
|
|
|
$row.append("<div class='col-md-1 oe_drop_to_remove'/>");
|
|
|
|
var $add_drop = $last.clone();
|
|
|
|
$row.append($add_drop);
|
|
|
|
self._drag_and_drop_active_drop_zone($add_drop);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2013-09-23 09:51:22 +00:00
|
|
|
},
|
|
|
|
_drag_and_drop_start: function () {
|
|
|
|
this._super();
|
|
|
|
this.$target.attr("class",this.$target.attr("class").replace(/\s*(col-lg-offset-|col-md-offset-)([0-9-]+)/g, ''));
|
|
|
|
},
|
|
|
|
_drag_and_drop_stop: function () {
|
|
|
|
this.$target.addClass("col-md-offset-" + this.$target.prevAll(".oe_drop_to_remove").length);
|
|
|
|
this._super();
|
|
|
|
},
|
2013-12-10 12:54:15 +00:00
|
|
|
hide_remove_button: function() {
|
2013-12-11 16:23:05 +00:00
|
|
|
this.$overlay.find('.oe_snippet_remove').toggleClass("hidden",
|
|
|
|
!this.$target.siblings().length && this.$target.parents("[data-snippet-id]:first").find("[data-snippet-id='colmd']").length > 1);
|
2013-12-10 12:54:15 +00:00
|
|
|
},
|
2013-11-29 16:00:45 +00:00
|
|
|
onFocus : function () {
|
|
|
|
this._super();
|
2013-12-10 12:54:15 +00:00
|
|
|
this.hide_remove_button();
|
2013-11-29 16:00:45 +00:00
|
|
|
},
|
|
|
|
on_clone: function () {
|
|
|
|
var $clone = this.$target.clone(false);
|
2013-12-04 11:54:24 +00:00
|
|
|
var _class = $clone.attr("class").replace(/\s*(col-lg-offset-|col-md-offset-)([0-9-]+)/g, '');
|
2013-11-29 16:00:45 +00:00
|
|
|
_class += ' col-md-1';
|
|
|
|
$clone.attr("class", _class);
|
|
|
|
this.$target.after($clone);
|
2013-12-10 12:54:15 +00:00
|
|
|
this.hide_remove_button();
|
2013-11-29 16:00:45 +00:00
|
|
|
return false;
|
|
|
|
},
|
|
|
|
on_remove: function () {
|
2013-12-11 16:23:05 +00:00
|
|
|
if (!this.$target.siblings().length) {
|
|
|
|
var $parent = this.$target.parents("[data-snippet-id]:first");
|
|
|
|
if($parent.find("[data-snippet-id='colmd']").length > 1) {
|
|
|
|
return false;
|
|
|
|
} else {
|
2014-01-22 09:46:54 +00:00
|
|
|
if (!$parent.data("snippet-editor")) {
|
|
|
|
this.parent.create_overlay($parent);
|
|
|
|
}
|
2013-12-11 16:23:05 +00:00
|
|
|
$parent.data("snippet-editor").on_remove();
|
|
|
|
}
|
2013-11-29 16:00:45 +00:00
|
|
|
}
|
2013-12-10 12:54:15 +00:00
|
|
|
this._super();
|
|
|
|
this.hide_remove_button();
|
|
|
|
return false;
|
2013-11-29 16:00:45 +00:00
|
|
|
},
|
2013-09-12 11:51:30 +00:00
|
|
|
on_resize: function (compass, beginClass, current) {
|
2013-09-12 15:17:18 +00:00
|
|
|
if (compass !== 'w')
|
2013-09-12 11:51:30 +00:00
|
|
|
return;
|
|
|
|
|
2013-10-22 15:11:44 +00:00
|
|
|
// don't change the right border position when we change the offset (replace col size)
|
2013-09-12 15:17:18 +00:00
|
|
|
var beginCol = Number(beginClass.match(/col-md-([0-9]+)|$/)[1] || 0);
|
2013-09-17 13:31:44 +00:00
|
|
|
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);
|
2013-09-20 14:49:22 +00:00
|
|
|
if (offset < 0) {
|
|
|
|
offset = 0;
|
|
|
|
}
|
|
|
|
var colSize = beginCol - (offset - beginOffset);
|
|
|
|
if (colSize <= 0) {
|
|
|
|
colSize = 1;
|
|
|
|
offset = beginOffset + beginCol - 1;
|
|
|
|
}
|
2013-09-17 13:31:44 +00:00
|
|
|
this.$target.attr("class",this.$target.attr("class").replace(/\s*(col-lg-offset-|col-md-offset-|col-md-)([0-9-]+)/g, ''));
|
2013-09-12 11:51:30 +00:00
|
|
|
|
2013-09-12 15:17:18 +00:00
|
|
|
this.$target.addClass('col-md-' + (colSize > 12 ? 12 : colSize));
|
|
|
|
if (offset > 0) {
|
2013-09-17 13:31:44 +00:00
|
|
|
this.$target.addClass('col-md-offset-' + offset);
|
2013-09-12 11:51:30 +00:00
|
|
|
}
|
2013-09-09 14:49:40 +00:00
|
|
|
},
|
2013-09-09 13:28:00 +00:00
|
|
|
});
|
2013-12-06 13:32:39 +00:00
|
|
|
|
2013-11-29 16:00:45 +00:00
|
|
|
website.snippet.editorRegistry.slider = website.snippet.editorRegistry.resize.extend({
|
2013-11-06 13:39:17 +00:00
|
|
|
drop_and_build_snippet: function() {
|
2013-12-04 13:07:40 +00:00
|
|
|
var id = $(".carousel").length;
|
2013-11-25 09:30:56 +00:00
|
|
|
this.id = "myCarousel" + id;
|
|
|
|
this.$target.attr("id", this.id);
|
2013-10-07 11:36:20 +00:00
|
|
|
this.$target.find(".carousel-control").attr("href", "#myCarousel" + id);
|
2013-11-29 16:00:45 +00:00
|
|
|
this.$target.find("[data-target]").attr("data-target", "#myCarousel" + id);
|
2013-10-30 10:30:48 +00:00
|
|
|
|
2013-10-07 11:36:20 +00:00
|
|
|
this.rebind_event();
|
|
|
|
},
|
2013-11-29 16:00:45 +00:00
|
|
|
// rebind event to active carousel on edit mode
|
|
|
|
rebind_event: function () {
|
|
|
|
var self = this;
|
|
|
|
this.$target.find('.carousel-indicators [data-target]').off('click').on('click', function () {
|
|
|
|
self.$target.carousel(+$(this).data('slide-to')); });
|
|
|
|
|
2013-12-03 09:26:41 +00:00
|
|
|
this.$target.attr('contentEditable', 'false');
|
2013-12-10 12:54:15 +00:00
|
|
|
this.$target.find('.oe_structure, .content>.row').attr('contentEditable', 'true');
|
2013-11-29 16:00:45 +00:00
|
|
|
|
2013-10-07 11:36:20 +00:00
|
|
|
this.$target.carousel('pause');
|
|
|
|
},
|
2013-10-15 15:43:51 +00:00
|
|
|
clean_for_save: function () {
|
|
|
|
this._super();
|
2013-10-16 07:02:37 +00:00
|
|
|
this.$target.find(".item").removeClass("next prev left right");
|
2013-10-15 15:43:51 +00:00
|
|
|
if(!this.$target.find(".item.active").length) {
|
|
|
|
this.$target.find(".item:first").addClass("active");
|
|
|
|
}
|
2013-11-29 16:00:45 +00:00
|
|
|
},
|
|
|
|
onFocus: function () {
|
|
|
|
this._super();
|
|
|
|
this.$target.carousel('pause');
|
|
|
|
},
|
|
|
|
onBlur: function () {
|
|
|
|
this._super();
|
|
|
|
this.$target.carousel('cycle');
|
2013-10-15 15:43:51 +00:00
|
|
|
},
|
2013-09-06 15:04:08 +00:00
|
|
|
start : function () {
|
2013-12-04 11:54:24 +00:00
|
|
|
var self = this;
|
2013-09-11 13:37:25 +00:00
|
|
|
this._super();
|
2013-10-07 11:36:20 +00:00
|
|
|
this.id = this.$target.attr("id");
|
|
|
|
this.$inner = this.$target.find('.carousel-inner');
|
|
|
|
this.$indicators = this.$target.find('.carousel-indicators');
|
|
|
|
|
2013-12-04 11:54:24 +00:00
|
|
|
this.$editor.find(".js_add").on('click', function () {self.on_add_slide(); return false;});
|
|
|
|
this.$editor.find(".js_remove").on('click', function () {self.on_remove_slide(); return false;});
|
2013-09-13 11:44:56 +00:00
|
|
|
|
2013-11-21 12:34:39 +00:00
|
|
|
this.rebind_event();
|
2013-11-29 16:00:45 +00:00
|
|
|
},
|
|
|
|
on_add_slide: function () {
|
|
|
|
var self = this;
|
|
|
|
var cycle = this.$inner.find('.item').length;
|
|
|
|
var $active = this.$inner.find('.item.active, .item.prev, .item.next').first();
|
|
|
|
var index = $active.index();
|
|
|
|
this.$target.find('.carousel-control, .carousel-indicators').removeClass("hidden");
|
|
|
|
this.$indicators.append('<li data-target="#' + this.id + '" data-slide-to="' + cycle + '"></li>');
|
|
|
|
|
|
|
|
var $clone = this.$el.find(".item.active").clone();
|
|
|
|
|
|
|
|
// insert
|
|
|
|
$clone.removeClass('active').insertAfter($active);
|
|
|
|
setTimeout(function() {
|
|
|
|
self.$target.carousel().carousel(++index);
|
|
|
|
self.rebind_event();
|
|
|
|
},0);
|
|
|
|
return $clone;
|
|
|
|
},
|
|
|
|
on_remove_slide: function () {
|
|
|
|
if (this.remove_process) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var self = this;
|
|
|
|
var new_index = 0;
|
|
|
|
var cycle = this.$inner.find('.item').length - 1;
|
|
|
|
var index = this.$inner.find('.item.active').index();
|
2013-12-06 13:32:39 +00:00
|
|
|
|
2013-11-29 16:00:45 +00:00
|
|
|
if (cycle > 0) {
|
|
|
|
this.remove_process = true;
|
|
|
|
var $el = this.$inner.find('.item.active');
|
|
|
|
self.$target.on('slid.bs.carousel', function (event) {
|
|
|
|
$el.remove();
|
|
|
|
self.$indicators.find("li:last").remove();
|
|
|
|
self.$target.off('slid.bs.carousel');
|
|
|
|
self.rebind_event();
|
|
|
|
self.remove_process = false;
|
|
|
|
if (cycle == 1) {
|
|
|
|
self.on_remove_slide(event);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
setTimeout(function () {
|
|
|
|
self.$target.carousel( index > 0 ? --index : cycle );
|
|
|
|
}, 500);
|
|
|
|
} else {
|
|
|
|
this.$target.find('.carousel-control, .carousel-indicators').addClass("hidden");
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
website.snippet.editorRegistry.carousel = website.snippet.editorRegistry.slider.extend({
|
2014-01-15 12:59:23 +00:00
|
|
|
getSize: function () {
|
|
|
|
this.grid = this._super();
|
|
|
|
this.grid.size = 8;
|
|
|
|
return this.grid;
|
|
|
|
},
|
2013-11-29 16:00:45 +00:00
|
|
|
clean_for_save: function () {
|
|
|
|
this._super();
|
|
|
|
this.$target.css("background-image", "");
|
|
|
|
this.$target.removeClass(this._class);
|
|
|
|
},
|
2014-01-08 12:21:13 +00:00
|
|
|
load_style_options : function () {
|
|
|
|
this._super();
|
|
|
|
$(".snippet-style-size li[data-class='']").remove();
|
|
|
|
},
|
2013-11-29 16:00:45 +00:00
|
|
|
start : function () {
|
|
|
|
var self = this;
|
|
|
|
this._super();
|
2013-10-07 11:36:20 +00:00
|
|
|
|
2013-11-21 12:34:39 +00:00
|
|
|
// set background and prepare to clean for save
|
|
|
|
var add_class = function (c){
|
|
|
|
if (c) self._class = (self._class || "").replace(new RegExp("[ ]+" + c.replace(" ", "|[ ]+")), '') + ' ' + c;
|
|
|
|
return self._class || "";
|
|
|
|
};
|
|
|
|
this.$target.on('snippet-style-change snippet-style-preview', function (event, style, np) {
|
|
|
|
var $active = self.$target.find(".item.active");
|
|
|
|
if (style['snippet-style-id'] === "size") return;
|
|
|
|
if (style['snippet-style-id'] === "background") {
|
|
|
|
$active.css("background-image", self.$target.css("background-image"));
|
|
|
|
}
|
|
|
|
if (np.$prev) {
|
|
|
|
$active.removeClass(np.$prev.data("class"));
|
|
|
|
}
|
|
|
|
if (np.$next) {
|
|
|
|
$active.addClass(np.$next.data("class"));
|
|
|
|
add_class(np.$next.data("class"));
|
|
|
|
}
|
2013-10-15 09:50:58 +00:00
|
|
|
});
|
2013-11-21 12:34:39 +00:00
|
|
|
this.$target.on('slid', function () { // slide.bs.carousel
|
|
|
|
var $active = self.$target.find(".item.active");
|
|
|
|
self.$target
|
|
|
|
.css("background-image", $active.css("background-image"))
|
|
|
|
.removeClass(add_class($active.attr("class")))
|
|
|
|
.addClass($active.attr("class"))
|
|
|
|
.trigger("snippet-style-reset");
|
2013-10-15 09:50:58 +00:00
|
|
|
|
2013-11-21 12:34:39 +00:00
|
|
|
self.$target.carousel();
|
|
|
|
});
|
|
|
|
this.$target.trigger('slid');
|
2013-10-07 11:36:20 +00:00
|
|
|
},
|
2013-11-29 16:00:45 +00:00
|
|
|
on_add_slide: function () {
|
|
|
|
var $clone = this._super();
|
2013-11-21 12:34:39 +00:00
|
|
|
|
|
|
|
// choose an other background
|
|
|
|
var $styles = this.$target.data("snippet-style-ids").background.$el.find("li[data-class]:not(.oe_custom_bg)");
|
|
|
|
var styles_index = $styles.index($styles.filter(".active")[0]);
|
|
|
|
var $select = $($styles[styles_index >= $styles.length-1 ? 0 : styles_index+1]);
|
|
|
|
$clone.css("background-image", $select.data("src") ? "url('"+ $select.data("src") +"')" : "");
|
|
|
|
$clone.addClass($select.data("class") || "");
|
|
|
|
|
2013-11-29 16:00:45 +00:00
|
|
|
return $clone;
|
2013-09-17 11:00:43 +00:00
|
|
|
},
|
2013-11-29 16:00:45 +00:00
|
|
|
// rebind event to active carousel on edit mode
|
|
|
|
rebind_event: function () {
|
2013-10-07 11:36:20 +00:00
|
|
|
var self = this;
|
2013-11-29 16:00:45 +00:00
|
|
|
this.$target.find('.carousel-control').off('click').on('click', function () {
|
|
|
|
self.$target.carousel( $(this).data('slide')); });
|
2013-12-03 09:26:41 +00:00
|
|
|
|
2013-12-10 12:08:57 +00:00
|
|
|
this.$target.find('.carousel-image, .content').attr('contentEditable', 'true');
|
2014-01-15 14:17:56 +00:00
|
|
|
this.$target.find('.carousel-image').attr('attributeEditable', 'true');
|
2013-11-29 16:00:45 +00:00
|
|
|
this._super();
|
2013-09-17 11:00:43 +00:00
|
|
|
},
|
2013-09-06 15:04:08 +00:00
|
|
|
});
|
|
|
|
|
2013-09-27 12:41:16 +00:00
|
|
|
website.snippet.editorRegistry.parallax = website.snippet.editorRegistry.resize.extend({
|
2014-01-15 12:59:23 +00:00
|
|
|
getSize: function () {
|
|
|
|
this.grid = this._super();
|
|
|
|
this.grid.size = 8;
|
|
|
|
return this.grid;
|
|
|
|
},
|
|
|
|
on_resize: function (compass, beginClass, current) {
|
|
|
|
this.$target.data("snippet-view").set_values();
|
|
|
|
},
|
2013-09-27 12:41:16 +00:00
|
|
|
start : function () {
|
2013-11-12 16:20:36 +00:00
|
|
|
var self = this;
|
2013-09-27 12:41:16 +00:00
|
|
|
this._super();
|
2013-11-21 12:34:39 +00:00
|
|
|
this.scroll();
|
|
|
|
this.$target.on('snippet-style-change snippet-style-preview', function () {
|
2013-11-12 16:20:36 +00:00
|
|
|
self.$target.data("snippet-view").set_values();
|
|
|
|
});
|
2013-12-04 11:54:24 +00:00
|
|
|
this.$target.attr('contentEditable', 'false');
|
|
|
|
this.$target.find('> div > .oe_structure').attr('contentEditable', 'true');
|
2013-09-27 12:41:16 +00:00
|
|
|
},
|
2013-11-08 15:00:25 +00:00
|
|
|
scroll: function () {
|
|
|
|
var self = this;
|
2013-10-18 14:53:13 +00:00
|
|
|
var $ul = this.$editor.find('ul[name="parallax-scroll"]');
|
|
|
|
var $li = $ul.find("li");
|
2013-11-08 15:00:25 +00:00
|
|
|
var speed = this.$target.data('scroll-background-ratio') || 0.6 ;
|
2013-10-18 14:53:13 +00:00
|
|
|
$ul.find('[data-value="' + speed + '"]').addClass('active');
|
|
|
|
$li.on('click', function (event) {
|
|
|
|
$li.removeClass("active");
|
|
|
|
$(this).addClass("active");
|
2013-10-22 15:11:44 +00:00
|
|
|
var speed = $(this).data('value');
|
2013-11-08 15:00:25 +00:00
|
|
|
self.$target.attr('data-scroll-background-ratio', speed);
|
|
|
|
self.$target.data("snippet-view").set_values();
|
2013-11-12 16:20:36 +00:00
|
|
|
return false;
|
2013-10-18 14:53:13 +00:00
|
|
|
});
|
2013-11-12 16:20:36 +00:00
|
|
|
this.$target.data("snippet-view").set_values();
|
2013-10-18 14:53:13 +00:00
|
|
|
},
|
2013-10-19 20:59:31 +00:00
|
|
|
clean_for_save: function () {
|
|
|
|
this._super();
|
2013-11-12 16:20:36 +00:00
|
|
|
this.$target.find(".parallax")
|
|
|
|
.css("background-position", '')
|
|
|
|
.removeAttr("data-scroll-background-offset");
|
2013-11-20 11:52:50 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2013-09-24 08:24:23 +00:00
|
|
|
/*
|
|
|
|
* data-snippet-id automatically setted
|
|
|
|
* Don't need to add data-snippet-id="..." into the views
|
|
|
|
*/
|
|
|
|
|
2013-11-29 16:00:45 +00:00
|
|
|
website.snippet.selector.push([".row > [class*='col-md-']", 'colmd']);
|
2013-09-17 11:00:43 +00:00
|
|
|
website.snippet.selector.push(['hr', 'hr']);
|
2013-11-29 16:00:45 +00:00
|
|
|
website.snippet.selector.push(['blockquote', 'quote']);
|
2013-09-17 11:00:43 +00:00
|
|
|
|
2013-09-14 22:13:01 +00:00
|
|
|
})();
|