/*--------------------------------------------------------- * OpenERP base library *---------------------------------------------------------*/ openerp.base$views = function(openerp) { // process all kind of actions openerp.base.ActionManager = openerp.base.Controller.extend({ init: function(session, element_id) { this._super(session, element_id); this.action = null; this.viewmanager = null; }, /** * Process an action * Supported actions: act_window */ do_action: function(action) { // instantiate the right controllers by understanding the action this.action = action; if(action.type == "ir.actions.act_window") { this.viewmanager = new openerp.base.ViewManager(this.session,this.element_id); this.viewmanager.do_action_window(action); this.viewmanager.start(); } } }); // This will be ViewManager Abstract/Common openerp.base.ViewManager = openerp.base.Controller.extend({ init: function(session, element_id) { this._super(session, element_id); this.action = null; this.dataset = null; this.searchview_id = false; this.searchview = null; this.search_visible = true; // this.views = { "list": { "view_id":1234, "controller": instance} } this.views = {}; }, start: function() { }, on_mode_switch: function(view_type) { for (var i in this.views) { if(i == view_type) { this.views[i].controller.$element.show(); } else { this.views[i].controller.$element.hide(); } } }, do_action_window: function(action) { var self = this; var prefix_id = "#" + this.element_id; this.action = action; this.dataset = new openerp.base.DataSet(this.session, action.res_model); this.dataset.start(); this.$element.html(QWeb.render("ViewManager", {"prefix": this.element_id, views: action.views})); this.searchview_id = false; if(this.search_visible && action.search_view_id) { this.searchview_id = action.search_view_id[0]; this.searchview = new openerp.base.SearchView(this.session, this.element_id + "_search", this.dataset, this.searchview_id); this.searchview.start(); } for(var i = 0; i < action.views.length; i++) { var view_id, controller; view_id = action.views[i][0]; if(action.views[i][1] == "tree") { controller = new openerp.base.ListView(this.session, this.element_id + "_view_tree", this.dataset, view_id); controller.start(); this.views.tree = { view_id: view_id, controller: controller }; this.$element.find(prefix_id + "_button_tree").bind('click',function(){ self.on_mode_switch("tree"); }); } else if(action.views[i][1] == "form") { controller = new openerp.base.FormView(this.session, this.element_id + "_view_form", this.dataset, view_id); controller.start(); this.views.form = { view_id: view_id, controller: controller }; this.$element.find(prefix_id + "_button_form").bind('click',function(){ self.on_mode_switch("form"); }); } } // switch to the first one in sequence this.on_mode_switch("tree"); }, // create when root, also add to parent when o2m on_create: function() { }, on_remove: function() { }, on_edit: function() { } }); // Extends view manager openerp.base.ViewManagerRoot = openerp.base.Controller.extend({ }); // Extends view manager openerp.base.ViewManagerUsedAsAMany2One = openerp.base.Controller.extend({ }); /** * Management interface between views and the collection of selected OpenERP * records (represents the view's state?) */ openerp.base.DataSet = openerp.base.Controller.extend({ init: function(session, model) { this._super(session); this.model = model; this._fields = null; this._ids = []; this._active_ids = null; this._active_id_index = 0; this._sort = []; this._domain = []; this._context = {}; }, start: function() { // TODO: fields_view_get fields selection? this.rpc("/base/dataset/fields", {"model":this.model}, this.on_fields); }, on_fields: function(result) { this._fields = result._fields; this.on_ready(); }, /** * Fetch all the records selected by this DataSet, based on its domain * and context. * * Fires the on_ids event. * * @param {Number} [offset=0] The index from which selected records should be returned * @param {Number} [limit=null] The maximum number of records to return * @returns itself */ fetch: function (offset, limit) { offset = offset || 0; limit = limit || null; this.rpc('/base/dataset/find', { model: this.model, fields: this._fields, domain: this._domain, context: this._context, sort: this._sort, offset: offset, limit: limit }, _.bind(function (records) { var data_records = _.map( records, function (record) { return new openerp.base.DataRecord( this.session, this.model, this._fields, record); }, this); this.on_fetch(data_records, { offset: offset, limit: limit, domain: this._domain, context: this._context, sort: this._sort }); }, this)); return this; }, /** * @event * * Fires after the DataSet fetched the records matching its internal ids selection * * @param {Array} records An array of the DataRecord fetched * @param event The on_fetch event object * @param {Number} event.offset the offset with which the original DataSet#fetch call was performed * @param {Number} event.limit the limit set on the original DataSet#fetch call * @param {Array} event.domain the domain set on the DataSet before DataSet#fetch was called * @param {Object} event.context the context set on the DataSet before DataSet#fetch was called * @param {Array} event.sort the sorting criteria used to get the ids */ on_fetch: function (records, event) { }, /** * Fetch all the currently active records for this DataSet (records selected via DataSet#select) * * @returns itself */ active_ids: function () { this.rpc('/base/dataset/get', { ids: this.get_active_ids(), model: this.model }, _.bind(function (records) { this.on_active_ids(_.map( records, function (record) { return new openerp.base.DataRecord( this.session, this.model, this._fields, record); }, this)); }, this)); return this; }, /** * @event * * Fires after the DataSet fetched the records matching its internal active ids selection * * @param {Array} records An array of the DataRecord fetched */ on_active_ids: function (records) { }, /** * Fetches the current active record for this DataSet * * @returns itself */ active_id: function () { this.rpc('/base/dataset/get', { ids: [this.get_active_id()], model: this.model }, _.bind(function (records) { var record = records[0]; this.on_active_id( record && new openerp.base.DataRecord( this.session, this.model, this._fields, record)); }, this)); return this; }, /** * Fires after the DataSet fetched the record matching the current active record * * @param record the record matching the provided id, or null if there is no record for this id */ on_active_id: function (record) { }, /** * Configures the DataSet * * @param options DataSet options * @param {Array} options.domain the domain to assign to this DataSet for filtering * @param {Object} options.context the context this DataSet should use during its calls * @param {Array} options.sort the sorting criteria for this DataSet * @returns itself */ set: function (options) { if (options.domain) { this._domain = _.clone(options.domain); } if (options.context) { this._context = _.clone(options.context); } if (options.sort) { this._sort = _.clone(options.sort); } return this; }, /** * Activates the previous id in the active sequence. If there is no previous id, wraps around to the last one * @returns itself */ prev: function () { this._active_id_index -= 1; if (this._active_id_index < 0) { this._active_id_index = this._active_ids.length - 1; } return this; }, /** * Activates the next id in the active sequence. If there is no next id, wraps around to the first one * @returns itself */ next: function () { this._active_id_index += 1; if (this._active_id_index >= this._active_ids.length) { this._active_id_index = 0; } return this; }, /** * Sets active_ids by value: * * * Activates all ids part of the current selection * * Sets active_id to be the first id of the selection * * @param {Array} ids the list of ids to activate * @returns itself */ select: function (ids) { this._active_ids = ids; this._active_id_index = 0; return this; }, /** * Fetches the ids of the currently selected records, if any. */ get_active_ids: function () { return this._active_ids; }, /** * Sets the current active_id by value * * If there are no active_ids selected, selects the provided id as the sole active_id * * If there are ids selected and the provided id is not in them, raise an error * * @param {Object} id the id to activate * @returns itself */ activate: function (id) { if(!this._active_ids) { this._active_ids = [id]; this._active_id_index = 0; } else { var index = _.indexOf(this._active_ids, id); if (index == -1) { throw new Error( "Could not find id " + id + " in array [" + this._active_ids.join(', ') + "]"); } this._active_id_index = index; } return this; }, /** * Fetches the id of the current active record, if any. * * @returns record? record id or null */ get_active_id: function () { if (!this._active_ids) { return null; } return this._active_ids[this._active_id_index]; } }); openerp.base.DataRecord = openerp.base.Controller.extend({ init: function(session, model, fields, values) { this._super(session, null); this.model = model; this.id = values.id || null; this.fields = fields; this.values = values; }, on_change: function() { }, on_reload: function() { } }); openerp.base.SearchView = openerp.base.Controller.extend({ init: function(session, element_id, dataset, view_id) { this._super(session, element_id); this.dataset = dataset; this.model = dataset.model; this.view_id = view_id; this.input_index = 0; this.input_ids = {}; this.domain = []; }, start: function() { //this.log('Starting SearchView '+this.model+this.view_id) this.rpc("/base/searchview/load", {"model": this.model, "view_id":this.view_id}, this.on_loaded); }, on_loaded: function(data) { this.fields_view = data.fields_view; this.log(this.fields_view); this.input_ids = {}; this.$element.html(QWeb.render("SearchView", {"fields_view": this.fields_view})); this.$element.find("#search").bind('click',this.on_search); // TODO bind click event on all button // TODO we don't do many2one yet, but in the future bind a many2one controller on them this.log(this.$element.find("#search")); }, register_input: function(node) { // self should be passed in the qweb dict to do: // // generate id var id = this.element_id + "_" + this.input_index++; // TODO construct a nice object // save it in our registry this.input_ids[id] = { node: node, type: "filter", domain: "", context: "", disabled: false }; return id; }, on_click: function() { // event catched on a button // flip the disabled flag // adjust the css class }, on_search: function() { this.log("on_search"); // collect all non disabled domains definitions, AND them // evaluate as python expression // save the result in this.domain this.dataset.fetch(); }, on_clear: function() { } }); openerp.base.SearchViewInput = openerp.base.Controller.extend({ // TODO not sure should we create a controller for every input ? // of we just keep a simple dict for each input in // openerp.base.SearchView#input_ids // and use if when we get an event depending on the type // i think it's less bloated to avoid useless controllers // but i think for many2one a controller would be nice // so simple dict for simple inputs // an controller for many2one ? }); openerp.base.FormView = openerp.base.Controller.extend({ init: function(session, element_id, dataset, view_id) { this._super(session, element_id); this.dataset = dataset; this.model = dataset.model; this.view_id = view_id; this.fields_views = {}; this.widgets = {}; this.widgets_counter = 0; this.fields = {}; this.datarecord = {}; }, start: function() { //this.log('Starting FormView '+this.model+this.view_id) this.rpc("/base/formview/load", {"model": this.model, "view_id": this.view_id}, this.on_loaded); }, on_loaded: function(data) { this.fields_view = data.fields_view; //this.log(this.fields_view); var frame = new openerp.base.WidgetFrame(this, this.fields_view.arch); this.$element.html(QWeb.render("FormView", { "frame": frame, "view": this })); for (var i in this.widgets) { this.widgets[i].start(); } // bind to all wdigets that have onchange ?? this.dataset.on_fetch.add(this.on_record_loaded); }, on_record_loaded: function(records) { this.datarecord = records[0]; for (var f in this.fields) { this.fields[f].set_value(this.datarecord.values[f]); } } }); openerp.base.ListView = openerp.base.Controller.extend({ init: function(session, element_id, dataset, view_id) { this._super(session, element_id); this.dataset = dataset; this.model = dataset.model; this.view_id = view_id; this.name = ""; this.cols = []; this.$table = null; this.colnames = []; this.colmodel = []; this.event_loading = false; // TODO in the future prevent abusive click by masking }, start: function() { //this.log('Starting ListView '+this.model+this.view_id) this.rpc("/base/listview/load", {"model": this.model, "view_id":this.view_id}, this.on_loaded); }, on_loaded: function(data) { this.fields_view = data.fields_view; //this.log(this.fields_view); this.name = "" + this.fields_view.arch.attrs.string; this.$element.html(QWeb.render("ListView", {"fields_view": this.fields_view})); this.$table = this.$element.find("table"); this.cols = []; this.colnames = []; this.colmodel = []; // TODO uss a object for each col, fill it with view and fallback to dataset.model_field var tree = this.fields_view.arch.children; for(var i = 0; i < tree.length; i++) { var col = tree[i]; if(col.tag == "field") { this.cols.push(col.attrs.name); this.colnames.push(col.attrs.name); this.colmodel.push({ name: col.attrs.name, index: col.attrs.name }); } } this.dataset.fields = this.cols; this.dataset.on_fetch.add(this.do_fill_table); }, do_fill_table: function(records) { this.log("do_fill_table"); var self = this; //this.log(this.dataset.data); var rows = []; for(var i = 0; i < records.length; i++) { // TODO very strange is sometimes non existing ? even as admin ? example ir.ui.menu var row = records[i].values; if(row) rows.push(row); // else // debugger; } //this.log(rows); var width = this.$element.width(); this.$table.jqGrid({ data: rows, datatype: "local", height: "100%", rowNum: 100, //rowList: [10,20,30], colNames: this.colnames, colModel: this.colmodel, //pager: "#plist47", viewrecords: true, caption: this.name }).setGridWidth(width); $(window).bind('resize', function() { self.$element.children().hide(); self.$table.setGridWidth(self.$element.width()); self.$element.children().show(); }).trigger('resize'); } }); openerp.base.TreeView = openerp.base.Controller.extend({ }); openerp.base.Widget = openerp.base.Controller.extend({ // TODO Change this to init: function(view, node) { and use view.session and a new element_id for the super // it means that widgets are special controllers init: function(view, node) { this.view = view; this.node = node; this.type = this.type || node.tag; this.element_name = this.element_name || this.type; this.element_id = [this.view.element_id, this.element_name, this.view.widgets_counter++].join("_"); this._super(this.view.session, this.element_id); this.view.widgets[this.element_id] = this; this.children = node.children; this.colspan = parseInt(node.attrs.colspan || 1); this.template = "Widget"; this.string = this.string || node.attrs.string; this.help = this.help || node.attrs.help; this.invisible = (node.attrs.invisible == '1'); }, start: function() { this.$element = $('#' + this.element_id); }, render: function() { var template = this.template; return QWeb.render(template, { "widget": this }); } }); openerp.base.WidgetFrame = openerp.base.Widget.extend({ init: function(view, node) { this._super(view, node); this.template = "WidgetFrame"; this.columns = node.attrs.col || 4; this.x = 0; this.y = 0; this.table = []; this.add_row(); for (var i = 0; i < node.children.length; i++) { var n = node.children[i]; if (n.tag == "newline") { this.add_row(); } else { this.handle_node(n); } } this.set_row_cells_with(this.table[this.table.length - 1]); }, add_row: function(){ if (this.table.length) { this.set_row_cells_with(this.table[this.table.length - 1]); } var row = []; this.table.push(row); this.x = 0; this.y += 1; return row; }, set_row_cells_with: function(row) { for (var i = 0; i < row.length; i++) { var w = row[i]; if (w.is_field_label) { w.width = "1%"; if (row[i + 1]) { row[i + 1].width = Math.round((100 / this.columns) * (w.colspan + 1) - 1) + '%'; } } else if (w.width === undefined) { w.width = Math.round((100 / this.columns) * w.colspan) + '%'; } } }, handle_node: function(n) { var type = this.view.fields_view.fields[n.attrs.name] || {}; var widget_type = n.attrs.widget || type.type || n.tag; if (openerp.base.widgets[widget_type]) { var widget = new openerp.base.widgets[widget_type](this.view, n); if (n.tag == 'field' && n.attrs.nolabel != '1') { var label = new openerp.base.widgets['label'](this.view, n); label["for"] = widget; this.add_widget(label); } this.add_widget(widget); } else { this.log("Unhandled widget type : " + widget_type, n); } }, add_widget: function(w) { if (!w.invisible) { var current_row = this.table[this.table.length - 1]; if (current_row.length && (this.x + w.colspan) > this.columns) { current_row = this.add_row(); } current_row.push(w); this.x += w.colspan; } return w; } }); openerp.base.WidgetNotebook = openerp.base.Widget.extend({ init: function(view, node) { this._super(view, node); this.template = "WidgetNotebook"; this.pages = []; for (var i = 0; i < node.children.length; i++) { var n = node.children[i]; if (n.tag == "page") { var page = new openerp.base.WidgetFrame(this.view, n); this.pages.push(page); } } }, start: function() { this._super.apply(this, arguments); this.$element.tabs(); } }); openerp.base.WidgetSeparator = openerp.base.Widget.extend({ init: function(view, node) { this._super(view, node); this.template = "WidgetSeparator"; } }); openerp.base.WidgetButton = openerp.base.Widget.extend({ init: function(view, node) { this._super(view, node); this.template = "WidgetButton"; } }); openerp.base.WidgetLabel = openerp.base.Widget.extend({ init: function(view, node) { this.is_field_label = true; this.element_name = 'label_' + node.attrs.name; this._super(view, node); this.template = "WidgetLabel"; this.colspan = 1; }, render: function () { if (this['for'] && this.type !== 'label') { return QWeb.render(this.template, {widget: this['for']}); } // Actual label widgets should not have a false and have type label return QWeb.render(this.template, {widget: this}); } }); openerp.base.Field = openerp.base.Widget.extend({ init: function(view, node) { this.name = node.attrs.name; view.fields[this.name] = this; this.type = node.attrs.widget || view.fields_view.fields[node.attrs.name].type; this.element_name = "field_" + this.name + "_" + this.type; this.original_value; this._super(view, node); if (node.attrs.nolabel != '1' && this.colspan > 1) { this.colspan--; } // this.datarecord = this.view.datarecord ?? this.field = view.fields_view.fields[node.attrs.name]; this.string = node.attrs.string || this.field.string; this.help = node.attrs.help || this.field.help; this.nolabel = (node.attrs.nolabel == '1'); }, set_value: function(value) { this.original_value = value; } }); openerp.base.FieldChar = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldChar"; }, start: function() { this._super.apply(this, arguments); // this.$element.bind('change',) ... blur, focus, ... }, set_value: function(value) { this._super.apply(this, arguments); if (value != null && value !== false) { this.$element.find('input').val(this.to_string(value)); } }, get_value: function() { return this.from_string(this.$element.find('input').val()); }, to_string: function(value) { return value.toString(); }, from_string: function(value) { return value.toString(); }, on_change: function() { //this.view.update_field(this.name,value); } }); openerp.base.FieldEmail = openerp.base.FieldChar.extend({ }); openerp.base.FieldUrl = openerp.base.FieldChar.extend({ }); openerp.base.FieldFloat = openerp.base.FieldChar.extend({ to_string: function(value) { return value.toFixed(2); } }); openerp.base.FieldText = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldText"; }, set_value: function(value) { this._super.apply(this, arguments); if (value != null && value !== false) { this.$element.find('textarea').val(value); } }, get_value: function() { return this.$element.find('textarea').val(); } }); openerp.base.FieldBoolean = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldBoolean"; }, set_value: function(value) { this._super.apply(this, arguments); this.$element.find('input')[0].checked = value; }, get_value: function() { this.$element.find('input')[0].checked; } }); openerp.base.FieldDate = openerp.base.FieldChar.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldDate"; } }); openerp.base.FieldDatetime = openerp.base.FieldChar.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldDatetime"; } }); openerp.base.FieldTextXml = openerp.base.Field.extend({ // to replace view editor }); openerp.base.FieldSelection = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldSelection"; }, set_value: function(value) { this._super.apply(this, arguments); if (value != null && value !== false) { this.$element.find('select').val(value); } }, get_value: function() { return this.$element.find('select').val(); } }); openerp.base.FieldMany2One = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldMany2One"; } }); openerp.base.FieldOne2Many = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldOne2Many"; } }); openerp.base.FieldMany2Many = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldMany2Many"; } }); openerp.base.FieldReference = openerp.base.Field.extend({ init: function(view, node) { this._super(view, node); this.template = "FieldReference"; } }); openerp.base.widgets = { 'group' : openerp.base.WidgetFrame, 'notebook' : openerp.base.WidgetNotebook, 'separator' : openerp.base.WidgetSeparator, 'label' : openerp.base.WidgetLabel, 'char' : openerp.base.FieldChar, 'email' : openerp.base.FieldEmail, 'url' : openerp.base.FieldUrl, 'text' : openerp.base.FieldText, 'date' : openerp.base.FieldDate, 'datetime' : openerp.base.FieldDatetime, 'selection' : openerp.base.FieldSelection, 'many2one' : openerp.base.FieldMany2One, 'many2many' : openerp.base.FieldMany2Many, 'one2many' : openerp.base.FieldOne2Many, 'one2many_list' : openerp.base.FieldOne2Many, 'reference' : openerp.base.FieldReference, 'boolean' : openerp.base.FieldBoolean, 'float' : openerp.base.FieldFloat, 'button' : openerp.base.WidgetButton }; openerp.base.CalendarView = openerp.base.Controller.extend({ // Dhtmlx scheduler ? }); openerp.base.GanttView = openerp.base.Controller.extend({ // Dhtmlx gantt ? }); openerp.base.DiagramView = openerp.base.Controller.extend({ // }); openerp.base.GraphView = openerp.base.Controller.extend({ }); openerp.base.ProcessView = openerp.base.Controller.extend({ }); openerp.base.HelpView = openerp.base.Controller.extend({ }); }; // DEBUG_RPC:rpc.request:('execute', 'addons-dsh-l10n_us', 1, '*', ('ir.filters', 'get_filters', u'res.partner')) // vim:et fdc=0 fdl=0 foldnestmax=3 fdm=syntax: