1 openerp.web.view_editor = function(openerp) {
2 var _t = openerp.web._t;
3 var QWeb = openerp.web.qweb;
4 openerp.web.ViewEditor = openerp.web.Widget.extend({
5 init: function(parent, element_id, dataset, view, options) {
7 this.element_id = element_id
9 this.dataset = new openerp.web.DataSetSearch(this, 'ir.ui.view', null, null),
10 this.model = dataset.model;
11 this.xml_element_id = 0;
12 this.property = openerp.web.ViewEditor.property_widget;
13 this.one_object = false;
16 this.init_view_editor();
18 init_view_editor: function() {
21 name: _.str.sprintf("Manage Views (%s)", this.model),
22 context: this.session.user_context,
23 domain: [["model", "=", this.model]],
24 res_model: 'ir.ui.view',
25 views: [[false, 'list']],
26 type: 'ir.actions.act_window',
28 limit: this.dataset.limit || 80,
33 views_switcher: false,
34 action_buttons: false,
38 select_view_id: self.parent.fields_view.view_id
41 this.view_edit_dialog = new openerp.web.Dialog(this, {
42 title: _t(_.str.sprintf("Manage Views (%s)", this.model)),
45 {text: _t("Create"), click: function() { self.on_create_view(); }},
46 {text: _t("Edit"), click: function() { self.xml_element_id = 0; self.get_arch(); }},
47 {text: _t("Remove"), click: function() { self.do_delete_view(); }},
48 {text: _t("Close"), click: function() { self.view_edit_dialog.close(); }}
51 this.main_view_id = this.parent.fields_view.view_id;
52 this.action_manager = new openerp.web.ActionManager(this);
53 this.action_manager.appendTo(this.view_edit_dialog);
54 $.when(this.action_manager.do_action(action)).then(function() {
55 var viewmanager = self.action_manager.inner_viewmanager,
56 controller = viewmanager.views[viewmanager.active_view].controller;
57 controller.on_loaded.add_last(function(){
58 $(controller.groups).bind({
59 'selected': function(e, ids, records) {
60 self.main_view_id = ids[0];
66 on_create_view: function() {
68 this.create_view_dialog = new openerp.web.Dialog(this, {
69 title: _.str.sprintf(_t("Create a view (%s)"), self.model),
71 {text: _t("Save"), click: function () {
74 _.each(self.create_view_widget, function(widget) {
75 if (widget.is_invalid) {
79 if (widget.dirty && !widget.is_invalid) {
80 view_values[widget.name] = widget.get_value();
84 self.on_valid_create_view(self.create_view_widget);
86 $.when(self.do_save_view(view_values)).then(function() {
87 self.create_view_dialog.close();
88 var controller = self.action_manager.inner_viewmanager.views[self.action_manager.inner_viewmanager.active_view].controller;
89 controller.reload_content();
93 {text: _t("Cancel"), click: function () { self.create_view_dialog.close(); }}
96 this.create_view_dialog.start().open();
97 var view_widget = [{'name': 'view_name', 'string':'View Name', 'type': 'char', 'required': true, 'value' : this.model + '.custom_' + Math.round(Math.random() * 1000)},
98 {'name': 'view_type', 'string': 'View Type', 'type': 'selection', 'required': true, 'value': 'Form', 'selection': [['',''],['tree', 'Tree'],['form', 'Form'],['graph', 'Graph'],['calendar', 'Calender']]},
99 {'name': 'proirity', 'string': 'Priority', 'type': 'float', 'required': true, 'value':'16'}];
100 this.create_view_dialog.$element.append('<table id="create_view" style="width:400px" class="oe_forms"></table>');
101 this.create_view_widget = [];
102 _.each(view_widget, function(widget) {
103 var type_widget = new (self.property.get_any([widget.type])) (self.create_view_dialog, widget);
104 self.create_view_dialog.$element.find('table[id=create_view]').append('<tr><td width="100px" align="right">' + widget.string + ':</td>' + type_widget.render()+'</tr>');
107 value = widget.value;
108 type_widget.dirty = true;
111 type_widget.set_value(value)
112 self.create_view_widget.push(type_widget);
115 do_save_view: function(values) {
117 var field_dataset = new openerp.web.DataSetSearch(this, this.model, null, null);
118 var model_dataset = new openerp.web.DataSetSearch(this, 'ir.model', null, null);
119 var view_string = "", field_name = false, self = this;
120 field_dataset.call( 'fields_get', [], function(fields) {
121 _.each(['name', 'x_name'], function(value) {
122 if (_.include(_.keys(fields), value)) {
128 model_dataset.read_slice(['name','field_id'], {"domain": [['model','=',self.model]]}, function(records) {
129 if (records) {view_string = records[0].name;}
130 var arch = _.str.sprintf("<?xml version='1.0'?>\n<%s string='%s'>\n\t<field name='%s'/>\n</%s>", values.view_type, view_string, field_name, values.view_type);
131 var vals = {'model': self.model, 'name': values.view_name, 'priority': values.priority, 'type': values.view_type, 'arch': arch};
132 self.dataset.create(vals, function(suc) {
138 return def.promise();
140 on_valid_create_view: function(widgets) {
142 _.each(widgets, function(widget) {
143 if (widget.is_invalid) {
144 msg += "<li>" + widget.name + "</li>";
148 this.do_warn("The following fields are invalid :", msg);
150 add_node_name : function(node) {
151 if(node.tagName.toLowerCase() == "button" || node.tagName.toLowerCase() == "field"){
152 return (node.getAttribute('name'))?
153 _.str.sprintf( "<%s name='%s'>",node.tagName.toLowerCase(), node.getAttribute('name')):
154 _.str.sprintf( "<%s>",node.tagName.toLowerCase());
155 }else if(node.tagName.toLowerCase() == "group"){
156 return (node.getAttribute('string'))?
157 _.str.sprintf( "<%s>",node.getAttribute('string')):
158 _.str.sprintf( "<%s>",node.tagName.toLowerCase());
160 return (node.getAttribute('string'))?
161 _.str.sprintf( "<%s string='%s'>",node.tagName.toLowerCase(), node.getAttribute('string')):
162 _.str.sprintf( "<%s>",node.tagName.toLowerCase());
165 do_delete_view: function() {
166 if (confirm(_t("Do you really want to remove this view?"))) {
167 var controller = this.action_manager.inner_viewmanager.views[this.action_manager.inner_viewmanager.active_view].controller;
168 this.dataset.unlink([this.main_view_id]).then(function() {
169 controller.reload_content();
170 this.main_view_id = self.parent.fields_view.view_id;
174 create_View_Node: function(node){
176 'level': ($(node).parents()).length + 1,
177 'id': this.xml_element_id += 1,
179 'name': this.add_node_name(node),
182 ViewNode.att_list.push(node.tagName.toLowerCase());
183 _.each(node.attributes, function(att) {
184 ViewNode.att_list.push([att.nodeName, att.nodeValue]);
188 append_child_object: function(main_object, parent_id, child_obj_list) {
190 if (main_object.id == parent_id) {
191 main_object.child_id = child_obj_list;
194 _.each(main_object.child_id, function(child_object) {
195 self.append_child_object(child_object, parent_id, child_obj_list);
199 convert_arch_to_obj: function(xml_Node, main_object, parent_id) {
201 var child_obj_list = [];
202 _.each(xml_Node, function(element) {
203 child_obj_list.push(self.create_View_Node(element));
205 this.append_child_object(main_object, parent_id, child_obj_list);
206 var obj_xml_list = _.zip(xml_Node, child_obj_list);
207 _.each(obj_xml_list, function(node) {
208 var children = _.filter(node[0].childNodes, function(child) {
209 return child.nodeType == 1;
212 self.convert_arch_to_obj(children, main_object, node[1].id);
217 parse_xml: function(arch, view_id) {
220 'id': this.xml_element_id +=1,
222 'name': _.str.sprintf("<view view_id = %s>", view_id),
225 var xml_arch = QWeb.load_xml(arch);
226 return [this.convert_arch_to_obj(xml_arch.childNodes, main_object, this.xml_element_id)];
228 get_arch: function() {
230 var view_arch_list = [];
231 this.dataset.read_ids([parseInt(self.main_view_id)], ['arch', 'type'], function(arch) {
233 var arch_object = self.parse_xml(arch[0].arch, self.main_view_id);
234 self.main_view_type = arch[0].type == 'tree'? 'list': arch[0].type;
235 view_arch_list.push({"view_id": self.main_view_id, "arch": arch[0].arch});
236 self.dataset.read_slice([], {domain: [['inherit_id','=', parseInt(self.main_view_id)]]}, function(result) {
237 _.each(result, function(res) {
238 view_arch_list.push({"view_id": res.id, "arch": res.arch});
239 self.inherit_view(arch_object, res);
241 return self.edit_view({"main_object": arch_object,
242 "parent_child_id": self.parent_child_list(arch_object, []),
243 "arch": view_arch_list});
246 self.do_warn("Please select view in list :");
250 parent_child_list : function(one_object, parent_list) {
252 _.each(one_object , function(element) {
253 if (element.child_id.length != 0) {
254 parent_list.push({"key": element.id, "value": _.pluck(element.child_id, 'id')});
255 self.parent_child_list(element.child_id, parent_list);
260 inherit_view : function(arch_object, result) {
261 var self = this, xml_list = [], xml_arch = QWeb.load_xml(result.arch);
262 if (xml_arch.childNodes[0].tagName == "data") {
263 xml_list = _.filter(xml_arch.childNodes[0].childNodes, function(child) {
264 return child.nodeType == 1;
267 xml_list.push( xml_arch.childNodes[0]);
269 _.each(xml_list, function(xml) {
270 var expr_to_list = [], xpath_arch_object = self.parse_xml(QWeb.tools.xml_node_to_string(xml), result.id);
271 if (xml.tagName == "xpath") {
272 var part_expr = _.without(xml.getAttribute('expr').split("/"), "");
273 _.each(part_expr, function(part) {
274 expr_to_list.push(_.without($.trim(part.replace(/[^a-zA-Z 0-9 _]+/g,'!')).split("!"), ""));
277 var temp = _.reject(xpath_arch_object[0].child_id[0].att_list, function(list) {
278 return list instanceof Array? _.include(list, "position"): false;
280 expr_to_list = [_.flatten(temp)];
282 self.inherit_apply(expr_to_list, arch_object ,xpath_arch_object);
285 inherit_apply: function(expr_list ,arch_object ,xpath_arch_object) {
287 if (xpath_arch_object.length) {
288 var check = expr_list[0], obj = false;
289 switch (check.length) {
291 if (parseInt(check[1])) {
293 var temp_list = _.select(arch_object, function(element) {
294 return _.include(_.flatten(element.att_list), check[0]);
296 obj = arch_object[_.indexOf(arch_object, temp_list[parseInt(check[1]) - 1])];
298 //for notebook[last()]
299 obj = _.detect(arch_object, function(element) {
300 return _.include(_.flatten(element.att_list), check[0]);
305 //for field[@name='type']
306 obj = _.detect(arch_object, function(element){
307 if ((_.intersection(_.flatten(element.att_list), _.uniq(check))).length == check.length) {
314 var temp_list = _.select(arch_object, function(element) {
315 return _.include(_.flatten(element.att_list), check[0]);
317 if (temp_list.length != 0) {
318 expr_list.length == 1 ? obj = temp_list[0] : expr_list.shift();
324 if (expr_list.length) {
325 self.inherit_apply(expr_list, obj.child_id, xpath_arch_object);
327 self.increase_level(xpath_arch_object[0], obj.level + 1);
328 obj.child_id.push(xpath_arch_object[0]);
329 xpath_arch_object.pop();
332 _.each(arch_object, function(element) {
333 self.inherit_apply(expr_list, element.child_id, xpath_arch_object);
338 increase_level: function(val, level) {
341 _.each(val.child_id, function(val, key) {
342 self.increase_level(val, level + 1);
345 do_select_row: function(row_id) {
346 this.edit_xml_dialog.$element.find("tr[id^='viewedit-']").removeClass('ui-selected');
347 this.edit_xml_dialog.$element.find("tr[id=viewedit-" + row_id + "]").addClass('ui-selected');
349 do_parent_img_hide_show: function(img) {
350 if ($(img).attr('src') == '/web/static/src/img/collapse.gif') {
351 $(img).attr('src', '/web/static/src/img/expand.gif');
354 $(img).attr('src', '/web/static/src/img/collapse.gif');
355 this.on_collapse(img);
358 edit_view: function(one_object) {
360 this.one_object = one_object;
361 this.edit_xml_dialog = new openerp.web.Dialog(this, {
362 title: _.str.sprintf(_t("View Editor %d - %s"), self.main_view_id, self.model),
365 {text: _t("Preview"), click: function() {
367 context: self.session.user_context,
368 res_model: self.model,
369 views: [[self.main_view_id, self.main_view_type]],
370 type: 'ir.actions.act_window',
374 views_switcher: false,
375 action_buttons: false
378 var action_manager = new openerp.web.ActionManager(self);
379 action_manager.do_action(action);
381 {text: _t("Close"), click: function(){
382 self.edit_xml_dialog.close();
386 var no_property_att = [];
387 _.each(_PROPERTIES, function(val, key) {
388 if (! val.length) no_property_att.push(key);
390 this.edit_xml_dialog.$element.html(QWeb.render('view_editor', {'data': one_object['main_object'], 'no_properties': no_property_att}));
391 this.edit_xml_dialog.$element.find("tr[id^='viewedit-']").click(function() {
392 self.do_select_row(this.id.split('-')[1]);
394 this.edit_xml_dialog.$element.find("img[id^='parentimg-']").click(function() {
395 self.do_parent_img_hide_show(this);
397 this.edit_xml_dialog.$element.find("img[id^='side-']").click(function() {
398 self.on_select_img(this);
401 on_select_img: function(element_img) {
403 var side = $(element_img).closest("tr[id^='viewedit-']");
404 this.one_object.clicked_tr_id = parseInt((side.attr('id')).split('-')[1]);
405 this.one_object.clicked_tr_level = parseInt(side.attr('level'));
406 var img = side.find("img[id='parentimg-" + this.one_object.clicked_tr_id + "']").attr('src');
407 var view_id = 0, view_xml_id = 0, view_find = side;
408 //for view id finding
409 var min_level = this.one_object.clicked_tr_id;
410 if (($(side).find('a').text()).search("view_id") != -1) {
411 view_id = parseInt(($(view_find).find('a').text()).replace(/[^0-9]+/g, ''));
412 view_xml_id = (view_find.attr('id')).split('-')[1];
413 this.one_object.clicked_tr_id += 1;
414 this.one_object.clicked_tr_level += 1;
417 view_find = view_find.prev();
418 if (view_find.length == 0 ||
419 (self.edit_xml_dialog.$element.find(view_find).find('a').text()).search("view_id") != -1
420 && parseInt(view_find.attr('level')) < min_level ) {
421 view_id = parseInt(($(view_find).find('a').text()).replace(/[^0-9]+/g, ''));
422 view_xml_id = parseInt((view_find.attr('id')).split('-')[1]);
425 if (view_find.attr('level') < min_level) {
426 min_level = parseInt(view_find.attr('level'));
430 this.one_object.clicked_tr_view = [view_id, view_xml_id];
431 switch (element_img.id) {
433 self.do_node_add(side);
436 if (confirm(_t("Do you really want to remove this node?"))) {
437 self.do_save_update_arch("remove_node");
441 self.do_node_edit(side);
444 self.do_node_up(side, img);
447 self.do_node_down(side, img);
451 do_node_add: function(side){
453 var tr = $(side).find('a').text();
454 var parent_tr = ($(side).prevAll("tr[level=" + String(this.one_object.clicked_tr_level - 1) + "]"))[0];
455 var field_dataset = new openerp.web.DataSetSearch(this, this.model, null, null);
456 parent_tr = $(parent_tr).find('a').text();
457 field_dataset.call( 'fields_get', [], function(result) {
458 var fields = _.keys(result);
459 fields.push(" "),fields.sort();
460 var property_to_check = [];
461 _.each([tr, parent_tr],function(element) {
462 property_to_check.push(
463 _.detect(_.keys(_CHILDREN),function(res){
464 return _.str.include(element, res);
467 self.on_add_node(property_to_check, fields);
470 do_node_edit: function(side) {
472 var result = self.get_object_by_id(this.one_object.clicked_tr_id, this.one_object['main_object'], []);
473 if (result.length && result[0] && result[0].att_list) {
474 var properties = _PROPERTIES[result[0].att_list[0]];
475 self.on_edit_node(properties);
478 do_node_down: function(cur_tr, img) {
479 var self = this, next_tr, last_tr, tr_to_move = [];
480 tr_to_move.push(cur_tr);
483 next_tr = cur_tr.next();
484 if ( parseInt(next_tr.attr('level')) <= this.one_object.clicked_tr_level || next_tr.length == 0) {
488 tr_to_move.push(next_tr);
493 last_tr = cur_tr.next();
495 if ((self.edit_xml_dialog.$element.find(last_tr).find('a').text()).search("view_id") != -1) {
498 if (last_tr.length != 0 && parseInt(last_tr.attr('level')) == this.one_object.clicked_tr_level) {
499 var last_tr_id = (last_tr.attr('id')).split('-')[1];
500 img = last_tr.find("img[id='parentimg-" + last_tr_id + "']").attr('src');
502 self.edit_xml_dialog.$element.find("img[id='parentimg-" + last_tr_id + "']").
503 attr('src', '/web/static/src/img/expand.gif');
505 var next_tr = last_tr.next();
506 if (next_tr.attr('level') <= this.one_object.clicked_tr_level || next_tr.length == 0) break;
511 tr_to_move.reverse();
512 _.each(tr_to_move, function(rec) {
513 $(last_tr).after(rec);
515 self.do_save_update_arch("down");
518 do_node_up: function(cur_tr, img) {
519 var self = this, side = cur_tr, tr_to_move = [];
520 tr_to_move.push(side);
522 var prev_tr = cur_tr.prev();
523 if (this.one_object.clicked_tr_level >= parseInt(prev_tr.attr('level')) || prev_tr.length == 0) {
530 self.edit_xml_dialog.$element.find("img[id='parentimg-" + this.one_object.clicked_tr_id + "']").
531 attr('src', '/web/static/src/img/expand.gif');
533 next_tr = side.next();
534 if (parseInt(next_tr.attr('level')) <= this.one_object.clicked_tr_level || next_tr.length == 0) {
538 tr_to_move.push(next_tr);
543 if (last_tr.length != 0 && parseInt(last_tr.attr('level')) == this.one_object.clicked_tr_level &&
544 (self.edit_xml_dialog.$element.find(last_tr).find('a').text()).search("view_id") == -1) {
545 _.each(tr_to_move, function(rec) {
546 $(last_tr).before(rec);
548 self.do_save_update_arch("up");
551 do_save_update_arch: function(move_direct, update_values) {
553 var arch = _.detect(self.one_object['arch'], function(element)
554 {return element.view_id == self.one_object.clicked_tr_view[0]});
555 var obj = self.get_object_by_id(this.one_object.clicked_tr_view[1],this.one_object['main_object'], []);
556 //for finding xpath tag from inherit view
557 var xml_arch = QWeb.load_xml(arch.arch);
558 if (xml_arch.childNodes[0].tagName == "data") {
559 var check_list = _.flatten(obj[0].child_id[0].att_list);
560 var children = _.filter(xml_arch.childNodes[0].childNodes, function (child) {
561 return child.nodeType == 1;
563 arch.arch = _.detect(children, function(xml_child) {
564 var temp_obj = self.create_View_Node(xml_child),
565 insert = _.intersection(_.flatten(temp_obj.att_list),_.uniq(check_list));
566 if (insert.length == check_list.length ) {return xml_child;}
569 arch_to_pass = _.filter($(arch.arch), function (child) {
570 return child.nodeType == 1;
572 return self.do_save_xml(arch_to_pass[0], obj[0].child_id[0],[], move_direct, update_values,arch);
574 get_object_by_id: function(id, one_object, result) {
576 if (result.length == 0 ) {
577 var check = _.detect(one_object , function(obj) {
580 if (check) {result.push(check);};
581 _.each(one_object, function(obj) {
582 self.get_object_by_id(id,obj.child_id, result);
587 create_clone: function(clone, new_node_obj, position){
589 clone.find('a').text(new_node_obj.name);
590 ($(clone.find('a').parent()).siblings('td')).css( "padding-left", 20 * new_node_obj.level);
591 clone.attr("id", "viewedit-" + new_node_obj.id);
592 clone.attr("level", new_node_obj.level);
593 clone.find("img[id^='parentimg-']").remove();
594 clone.bind("click",function(){
595 self.do_select_row(this.id.split('-')[1]);
597 clone.find("img[id^='side-']").click(function() {
598 self.on_select_img(this);
602 do_save_xml: function(arch1, obj, child_list, move_direct, update_values, arch){
603 var self = this, children_list = $(arch1).children(), list_obj_xml = _.zip(children_list, obj.child_id);
604 if (this.one_object.clicked_tr_id) {
605 if (obj.id == this.one_object.clicked_tr_id) {
606 var parent = false, index = _.indexOf(child_list, obj);
607 if (move_direct == "down") {
608 var next = $(arch1).next();
609 $(next).after(arch1);
610 var re_insert_obj = child_list.splice(index, 1);
611 child_list.splice(index+1, 0, re_insert_obj[0]);
612 parent = $(arch1).parents();
613 } else if (move_direct == "up") {
614 var prev = $(arch1).prev();
615 $(prev).before(arch1);
616 var re_insert_obj = child_list.splice(index, 1);
617 child_list.splice(index-1, 0, re_insert_obj[0]);
618 parent = $(arch1).parents();
619 } else if (move_direct == "update_node") {
620 _.each(update_values, function(val){
621 if (val[1]) $(arch1)[0].setAttribute(val[0], val[1]);
622 else $(arch1)[0].removeAttribute(val[0]);
624 var new_obj = self.create_View_Node(arch1);
625 new_obj.id = obj.id,new_obj.child_id = obj.child_id;
626 self.edit_xml_dialog.$element.
627 find("tr[id='viewedit-"+this.one_object.clicked_tr_id+"']").
628 find('a').text(new_obj.name);
629 child_list.splice(index, 1, new_obj);
630 parent = $(arch1).parents();
631 } else if(move_direct == "add_node") {
632 var tr_click = self.edit_xml_dialog.$element.find("tr[id='viewedit-"+self.one_object.clicked_tr_id+"']"),
633 temp_xml = QWeb.load_xml(update_values[0]),
634 object_xml = self.create_View_Node(temp_xml.childNodes[0]);
635 (update_values[1] == "Inside")? object_xml.level = obj.level + 1:object_xml.level = obj.level;
636 var clone = self.create_clone(tr_click.clone(),object_xml),
637 after_append = _.detect(self.one_object['parent_child_id'],function(ele){
638 return self.one_object.clicked_tr_id == ele.key;
640 after_append = (after_append)?_.last(after_append.value):self.one_object.clicked_tr_id;
641 switch (update_values[1]) {
643 self.edit_xml_dialog.$element.
644 find("tr[id='viewedit-"+after_append+"']").after(clone);
645 $(arch1).after(update_values[0]);
646 child_list.splice(index + 1, 0, object_xml);
649 tr_click.before(clone);
650 $(arch1).before(update_values[0]);
651 child_list.splice(index - 1, 0, object_xml);
654 if (tr_click.find("img[id^='parentimg-']").length == 0) {
655 ($(tr_click.find('a').parent()).siblings('td'))
656 .append($('<img width="16" height="16"></img>').attr('src', '/web/static/src/img/collapse.gif').
657 attr('id','parentimg-'+ self.one_object.clicked_tr_id).click(function(){
658 self.do_parent_img_hide_show(this);
661 $(arch1).append(update_values[0]);
662 self.edit_xml_dialog.$element.
663 find("tr[id='viewedit-"+after_append+"']").after(clone);
664 obj.child_id.push(object_xml);
667 self.edit_xml_dialog.$element.
668 find("tr[id='viewedit-" + object_xml.id + "']").removeClass('ui-selected');
669 parent = $(arch1).parents();
670 } else if (move_direct == "remove_node") {
671 parent = $(arch1).parents();
672 if (parent.length == 0 || (parent[0].tagName.toLowerCase() == "data")) {
673 self.one_object.clicked_tr_id = self.one_object.clicked_tr_id -1;
674 self.one_object.clicked_tr_level = self.one_object.clicked_tr_level - 1;
675 (parent.length == 0)?parent.push("remove_view"):false;
678 child_list.splice(index,1);
679 var cur_tr = self.edit_xml_dialog.$element.
680 find("tr[id='viewedit-" + self.one_object.clicked_tr_id + "']");
681 _.each(self.get_list_tr(cur_tr,self.one_object.clicked_tr_level), function(tr_element){
685 var parent_img = _.detect(self.one_object['parent_child_id'],function(element){
686 return _.include(element.value, self.one_object.clicked_tr_id);
688 if(parent_img.value.length == 1){
689 self.edit_xml_dialog.$element.
690 find("tr[id='viewedit-"+parent_img.key+"']").
691 find("img[id^='parentimg-']").remove();
693 self.one_object['parent_child_id'] = self.parent_child_list(self.one_object['main_object'],[]);
695 var convert_to_utf = (parent.length != 0)? parent[parent.length-1]: arch1;
696 if (convert_to_utf != "remove_view") {
697 convert_to_utf = QWeb.tools.xml_node_to_string(convert_to_utf);
698 convert_to_utf = convert_to_utf.replace('xmlns="http://www.w3.org/1999/xhtml"', "");
699 convert_to_utf = '<?xml version="1.0"?>' + convert_to_utf;
700 arch.arch = convert_to_utf;
701 this.dataset.write(this.one_object.clicked_tr_view[0] ,{"arch":convert_to_utf}, function(r) {});
703 this.dataset.unlink([this.one_object.clicked_tr_view[0]],function(res) {});
705 if(move_direct == "add_node"){
706 self.add_node_dialog.close();
707 self.on_select_img(clone.find("img[id='side-edit']")[0]);
708 self.one_object['parent_child_id'] = self.parent_child_list(self.one_object['main_object'],[]);
711 if (obj.level <= this.one_object.clicked_tr_level) {
712 _.each(list_obj_xml, function(child_node) {
713 self.do_save_xml(child_node[0], child_node[1], obj.child_id, move_direct, update_values, arch);
718 on_expand: function(expand_img){
719 var level = parseInt($(expand_img).closest("tr[id^='viewedit-']").attr('level'));
720 var cur_tr = $(expand_img).closest("tr[id^='viewedit-']");
721 _.each(this.get_list_tr(cur_tr,level), function(tr_element){
725 get_list_tr: function(cur_tr,level){
728 var nxt_tr = cur_tr.next();
729 if (parseInt(nxt_tr.attr('level')) > level) {
731 tr_list.push(nxt_tr);
732 } else return tr_list;
735 on_collapse: function(collapse_img) {
736 var self = this, id = collapse_img.id.split('-')[1];
737 var datas = _.detect(self.one_object['parent_child_id'] , function(res) {
738 return res.key == id;
740 _.each(datas.value, function (rec) {
741 var tr = self.edit_xml_dialog.$element.find("tr[id='viewedit-" + rec + "']");
742 tr.find("img[id='parentimg-" + rec + "']").attr('src', '/web/static/src/img/expand.gif');
746 on_edit_node: function(properties){
748 this.edit_node_dialog = new openerp.web.Dialog(this,{
749 title: _t("Properties"),
752 {text: _t("Update"), click: function () {
753 var warn = false, update_values = [];
754 _.each(self.edit_widget, function(widget) {
755 if (widget.is_invalid) {
759 if (widget.dirty && !widget.is_invalid) {
760 update_values.push([widget.name, widget.get_value()]);
764 self.on_valid_create_view(self.edit_widget);
766 self.do_save_update_arch("update_node", update_values);
767 self.edit_node_dialog.close();
770 {text: _t("Cancel"), click: function () { self.edit_node_dialog.close(); }}
773 this.edit_node_dialog.start().open();
774 var _PROPERTIES_ATTRIBUTES = {
775 'name' : {'name':'name', 'string': 'Name', 'type': 'char'},
776 'string' : {'name':'string', 'string': 'String', 'type': 'char'},
777 'required' : {'name':'required', 'string': 'Required', 'type': 'boolean'},
778 'readonly' : {'name':'readonly', 'string': 'Readonly', 'type': 'boolean'},
779 'domain' : {'name':'domain', 'string': 'Domain', 'type': 'char'},
780 'context' : {'name':'context', 'string': 'Context', 'type': 'char'},
781 'limit' : {'name':'limit', 'string': 'Limit', 'type': 'float'},
782 'min_rows' : {'name':'min_rows', 'string': 'Minimum rows', 'type': 'float'},
783 'date_start' : {'name':'date_start', 'string': 'Start date', 'type': 'char'},
784 'date_delay' : {'name':'date_delay', 'string': 'Delay date', 'type': 'char'},
785 'day_length' : {'name':'day_length', 'string': 'Day length', 'type': 'char'},
786 'mode' : {'name':'mode', 'string': 'Mode', 'type': 'char'},
787 'align' : {'name':'align', 'string': 'Alignment ', 'type': 'selection', 'selection': [['', ''], ['0.0', 'Left'], ['0.5', 'Center'], ['1.0', 'Right']]},
788 'icon' : {'name':'icon', 'string': 'Icon', 'type': 'selection', 'selection': _ICONS},
789 'type' : {'name':'type', 'string': 'Type', 'type': 'selection', 'selection': [['', ''], ['action', 'Action'], ['object', 'Object'], ['workflow', 'Workflow'], ['server_action', 'Server Action']]},
790 'special' : {'name':'special', 'string': 'Special', 'type': 'selection', 'selection': [['',''],['save', 'Save Button'], ['cancel', 'Cancel Button'], ['open', 'Open Button']]},
791 'target' : {'name':'target', 'string': 'Target', 'type': 'selection', 'selection': [['', ''], ['new', 'New Window']]},
792 'confirm' : {'name':'confirm', 'string': 'Confirm', 'type': 'char'},
793 'style' : {'name':'style', 'string': 'Style', 'type': 'selection', 'selection':[["",""],["1", "1"],["1-1", "1-1"],["1-2", "1-2"],["2-1", "2-1"],["1-1-1", "1-1-1"]]},
794 'filename' : {'name':'filename', 'string': 'File Name', 'type': 'char'},
795 'width' : {'name':'width', 'string': 'Width', 'type': 'float'},
796 'height' : {'name':'height', 'string': 'Height', 'type': 'float'},
797 'attrs' : {'name':'attrs', 'string': 'Attrs', 'type': 'char'},
798 'col' : {'name':'col', 'string': 'col', 'type': 'float'},
799 'link' : {'name':'link', 'string': 'Link', 'type': 'char'},
800 'position' : {'name':'position', 'string': 'Position', 'type': 'selection', 'selection': [['',''],['after', 'After'],['before', 'Before'],['inside', 'Inside'],['replace', 'Replace']]},
801 'states' : {'name':'states', 'string': 'states', 'type': 'char'},
802 'eval' : {'name':'eval', 'string': 'Eval', 'type': 'char'},
803 'ref' : {'name':'ref', 'string': 'Ref', 'type': 'char'},
804 'on_change' : {'name':'on_change', 'string': 'On change', 'type': 'char'},
805 'nolabel' : {'name':'nolabel', 'string': 'No label', 'type': 'boolean'},
806 'completion' : {'name':'completion', 'string': 'Completion', 'type': 'boolean'},
807 'colspan' : {'name':'colspan', 'string': 'Colspan', 'type': 'float'},
808 'widget' : {'name':'widget', 'string': 'widget', 'type': 'selection'},
809 'colors' : {'name':'colors', 'string': 'Colors', 'type': 'char'},
810 'editable' : {'name':'editable', 'string': 'Editable', 'type': 'selection', 'selection': [["",""],["top","Top"],["bottom", "Bottom"]]},
811 'groups' : {'name':'groups', 'string': 'Groups', 'type': 'seleciton_multi'},
813 var arch_val = self.get_object_by_id(this.one_object.clicked_tr_id,this.one_object['main_object'], []);
814 this.edit_node_dialog.$element.append('<table id="rec_table" style="width:400px" class="oe_forms"></table>');
815 this.edit_widget = [];
816 self.ready = $.when(self.on_groups(properties)).then(function () {
817 _PROPERTIES_ATTRIBUTES['groups']['selection'] = self.groups;
818 var values = _.keys( openerp.web.form.widgets.map);
821 _PROPERTIES_ATTRIBUTES['widget']['selection'] = values;
822 var widgets = _.filter(_PROPERTIES_ATTRIBUTES, function(property){ return _.include(properties, property.name)})
823 _.each(widgets, function(widget) {
824 var type_widget = new (self.property.get_any([widget.type])) (self.edit_node_dialog, widget);
825 var value = _.detect(arch_val[0]['att_list'],function(res) {
826 return res instanceof Array? _.include(res, widget.name): false;
828 value = value instanceof Array ? value[1] : value;
829 self.edit_node_dialog.$element.find('table[id=rec_table]').append('<tr><td align="right">' + widget.string + ':</td>' + type_widget.render() + '</tr>');
831 type_widget.set_value(value);
832 self.edit_widget.push(type_widget);
837 on_groups: function(properties){
840 if (!_.include(properties, 'groups')) {
844 var group_ids = [], group_names = {}, groups = [];
845 var res_groups = new openerp.web.DataSetSearch(this,'res.groups', null, null),
846 model_data = new openerp.web.DataSetSearch(self,'ir.model.data', null, null);
849 .done(function(res_grp) {
850 _.each(res_grp,function(res){
852 group_names[key]=res.name;
853 group_ids.push(res.id);
856 .read_slice([],{domain:[['res_id', 'in', group_ids],['model','=','res.groups']]})
857 .done(function(model_grp) {
858 _.each(model_grp, function(res_group) {
859 groups.push([res_group.module + "." + res_group.name,group_names[res_group.res_id]]);
861 self.groups = groups;
865 return def.promise();
867 on_add_node: function(properties, fields){
869 var render_list = [{'name': 'node_type','selection': _.keys(_CHILDREN).sort(), 'value': 'field', 'string': 'Node Type','type': 'selection'},
870 {'name': 'field_value','selection': fields, 'value': false, 'string': '','type': 'selection'},
871 {'name': 'position','selection': ['After','Before','Inside'], 'value': false, 'string': 'Position','type': 'selection'}];
872 this.add_widget = [];
873 this.add_node_dialog = new openerp.web.Dialog(this,{
874 title: _t("Properties"),
877 {text: _t("Update"), click: function() {
878 var check_add_node = true, values = {};
879 _.each(self.add_widget, function(widget) {
880 values[widget.name] = widget.get_value() || false;
882 (values.position == "Inside")?
883 check_add_node =(_.include(_CHILDREN[properties[0]],values.node_type))?true:false:
884 check_add_node =(_.include(_CHILDREN[properties[1]],values.node_type))?true:false;
885 if(values.node_type == "field" && check_add_node )
886 {check_add_node = (values.field_value != " ")?true:false;
889 var tag = (values.node_type == "field")?
890 _.str.sprintf("<%s name='%s'> </%s>",values.node_type,values.field_value,values.node_type):
891 _.str.sprintf("<%s> </%s>",values.node_type,values.node_type);
892 self.do_save_update_arch("add_node", [tag, values.position]);
894 alert("Can't Update View");
897 {text: _t("Cancel"), click: function() { self.add_node_dialog.close(); }}
900 this.add_node_dialog.$element.append('<table id="rec_table" style="width:420px" class="oe_forms"><tbody><tr></tbody></table>');
901 var table_selector = self.add_node_dialog.$element.find('table[id=rec_table] tbody');
902 _.each(render_list, function(node) {
903 type_widget = new (self.property.get_any([node.type])) (self.add_node_dialog, node);
904 if (node.name == "position") {
905 table_selector.append('</tr><tr><td align="right" width="100px">' + node.string + '</td>' + type_widget.render() + '</tr>');
907 table_selector.append('<td align="right">' + node.string + '</td>' + type_widget.render() );
908 if (node.name == "field_value") {
909 table_selector.append('<td id="new_field" align="right" width="100px"> <button>New Field</button></td>');
913 type_widget.set_value(node.value);
914 self.add_widget.push(type_widget);
916 table_selector.find("td[id^=]").attr("width","100px");
917 self.add_node_dialog.$element.find('#new_field').click(function() {
918 model_data = new openerp.web.DataSetSearch(self,'ir.model', null, null);
919 model_data.read_slice([], {domain: [['model','=', self.model]]}, function(result) {
920 self.render_new_field(result[0].id);
924 render_new_field :function(id){
927 context: {'default_model_id': id, 'manual':true},
928 res_model: "ir.model.fields",
929 views: [[false, 'form']],
930 type: 'ir.actions.act_window',
933 action_buttons: true,
936 var action_manager = new openerp.web.ActionManager(self);
937 $.when(action_manager.do_action(action)).then(function() {
938 var controller = action_manager.dialog_viewmanager.views['form'].controller;
939 // TODO NIV: use page view
940 controller.do_set_readonly.add_last(function(){
941 action_manager.stop();
942 new_fields_name = new openerp.web.DataSetSearch(self,'ir.model.fields', null, null);
943 new_fields_name.read_ids([controller.datarecord.id], ['name'], function(result) {
944 self.add_node_dialog.$element.find('select[id=field_value]').append($("<option selected></option>").attr("value", result[0].name).text(result[0].name));
945 _.detect(self.add_widget,function(widget){
946 widget.name == "field_value"? widget.selection.push(result[0].name): false;
953 openerp.web.ViewEditor.Field = openerp.web.Class.extend({
954 init: function(view, widget) {
955 this.$element = view.$element;
957 this.name = widget.name;
958 this.selection = widget.selection || [];
959 this.required = widget.required || false;
960 this.string = widget.string || "";
961 this.type = widget.type;
962 this.is_invalid = false;
967 update_dom: function() {
968 this.$element.find("td[id=" + this.name + "]").toggleClass('invalid', this.is_invalid);
969 this.$element.find("td[id=" + this.name + "]").toggleClass('required', this.required);
971 on_ui_change: function() {
976 validate: function() {
977 this.is_invalid = false;
979 var value = openerp.web.parse_value(this.get_value(), this, '');
980 this.is_invalid = this.required && value === '';
982 this.is_invalid = true;
986 return _.str.sprintf("<td id = %s>%s</td>", this.name, QWeb.render(this.template, {widget: this}))
989 openerp.web.ViewEditor.FieldBoolean = openerp.web.ViewEditor.Field.extend({
990 template : "vieweditor_boolean",
994 this.$element.find("input[id="+ self.name+"]").change(function() {
998 set_value: function(value) {
1000 this.$element.find("input[id=" + this.name+ "]").attr('checked', true);
1003 get_value: function() {
1004 return this.$element.find("input[id=" + this.name + "]").is(':checked')? "1" : null;
1007 openerp.web.ViewEditor.FieldChar = openerp.web.ViewEditor.Field.extend({
1008 template : "vieweditor_char",
1009 start: function () {
1012 this.$element.find("input[id="+ this.name+"]").css('width','100%').change(function() {
1013 self.on_ui_change();
1016 set_value: function(value) {
1017 this.$element.find("input[id=" + this.name + "]").val(value);
1019 get_value: function() {
1020 return this.$element.find("input[id=" + this.name + "]").val();
1023 openerp.web.ViewEditor.FieldSelect = openerp.web.ViewEditor.Field.extend({
1024 template : "vieweditor_selection",
1025 start: function () {
1028 this.$element.find("select[id=" + this.name + "]").css('width', '100%').change(function() {
1029 self.on_ui_change();
1030 if (self.name == "node_type") {
1031 if (self.get_value() == "field") {
1032 self.$element.find('#new_field').show();
1033 self.$element.find("select[id=field_value]").show();
1035 self.$element.find('#new_field').hide();
1036 self.$element.find("select[id=field_value]").hide();
1041 set_value: function(value) {
1043 value = value === null? false: value;
1044 for (var i = 0, ii = this.selection.length; i < ii; i++) {
1045 if ((this.selection[i] instanceof Array && this.selection[i][1] === value) || this.selection[i] === value) index = i;
1047 this.$element.find("select[id=" + this.name + "]")[0].selectedIndex = index;
1049 get_value: function() {
1050 return this.$element.find("select[id=" + this.name + "]").val();
1053 openerp.web.ViewEditor.FieldSelectMulti = openerp.web.ViewEditor.FieldSelect.extend({
1054 start: function () {
1056 this.$element.find("select[id=" + this.name + "]").css('height', '100px').attr("multiple", true);
1058 set_value: function(value) {
1060 self.$element.find("#groups option").attr("selected",false);
1061 if (!value) return false;
1062 _.each(this.selection, function(item) {
1063 if (_.include(value.split(','), item[0])) {
1064 self.$element.find("select[id="+self.name+"] option[value='" + item[0] +"']").attr("selected",1)
1069 openerp.web.ViewEditor.FieldFloat = openerp.web.ViewEditor.FieldChar.extend({
1073 'field' : ['name', 'string', 'required', 'readonly', 'domain', 'context', 'nolabel', 'completion',
1074 'colspan', 'widget', 'eval', 'ref', 'on_change', 'attrs', 'groups'],
1075 'form' : ['string', 'col', 'link'],
1076 'notebook' : ['colspan', 'position', 'groups'],
1077 'page' : ['string', 'states', 'attrs', 'groups'],
1078 'group' : ['string', 'col', 'colspan', 'states', 'attrs', 'groups'],
1079 'image' : ['filename', 'width', 'height', 'groups'],
1080 'separator' : ['string', 'colspan', 'groups'],
1081 'label': ['string', 'align', 'colspan', 'groups'],
1082 'button': ['name', 'string', 'icon', 'type', 'states', 'readonly', 'special', 'target', 'confirm', 'context', 'attrs', 'colspan', 'groups'],
1086 'action' : ['name', 'string', 'colspan', 'groups'],
1087 'tree' : ['string', 'colors', 'editable', 'link', 'limit', 'min_rows'],
1088 'graph' : ['string', 'type'],
1089 'calendar' : ['string', 'date_start', 'date_stop', 'date_delay', 'day_length', 'color', 'mode'],
1092 'form': ['notebook', 'group', 'field', 'label', 'button','board', 'newline', 'separator'],
1095 'calendar': ['field'],
1096 'notebook': ['page'],
1097 'page': ['notebook', 'group', 'field', 'label', 'button', 'newline', 'separator'],
1098 'group': ['field', 'label', 'button', 'separator', 'newline'],
1099 'board': ['column'],
1101 'field': ['form', 'tree', 'graph'],
1107 var _ICONS = ['','STOCK_ABOUT', 'STOCK_ADD', 'STOCK_APPLY', 'STOCK_BOLD',
1108 'STOCK_CANCEL', 'STOCK_CDROM', 'STOCK_CLEAR', 'STOCK_CLOSE', 'STOCK_COLOR_PICKER',
1109 'STOCK_CONNECT', 'STOCK_CONVERT', 'STOCK_COPY', 'STOCK_CUT', 'STOCK_DELETE',
1110 'STOCK_DIALOG_AUTHENTICATION', 'STOCK_DIALOG_ERROR', 'STOCK_DIALOG_INFO',
1111 'STOCK_DIALOG_QUESTION', 'STOCK_DIALOG_WARNING', 'STOCK_DIRECTORY', 'STOCK_DISCONNECT',
1112 'STOCK_DND', 'STOCK_DND_MULTIPLE', 'STOCK_EDIT', 'STOCK_EXECUTE', 'STOCK_FILE',
1113 'STOCK_FIND', 'STOCK_FIND_AND_REPLACE', 'STOCK_FLOPPY', 'STOCK_GOTO_BOTTOM',
1114 'STOCK_GOTO_FIRST', 'STOCK_GOTO_LAST', 'STOCK_GOTO_TOP', 'STOCK_GO_BACK',
1115 'STOCK_GO_DOWN', 'STOCK_GO_FORWARD', 'STOCK_GO_UP', 'STOCK_HARDDISK',
1116 'STOCK_HELP', 'STOCK_HOME', 'STOCK_INDENT', 'STOCK_INDEX', 'STOCK_ITALIC',
1117 'STOCK_JUMP_TO', 'STOCK_JUSTIFY_CENTER', 'STOCK_JUSTIFY_FILL',
1118 'STOCK_JUSTIFY_LEFT', 'STOCK_JUSTIFY_RIGHT', 'STOCK_MEDIA_FORWARD',
1119 'STOCK_MEDIA_NEXT', 'STOCK_MEDIA_PAUSE', 'STOCK_MEDIA_PLAY',
1120 'STOCK_MEDIA_PREVIOUS', 'STOCK_MEDIA_RECORD', 'STOCK_MEDIA_REWIND',
1121 'STOCK_MEDIA_STOP', 'STOCK_MISSING_IMAGE', 'STOCK_NETWORK', 'STOCK_NEW',
1122 'STOCK_NO', 'STOCK_OK', 'STOCK_OPEN', 'STOCK_PASTE', 'STOCK_PREFERENCES',
1123 'STOCK_PRINT', 'STOCK_PRINT_PREVIEW', 'STOCK_PROPERTIES', 'STOCK_QUIT',
1124 'STOCK_REDO', 'STOCK_REFRESH', 'STOCK_REMOVE', 'STOCK_REVERT_TO_SAVED',
1125 'STOCK_SAVE', 'STOCK_SAVE_AS', 'STOCK_SELECT_COLOR', 'STOCK_SELECT_FONT',
1126 'STOCK_SORT_ASCENDING', 'STOCK_SORT_DESCENDING', 'STOCK_SPELL_CHECK',
1127 'STOCK_STOP', 'STOCK_STRIKETHROUGH', 'STOCK_UNDELETE', 'STOCK_UNDERLINE',
1128 'STOCK_UNDO', 'STOCK_UNINDENT', 'STOCK_YES', 'STOCK_ZOOM_100',
1129 'STOCK_ZOOM_FIT', 'STOCK_ZOOM_IN', 'STOCK_ZOOM_OUT',
1130 'terp-account', 'terp-crm', 'terp-mrp', 'terp-product', 'terp-purchase',
1131 'terp-sale', 'terp-tools', 'terp-administration', 'terp-hr', 'terp-partner',
1132 'terp-project', 'terp-report', 'terp-stock', 'terp-calendar', 'terp-graph'
1134 openerp.web.ViewEditor.property_widget = new openerp.web.Registry({
1135 'boolean' : 'openerp.web.ViewEditor.FieldBoolean',
1136 'seleciton_multi' : 'openerp.web.ViewEditor.FieldSelectMulti',
1137 'selection' : 'openerp.web.ViewEditor.FieldSelect',
1138 'char' : 'openerp.web.ViewEditor.FieldChar',
1139 'float' : 'openerp.web.ViewEditor.FieldFloat',