[IMP] mark a bunch of dialog stuff as exportable (mostly buttons, but also dialog...
[odoo/odoo.git] / addons / web / static / src / js / chrome.js
1 /*---------------------------------------------------------
2  * OpenERP Web chrome
3  *---------------------------------------------------------*/
4 openerp.web.chrome = function(openerp) {
5 var QWeb = openerp.web.qweb,
6     _t = openerp.web._t;
7
8 openerp.web.Notification =  openerp.web.Widget.extend(/** @lends openerp.web.Notification# */{
9     template: 'Notification',
10     identifier_prefix: 'notification-',
11
12     init: function() {
13         this._super.apply(this, arguments);
14         openerp.notification = this;
15     },
16
17     start: function() {
18         this._super.apply(this, arguments);
19         this.$element.notify({
20             speed: 500,
21             expires: 1500
22         });
23     },
24     notify: function(title, text) {
25         this.$element.notify('create', {
26             title: title,
27             text: text
28         });
29     },
30     warn: function(title, text) {
31         this.$element.notify('create', 'oe_notification_alert', {
32             title: title,
33             text: text
34         });
35     }
36
37 });
38
39 openerp.web.Dialog = openerp.web.OldWidget.extend(/** @lends openerp.web.Dialog# */{
40     dialog_title: "",
41     identifier_prefix: 'dialog',
42     /**
43      * @constructs openerp.web.Dialog
44      * @extends openerp.web.OldWidget
45      *
46      * @param parent
47      * @param dialog_options
48      */
49     init: function (parent, dialog_options) {
50         var self = this;
51         this._super(parent);
52         this.dialog_options = {
53             modal: true,
54             width: 'auto',
55             min_width: 0,
56             max_width: '100%',
57             height: 'auto',
58             min_height: 0,
59             max_height: '100%',
60             autoOpen: false,
61             buttons: {},
62             beforeClose: function () { self.on_close(); }
63         };
64         for (var f in this) {
65             if (f.substr(0, 10) == 'on_button_') {
66                 this.dialog_options.buttons[f.substr(10)] = this[f];
67             }
68         }
69         if (dialog_options) {
70             this.set_options(dialog_options);
71         }
72     },
73     set_options: function(options) {
74         options = options || {};
75         options.width = this.get_width(options.width || this.dialog_options.width);
76         options.min_width = this.get_width(options.min_width || this.dialog_options.min_width);
77         options.max_width = this.get_width(options.max_width || this.dialog_options.max_width);
78         options.height = this.get_height(options.height || this.dialog_options.height);
79         options.min_height = this.get_height(options.min_height || this.dialog_options.min_height);
80         options.max_height = this.get_height(options.max_height || this.dialog_options.max_width);
81
82         if (options.width !== 'auto') {
83             if (options.width > options.max_width) options.width = options.max_width;
84             if (options.width < options.min_width) options.width = options.min_width;
85         }
86         if (options.height !== 'auto') {
87             if (options.height > options.max_height) options.height = options.max_height;
88             if (options.height < options.min_height) options.height = options.min_height;
89         }
90         if (!options.title && this.dialog_title) {
91             options.title = this.dialog_title;
92         }
93         _.extend(this.dialog_options, options);
94     },
95     get_width: function(val) {
96         return this.get_size(val.toString(), $(window.top).width());
97     },
98     get_height: function(val) {
99         return this.get_size(val.toString(), $(window.top).height());
100     },
101     get_size: function(val, available_size) {
102         if (val === 'auto') {
103             return val;
104         } else if (val.slice(-1) == "%") {
105             return Math.round(available_size / 100 * parseInt(val.slice(0, -1), 10));
106         } else {
107             return parseInt(val, 10);
108         }
109     },
110     start: function () {
111         this.$dialog = $(this.$element).dialog(this.dialog_options);
112         this._super();
113         return this;
114     },
115     open: function(dialog_options) {
116         // TODO fme: bind window on resize
117         if (this.template) {
118             this.$element.html(this.render());
119         }
120         this.set_options(dialog_options);
121         this.$dialog.dialog(this.dialog_options).dialog('open');
122         return this;
123     },
124     close: function() {
125         // Closes the dialog but leave it in a state where it could be opened again.
126         this.$dialog.dialog('close');
127     },
128     on_close: function() {
129     },
130     stop: function () {
131         // Destroy widget
132         this.close();
133         this.$dialog.dialog('destroy');
134         this._super();
135     }
136 });
137
138 openerp.web.CrashManager = openerp.web.CallbackEnabled.extend({
139     init: function() {
140         this._super();
141         openerp.connection.on_rpc_error.add(this.on_rpc_error);
142     },
143     on_rpc_error: function(error) {
144         this.error = error;
145         if (error.data.fault_code) {
146             var split = ("" + error.data.fault_code).split('\n')[0].split(' -- ');
147             if (split.length > 1) {
148                 error.type = split.shift();
149                 error.data.fault_code = error.data.fault_code.substr(error.type.length + 4);
150             }
151         }
152         if (error.code === 200 && error.type) {
153             this.on_managed_error(error);
154         } else {
155             this.on_traceback(error);
156         }
157     },
158     on_managed_error: function(error) {
159         $('<div>' + QWeb.render('DialogWarning', {error: error}) + '</div>').dialog({
160             title: "OpenERP " + _.str.capitalize(error.type),
161             buttons: [
162                 {text: _t("Ok"), click: function() { $(this).dialog("close"); }}
163             ]
164         });
165     },
166     on_traceback: function(error) {
167         var dialog = new openerp.web.Dialog(this, {
168             title: "OpenERP " + _.str.capitalize(error.type),
169             autoOpen: true,
170             width: '90%',
171             height: '90%',
172             min_width: '800px',
173             min_height: '600px',
174             buttons: [
175                 {text: _t("Ok"), click: function() { $(this).dialog("close"); }}
176             ]
177         }).start();
178         dialog.$element.html(QWeb.render('DialogTraceback', {error: error}));
179     }
180 });
181
182 openerp.web.Loading = openerp.web.Widget.extend(/** @lends openerp.web.Loading# */{
183     template: 'Loading',
184     /**
185      * @constructs openerp.web.Loading
186      * @extends openerp.web.Widget
187      *
188      * @param parent
189      * @param element_id
190      */
191     init: function(parent) {
192         this._super(parent);
193         this.count = 0;
194         this.blocked_ui = false;
195         this.session.on_rpc_request.add_first(this.on_rpc_event, 1);
196         this.session.on_rpc_response.add_last(this.on_rpc_event, -1);
197     },
198     on_rpc_event : function(increment) {
199         var self = this;
200         if (!this.count && increment === 1) {
201             // Block UI after 3s
202             this.long_running_timer = setTimeout(function () {
203                 self.blocked_ui = true;
204                 $.blockUI();
205             }, 3000);
206         }
207
208         this.count += increment;
209         if (this.count > 0) {
210             //this.$element.html(QWeb.render("Loading", {}));
211             this.$element.html("Loading ("+this.count+")");
212             this.$element.show();
213             this.widget_parent.$element.addClass('loading');
214         } else {
215             this.count = 0;
216             clearTimeout(this.long_running_timer);
217             // Don't unblock if blocked by somebody else
218             if (self.blocked_ui) {
219                 this.blocked_ui = false;
220                 $.unblockUI();
221             }
222             this.$element.fadeOut();
223             this.widget_parent.$element.removeClass('loading');
224         }
225     }
226 });
227
228 openerp.web.Database = openerp.web.Widget.extend(/** @lends openerp.web.Database# */{
229     /**
230      * @constructs openerp.web.Database
231      * @extends openerp.web.Widget
232      *
233      * @param parent
234      * @param element_id
235      * @param option_id
236      */
237     init: function(parent, element_id, option_id) {
238         this._super(parent, element_id);
239         this.$option_id = $('#' + option_id);
240         this.unblockUIFunction = $.unblockUI;
241     },
242     start: function() {
243         this._super();
244         this.$element.html(QWeb.render("Database", this));
245
246         var self = this;
247         var fetch_db = this.rpc("/web/database/get_list", {}, function(result) {
248             self.db_list = result.db_list;
249         });
250         var fetch_langs = this.rpc("/web/session/get_lang_list", {}, function(result) {
251             if (result.error) {
252                 self.display_error(result);
253                 return;
254             }
255             self.lang_list = result.lang_list;
256         });
257         $.when(fetch_db, fetch_langs).then(function () {self.do_create();});
258
259         this.$element.find('#db-create').click(this.do_create);
260         this.$element.find('#db-drop').click(this.do_drop);
261         this.$element.find('#db-backup').click(this.do_backup);
262         this.$element.find('#db-restore').click(this.do_restore);
263         this.$element.find('#db-change-password').click(this.do_change_password);
264         this.$element.find('#back-to-login').click(function() {
265             self.hide();
266         });
267     },
268     stop: function () {
269         this.hide();
270         this.$option_id.empty();
271
272         this.$element
273             .find('#db-create, #db-drop, #db-backup, #db-restore, #db-change-password, #back-to-login')
274                 .unbind('click')
275             .end()
276             .empty();
277         this._super();
278     },
279     show: function () {
280         this.$element.closest(".openerp")
281                 .removeClass("login-mode")
282                 .addClass("database_block");
283     },
284     hide: function () {
285         this.$element.closest(".openerp")
286                 .addClass("login-mode")
287                 .removeClass("database_block")
288     },
289     /**
290      * Converts a .serializeArray() result into a dict. Does not bother folding
291      * multiple identical keys into an array, last key wins.
292      *
293      * @param {Array} array
294      */
295     to_object: function (array) {
296         var result = {};
297         _(array).each(function (record) {
298             result[record.name] = record.value;
299         });
300         return result;
301     },
302     /**
303      * Waits until the new database is done creating, then unblocks the UI and
304      * logs the user in as admin
305      *
306      * @param {Number} db_creation_id identifier for the db-creation operation, used to fetch the current installation progress
307      * @param {Object} info info fields for this database creation
308      * @param {String} info.db name of the database being created
309      * @param {String} info.password super-admin password for the database
310      */
311     wait_for_newdb: function (db_creation_id, info) {
312         var self = this;
313         self.rpc('/web/database/progress', {
314             id: db_creation_id,
315             password: info.password
316         }, function (result) {
317             var progress = result[0];
318             // I'd display a progress bar, but turns out the progress status
319             // the server report kind-of blows goats: it's at 0 for ~75% of
320             // the installation, then jumps to 75%, then jumps down to either
321             // 0 or ~40%, then back up to 75%, then terminates. Let's keep that
322             // mess hidden behind a not-very-useful but not overly weird
323             // message instead.
324             if (progress < 1) {
325                 setTimeout(function () {
326                     self.wait_for_newdb(db_creation_id, info);
327                 }, 500);
328                 return;
329             }
330
331             var admin = result[1][0];
332             setTimeout(function () {
333                 self.widget_parent.do_login(
334                         info.db, admin.login, admin.password);
335                 self.stop();
336                 self.unblockUI();
337             });
338         });
339     },
340     /**
341      * Blocks UI and replaces $.unblockUI by a noop to prevent third parties
342      * from unblocking the UI
343      */
344     blockUI: function () {
345         $.blockUI();
346         $.unblockUI = function () {};
347     },
348     /**
349      * Reinstates $.unblockUI so third parties can play with blockUI, and
350      * unblocks the UI
351      */
352     unblockUI: function () {
353         $.unblockUI = this.unblockUIFunction;
354         $.unblockUI();
355     },
356     /**
357      * Displays an error dialog resulting from the various RPC communications
358      * failing over themselves
359      *
360      * @param {Object} error error description
361      * @param {String} error.title title of the error dialog
362      * @param {String} error.error message of the error dialog
363      */
364     display_error: function (error) {
365         return $('<div>').dialog({
366             modal: true,
367             title: error.title,
368             buttons: [
369                 {text: _t("Ok"), click: function() { $(this).dialog("close"); }}
370             ]
371         }).html(error.error);
372     },
373     do_create: function() {
374         var self = this;
375         self.$option_id.html(QWeb.render("Database.CreateDB", self));
376         self.$option_id.find("form[name=create_db_form]").validate({
377             submitHandler: function (form) {
378                 var fields = $(form).serializeArray();
379                 self.blockUI();
380                 self.rpc("/web/database/create", {'fields': fields}, function(result) {
381                     if (result.error) {
382                         self.unblockUI();
383                         self.display_error(result);
384                         return;
385                     }
386                     self.db_list.push(self.to_object(fields)['db_name']);
387                     self.db_list.sort();
388                     self.widget_parent.set_db_list(self.db_list);
389                     var form_obj = self.to_object(fields);
390                     self.wait_for_newdb(result, {
391                         password: form_obj['super_admin_pwd'],
392                         db: form_obj['db_name']
393                     });
394                 });
395             }
396         });
397     },
398     do_drop: function() {
399         var self = this;
400         self.$option_id.html(QWeb.render("DropDB", self));
401         self.$option_id.find("form[name=drop_db_form]").validate({
402             submitHandler: function (form) {
403                 var $form = $(form),
404                     fields = $form.serializeArray(),
405                     $db_list = $form.find('select[name=drop_db]'),
406                     db = $db_list.val();
407
408                 if (!confirm("Do you really want to delete the database: " + db + " ?")) {
409                     return;
410                 }
411                 self.rpc("/web/database/drop", {'fields': fields}, function(result) {
412                     if (result.error) {
413                         self.display_error(result);
414                         return;
415                     }
416                     $db_list.find(':selected').remove();
417                     self.db_list.splice(_.indexOf(self.db_list, db, true), 1);
418                     self.widget_parent.set_db_list(self.db_list);
419                     self.do_notify("Dropping database", "The database '" + db + "' has been dropped");
420                 });
421             }
422         });
423     },
424     do_backup: function() {
425         var self = this;
426         self.$option_id
427             .html(QWeb.render("BackupDB", self))
428             .find("form[name=backup_db_form]").validate({
429             submitHandler: function (form) {
430                 self.blockUI();
431                 self.session.get_file({
432                     form: form,
433                     error: function (body) {
434                         var error = body.firstChild.data.split('|');
435                         self.display_error({
436                             title: error[0],
437                             error: error[1]
438                         });
439                     },
440                     complete: $.proxy(self, 'unblockUI')
441                 });
442             }
443         });
444     },
445     do_restore: function() {
446         var self = this;
447         self.$option_id.html(QWeb.render("RestoreDB", self));
448
449         self.$option_id.find("form[name=restore_db_form]").validate({
450             submitHandler: function (form) {
451                 self.blockUI();
452                 $(form).ajaxSubmit({
453                     url: '/web/database/restore',
454                     type: 'POST',
455                     resetForm: true,
456                     success: function (body) {
457                         // TODO: ui manipulations
458                         // note: response objects don't work, but we have the
459                         // HTTP body of the response~~
460
461                         // If empty body, everything went fine
462                         if (!body) { return; }
463
464                         if (body.indexOf('403 Forbidden') !== -1) {
465                             self.display_error({
466                                 title: 'Access Denied',
467                                 error: 'Incorrect super-administrator password'
468                             })
469                         } else {
470                             self.display_error({
471                                 title: 'Restore Database',
472                                 error: 'Could not restore the database'
473                             })
474                         }
475                     },
476                     complete: $.proxy(self, 'unblockUI')
477                 });
478             }
479         });
480     },
481     do_change_password: function() {
482         var self = this;
483         self.$option_id.html(QWeb.render("Change_DB_Pwd", self));
484
485         self.$option_id.find("form[name=change_pwd_form]").validate({
486             messages: {
487                 old_pwd: "Please enter your previous password",
488                 new_pwd: "Please enter your new password",
489                 confirm_pwd: {
490                     required: "Please confirm your new password",
491                     equalTo: "The confirmation does not match the password"
492                 }
493             },
494             submitHandler: function (form) {
495                 self.rpc("/web/database/change_password", {
496                     'fields': $(form).serializeArray()
497                 }, function(result) {
498                     if (result.error) {
499                         self.display_error(result);
500                         return;
501                     }
502                     self.do_notify("Changed Password", "Password has been changed successfully");
503                 });
504             }
505         });
506     }
507 });
508
509 openerp.web.Login =  openerp.web.Widget.extend(/** @lends openerp.web.Login# */{
510     remember_credentials: true,
511     
512     template: "Login",
513     identifier_prefix: 'oe-app-login-',
514     /**
515      * @constructs openerp.web.Login
516      * @extends openerp.web.Widget
517      *
518      * @param parent
519      * @param element_id
520      */
521
522     init: function(parent) {
523         this._super(parent);
524         this.has_local_storage = typeof(localStorage) != 'undefined';
525         this.selected_db = null;
526         this.selected_login = null;
527
528         if (this.has_local_storage && this.remember_credentials) {
529             this.selected_db = localStorage.getItem('last_db_login_success');
530             this.selected_login = localStorage.getItem('last_login_login_success');
531             if (jQuery.deparam(jQuery.param.querystring()).debug != undefined) {
532                 this.selected_password = localStorage.getItem('last_password_login_success');
533             }
534         }
535         
536         var qs = jQuery.deparam(jQuery.param.querystring());
537         if (qs.db) {
538             this.selected_db = qs.db;
539         }
540         if (qs.login) {
541             this.selected_login = qs.login;
542         }
543
544     },
545     start: function() {
546         var self = this;
547         this.database = new openerp.web.Database(
548                 this, "oe_database", "oe_db_options");
549         this.database.start();
550
551         this.$element.find('#oe-db-config').click(function() {
552             self.database.show();
553         });
554
555         this.$element.find("form").submit(this.on_submit);
556
557         this.rpc("/web/database/get_list", {}, function(result) {
558             self.set_db_list(result.db_list);
559         }, 
560         function(error, event) {
561             if (error.data.fault_code === 'AccessDenied') {
562                 event.preventDefault();
563             }
564         });
565
566     },
567     stop: function () {
568         this.database.stop();
569         this._super();
570     },
571     set_db_list: function (list) {
572         this.$element.find("[name=db]").replaceWith(
573             openerp.web.qweb.render('Login_dblist', {
574                 db_list: list, selected_db: this.selected_db}))
575     },
576     on_login_invalid: function() {
577         this.$element.closest(".openerp").addClass("login-mode");
578     },
579     on_login_valid: function() {
580         this.$element.closest(".openerp").removeClass("login-mode");
581     },
582     on_submit: function(ev) {
583         ev.preventDefault();
584         var $e = this.$element;
585         var db = $e.find("form [name=db]").val();
586         var login = $e.find("form input[name=login]").val();
587         var password = $e.find("form input[name=password]").val();
588
589         this.do_login(db, login, password);
590     },
591     /**
592      * Performs actual login operation, and UI-related stuff
593      *
594      * @param {String} db database to log in
595      * @param {String} login user login
596      * @param {String} password user password
597      */
598     do_login: function (db, login, password) {
599         var self = this;
600         this.session.session_authenticate(db, login, password, function() {
601             if(self.session.session_is_valid()) {
602                 if (self.has_local_storage) {
603                     if(self.remember_credentials) {
604                         localStorage.setItem('last_db_login_success', db);
605                         localStorage.setItem('last_login_login_success', login);
606                         if (jQuery.deparam(jQuery.param.querystring()).debug != undefined) {
607                             localStorage.setItem('last_password_login_success', password);
608                         }
609                     } else {
610                         localStorage.setItem('last_db_login_success', '');
611                         localStorage.setItem('last_login_login_success', '');
612                         localStorage.setItem('last_password_login_success', '');
613                     }
614                 }
615                 self.on_login_valid();
616             } else {
617                 self.$element.addClass("login_invalid");
618                 self.on_login_invalid();
619             }
620         });
621     },
622     do_ask_login: function(continuation) {
623         this.on_login_invalid();
624         this.$element
625             .removeClass("login_invalid");
626         this.on_login_valid.add({
627             position: "last",
628             unique: true,
629             callback: continuation || function() {}
630         });
631     },
632     on_logout: function() {
633         this.session.logout();
634     }
635 });
636
637 openerp.web.Header =  openerp.web.Widget.extend(/** @lends openerp.web.Header# */{
638     template: "Header",
639     identifier_prefix: 'oe-app-header-',
640     /**
641      * @constructs openerp.web.Header
642      * @extends openerp.web.Widget
643      *
644      * @param parent
645      */
646     init: function(parent) {
647         this._super(parent);
648         this.qs = "?" + jQuery.param.querystring();
649         this.$content = $();
650         this.update_promise = $.Deferred().resolve();
651     },
652     start: function() {
653         this._super();
654     },
655     do_update: function () {
656         var self = this;
657         var fct = function() {
658             self.$content.remove();
659             if (!self.session.uid)
660                 return;
661             var func = new openerp.web.Model("res.users").get_func("read");
662             return func(self.session.uid, ["name", "company_id"]).pipe(function(res) {
663                 self.$content = $(QWeb.render("Header-content", {widget: self, user: res}));
664                 self.$content.appendTo(self.$element);
665                 self.$element.find(".logout").click(self.on_logout);
666                 self.$element.find("a.preferences").click(self.on_preferences);
667                 self.$element.find(".about").click(self.on_about);
668                 return self.shortcut_load();
669             });
670         };
671         this.update_promise = this.update_promise.pipe(fct, fct);
672     },
673     on_about: function() {
674         var self = this;
675         self.rpc("/web/webclient/version_info", {}).then(function(res) {
676             var $help = $(QWeb.render("About-Page", {version_info: res}));
677             $help.dialog({autoOpen: true,
678                 modal: true, width: 960, title: _t("About")});
679         });
680     },
681     shortcut_load :function(){
682         var self = this,
683             sc = self.session.shortcuts,
684             shortcuts_ds = new openerp.web.DataSet(this, 'ir.ui.view_sc');
685         // TODO: better way to communicate between sections.
686         // sc.bindings, because jquery does not bind/trigger on arrays...
687         if (!sc.binding) {
688             sc.binding = {};
689             $(sc.binding).bind({
690                 'add': function (e, attrs) {
691                     shortcuts_ds.create(attrs, function (out) {
692                         $('<li>', {
693                             'data-shortcut-id':out.result,
694                             'data-id': attrs.res_id
695                         }).text(attrs.name)
696                           .appendTo(self.$element.find('.oe-shortcuts ul'));
697                         attrs.id = out.result;
698                         sc.push(attrs);
699                     });
700                 },
701                 'remove-current': function () {
702                     var menu_id = self.session.active_id;
703                     var $shortcut = self.$element
704                         .find('.oe-shortcuts li[data-id=' + menu_id + ']');
705                     var shortcut_id = $shortcut.data('shortcut-id');
706                     $shortcut.remove();
707                     shortcuts_ds.unlink([shortcut_id]);
708                     var sc_new = _.reject(sc, function(shortcut){ return shortcut_id === shortcut.id});
709                     sc.splice(0, sc.length);
710                     sc.push.apply(sc, sc_new);
711                     }
712             });
713         }
714         return this.rpc('/web/session/sc_list', {}, function(shortcuts) {
715             sc.splice(0, sc.length);
716             sc.push.apply(sc, shortcuts);
717
718             self.$element.find('.oe-shortcuts')
719                 .html(QWeb.render('Shortcuts', {'shortcuts': shortcuts}))
720                 .undelegate('li', 'click')
721
722                 .delegate('li', 'click', function(e) {
723                     e.stopPropagation();
724                     var id = $(this).data('id');
725                     self.session.active_id = id;
726                     self.rpc('/web/menu/action', {'menu_id':id}, function(ir_menu_data) {
727                         if (ir_menu_data.action.length){
728                             self.on_action(ir_menu_data.action[0][2]);
729                         }
730                     });
731                 });
732         });
733     },
734
735     on_action: function(action) {
736     },
737     on_preferences: function(){
738         var self = this;
739         var action_manager = new openerp.web.ActionManager(this);
740         var dataset = new openerp.web.DataSet (this,'res.users',this.context);
741         dataset.call ('action_get','',function (result){
742             self.rpc('/web/action/load', {action_id:result}, function(result){
743                 action_manager.do_action(_.extend(result['result'], {
744                     res_id: self.session.uid,
745                     res_model: 'res.users',
746                     flags: {
747                         action_buttons: false,
748                         search_view: false,
749                         sidebar: false,
750                         views_switcher: false,
751                         pager: false
752                     }
753                 }));
754             });
755         });
756         this.dialog = new openerp.web.Dialog(this,{
757             modal: true,
758             title: _t("Preferences"),
759             width: 600,
760             height: 500,
761             buttons: [
762                 {text: _t("Change password"), click: function(){ self.change_password(); }},
763                 {text: _t("Cancel"), click: function(){ $(this).dialog('destroy'); }},
764                 {text: _t("Save"), click: function(){
765                         var inner_viewmanager = action_manager.inner_viewmanager;
766                         inner_viewmanager.views[inner_viewmanager.active_view].controller.do_save()
767                         .then(function() {
768                             self.dialog.stop();
769                             window.location.reload();
770                         });
771                     }
772                 }
773             ]
774         });
775        this.dialog.start().open();
776        action_manager.appendTo(this.dialog);
777        action_manager.render(this.dialog);
778     },
779
780     change_password :function() {
781         var self = this;
782         this.dialog = new openerp.web.Dialog(this,{
783             modal : true,
784             title: _t("Change Password"),
785             width : 'auto',
786             height : 'auto'
787         });
788         this.dialog.start().open();
789         this.dialog.$element.html(QWeb.render("Change_Pwd", self));
790         this.dialog.$element.find("form[name=change_password_form]").validate({
791             submitHandler: function (form) {
792                 self.rpc("/web/session/change_password",{
793                     'fields': $(form).serializeArray()
794                 }, function(result) {
795                     if (result.error) {
796                         self.display_error(result);
797                         return;
798                     } else {
799                         self.session.logout();
800                     }
801                 });
802             }
803         });
804     },
805     display_error: function (error) {
806         return $('<div>').dialog({
807             modal: true,
808             title: error.title,
809             buttons: [
810                 {text: _("Ok"), click: function() { $(this).dialog("close"); }}
811             ]
812         }).html(error.error);
813     },
814     on_logout: function() {
815     }
816 });
817
818 openerp.web.Menu =  openerp.web.Widget.extend(/** @lends openerp.web.Menu# */{
819     /**
820      * @constructs openerp.web.Menu
821      * @extends openerp.web.Widget
822      *
823      * @param parent
824      * @param element_id
825      * @param secondary_menu_id
826      */
827     init: function(parent, element_id, secondary_menu_id) {
828         this._super(parent, element_id);
829         this.secondary_menu_id = secondary_menu_id;
830         this.$secondary_menu = $("#" + secondary_menu_id);
831         this.menu = false;
832         this.folded = false;
833         if (window.localStorage) {
834             this.folded = localStorage.getItem('oe_menu_folded') === 'true';
835         }
836         this.float_timeout = 700;
837     },
838     start: function() {
839         this.$secondary_menu.addClass(this.folded ? 'oe_folded' : 'oe_unfolded');
840     },
841     do_reload: function() {
842         return this.rpc("/web/menu/load", {}, this.on_loaded);
843     },
844     on_loaded: function(data) {
845         this.data = data;
846         this.$element.html(QWeb.render("Menu", { widget : this }));
847         this.$secondary_menu.html(QWeb.render("Menu.secondary", { widget : this }));
848         this.$element.add(this.$secondary_menu).find("a").click(this.on_menu_click);
849         this.$secondary_menu.find('.oe_toggle_secondary_menu').click(this.on_toggle_fold);
850     },
851     on_toggle_fold: function() {
852         this.$secondary_menu.toggleClass('oe_folded').toggleClass('oe_unfolded');
853         if (this.folded) {
854             this.$secondary_menu.find('.oe_secondary_menu.active').show();
855         } else {
856             this.$secondary_menu.find('.oe_secondary_menu').hide();
857         }
858         this.folded = !this.folded;
859         if (window.localStorage) {
860             localStorage.setItem('oe_menu_folded', this.folded.toString());
861         }
862     },
863     on_menu_click: function(ev, id) {
864         id = id || 0;
865         var $clicked_menu, manual = false;
866
867         if (id) {
868             // We can manually activate a menu with it's id (for hash url mapping)
869             manual = true;
870             $clicked_menu = this.$element.find('a[data-menu=' + id + ']');
871             if (!$clicked_menu.length) {
872                 $clicked_menu = this.$secondary_menu.find('a[data-menu=' + id + ']');
873             }
874         } else {
875             $clicked_menu = $(ev.currentTarget);
876             id = $clicked_menu.data('menu');
877         }
878
879         if (this.do_menu_click($clicked_menu, manual) && id) {
880             this.session.active_id = id;
881             this.rpc('/web/menu/action', {'menu_id': id}, this.on_menu_action_loaded);
882         }
883         if (ev) {
884             ev.stopPropagation();
885         }
886         return false;
887     },
888     do_menu_click: function($clicked_menu, manual) {
889         var $sub_menu, $main_menu,
890             active = $clicked_menu.is('.active'),
891             sub_menu_visible = false;
892
893         if (this.$secondary_menu.has($clicked_menu).length) {
894             $sub_menu = $clicked_menu.parents('.oe_secondary_menu');
895             $main_menu = this.$element.find('a[data-menu=' + $sub_menu.data('menu-parent') + ']');
896         } else {
897             $sub_menu = this.$secondary_menu.find('.oe_secondary_menu[data-menu-parent=' + $clicked_menu.attr('data-menu') + ']');
898             $main_menu = $clicked_menu;
899         }
900
901         sub_menu_visible = $sub_menu.is(':visible');
902         this.$secondary_menu.find('.oe_secondary_menu').hide();
903
904         $('.active', this.$element.add(this.$secondary_menu)).removeClass('active');
905         $main_menu.add($clicked_menu).add($sub_menu).addClass('active');
906
907         if (!(this.folded && manual)) {
908             this.do_show_secondary($sub_menu, $main_menu);
909         } else {
910             this.do_show_secondary();
911         }
912
913         if ($main_menu != $clicked_menu) {
914             if ($clicked_menu.is('.submenu')) {
915                 $sub_menu.find('.submenu.opened').each(function() {
916                     if (!$(this).next().has($clicked_menu).length && !$(this).is($clicked_menu)) {
917                         $(this).removeClass('opened').next().hide();
918                     }
919                 });
920                 $clicked_menu.toggleClass('opened').next().toggle();
921             } else if ($clicked_menu.is('.leaf')) {
922                 $sub_menu.toggle(!this.folded);
923                 return true;
924             }
925         } else if (this.folded) {
926             if (active && sub_menu_visible) {
927                 $sub_menu.hide();
928                 return true;
929             }
930             return manual;
931         } else {
932             return true;
933         }
934         return false;
935     },
936     do_hide_secondary: function() {
937         this.$secondary_menu.hide();
938     },
939     do_show_secondary: function($sub_menu, $main_menu) {
940         var self = this;
941         this.$secondary_menu.show();
942         if (!arguments.length) {
943             return;
944         }
945         if (this.folded) {
946             var css = $main_menu.position(),
947                 fold_width = this.$secondary_menu.width() + 2,
948                 window_width = $(window).width();
949             css.top += 33;
950             css.left -= Math.round(($sub_menu.width() - $main_menu.width()) / 2);
951             css.left = css.left < fold_width ? fold_width : css.left;
952             if ((css.left + $sub_menu.width()) > window_width) {
953                 delete(css.left);
954                 css.right = 1;
955             }
956             $sub_menu.css(css);
957             $sub_menu.mouseenter(function() {
958                 clearTimeout($sub_menu.data('timeoutId'));
959             }).mouseleave(function(evt) {
960                 var timeoutId = setTimeout(function() {
961                     if (self.folded) {
962                         $sub_menu.hide();
963                     }
964                 }, self.float_timeout);
965                 $sub_menu.data('timeoutId', timeoutId);
966             });
967         }
968         $sub_menu.show();
969     },
970     on_menu_action_loaded: function(data) {
971         var self = this;
972         if (data.action.length) {
973             var action = data.action[0][2];
974             self.on_action(action);
975         } else {
976             self.on_action({type: 'null_action'});
977         }
978     },
979     on_action: function(action) {
980     }
981 });
982
983 openerp.web.WebClient = openerp.web.Widget.extend(/** @lends openerp.web.WebClient */{
984     /**
985      * @constructs openerp.web.WebClient
986      * @extends openerp.web.Widget
987      *
988      * @param element_id
989      */
990     init: function(element_id) {
991         this._super(null, element_id);
992         openerp.webclient = this;
993
994         this.notification = new openerp.web.Notification(this);
995         this.loading = new openerp.web.Loading(this);
996         this.crashmanager =  new openerp.web.CrashManager();
997
998         this.header = new openerp.web.Header(this);
999         this.login = new openerp.web.Login(this);
1000         this.header.on_logout.add(this.login.on_logout);
1001         this.header.on_action.add(this.on_menu_action);
1002
1003         this.session.on_session_invalid.add(this.login.do_ask_login);
1004         this.session.on_session_valid.add_last(this.header.do_update);
1005         this.session.on_session_invalid.add_last(this.header.do_update);
1006         this.session.on_session_valid.add_last(this.on_logged);
1007         this.session.on_session_invalid.add_last(this.on_logged_out);
1008
1009
1010         this._current_state = null;
1011
1012     },
1013     start: function() {
1014         this._super.apply(this, arguments);
1015         var self = this;
1016         openerp.connection.bind(function() {
1017        
1018             var params = {};
1019             if(jQuery.param != undefined && jQuery.deparam(jQuery.param.querystring()).kitten != undefined) {
1020                 self.$element.addClass("kitten-mode-activated");
1021             }
1022             self.$element.html(QWeb.render("Interface", params));
1023             openerp.connection.session_restore();
1024
1025
1026             // TODO nivification of menu Widget !!!
1027             self.menu = new openerp.web.Menu(self, "oe_menu", "oe_secondary_menu");
1028             self.menu.on_action.add(self.on_menu_action);
1029
1030
1031             self.notification.prependTo(self.$element);
1032             self.loading.appendTo($('#oe_loading'));
1033             self.header.appendTo($("#oe_header"));
1034             self.login.appendTo($('#oe_login'));
1035             self.menu.start();
1036         });
1037     },
1038     do_reload: function() {
1039         return $.when(this.session.session_restore(),this.menu.do_reload());
1040     },
1041     do_notify: function() {
1042         var n = this.notification;
1043         n.notify.apply(n, arguments);
1044     },
1045     do_warn: function() {
1046         var n = this.notification;
1047         n.warn.apply(n, arguments);
1048     },
1049     on_logged: function() {
1050         this.menu.do_reload();
1051         if(this.action_manager)
1052             this.action_manager.stop();
1053         this.action_manager = new openerp.web.ActionManager(this);
1054         this.action_manager.appendTo($("#oe_app"));
1055
1056         if (openerp._modules_loaded) { // TODO: find better option than this
1057             this.bind_hashchange();
1058         } else {
1059             this.session.on_modules_loaded.add({        // XXX what about a $.Deferred ?
1060                 callback: $.proxy(this, 'bind_hashchange'),
1061                 unique: true,
1062                 position: 'last'
1063             })
1064         }
1065     },
1066     on_logged_out: function() {
1067         $(window).unbind('hashchange', this.on_hashchange);
1068         this.do_push_state({});
1069         if(this.action_manager)
1070             this.action_manager.stop();
1071         this.action_manager = null;
1072     },
1073     bind_hashchange: function() {
1074         $(window).bind('hashchange', this.on_hashchange);
1075
1076         var state = $.bbq.getState(true);
1077         if (! _.isEmpty(state)) {
1078             $(window).trigger('hashchange');
1079         } else {
1080             this.action_manager.do_action({type: 'ir.actions.client', tag: 'default_home'});
1081         }
1082     },
1083     on_hashchange: function(event) {
1084         var state = event.getState(true);
1085         if (!_.isEqual(this._current_state, state)) {
1086             this.action_manager.do_load_state(state);
1087         }
1088         this._current_state = state;
1089     },
1090     do_push_state: function(state) {
1091         var url = '#' + $.param(state);
1092         this._current_state = _.clone(state);
1093         $.bbq.pushState(url);
1094     },
1095     on_menu_action: function(action) {
1096         this.action_manager.do_action(action);
1097     },
1098     do_action: function(action) {
1099         var self = this;
1100         // TODO replace by client action menuclick 
1101         if(action.type === "ir.ui.menu") {
1102             this.do_reload().then(function () {
1103                 self.menu.on_menu_click(null, action.menu_id);
1104             });
1105         }
1106     },
1107 });
1108
1109
1110 openerp.currentScript = function() {
1111     var currentScript = document.currentScript;
1112     if (!currentScript) {
1113         var sc = document.getElementsByTagName('script');
1114         currentScript = sc[sc.length-1];
1115     }
1116     return currentScript;
1117 };
1118
1119 openerp.web.EmbeddedClient = openerp.web.Widget.extend({
1120     template: 'EmptyComponent',
1121     init: function(action_id, options) {
1122         this._super();
1123         // TODO take the xmlid of a action instead of its id 
1124         this.action_id = action_id;
1125         this.options = options || {};
1126         this.am = new openerp.web.ActionManager(this);
1127     },
1128
1129     start: function() {
1130         var self = this;
1131
1132         this.am.appendTo(this.$element.addClass('openerp'));
1133
1134         return this.rpc("/web/action/load", { action_id: this.action_id }, function(result) {
1135             var action = result.result;
1136             action.flags = _.extend({
1137                 //views_switcher : false,
1138                 search_view : false,
1139                 action_buttons : false,
1140                 sidebar : false
1141                 //pager : false
1142             }, self.options, action.flags || {});
1143
1144             self.am.do_action(action);
1145         });
1146     },
1147
1148 });
1149
1150
1151 };
1152
1153 // vim:et fdc=0 fdl=0 foldnestmax=3 fdm=syntax: