Files

2883 lines
115 KiB
JavaScript
Raw Permalink Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/* This file is part of Tryton. The COPYRIGHT file at the top level of
this repository contains the full copyright notices and license terms. */
(function() {
'use strict';
var ENCODINGS = ["866", "ansi_x3.4-1968", "arabic", "ascii",
"asmo-708", "big5", "big5-hkscs", "chinese", "cn-big5", "cp1250",
"cp1251", "cp1252", "cp1253", "cp1254", "cp1255", "cp1256",
"cp1257", "cp1258", "cp819", "cp866", "csbig5", "cseuckr",
"cseucpkdfmtjapanese", "csgb2312", "csibm866", "csiso2022jp",
"csiso2022kr", "csiso58gb231280", "csiso88596e", "csiso88596i",
"csiso88598e", "csiso88598i", "csisolatin1", "csisolatin2",
"csisolatin3", "csisolatin4", "csisolatin5", "csisolatin6",
"csisolatin9", "csisolatinarabic", "csisolatincyrillic",
"csisolatingreek", "csisolatinhebrew", "cskoi8r", "csksc56011987",
"csmacintosh", "csshiftjis", "cyrillic", "dos-874", "ecma-114",
"ecma-118", "elot_928", "euc-jp", "euc-kr", "gb18030", "gb2312",
"gb_2312", "gb_2312-80", "gbk", "greek", "greek8", "hebrew",
"hz-gb-2312", "ibm819", "ibm866", "iso-2022-cn", "iso-2022-cn-ext",
"iso-2022-jp", "iso-2022-kr", "iso-8859-1", "iso-8859-10",
"iso-8859-11", "iso-8859-13", "iso-8859-14", "iso-8859-15",
"iso-8859-16", "iso-8859-2", "iso-8859-3", "iso-8859-4",
"iso-8859-5", "iso-8859-6", "iso-8859-6-e", "iso-8859-6-i",
"iso-8859-7", "iso-8859-8", "iso-8859-8-e", "iso-8859-8-i",
"iso-8859-9", "iso-ir-100", "iso-ir-101", "iso-ir-109",
"iso-ir-110", "iso-ir-126", "iso-ir-127", "iso-ir-138",
"iso-ir-144", "iso-ir-148", "iso-ir-149", "iso-ir-157", "iso-ir-58",
"iso8859-1", "iso8859-10", "iso8859-11", "iso8859-13", "iso8859-14",
"iso8859-15", "iso8859-2", "iso8859-3", "iso8859-4", "iso8859-5",
"iso8859-6", "iso8859-7", "iso8859-8", "iso8859-9", "iso88591",
"iso885910", "iso885911", "iso885913", "iso885914", "iso885915",
"iso88592", "iso88593", "iso88594", "iso88595", "iso88596",
"iso88597", "iso88598", "iso88599", "iso_8859-1", "iso_8859-15",
"iso_8859-1:1987", "iso_8859-2", "iso_8859-2:1987", "iso_8859-3",
"iso_8859-3:1988", "iso_8859-4", "iso_8859-4:1988", "iso_8859-5",
"iso_8859-5:1988", "iso_8859-6", "iso_8859-6:1987", "iso_8859-7",
"iso_8859-7:1987", "iso_8859-8", "iso_8859-8:1988", "iso_8859-9",
"iso_8859-9:1989", "koi", "koi8", "koi8-r", "koi8-ru", "koi8-u",
"koi8_r", "korean", "ks_c_5601-1987", "ks_c_5601-1989", "ksc5601",
"ksc_5601", "l1", "l2", "l3", "l4", "l5", "l6", "l9", "latin1",
"latin2", "latin3", "latin4", "latin5", "latin6", "logical", "mac",
"macintosh", "ms932", "ms_kanji", "shift-jis", "shift_jis", "sjis",
"sun_eu_greek", "tis-620", "unicode-1-1-utf-8", "us-ascii",
"utf-16", "utf-16be", "utf-16le", "utf-8", "utf8", "visual",
"windows-1250", "windows-1251", "windows-1252", "windows-1253",
"windows-1254", "windows-1255", "windows-1256", "windows-1257",
"windows-1258", "windows-31j", "windows-874", "windows-949",
"x-cp1250", "x-cp1251", "x-cp1252", "x-cp1253", "x-cp1254",
"x-cp1255", "x-cp1256", "x-cp1257", "x-cp1258", "x-euc-jp", "x-gbk",
"x-mac-cyrillic", "x-mac-roman", "x-mac-ukrainian", "x-sjis",
"x-user-defined", "x-x-big5"];
Sao.Window = {};
Sao.Window.InfoBar = Sao.class_(Object, {
init: function() {
this.el = jQuery('<div/>', {
'class': 'infobar',
});
this.__messages = new Set();
},
add: function(message, type, kind) {
kind = kind || null;
if (!message) {
return;
}
var key = JSON.stringify([message, type]);
if (!this.__messages.has(key)) {
var infobar = jQuery('<div/>', {
'class': 'alert alert-dismissible alert-' + (
type || 'error'),
'role': 'alert',
}).append(jQuery('<button/>', {
'type': 'button',
'class': 'close',
'aria-label': Sao.i18n.gettext("Close"),
'title': Sao.i18n.gettext("Close"),
'data-dismiss': 'alert',
}).append(jQuery('<span/>', {
'aria-hidden': true,
}).append('&times;'))
).append(jQuery('<span/>')
.css('white-space','pre-wrap')
.text(message))
.on('close.bs.alert',
null, key, this.__response.bind(this));
this.el.append(infobar);
infobar.data('kind', kind);
}
},
__response: function(evt) {
this.__messages.add(evt.data);
},
refresh: function(kind) {
kind = kind || null;
this.el.children().each((i, el) => {
el = jQuery(el);
if (el.data('kind') === kind) {
el.remove();
}
});
},
clear: function() {
let kinds = new Set();
this.el.children().each(
(i, el) => kinds.add(jQuery(el).data('kind')));
kinds.forEach(kind => {
this.refresh(kind);
});
this.__messages.clear();
},
});
Sao.Window.Form = Sao.class_(Object, {
init: function(screen, callback, kwargs) {
kwargs = kwargs || {};
this._position = undefined;
this._length = 0;
this.screen = screen;
this.callback = callback;
this.many = kwargs.many || 0;
this.domain = kwargs.domain || null;
this.context = kwargs.context || null;
this.save_current = kwargs.save_current;
var title_prm = jQuery.when(kwargs.title || '').then(
title => {
if (screen.breadcrumb.length) {
var breadcrumb = jQuery.extend([], screen.breadcrumb);
if (title) {
breadcrumb.push(title);
}
this.title = breadcrumb.slice(-3, -1).map(function(x) {
return Sao.common.ellipsize(x, 30);
}).concat(breadcrumb.slice(-1)).join(' ');
if (breadcrumb.length > 3) {
this.title = '... ' + this.title;
}
} else {
if (!title) {
title = Sao.common.MODELNAME.get(this.screen.model_name);
}
this.title = title;
}
var revision = this.screen.context._datetime;
var label;
if (revision &&
Sao.common.MODELHISTORY.contains(this.screen.model_name)) {
var date_format = Sao.common.date_format(
this.screen.context.date_format);
var time_format = '%H:%M:%S.%f';
var revision_label = ' @ ' + Sao.common.format_datetime(
date_format + ' ' + time_format, revision);
label = Sao.common.ellipsize(
this.title, 80 - revision_label.length) +
revision_label;
title = this.title + revision_label;
} else {
label = Sao.common.ellipsize(this.title, 80);
}
return label;
});
this.prev_view = screen.current_view;
this.screen.screen_container.alternate_view = true;
this.info_bar = new Sao.Window.InfoBar();
var view_type = kwargs.view_type || 'form';
this.switch_prm = this.screen.switch_view(view_type)
.done(() => {
if (kwargs.new_ &&
(this.screen.current_view.view_type == view_type)) {
this.screen.new_(undefined, kwargs.defaults);
}
});
var dialog = new Sao.Dialog('', 'window-form', 'lg', false);
this.dialog = dialog;
this.el = dialog.modal;
this.el.on('keydown', e => {
if (e.which == Sao.common.ESC_KEYCODE) {
e.preventDefault();
this.response('RESPONSE_CANCEL');
}
});
var readonly = this.screen.group.readonly;
this.but_ok = null;
this.but_new = null;
this._initial_value = null;
this.view_type = view_type;
if ((view_type == 'form') && !readonly) {
let label;
if (kwargs.new_) {
label = Sao.i18n.gettext('Delete');
} else {
label = Sao.i18n.gettext("Discard changes");
var record = this.screen.current_record;
this._initial_value = record.get_on_change_value();
if (record.group.parent &&
record.model.fields[record.group.parent_name]) {
var parent_field = record.model.fields[
record.group.parent_name];
this._initial_value[record.group.parent_name] = (
parent_field.get_eval(record));
}
}
dialog.footer.append(jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': label,
}).text(label).click(() => {
this.response('RESPONSE_CANCEL');
}));
}
if (kwargs.new_ && this.many) {
let label;
if (this.save_current && !readonly) {
label = Sao.i18n.gettext("Save and New");
} else {
label = Sao.i18n.gettext("Add and New");
}
dialog.footer.append(jQuery('<button/>', {
'class': 'btn btn-default',
'type': 'button',
'title': label,
}).text(label).click(() => {
this.response('RESPONSE_ACCEPT');
}));
}
if (this.save_current && !readonly) {
let label = Sao.i18n.gettext("Save");
this.but_ok = jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': label,
}).text(label).appendTo(dialog.footer);
} else {
let label;
if (readonly || (view_type == 'tree')) {
label = Sao.i18n.gettext("Close");
} else if (kwargs.new_) {
label = Sao.i18n.gettext("Add");
} else {
label = Sao.i18n.gettext("Apply changes");
}
this.but_ok = jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': label,
}).text(label).appendTo(dialog.footer);
}
dialog.content.submit(e => {
e.preventDefault();
this.response('RESPONSE_OK');
});
if (view_type == 'tree') {
var menu = jQuery('<div/>', {
'class': 'window-form-toolbar'
}).appendTo(dialog.body);
var group = jQuery('<div/>', {
'class': 'input-group input-group-sm'
}).appendTo(menu);
this.wid_text = jQuery('<input/>', {
type: 'input'
}).appendTo(menu);
this.wid_text.hide();
var buttons = jQuery('<div/>', {
'class': 'input-group-btn'
}).appendTo(group);
var access = Sao.common.MODELACCESS.get(this.screen.model_name);
var disable_during = function(callback) {
return function(evt) {
var button = jQuery(evt.target);
button.prop('disabled', true);
(callback(evt) || jQuery.when())
.always(function() {
button.prop('disabled', false);
});
};
};
this.but_switch = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Switch"),
'title': Sao.i18n.gettext("Switch"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-switch')
).appendTo(buttons);
this.but_switch.click(
disable_during(this.switch_.bind(this)));
this.but_previous = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Previous"),
'title': Sao.i18n.gettext("Previous"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-back')
).appendTo(buttons);
this.but_previous.click(
disable_during(this.previous.bind(this)));
this.label = jQuery('<span/>', {
'class': 'badge'
}).appendTo(jQuery('<span/>', {
'class': 'btn hidden-xs',
}).appendTo(buttons));
this.but_next = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Next"),
'title': Sao.i18n.gettext("Next"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-forward')
).appendTo(buttons);
this.but_next.click(disable_during(this.next.bind(this)));
if (this.domain) {
this.wid_text.show();
this.but_add = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Add"),
'title': Sao.i18n.gettext("Add"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-add')
).appendTo(buttons);
this.but_add.click(disable_during(this.add.bind(this)));
this.but_add.prop('disabled', !access.read || readonly);
this.but_remove = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Remove"),
'title': Sao.i18n.gettext("Remove"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-remove')
).appendTo(buttons);
this.but_remove.click(
disable_during(this.remove.bind(this)));
this.but_remove.prop('disabled', !access.read || readonly);
}
this.but_new = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("New"),
'title': Sao.i18n.gettext("New"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-create')
).appendTo(buttons);
this.but_new.click(disable_during(this.new_.bind(this)));
this.but_new.prop('disabled', !access.create || readonly);
this.but_del = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Delete"),
'title': Sao.i18n.gettext("Delete"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-delete')
).appendTo(buttons);
this.but_del.click(disable_during(this.delete_.bind(this)));
this.but_del.prop('disabled', !access['delete'] || readonly);
this.but_undel = jQuery('<button/>', {
'class': 'btn btn-default btn-sm',
'type': 'button',
'aria-label': Sao.i18n.gettext("Undelete"),
'title': Sao.i18n.gettext("Undelete"),
}).append(Sao.common.ICONFACTORY.get_icon_img('tryton-undo')
).appendTo(buttons);
this.but_undel.click(disable_during(this.undelete.bind(this)));
this.but_undel.prop('disabled', !access['delete'] || readonly);
}
var content = jQuery('<div/>').appendTo(dialog.body);
dialog.body.append(this.info_bar.el);
this.screen.windows.push(this);
this.switch_prm.done(() => {
if (this.screen.current_view.view_type != view_type) {
this.destroy();
} else {
title_prm.done(dialog.add_title.bind(dialog));
content.append(this.screen.screen_container.alternate_viewport);
this.el.modal('show');
}
});
this.el.on('shown.bs.modal', event => {
this.screen.display().done(() => {
this.screen.set_cursor();
});
});
this.el.on('hidden.bs.modal', function(event) {
jQuery(this).remove();
});
},
record_message: function(position, size) {
this._position = position;
this._length = size;
let name = '_';
if (position) {
let selected = this.screen.selected_records.length;
name = ' ' + position;
if (selected > 1) {
name += '#' + selected;
}
}
let message = name + '/' + Sao.common.humanize(size);
if (this.label) {
this.label.text(message).attr('title', message);
}
this._set_button_sensitive();
},
record_modified: function() {
this.info_bar.refresh();
this._set_button_sensitive();
},
_set_button_sensitive: function() {
if (this.view_type != 'tree') {
return;
}
let access = Sao.common.MODELACCESS.get(this.screen.model_name);
let first = false,
last = false;
if (typeof this._position == 'number') {
first = this._position <= 1;
last = this._position >= this._length;
}
var deletable =
this.screen.deletable &&
this.screen.selected_records.some((r) => !r.deleted && !r.removed);
let undeletable =
this.screen.selected_records.some((r) => r.deleted || r.removed);
let view_type = this.screen.current_view.view_type;
let has_views = this.screen.number_of_views > 1;
let readonly = this.screen.group.readonly;
this.but_switch.prop(
'disabled',
!((this._position || (view_type == 'form')) && has_views));
this.but_new.prop(
'disabled',
readonly ||
!access.create);
this.but_del.prop(
'disabled',
readonly ||
!access.delete ||
!deletable);
this.but_undel.prop(
'disabled',
readonly ||
!undeletable ||
(typeof this._position != 'number'));
this.but_next.prop(
'disabled',
!this._length ||
last);
this.but_previous.prop(
'disabled',
!this._length ||
first);
if (this.domain) {
this.but_add.prop(
'disabled',
readonly ||
!access.write_access ||
!access.readonly);
this.wid_text.prop('disabled', this.but_add.prop('disabled'));
this.but_remove.prop(
'disabled',
readonly ||
(typeof this._position != 'number') ||
!access.write ||
!access.read);
}
},
add: function() {
var domain = jQuery.extend([], this.domain);
var model_name = this.screen.model_name;
var value = this.wid_text.val();
const callback = result => {
var prm = jQuery.when();
if (!jQuery.isEmptyObject(result)) {
var ids = [];
for (const record of result) {
ids.push(record[0]);
}
this.screen.group.load(ids, true, -1, null);
prm = this.screen.display();
}
prm.done(() => {
this.screen.set_cursor();
});
this.entry.val('');
};
var parser = new Sao.common.DomainParser();
new Sao.Window.Search(model_name, callback, {
sel_multi: true,
context: this.context,
domain: domain,
search_filter: parser.quote(value)
});
},
remove: function() {
this.screen.remove(false, true, false);
},
new_: function() {
this.screen.new_();
this._initial_value = null;
this.many -= 1;
if (this.many == 0) {
this.but_new.addClass('disabled');
}
},
delete_: function() {
this.screen.remove(false, false, false);
},
undelete: function() {
this.screen.unremove();
},
previous: function() {
return this.screen.display_previous();
},
next: function() {
return this.screen.display_next();
},
switch_: function() {
return this.screen.switch_view();
},
response: function(response_id) {
var result;
this.screen.current_view.set_value();
var readonly = this.screen.group.readonly;
if (~['RESPONSE_OK', 'RESPONSE_ACCEPT'].indexOf(response_id) &&
!readonly &&
this.screen.current_record) {
jQuery.when(this.screen.current_record.validate()).then(validate => {
if (validate && this.screen.attributes.pre_validate) {
return this.screen.current_record.pre_validate().then(
() => true, () => false);
}
return validate;
}).then(validate => {
var closing_prm = jQuery.Deferred();
if (validate && this.save_current) {
this.screen.save_current().then(closing_prm.resolve,
closing_prm.reject);
} else if (validate &&
this.screen.current_view.view_type == 'form') {
var view = this.screen.current_view;
var validate_prms = [];
for (var name in view.widgets) {
var widget = view.widgets[name];
if (widget.screen &&
widget.screen.attributes.pre_validate) {
var record = widget.screen.current_record;
if (record) {
validate_prms.push(record.pre_validate());
}
}
}
jQuery.when.apply(jQuery, validate_prms).then(
closing_prm.resolve, closing_prm.reject);
} else if (!validate) {
this.info_bar.add(
this.screen.invalid_message(), 'danger');
closing_prm.reject();
} else {
this.info_bar.clear();
closing_prm.resolve();
}
closing_prm.fail(() => {
this.screen.display().done(() => {
this.screen.set_cursor();
});
});
// TODO Add support for many
closing_prm.done(() => {
if (response_id == 'RESPONSE_ACCEPT') {
this.screen.new_();
this.screen.current_view.display().done(() => {
this.screen.set_cursor();
});
this.many -= 1;
if (this.many === 0) {
this.but_new.prop('disabled', true);
}
} else {
result = true;
if (this.callback) {
this.callback(result);
}
this.destroy();
}
});
});
return;
}
var cancel_prm = null;
if (response_id == 'RESPONSE_CANCEL' &&
!readonly &&
this.screen.current_record) {
result = false;
var record = this.screen.current_record;
var added = record.modified_fields.id;
if ((record.id < 0) || this.save_current) {
cancel_prm = this.screen.cancel_current(
this._initial_value);
} else if (record.modified) {
record.cancel();
cancel_prm = record.reload().then(() => {
this.screen.display();
});
}
if (added) {
record.modified_fields.id = added;
}
} else {
result = (response_id != 'RESPONSE_CANCEL') && !readonly;
}
(cancel_prm || jQuery.when()).then(() => {
if (this.callback) {
this.callback(result);
}
this.destroy();
});
},
destroy: function() {
this.screen.windows.splice(this.screen.windows.indexOf(this), 1);
this.screen.screen_container.alternate_view = false;
this.screen.screen_container.alternate_viewport.children()
.detach();
if (this.prev_view) {
// Empty when opening from Many2One
this.screen.switch_view(this.prev_view.view_type);
}
this.el.modal('hide');
}
});
Sao.Window.Attachment = Sao.class_(Sao.Window.Form, {
init: function(record, callback) {
this.resource = record.model.name + ',' + record.id;
this.attachment_callback = callback;
var context = jQuery.extend({}, record.get_context());
var screen = new Sao.Screen('ir.attachment', {
domain: [['resource', '=', this.resource]],
mode: ['tree', 'form'],
context: context,
});
var title = record.rec_name().then(function(rec_name) {
return Sao.i18n.gettext('Attachments (%1)', rec_name);
});
Sao.Window.Attachment._super.init.call(this, screen, this.callback,
{view_type: 'tree', title: title});
this.switch_prm = this.switch_prm.then(function() {
return screen.search_filter();
});
},
callback: function(result) {
var prm = jQuery.when();
if (result) {
prm = this.screen.save_current();
}
if (this.attachment_callback) {
prm.always(this.attachment_callback.bind(this));
}
},
add_data: function(data, filename) {
var screen = this.screen;
this.switch_prm.then(function() {
screen.new_().then(function(record) {
var data_field = record.model.fields.data;
record.field_set_client(
data_field.description.filename, filename);
record.field_set_client('data', data);
screen.display();
});
});
},
add_uri: function(uri) {
var screen = this.screen;
this.switch_prm.then(function() {
screen.current_record = null;
screen.switch_view('form').then(function() {
screen.new_().then(function(record) {
record.field_set_client('link', uri);
record.field_set_client('type', 'link');
screen.display();
});
});
});
},
add_text: function(text) {
var screen = this.screen;
this.switch_prm.then(function() {
screen.current_record = null;
screen.switch_view('form').then(function() {
screen.new_().then(function(record) {
record.field_set_client('description', text);
screen.display();
});
});
});
},
});
Sao.Window.Attachment.get_attachments = function(record) {
var prm;
if (record && (record.id >= 0)) {
var context = record.get_context();
prm = Sao.rpc({
'method': 'model.ir.attachment.search_read',
'params': [
[['resource', '=', record.model.name + ',' + record.id]],
0, 20, null, ['rec_name', 'name', 'type', 'link'],
context],
}, record.model.session);
} else {
prm = jQuery.when([]);
}
var partial = function(callback, attachment, context, session) {
return function() {
return callback(attachment, context, session);
};
};
return prm.then(function(attachments) {
return attachments.map(function(attachment) {
var name = attachment.rec_name;
if (attachment.type == 'link') {
return [name, attachment.link];
} else {
var callback = Sao.Window.Attachment[
'open_' + attachment.type];
return [name, partial(
callback, attachment, context, record.model.session)];
}
});
});
};
Sao.Window.Attachment.open_data = function(attachment, context, session) {
Sao.rpc({
'method': 'model.ir.attachment.read',
'params': [
[attachment.id], ['data'], context],
}, session).then(function(values) {
Sao.common.download_file(values[0].data, attachment.name);
});
};
Sao.Window.Note = Sao.class_(Sao.Window.Form, {
init: function(record, callback) {
this.resource = record.model.name + ',' + record.id;
this.note_callback = callback;
var context = jQuery.extend({}, record.get_context());
var screen = new Sao.Screen('ir.note', {
domain: [['resource', '=', this.resource]],
mode: ['tree', 'form'],
context: context,
});
var title = record.rec_name().then(function(rec_name) {
return Sao.i18n.gettext('Notes (%1)', rec_name);
});
Sao.Window.Note._super.init.call(this, screen, this.callback,
{view_type: 'tree', title: title});
this.switch_prm = this.switch_prm.then(function() {
return screen.search_filter();
});
},
callback: function(result) {
var prm = jQuery.when();
if (result) {
var unread = this.screen.group.model.fields.unread;
for (const record of this.screen.group) {
if (record.get_loaded() || record.id < 0) {
if (!record.modified_fields.unread) {
unread.set_client(record, false);
}
}
}
prm = this.screen.save_current();
}
if (this.note_callback) {
prm.always(this.note_callback.bind(this));
}
}
});
Sao.Window.Log = Sao.class_(Sao.Window.Form, {
init: function(record) {
this.resource = record.model.name + ',' + record.id;
var title = record.rec_name().then(function(rec_name) {
return Sao.i18n.gettext("Logs (%1)", rec_name);
});
var context = jQuery.extend({}, record.get_context());
var form = jQuery('<div/>', {
'class': 'form',
});
var table = jQuery('<table/>', {
'class': 'form-container responsive responsive-noheader',
}).appendTo(form);
var body = jQuery('<tbody/>').appendTo(table);
record.model.execute(
'read', [[record.id],
['create_uid.rec_name', 'create_date',
'write_uid.rec_name', 'write_date',
'xml_id']], context
).then(([log]) => {
var row, cell, label, input;
row = jQuery('<tr/>').appendTo(body);
cell = jQuery('<td/>').css('text-align', 'end').appendTo(row);
label = jQuery('<label/>', {
'class': 'form-label',
'text': Sao.i18n.gettext("Model:"),
}).appendTo(cell);
label.uniqueId();
cell = jQuery('<td/>').appendTo(row);
input = jQuery('<input/>', {
'readonly': true,
'class': 'form-control input-sm',
'aria-labelledby': label.attr('id'),
}).val(record.model.name).appendTo(cell);
input.uniqueId();
cell = jQuery('<td/>').css('text-align', 'end').appendTo(row);
label = jQuery('<label/>', {
'class': 'form-label',
'text': Sao.i18n.gettext("ID:"),
}).appendTo(cell);
label.uniqueId();
cell = jQuery('<td/>').appendTo(row);
input = jQuery('<input/>', {
'type': 'integer',
'readonly': true,
'class': 'form-control input-sm',
'aria-labelledby': label.attr('id'),
}).val(record.id).appendTo(cell);
input.css('text-align', 'end');
input.uniqueId();
if (log.xml_id) {
const [module, xml_id] = log.xml_id.split('.', 2);
row = jQuery('<tr/>').appendTo(body);
cell = jQuery('<td/>').css('text-align', 'end').appendTo(row);
label = jQuery('<label/>', {
'class': 'form-label',
'text': Sao.i18n.gettext("Module:"),
}).appendTo(cell);
label.uniqueId();
cell = jQuery('<td/>').appendTo(row);
input = jQuery('<input/>', {
'readonly': true,
'class': 'form-control input-sm',
'aria-labelledby': label.attr('id'),
}).val(module).appendTo(cell);
input.uniqueId();
cell = jQuery('<td/>').css('text-align', 'end').appendTo(row);
label = jQuery('<label/>', {
'class': 'form-label',
'text': Sao.i18n.gettext("XML ID:"),
}).appendTo(cell);
label.uniqueId();
cell = jQuery('<td/>').appendTo(row);
input = jQuery('<input/>', {
'readonly': true,
'class': 'form-control input-sm',
'aria-labelledby': label.attr('id'),
}).val(xml_id).appendTo(cell);
input.uniqueId();
}
[['create_uid.', Sao.i18n.gettext("Created by:"),
'create_date', Sao.i18n.gettext("Created at:")],
['write_uid.', Sao.i18n.gettext("Last Modified by:"),
'write_date', Sao.i18n.gettext("Last Modified at:")],
].forEach(([user, user_label, date, date_label]) => {
row = jQuery('<tr/>').appendTo(body);
cell = jQuery('<td/>').css('text-align', 'end').appendTo(row);
label = jQuery('<label/>', {
'class': 'form-label',
'text': user_label,
}).appendTo(cell);
label.uniqueId();
cell = jQuery('<td/>').appendTo(row);
user = log[user];
if (user) {
user = user.rec_name;
}
input = jQuery('<input/>', {
'readonly': true,
'class': 'form-control input-sm',
'aria-labelledby': label.attr('id'),
}).val(user || '').appendTo(cell);
input.css('width', '50ch');
input.uniqueId();
cell = jQuery('<td/>').css('text-align', 'end').appendTo(row);
label = jQuery('<label/>', {
'class': 'form-label',
'text': date_label,
}).appendTo(cell);
label.uniqueId();
cell = jQuery('<td/>').appendTo(row);
date = log[date];
if (date) {
date = Sao.common.format_datetime(
Sao.common.date_format() + ' %H:%M:%S', date);
}
input = jQuery('<input/>', {
'readonly': true,
'class': 'form-control input-sm',
'aria-labelledby': label.attr('id'),
}).val(date || '').appendTo(cell);
input.css('width', date.length + 'ch');
input.uniqueId();
});
});
var screen = new Sao.Screen('ir.model.log', {
domain: [['resource', '=', this.resource]],
mode: ['tree', 'form'],
context: context,
});
Sao.Window.Log._super.init.call(
this, screen, null, {view_type: 'tree', title: title});
this.switch_prm = this.switch_prm.then(function() {
return screen.search_filter();
});
this.dialog.body.prepend(form);
},
});
Sao.Window.Search = Sao.class_(Object, {
init: function(model, callback, kwargs) {
kwargs = kwargs || {};
var views_preload = kwargs.views_preload || {};
this.model_name = model;
this.domain = kwargs.domain || [];
this.context = kwargs.context || {};
this.order = kwargs.order || null;
this.view_ids = kwargs.view_ids;
this.views_preload = views_preload;
this.sel_multi = kwargs.sel_multi;
this.callback = callback;
var title = kwargs.title;
if (!title) {
title = Sao.common.MODELNAME.get(model);
}
this.title = title;
this.exclude_field = kwargs.exclude_field || null;
var dialog = new Sao.Dialog(Sao.i18n.gettext(
'Search %1', this.title), '', 'lg', false);
this.el = dialog.modal;
this.el.on('keydown', e => {
if (e.which == Sao.common.ESC_KEYCODE) {
e.preventDefault();
this.response('RESPONSE_CANCEL');
}
});
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Cancel"),
}).text(Sao.i18n.gettext('Cancel')).click(() => {
this.response('RESPONSE_CANCEL');
}).appendTo(dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-default',
'type': 'button',
'title': Sao.i18n.gettext("Find"),
}).text(Sao.i18n.gettext('Find')).click(() => {
this.response('RESPONSE_APPLY');
}).appendTo(dialog.footer);
if (kwargs.new_ && Sao.common.MODELACCESS.get(model).create) {
jQuery('<button/>', {
'class': 'btn btn-default',
'type': 'button',
'title': Sao.i18n.gettext("New"),
}).text(Sao.i18n.gettext('New')).click(() => {
this.response('RESPONSE_ACCEPT');
}).appendTo(dialog.footer);
}
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("OK"),
}).text(Sao.i18n.gettext('OK')).appendTo(dialog.footer);
dialog.content.submit(e => {
e.preventDefault();
this.response('RESPONSE_OK');
});
this.screen = new Sao.Screen(model, {
mode: ['tree'],
context: this.context,
domain: this.domain,
order: this.order,
view_ids: kwargs.view_ids,
views_preload: views_preload,
row_activate: this.activate.bind(this),
readonly: true,
breadcrumb: [this.title],
});
this.screen.load_next_view().done(() => {
this.screen.switch_view().done(() => {
if (!this.sel_multi) {
this.screen.current_view.selection_mode = (
Sao.common.SELECTION_SINGLE);
} else {
this.screen.current_view.selection_mode = (
Sao.common.SELECTION_MULTIPLE);
}
dialog.body.append(this.screen.screen_container.el);
this.el.modal('show');
this.screen.display();
if (kwargs.search_filter !== undefined) {
this.screen.search_filter(kwargs.search_filter);
}
});
});
this.el.on('shown.bs.modal', () => {
this.screen.screen_container.search_entry.focus();
});
this.el.on('hidden.bs.modal', function(event) {
jQuery(this).remove();
});
},
activate: function() {
this.response('RESPONSE_OK');
},
response: function(response_id) {
var records;
var value = [];
if (response_id == 'RESPONSE_OK') {
records = this.screen.current_view.selected_records;
} else if (response_id == 'RESPONSE_APPLY') {
this.screen.search_filter(
this.screen.screen_container.get_text());
return;
} else if (response_id == 'RESPONSE_ACCEPT') {
var view_ids = jQuery.extend([], this.view_ids);
if (!jQuery.isEmptyObject(view_ids)) {
// Remove the first tree view as mode is form only
view_ids.shift();
}
var screen = new Sao.Screen(this.model_name, {
domain: this.domain,
context: this.context,
order: this.order,
mode: ['form'],
view_ids: view_ids,
views_preload: this.views_preload,
exclude_field: this.exclude_field,
});
var callback = function(result) {
if (result) {
var record = screen.current_record;
this.callback([[record.id,
record._values.rec_name || '']]);
} else {
this.callback(null);
}
};
this.el.modal('hide');
new Sao.Window.Form(screen, callback.bind(this), {
new_: true,
save_current: true,
});
return;
}
if (records) {
var index, record;
for (index in records) {
record = records[index];
value.push([record.id, record._values.rec_name || '']);
}
}
this.callback(value);
this.el.modal('hide');
}
});
Sao.Window.Preferences = Sao.class_(Object, {
init: function(callback) {
var dialog = new Sao.Dialog('Preferences', '', 'lg');
this.el = dialog.modal;
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Cancel"),
}).text(Sao.i18n.gettext('Cancel')).click(() => {
this.response('RESPONSE_CANCEL');
}).appendTo(dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("OK"),
}).text(Sao.i18n.gettext('OK')).appendTo(dialog.footer);
dialog.content.submit(e => {
e.preventDefault();
this.response('RESPONSE_OK');
});
this.screen = new Sao.Screen('res.user', {
mode: []
});
// Reset readonly set automaticly by MODELACCESS
this.screen.attributes.readonly = false;
this.screen.group.readonly = false;
this.screen.group.skip_model_access = true;
const set_view = view => {
this.screen.add_view(view);
this.screen.switch_view().done(() => {
this.screen.new_(false);
this.screen.model.execute('get_preferences', [false], {})
.then(set_preferences, this.destroy);
});
};
const set_preferences = preferences => {
this.screen.current_record.cancel();
this.screen.current_record.set(preferences);
this.screen.current_record.id =
this.screen.model.session.user_id;
this.screen.current_record.validate(null, true);
this.screen.display(true);
dialog.body.append(this.screen.screen_container.el);
this.el.modal('show');
};
this.el.on('hidden.bs.modal', function(event) {
callback();
jQuery(this).remove();
});
this.screen.model.execute('get_preferences_fields_view', [], {})
.then(set_view, this.destroy);
},
response: function(response_id) {
if (response_id == 'RESPONSE_OK') {
if (this.screen.current_record.validate()) {
var values = jQuery.extend({}, this.screen.get());
return this.screen.model.execute(
'set_preferences', [values], {})
.then(() => this.destroy());
}
return;
}
this.destroy();
},
destroy: function() {
this.el.modal('hide');
}
});
Sao.Window.Revision = Sao.class_(Object, {
init: function(revisions, revision, callback) {
this.callback = callback;
var dialog = new Sao.Dialog(
Sao.i18n.gettext('Revision'), '', 'lg');
this.el = dialog.modal;
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Cancel"),
}).text(Sao.i18n.gettext('Cancel')).click(() => {
this.response('RESPONSE_CANCEL');
}).appendTo(dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("OK"),
}).text(Sao.i18n.gettext('OK')).appendTo(dialog.footer);
dialog.content.submit(e => {
e.preventDefault();
this.response('RESPONSE_OK');
});
var group = jQuery('<div/>', {
'class': 'form-group'
}).appendTo(dialog.body);
jQuery('<label/>', {
'for': 'revision',
'text': 'Revision'
}).appendTo(group);
this.select = jQuery('<select/>', {
'class': 'form-control',
id: 'revision',
'placeholder': Sao.i18n.gettext('Revision')
}).appendTo(group);
var date_format = Sao.common.date_format();
var time_format = '%H:%M:%S.%f';
this.select.append(jQuery('<option/>', {
value: null,
text: ''
}));
for (let rev of revisions) {
var name = rev[2];
rev = rev[0];
this.select.append(jQuery('<option/>', {
value: rev.valueOf(),
text: Sao.common.format_datetime(
date_format + ' ' + time_format, rev) + ' ' + name,
}));
}
if (revision) {
this.select.val(revision.valueOf());
}
this.el.modal('show');
this.el.on('hidden.bs.modal', function(event) {
jQuery(this).remove();
});
},
response: function(response_id) {
var revision = null;
if (response_id == 'RESPONSE_OK') {
revision = this.select.val();
if (revision) {
revision = Sao.DateTime(parseInt(revision, 10));
}
}
this.el.modal('hide');
this.callback(revision);
}
});
Sao.Window.CSV = Sao.class_(Object, {
init: function(title) {
this.languages = Sao.rpc({
'method': 'model.ir.lang.search_read',
'params': [
[['translatable', '=', true]],
0, null, null, ['code', 'name'],
{},
],
}, this.session, false);
this.dialog = new Sao.Dialog(title, 'csv', 'lg');
this.el = this.dialog.modal;
this.fields = {};
this.fields_model = {};
var row_fields = jQuery('<div/>', {
'class': 'row'
}).appendTo(this.dialog.body);
var column_fields_all = jQuery('<div/>', {
'class': 'col-md-5',
}).append(jQuery('<div/>', {
'class': 'panel panel-default',
}).append(jQuery('<div/>', {
'class': 'panel-heading',
}).append(jQuery('<h3/>', {
'class': 'panel-title',
'text': Sao.i18n.gettext('All Fields')
})))).appendTo(row_fields);
this.fields_all = jQuery('<ul/>', {
'class': 'list-unstyled column-fields panel-body'
}).css('cursor', 'pointer')
.appendTo(column_fields_all.find('.panel'));
this.model_populate(this._get_fields(this.screen.model_name));
this.view_populate(this.fields_model, this.fields_all);
this.column_buttons = jQuery('<div/>', {
'class': 'col-md-2'
}).appendTo(row_fields);
jQuery('<button/>', {
'class': 'btn btn-default btn-block',
'type': 'button',
'title': Sao.i18n.gettext("Add"),
}).text(' ' + Sao.i18n.gettext('Add')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-add')
).click(() => {
this.fields_all.find('.bg-primary').each((i, el_field) => {
this.sig_sel_add(el_field);
});
})
.appendTo(this.column_buttons);
jQuery('<button/>', {
'class': 'btn btn-default btn-block',
'type': 'button',
'title': Sao.i18n.gettext("Remove"),
}).text(' ' + Sao.i18n.gettext('Remove')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-remove')
).click(() => {
// sig_unsel
this.fields_selected.children('li.bg-primary').remove();
})
.appendTo(this.column_buttons);
jQuery('<button/>', {
'class': 'btn btn-default btn-block',
'type': 'button',
'title': Sao.i18n.gettext("Clear"),
}).text(' ' + Sao.i18n.gettext('Clear')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-clear')
).click(() => {
this.fields_selected.empty();
})
.appendTo(this.column_buttons);
jQuery('<hr>').appendTo(this.column_buttons);
var column_fields_selected = jQuery('<div/>', {
'class': 'col-md-5',
}).append(jQuery('<div/>', {
'class': 'panel panel-default',
}).append(jQuery('<div/>', {
'class': 'panel-heading',
}).append(jQuery('<h3/>', {
'class': 'panel-title',
'text': Sao.i18n.gettext('Fields Selected')
})))).appendTo(row_fields);
this.fields_selected = jQuery('<ul/>', {
'class': 'list-unstyled column-fields panel-body',
}).css('cursor', 'pointer')
.appendTo(column_fields_selected.find('.panel'));
this.chooser_form = jQuery('<div/>', {
'class': 'form-inline'
}).appendTo(this.dialog.body);
var row_csv_param = jQuery('<div/>', {
}).appendTo(this.dialog.body);
jQuery('<label/>', {
'text': Sao.i18n.gettext('CSV Parameters')
}).append(jQuery('<span/>', {
'class': 'caret',
}).html('&nbsp;'))
.css('cursor', 'pointer')
.on('click', () => {
this.expander_csv.collapse('toggle');
}).appendTo(row_csv_param);
this.expander_csv = jQuery('<div/>', {
'id': 'expander_csv',
'class': 'collapse form-inline'
}).appendTo(row_csv_param);
var delimiter_label = jQuery('<label/>', {
'text': Sao.i18n.gettext('Delimiter:'),
'class': 'control-label',
'for': 'input-delimiter'
});
var separator = ',';
if (navigator.platform &&
navigator.platform.slice(0, 3) == 'Win') {
separator = ';';
}
this.el_csv_delimiter = jQuery('<input/>', {
'type': 'text',
'class': 'form-control',
'id': 'input-delimiter',
'size': '1',
'maxlength': '1',
'value': separator
});
jQuery('<div/>', {
'class': 'form-group'
}).append(delimiter_label)
.append(this.el_csv_delimiter)
.appendTo(this.expander_csv);
this.expander_csv.append(' ');
var quotechar_label = jQuery('<label/>', {
'text': Sao.i18n.gettext('Quote Char:'),
'class': 'control-label',
'for': 'input-quotechar'
});
this.el_csv_quotechar = jQuery('<input/>', {
'type': 'text',
'class': 'form-control',
'id': 'input-quotechar',
'size': '1',
'maxlength': '1',
'value': '"',
});
jQuery('<div/>', {
'class': 'form-group'
}).append(quotechar_label)
.append(this.el_csv_quotechar)
.appendTo(this.expander_csv);
this.expander_csv.append(' ');
this.el.modal('show');
this.el.on('hidden.bs.modal', function() {
jQuery(this).remove();
});
},
_get_fields: function(model) {
return Sao.rpc({
'method': 'model.' + model + '.fields_get'
}, this.session, false);
},
on_row_expanded: function(node) {
var container_view = jQuery('<ul/>').css('list-style', 'none')
.insertAfter(node.view);
this.children_expand(node);
this.view_populate(node.children, container_view);
},
destroy: function() {
this.el.modal('hide');
}
});
Sao.Window.Import = Sao.class_(Sao.Window.CSV, {
init: function(name, screen) {
this.name = name;
this.screen = screen;
this.session = Sao.Session.current_session;
this.fields_data = {}; // Ask before Removing this.
this.fields_invert = {};
Sao.Window.Import._super.init.call(this,
Sao.i18n.gettext('CSV Import: %1', name));
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Cancel"),
}).text(Sao.i18n.gettext("Cancel")).click(() => {
this.response('RESPONSE_CANCEL');
}).appendTo(this.dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("Import"),
}).text(Sao.i18n.gettext("Import")).click(e => {
e.preventDefault();
this.response('RESPONSE_OK');
}).appendTo(this.dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-default btn-block',
'type': 'button',
'title': Sao.i18n.gettext("Auto-Detect"),
}).text(' ' + Sao.i18n.gettext('Auto-Detect')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-search')
).click(() => {
this.autodetect();
})
.appendTo(this.column_buttons);
var chooser_label = jQuery('<label/>', {
'text': Sao.i18n.gettext('File to Import'),
'class': 'col-sm-6 control-label',
'for': 'input-csv-file'
});
this.file_input = jQuery('<input/>', {
'type': 'file',
'id': 'input-csv-file'
});
jQuery('<div/>', {
'class': 'form-group'
}).append(chooser_label).append(jQuery('<div/>', {
'class': 'col-sm-6'
}).append(this.file_input))
.appendTo(this.chooser_form);
var encoding_label = jQuery('<label/>', {
'text': Sao.i18n.gettext('Encoding:'),
'class': 'control-label',
'for': 'input-encoding'
});
this.el_csv_encoding = jQuery('<select/>', {
'class': 'form-control',
'id': 'input-encoding'
});
for (const encoding of ENCODINGS) {
jQuery('<option/>', {
'val': encoding,
}).append(encoding).appendTo(this.el_csv_encoding);
}
var enc = 'utf-8';
if (navigator.platform &&
navigator.platform.slice(0, 3) == 'Win') {
enc = 'cp1252';
}
this.el_csv_encoding.children('option[value="' + enc + '"]')
.attr('selected', 'selected');
jQuery('<div/>', {
'class': 'form-group'
}).append(encoding_label)
.append(this.el_csv_encoding)
.appendTo(this.expander_csv);
this.expander_csv.append(' ');
var skip_label = jQuery('<label/>', {
'text': Sao.i18n.gettext('Lines to Skip:'),
'class': 'control-label',
'for': 'input-skip'
});
this.el_csv_skip = jQuery('<input/>', {
'type': 'number',
'class': 'form-control',
'id': 'input-skip',
'value': '0'
});
jQuery('<div/>', {
'class': 'form-group'
}).append(skip_label)
.append(this.el_csv_skip)
.appendTo(this.expander_csv);
this.expander_csv.append(' ');
Sortable.create(this.fields_selected.get(0), {
handle: '.draggable-handle',
ghostClass: 'dragged-row'
});
},
sig_sel_add: function(el_field) {
el_field = jQuery(el_field);
this._add_node(el_field.attr('field'), el_field.attr('name'));
},
_add_node: function(field, name) {
jQuery('<li/>', {
'field': field,
'class': 'draggable-handle',
}).text(name).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-drag')
).click(function(evt) {
const node = jQuery(evt.target);
if (evt.ctrlKey || evt.metaKey) {
node.toggleClass('bg-primary');
} else {
node.addClass('bg-primary');
node.siblings().removeClass('bg-primary');
}
}).appendTo(this.fields_selected);
},
view_populate: function (parent_node, parent_view) {
var fields_order = Object.keys(parent_node).sort(function(a,b) {
if (parent_node[b].string < parent_node[a].string) {
return -1;
}
else {
return 1;
}
}).reverse();
fields_order.forEach(field => {
var name = parent_node[field].string || field;
var node = jQuery('<li/>', {
'field': parent_node[field].field,
'name': parent_node[field].name
}).text(name).click(e => {
if (e.ctrlKey || e.metaKey) {
node.toggleClass('bg-primary');
} else {
this.fields_all.find('li').removeClass('bg-primary');
node.addClass('bg-primary');
}
}).appendTo(parent_view);
parent_node[field].view = node;
var expander_icon = Sao.common.ICONFACTORY
.get_icon_img('tryton-arrow-right')
.data('expanded', false)
.click(e => {
e.stopPropagation();
var icon;
var expanded = expander_icon.data('expanded');
expander_icon.data('expanded', !expanded);
if (expanded) {
icon = 'tryton-arrow-right';
node.next('ul').remove();
} else {
icon = 'tryton-arrow-down';
this.on_row_expanded(parent_node[field]);
}
Sao.common.ICONFACTORY.get_icon_url(icon)
.then(function(url) {
expander_icon.attr('src', url);
});
}).prependTo(node);
expander_icon.css(
'visibility',
parent_node[field].children ? 'visible' : 'hidden');
});
},
model_populate: function (fields, parent_node, prefix_field,
prefix_name) {
parent_node = parent_node || this.fields_model;
prefix_field = prefix_field || '';
prefix_name = prefix_name || '';
for (const field_name of Object.keys(fields)) {
const field = fields[field_name];
if(!field.readonly || field_name == 'id') {
var name = field.string || field_name;
name = prefix_name + name;
// Only One2Many can be nested for import
var relation;
if (field.type == 'one2many') {
relation = field.relation;
} else {
relation = null;
}
var node = {
name: name,
field: prefix_field + field_name,
relation: relation,
string: field.string
};
parent_node[field_name] = node;
this.fields[prefix_field + field_name] = node;
this.fields_invert[name] = prefix_field + field_name;
if (relation) {
node.children = {};
} else if (field.translate) {
node.children = {};
for (const language of this.languages) {
const l_field_name = (
`${field_name}:lang=${language.code}`);
const l_name = (
prefix_name + name + ` (${language.name})`);
const l_node = {
name: l_name,
field: prefix_field + l_field_name,
relation: null,
string: language.name,
};
node.children[l_field_name] = l_node;
this.fields[prefix_field + l_field_name] = l_node;
this.fields_invert[l_name] = (
prefix_field + l_field_name);
}
}
}
}
},
children_expand: function(node) {
if (jQuery.isEmptyObject(node.children) && node.relation) {
this.model_populate(
this._get_fields(node.relation), node.children,
node.field + '/', node.name + '/');
}
},
autodetect: function() {
var fname = this.file_input.val();
if(!fname) {
Sao.common.message.run(
Sao.i18n.gettext('You must select an import file first.'));
return;
}
this.fields_selected.empty();
this.el_csv_skip.val(1);
Papa.parse(this.file_input[0].files[0], {
delimiter: this.el_csv_delimiter.val(),
quoteChar: this.el_csv_quotechar.val(),
preview: 1,
encoding: this.el_csv_encoding.val(),
error: (err, file, inputElem, reason) => {
Sao.common.warning.run(
reason,
Sao.i18n.gettext("Detection failed"));
},
complete: results => {
results.data[0].every(word => {
if (!(word in this.fields_invert) && !(word in this.fields)) {
var fields = this.fields_model;
var prefix = '';
var parents = word.split('/').slice(0, -1);
this._traverse(fields, prefix, parents, 0);
}
return this._auto_select(word);
});
}
});
},
_auto_select: function(word) {
var name,field;
if(word in this.fields_invert) {
name = word;
field = this.fields_invert[word];
}
else if (word in this.fields) {
name = this.fields[word].name;
field = [word];
}
else {
Sao.common.warning.run(
Sao.i18n.gettext('Unknown column header "%1"', word),
Sao.i18n.gettext('Error'));
return false;
}
this._add_node(field, name);
return true;
},
_traverse: function(fields, prefix, parents, i) {
var field, item;
var names = Object.keys(fields);
for (item = 0; item<names.length; item++) {
field = fields[names[item]];
if (field.name == (prefix + parents[i]) ||
field.field == (prefix + parents[i])) {
this.children_expand(field);
prefix += parents[i] + '/';
if (field.children) {
this._traverse(field.children, prefix, parents, ++i);
}
break;
}
}
},
response: function(response_id) {
if(response_id == 'RESPONSE_OK') {
var fields = [];
this.fields_selected.children('li').each((i, field_el) => {
fields.push(field_el.getAttribute('field'));
});
var fname = this.file_input.val();
if(fname) {
this.import_csv(fname, fields).then(() => {
this.destroy();
});
} else {
this.destroy();
}
}
else {
this.destroy();
}
},
import_csv: function(fname, fields) {
var skip = this.el_csv_skip.val();
var encoding = this.el_csv_encoding.val();
var prm = jQuery.Deferred();
Papa.parse(this.file_input[0].files[0], {
delimiter: this.el_csv_delimiter.val(),
quoteChar: this.el_csv_quotechar.val(),
encoding: encoding,
error: (err, file, inputElem, reason) => {
Sao.common.warning.run(
reason,
Sao.i18n.gettext("Import failed"))
.always(prm.reject);
},
complete: results => {
var data = results.data.slice(skip, results.data.length - 1);
Sao.rpc({
'method': 'model.' + this.screen.model_name +
'.import_data',
'params': [fields, data, {}]
}, this.session).then(count => {
return Sao.common.message.run(
Sao.i18n.ngettext('%1 record imported',
'%1 records imported', count));
}).then(prm.resolve, prm.reject);
}
});
return prm.promise();
}
});
Sao.Window.Export = Sao.class_(Sao.Window.CSV, {
init: function(name, screen, names) {
this.name = name;
this.screen = screen;
this.session = Sao.Session.current_session;
Sao.Window.Export._super.init.call(this,
Sao.i18n.gettext('CSV Export: %1',name));
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Close"),
}).text(Sao.i18n.gettext("Close")).click(() => {
this.response('RESPONSE_CLOSE');
}).appendTo(this.dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("Save As..."),
}).text(Sao.i18n.gettext("Save As...")).click(e => {
e.preventDefault();
this.response('RESPONSE_OK');
}).appendTo(this.dialog.footer);
this.info_bar = new Sao.Window.InfoBar();
this.dialog.body.append(this.info_bar.el);
var fields = this.screen.model.fields;
for (const name of names) {
var type = fields[name].description.type;
if (type == 'selection') {
this.sel_field(name + '.translated');
} else if (type == 'reference') {
this.sel_field(name + '.translated');
this.sel_field(name + '/rec_name');
} else {
this.sel_field(name);
}
}
this.predef_exports = {};
this.fill_predefwin();
jQuery('<button/>', {
'class': 'btn btn-default btn-block',
'type': 'button',
'title': Sao.i18n.gettext("Save Export"),
}).text(' ' + Sao.i18n.gettext('Save Export')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-save')
).click(() => {
this.addreplace_predef();
})
.appendTo(this.column_buttons);
this.button_url = jQuery('<a/>', {
'class': 'btn btn-default btn-block',
'target': '_blank',
'rel': 'noreferrer noopener',
'title': Sao.i18n.gettext("URL Export"),
}).text(' ' + Sao.i18n.gettext("URL Export")).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-public')
)
.appendTo(this.column_buttons);
this.dialog.body.on('change click', this.set_url.bind(this));
jQuery('<button/>', {
'class': 'btn btn-default btn-block',
'type': 'button',
'title': Sao.i18n.gettext("Delete Export"),
}).text(' ' + Sao.i18n.gettext('Delete Export')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-delete')
).click(() => {
this.remove_predef();
})
.appendTo(this.column_buttons);
var predefined_exports_column = jQuery('<div/>', {
'class': 'panel panel-default',
}).append(jQuery('<div/>', {
'class': 'panel-heading',
}).append(jQuery('<h3/>', {
'class': 'panel-title',
'text': Sao.i18n.gettext('Predefined Exports')
}))).appendTo(this.column_buttons);
this.predef_exports_list = jQuery('<ul/>', {
'class': 'list-unstyled predef-exports panel-body'
}).css('cursor', 'pointer')
.appendTo(predefined_exports_column);
var selected = !this.screen_is_tree && this.screen_has_selected
this.selected_records = jQuery('<select/>', {
'class': 'form-control',
'id': 'input-records',
}).append(jQuery('<option/>', {
'val': true,
'selected': selected,
}).text(Sao.i18n.gettext("Selected Records")))
.append(jQuery('<option/>', {
'val': false,
'selected': !selected,
}).text(Sao.i18n.gettext("Listed Records")));
this.ignore_search_limit = jQuery('<input/>', {
'type': 'checkbox',
});
this.selected_records.change(() => {
this.ignore_search_limit.parents('.form-group').first().toggle(
!JSON.parse(this.selected_records.val()) &&
!this.screen_is_tree);
});
jQuery('<div/>', {
'class': 'form-group',
}).appendTo(this.chooser_form)
.append(jQuery('<label/>', {
'text': Sao.i18n.gettext("Export:"),
'class': 'control-label',
'for': 'input-records',
})).append(this.selected_records);
jQuery('<div/>', {
'class': 'form-group',
}).appendTo(this.chooser_form)
.append(jQuery('<div/>', {
'class': 'checkbox',
}).append(jQuery('<label/>', {
'text': ' ' + Sao.i18n.gettext("Ignore search limit")
}).prepend(this.ignore_search_limit)))
.toggle(!selected && !this.screen_is_tree);
this.el_csv_locale = jQuery('<input/>', {
'type': 'checkbox',
'checked': 'checked',
});
jQuery('<div/>', {
'class': 'checkbox',
}).append(jQuery('<label/>', {
'text': ' ' + Sao.i18n.gettext("Use locale format"),
}).prepend(this.el_csv_locale)).appendTo(this.expander_csv);
this.expander_csv.append(' ');
this.el_add_field_names = jQuery('<input/>', {
'type': 'checkbox',
'checked': 'checked'
});
jQuery('<div/>', {
'class': 'checkbox',
}).append(jQuery('<label/>', {
'text': ' '+Sao.i18n.gettext('Add Field Names')
}).prepend(this.el_add_field_names)).appendTo(this.expander_csv);
this.expander_csv.append(' ');
this.set_url();
Sortable.create(this.fields_selected.get(0), {
handle: '.draggable-handle',
ghostClass: 'dragged-row'
});
},
get context() {
return this.screen.context;
},
get screen_is_tree() {
return Boolean(
this.screen.current_view &&
(this.screen.current_view.view_type == 'tree') &&
this.screen.current_view.children_field);
},
get screen_has_selected() {
return Boolean(this.screen.selected_records.length);
},
view_populate: function(parent_node, parent_view) {
var names = Object.keys(parent_node).sort(function(a, b) {
if (parent_node[b].string < parent_node[a].string) {
return -1;
}
else {
return 1;
}
}).reverse();
names.forEach(name => {
var path = parent_node[name].path;
var node = jQuery('<li/>', {
'path': path
}).text(parent_node[name].string).click(e => {
if (e.ctrlKey || e.metaKey) {
node.toggleClass('bg-primary');
} else {
this.fields_all.find('li')
.removeClass('bg-primary');
node.addClass('bg-primary');
}
}).appendTo(parent_view);
parent_node[name].view = node;
var expander_icon = Sao.common.ICONFACTORY
.get_icon_img('tryton-arrow-right')
.data('expanded', false)
.click(e => {
e.stopPropagation();
var icon;
var expanded = expander_icon.data('expanded');
expander_icon.data('expanded', !expanded);
if (expanded) {
icon = 'tryton-arrow-right';
node.next('ul').remove();
} else {
icon = 'tryton-arrow-down';
this.on_row_expanded(parent_node[name]);
}
Sao.common.ICONFACTORY.get_icon_url(icon)
.then(function(url) {
expander_icon.attr('src', url);
});
}).prependTo(node);
expander_icon.css(
'visibility',
parent_node[name].children ? 'visible' : 'hidden');
});
},
model_populate: function(fields, parent_node, prefix_field,
prefix_name) {
parent_node = parent_node || this.fields_model;
prefix_field = prefix_field || '';
prefix_name = prefix_name || '';
for (const name of Object.keys(fields)) {
var field = fields[name];
var string = field.string || name;
var items = [{ name: name, field: field, string: string }];
if (field.type == 'selection') {
items.push({
name: name+'.translated',
field: field,
string: Sao.i18n.gettext('%1 (string)', string)
});
} else if (field.type == 'reference') {
items.push({
name: name + '.translated',
field: field,
string: Sao.i18n.gettext("%1 (model name)", string),
});
items.push({
name: name + '/rec_name',
field: field,
string: Sao.i18n.gettext("%1/Record Name", string),
});
}
for (const item of items) {
var path = prefix_field + item.name;
var long_string = prefix_name + item.string;
var node = {
path: path,
string: item.string,
long_string: long_string,
relation: item.field.relation
};
parent_node[item.name] = node;
this.fields[path] = node;
// Insert relation only to real field
if (item.name.indexOf('.') == -1) {
if (item.field.relation) {
node.children = {};
} else if (item.field.translate) {
node.children = {};
for (const language of this.languages) {
const l_path = `${path}:lang=${language.code}`;
const l_string = (
`${long_string} (${language.name})`);
const l_node = {
path: l_path,
string: language.name,
long_string: l_string,
relation: null,
};
node.children[language.code] = l_node;
this.fields[l_path] = l_node;
}
}
}
}
}
},
children_expand: function(node) {
if (jQuery.isEmptyObject(node.children) && node.relation) {
this.model_populate(
this._get_fields(node.relation), node.children,
node.path + '/', node.long_string + '/');
}
},
sig_sel_add: function(el_field) {
el_field = jQuery(el_field);
var name = el_field.attr('path');
this.sel_field(name);
},
fill_predefwin: function() {
Sao.rpc({
'method': 'model.ir.export.get',
'params': [
this.screen.model_name, [
'name', 'header', 'records', 'ignore_search_limit',
'export_fields.name'],
this.context,
],
}, this.session).done(exports => {
for (const export_ of exports) {
this.predef_exports[export_.id] = {
'fields': export_['export_fields.'].map(
field => field.name),
'values': export_,
};
this.add_to_predef(export_.id, export_.name);
this.predef_exports_list.children('li').first().focus();
}
});
},
add_to_predef: function(id, name) {
var node = jQuery('<li/>', {
'text': name,
'export_id': id,
'tabindex': 0
}).on('keypress', function(e) {
var keyCode = (e.keyCode ? e.keyCode : e.which);
if(keyCode == 13 || keyCode == 32) {
node.click();
}
}).click(event => {
node.toggleClass('bg-primary')
.siblings().removeClass('bg-primary');
if (node.hasClass('bg-primary')) {
this.sel_predef(node.attr('export_id'));
}
});
this.predef_exports_list.append(node);
},
addreplace_predef: function() {
var fields = [];
var selected_fields = this.fields_selected.children('li');
for (const field of selected_fields) {
fields.push(field.getAttribute('path'));
}
if(fields.length === 0) {
return;
}
var pref_id;
const save = name => {
var prm;
var values = {
'header': this.el_add_field_names.is(':checked'),
'records': (
JSON.parse(this.selected_records.val()) ?
'selected' : 'listed'),
'ignore_search_limit': this.ignore_search_limit.is(
':checked'),
};
if (!pref_id) {
values.name = name;
values.resource = this.screen.model_name;
values.export_fields = fields.map(f => ({'name': f}));
prm = Sao.rpc({
method: 'model.ir.export.set',
params: [values, this.context],
}, this.session);
} else {
prm = Sao.rpc({
method: 'model.ir.export.update',
params: [pref_id, values, fields, this.context],
}, this.session).then(() => pref_id);
}
return prm.then(pref_id => {
this.session.cache.clear(
'model.' + this.screen.model_name + '.view_toolbar_get');
this.predef_exports[pref_id] = {
'fields': fields,
'values': values,
};
if (selection.length === 0) {
this.add_to_predef(pref_id, name);
}
});
};
var selection = this.predef_exports_list.children('li.bg-primary');
if (selection.length === 0) {
pref_id = null;
Sao.common.ask.run(
Sao.i18n.gettext('What is the name of this export?'),
'export')
.then(save);
}
else {
pref_id = selection.attr('export_id');
Sao.common.sur.run(
Sao.i18n.gettext(
'Override %1 definition?', selection.text()))
.then(save);
}
},
remove_predef: function() {
var selection = this.predef_exports_list.children('li.bg-primary');
if (selection.length === 0) {
return;
}
var export_id = jQuery(selection).attr('export_id');
Sao.rpc({
'method': 'model.ir.export.unset',
'params': [export_id, this.context]
}, this.session).then(() => {
this.session.cache.clear(
'model.' + this.screen.model_name + '.view_toolbar_get');
delete this.predef_exports[export_id];
selection.remove();
});
},
sel_predef: function(export_id) {
this.fields_selected.empty();
const export_ = this.predef_exports[export_id];
for (const name of export_.fields) {
if (!(name in this.fields)) {
var fields = this.fields_model;
var prefix = '';
var parents = name.split('/').slice(0, -1);
this._traverse(fields, prefix, parents, 0);
}
if(!(name in this.fields)) {
return;
}
this.sel_field(name);
}
this.el_add_field_names.prop('checked', export_.values.header);
this.selected_records.val(
JSON.stringify(export_.values.records == 'selected'));
this.selected_records.change();
this.ignore_search_limit.prop(
'checked', export_.values.ignore_search_limit);
},
_traverse: function(fields, prefix, parents, i) {
var field, item;
var names = Object.keys(fields);
for (item = 0; item < names.length; item++) {
field = fields[names[item]];
if (field.path == (prefix + parents[i])) {
this.children_expand(field);
prefix += parents[i] + '/';
if (field.children) {
this._traverse(field.children, prefix, parents, ++i);
}
break;
}
}
},
sel_field: function(name) {
var long_string = this.fields[name].long_string;
var relation = this.fields[name].relation;
if (relation) {
name += '/rec_name';
}
var node = jQuery('<li/>', {
'path': name,
'class': 'draggable-handle',
}).text(long_string).click(function(e) {
if (e.ctrlKey || e.metaKey) {
node.toggleClass('bg-primary');
} else {
jQuery(e.target).addClass('bg-primary')
.siblings().removeClass('bg-primary');
}
}).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-drag')
).appendTo(this.fields_selected);
},
response: function(response_id) {
this.info_bar.clear();
if(response_id == 'RESPONSE_OK') {
var fields = [];
this.fields_selected.children('li').each(function(i, field) {
fields.push(field.getAttribute('path'));
});
var header = this.el_add_field_names.is(':checked');
var prm, ids, paths;
if (JSON.parse(this.selected_records.val())) {
ids = this.screen.selected_records.map(function(r) {
return r.id;
});
paths = this.screen.selected_paths;
prm = Sao.rpc({
'method': (
'model.' + this.screen.model_name +
'.export_data'),
'params': [ids, fields, header, this.context]
}, this.session);
} else if (this.screen_is_tree) {
ids = this.screen.listed_records.map(function(r) {
return r.id;
});
paths = this.screen.listed_paths;
prm = Sao.rpc({
'method': (
'model.' + this.screen.model_name +
'.export_data'),
'params': [ids, fields, header, this.context]
}, this.session);
} else {
var domain = this.screen.search_domain(
this.screen.screen_container.get_text());
var offset, limit;
if (this.ignore_search_limit.prop('checked')) {
offset = 0;
limit = null;
} else {
offset = this.screen.offset;
limit = this.screen.limit;
}
prm = Sao.rpc({
'method': (
'model.' + this.screen.model_name +
'.export_data_domain'),
'params': [
domain, fields, offset, limit, this.screen.order,
header, this.context],
}, this.session);
}
prm.then(data => {
this.export_csv(data, paths, header);
});
} else {
this.destroy();
}
},
export_csv: function(data, paths, header=false) {
var locale_format = this.el_csv_locale.prop('checked');
var unparse_obj = {};
unparse_obj.data = data.map(function(row, i) {
var indent = paths && paths[i] ? paths[i].length -1 : 0;
return Sao.Window.Export.format_row(
row, indent, locale_format);
});
var csv = Papa.unparse(unparse_obj, {
quoteChar: this.el_csv_quotechar.val(),
delimiter: this.el_csv_delimiter.val()
});
if (navigator.platform &&
navigator.platform.slice(0, 3) == 'Win') {
csv = Sao.BOM_UTF8 + csv;
}
Sao.common.download_file(
csv, this.name + '.csv', {type: 'text/csv;charset=utf-8'});
var size = data.length;
if (header) {
size -= 1;
}
this.info_bar.add(
Sao.i18n.ngettext(
"%1 record saved", "%1 records saved", size), 'info');
},
set_url: function() {
var path = [this.session.database, 'data', this.screen.model_name];
var query_string = [];
var domain;
if (JSON.parse(this.selected_records.val())) {
domain = this.screen.current_view.selected_records.map(function(r) {
return r.id;
});
} else {
domain = this.screen.search_domain(
this.screen.screen_container.get_text());
if (!this.ignore_search_limit.prop('checked') &&
this.screen.limit !== null) {
query_string.push(['s', this.screen.limit.toString()]);
query_string.push(
['p', Math.floor(
this.screen.offset / this.screen.limit).toString()]);
}
if (this.screen.order) {
for (const expr of this.screen.order) {
query_string.push(['o', expr.map(function(e) {
return e;
}).join(',')]);
}
}
}
query_string.splice(
0, 0, ['d', JSON.stringify(Sao.rpc.prepareObject(domain))]);
if (!jQuery.isEmptyObject(this.screen.local_context)) {
query_string.push(
['c', JSON.stringify(Sao.rpc.prepareObject(
this.screen.local_context))]);
}
this.fields_selected.children('li').each(function(i, field) {
query_string.push(['f', field.getAttribute('path')]);
});
query_string.push(['dl', this.el_csv_delimiter.val()]);
query_string.push(['qc', this.el_csv_quotechar.val()]);
if (!this.el_add_field_names.is(':checked')) {
query_string.push(['h', '0']);
}
if (this.el_csv_locale.prop('checked')) {
query_string.push(['loc', '1']);
}
query_string = query_string.map(function(param) {
return param.map(encodeURIComponent).join('=');
}).join('&');
this.button_url.attr('href', '/' + path.join('/') + '?' + query_string);
},
});
Sao.Window.Export.format_row = function(
line, indent=0, locale_format=true) {
var row = [];
line.forEach(function(val, i) {
if (locale_format) {
if (val.isDateTime) {
val = val.local().format(
Sao.common.date_format() + ' ' +
Sao.common.moment_format('%X'));
} else if (val.isDate) {
val = val.format(Sao.common.date_format());
} else if (val.isTimeDelta) {
val = Sao.common.timedelta.format(
val, {'s': 1, 'm': 60, 'h': 60 * 60});
} else if (!isNaN(Number(val))) {
val = val.toLocaleString(
Sao.i18n.BC47(Sao.i18n.getlang()), {
'minimumFractionDigits': 0,
'maximumFractionDigits': 20,
});
}
} else if (val.isDateTime) {
val = val.utc();
} else if (val.isTimeDelta) {
val = val.asSeconds();
} else if (typeof(val) == 'boolean') {
val += 0;
}
if ((i === 0) && indent && (typeof(val) == 'string')) {
val = ' '.repeat(indent) + val;
}
if (val instanceof Uint8Array) {
val = Sao.common.btoa(val);
}
row.push(val);
});
return row;
};
Sao.Window.EmailEntry = Sao.class_(Sao.common.InputCompletion, {
init: function(el, session) {
this.session = session;
Sao.Window.EmailEntry._super.init.call(
this, el,
this._email_source,
this._email_match_selected,
this._email_format);
},
_email_match_selected: function(value) {
this.input.val(value[2]);
},
_email_source: function(text) {
if (this.input[0].selectionStart < this.input.val().length) {
return jQuery.when([]);
}
return Sao.rpc({
'method': 'model.ir.email.complete',
'params': [text, Sao.config.limit, {}],
}, this.session)
.fail(function() {
Sao.Logger.warn("Unable to complete email entry");
});
},
_email_format: function(value) {
return value[1];
},
});
Sao.Window.Email = Sao.class_(Object, {
init: function(name, record, prints, template) {
this.record = record;
this.dialog = new Sao.Dialog(
Sao.i18n.gettext('Email %1', name), 'email', 'lg');
this.el = this.dialog.modal;
this.dialog.content.addClass('form-horizontal');
var body = this.dialog.body;
function add_group(name, label, required) {
var group = jQuery('<div/>', {
'class': 'form-group',
}).appendTo(body);
jQuery('<label/>', {
'class': 'control-label col-sm-1',
'text': label,
'for': 'email-' + name,
}).appendTo(group);
var input = jQuery('<input/>', {
'type': 'text',
'class':'form-control input-sm',
'id': 'email-' + name,
}).appendTo(jQuery('<div/>', {
'class': 'col-sm-11',
}).appendTo(group));
if (required) {
input.attr('required', true);
}
return input;
}
this.to = add_group('to', Sao.i18n.gettext('To:'), true);
this.cc = add_group('cc', Sao.i18n.gettext('Cc:'));
this.bcc = add_group('bcc', Sao.i18n.gettext('Bcc:'));
for (const input of [this.to, this.cc, this.bcc]) {
new Sao.Window.EmailEntry(input, this.record.model.session);
}
this.subject = add_group(
'subject', Sao.i18n.gettext('Subject:'), true);
var panel = jQuery('<div/>', {
'class': 'panel panel-default',
}).appendTo(body
).append(jQuery('<div/>', {
'class': 'panel-heading',
}).append(Sao.common.richtext_toolbar()));
this.body = jQuery('<div>', {
'class': 'email-richtext form-control input-sm mousetrap',
'contenteditable': true,
'spellcheck': true,
'id': 'email-body',
}).appendTo(jQuery('<div/>', {
'class': 'panel-body',
}).appendTo(panel));
var print_frame = jQuery('<div/>', {
'class': 'col-md-4',
}).appendTo(body);
jQuery('<label/>', {
'text': Sao.i18n.gettext("Reports"),
}).appendTo(print_frame);
this.print_actions = {};
for (const print of prints) {
var print_check = jQuery('<input/>', {
'type': 'checkbox',
});
jQuery('<div/>', {
'class': 'checkbox',
}).append(jQuery('<label/>'
).text(Sao.i18n.gettext(print.name)
).prepend(print_check)).appendTo(print_frame);
this.print_actions[print.id] = print_check;
}
var attachment_frame = jQuery('<div/>', {
'class': 'col-md-4',
}).appendTo(body);
jQuery('<label/>', {
'text': Sao.i18n.gettext("Attachments"),
}).appendTo(attachment_frame);
this.attachments = jQuery('<select/>', {
'class': 'form-control input-sm',
'name': 'attachments',
'multiple': true,
}).appendTo(attachment_frame);
Sao.rpc({
'method': 'model.ir.attachment.search_read',
'params': [
[
['resource', '=', record.model.name + ',' + record.id],
['OR',
['data', '!=', null],
['file_id', '!=', null],
],
],
0, null, null, ['rec_name'], record.get_context()],
}, record.model.session).then(attachments => {
for (const attachment of attachments) {
this.attachments.append(jQuery('<option/>', {
'value': JSON.stringify(attachment.id),
'text': attachment.rec_name,
}));
}
})
.fail(function() {
Sao.Logger.error(
"Could not fetch attachment for", record);
});
this.files = jQuery('<div/>', {
'class': 'col-md-4',
}).appendTo(body);
jQuery('<label/>', {
'text': Sao.i18n.gettext("Files"),
}).appendTo(this.files);
this._add_file_button();
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Cancel"),
}).text(' ' + Sao.i18n.gettext('Cancel')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-cancel')
).click(() => {
this.response('RESPONSE_CANCEL');
}).appendTo(this.dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("Send"),
}).text(' ' + Sao.i18n.gettext('Send')).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-send')
).appendTo(this.dialog.footer);
this.dialog.content.submit(e => {
e.preventDefault();
this.response('RESPONSE_OK');
});
this._fill_with(template);
this.el.modal('show');
this.el.on('hidden.bs.modal', function() {
jQuery(this).remove();
});
},
_add_file_button: function() {
var row = jQuery('<div/>').appendTo(this.files);
var file = jQuery('<input/>', {
'type': 'file',
}).appendTo(row);
var button = jQuery('<a/>', {
'class': 'close',
'title': Sao.i18n.gettext("Remove File"),
}).append(jQuery('<span/>', {
'aria-hidden': true,
'text': 'x',
})).append(jQuery('<span/>', {
'class': 'sr-only',
}).text(Sao.i18n.gettext("Remove")));
button.hide();
button.appendTo(row);
file.on('change', () => {
button.show();
this._add_file_button();
});
button.click(function() {
row.remove();
});
},
get_files: function() {
var prms = [];
var files = [];
this.files.find('input[type=file]').each(function(i, input) {
if (input.files.length) {
var dfd = jQuery.Deferred();
prms.push(dfd);
Sao.common.get_file_data(
input.files[0], function(data, filename) {
files.push([filename, data]);
dfd.resolve();
});
}
});
return jQuery.when.apply(jQuery, prms).then(function() {
return files;
});
},
get_attachments: function() {
var attachments = this.attachments.val();
if (attachments) {
return attachments.map(function(e) { return JSON.parse(e); });
}
return [];
},
_fill_with: function(template) {
var prm;
if (template) {
prm = Sao.rpc({
'method': 'model.ir.email.template.get',
'params': [template, this.record.id, {}],
}, this.record.model.session);
} else {
prm = Sao.rpc({
'method': 'model.ir.email.template.get_default',
'params': [this.record.model.name, this.record.id, {}],
}, this.record.model.session);
}
prm.then(values => {
this.to.val((values.to || []).join(', '));
this.cc.val((values.cc || []).join(', '));
this.bcc.val((values.bcc || []).join(', '));
this.subject.val(values.subject || '');
this.body.html(Sao.HtmlSanitizer.sanitize(values.body || ''));
var print_ids = (values.reports || []);
for (var print_id in this.print_actions) {
var check = this.print_actions[print_id];
check.prop(
'checked', ~print_ids.indexOf(parseInt(print_id, 10)));
}
});
},
response: function(response_id) {
if (response_id == 'RESPONSE_OK') {
var to = this.to.val();
var cc = this.cc.val();
var bcc = this.bcc.val();
var subject = this.subject.val();
var body = Sao.common.richtext_normalize(this.body.html());
var reports = [];
for (var id in this.print_actions) {
var check = this.print_actions[id];
if (check.prop('checked')) {
reports.push(id);
}
}
var attachments = this.get_attachments();
var record = this.record;
this.get_files().then(function(files) {
return Sao.rpc({
'method': 'model.ir.email.send',
'params': [
to, cc, bcc, subject, body,
files,
[record.model.name, record.id],
reports,
attachments,
{}],
}, record.model.session);
}).then(() => {
this.destroy();
});
} else {
this.destroy();
}
},
destroy: function() {
this.el.modal('hide');
},
});
Sao.Window.CodeScanner = Sao.class_(Object, {
init: function(callback, loop=false) {
this.callback = callback;
this.loop = loop;
this.submitting = false;
this.dialog = new Sao.Dialog(
Sao.i18n.gettext("Code Scanner"), 'code-scanner', 'md');
this.el = this.dialog.modal;
this.input = jQuery('<input/>', {
'type': 'text',
'class': 'form-control input-sm mousetrap',
'aria-label': Sao.i18n.gettext("Code"),
'placeholder': Sao.i18n.gettext("Code"),
}).appendTo(jQuery('<div/>', {
'class': 'col-sm-12',
}).appendTo(this.dialog.body));
var sound_btn = jQuery('<button/>', {
'class': 'btn btn-default pull-right',
'type': 'button',
'title': Sao.i18n.gettext("Toggle Sound"),
}).prependTo(this.dialog.header);
var sound_icon = jQuery('<img/>', {
'class': 'icon',
}).appendTo(sound_btn);
var sound_set_state = function(state) {
var prm;
if (state) {
sound_btn.addClass('active');
sound_btn.attr('aria-pressed', state);
prm = Sao.common.ICONFACTORY.get_icon_url('tryton-sound-on')
.done(url => {
sound_icon.attr('src', url);
});
} else {
sound_btn.removeClass('active');
sound_btn.attr('aria-pressed', state);
prm = Sao.common.ICONFACTORY.get_icon_url('tryton-sound-off')
.done(url => {
sound_icon.attr('src', url);
});
}
localStorage.setItem('sao_code_scanner_sound', state);
return prm;
};
var sound_state = JSON.parse(
localStorage.getItem('sao_code_scanner_sound'));
if (sound_state === null) {
sound_state = true;
}
var sound_prm = sound_set_state(sound_state);
sound_btn.click(
() => sound_set_state(!sound_btn.hasClass('active')));
jQuery('<button/>', {
'class': 'btn btn-link',
'type': 'button',
'title': Sao.i18n.gettext("Close"),
}).text(' ' + Sao.i18n.gettext("Close")).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-close')
).click(() => {
this.response('RESPONSE_CLOSE');
}).appendTo(this.dialog.footer);
jQuery('<button/>', {
'class': 'btn btn-primary',
'type': 'submit',
'title': Sao.i18n.gettext("OK"),
}).text(' ' + Sao.i18n.gettext("OK")).prepend(
Sao.common.ICONFACTORY.get_icon_img('tryton-ok')
).appendTo(this.dialog.footer);
this.dialog.content.submit(e => {
e.preventDefault();
this.response('RESPONSE_OK');
});
this.el.on('shown.bs.modal', () => {
this.input.on('blur', this._keep_focus);
this.input.focus();
});
this.el.on('hidden.bs.modal', function() {
jQuery(this).remove();
});
// show modal after sound icons have been set
// because they can trigger a renew session modal
sound_prm.then(() => this.el.modal('show'));
},
_play_sound: function(sound) {
if (JSON.parse(localStorage.getItem('sao_code_scanner_sound'))) {
Sao.common.play_sound(sound);
}
},
_keep_focus: function() {
setTimeout(() => this.focus(), 1);
},
response: function(response_id) {
if (this.submitting) return;
if (response_id == 'RESPONSE_OK') {
var code = this.input.val();
this.input.val(''); // clear input to prevent multiple calls
if (code) {
this.submitting = true;
this.input.off('blur');
return this.callback(code)
.always(() => this.submitting = false)
.then((modified) => {
this._play_sound('success');
if (!this.loop || !modified) {
this.destroy();
}
this.input.on('blur', this._keep_focus);
this.input.focus();
}, (error) => {
this._play_sound('danger');
if (error[0] == 'UserError') {
Sao.common.warning.run(error[1][1], error[1][0]);
} else {
Sao.common.error.run(error[0], error[1]);
}
this.destroy();
});
}
}
if (!this.loop || response_id != 'RESPONSE_OK') {
this.destroy();
}
},
destroy: function() {
this.el.modal('hide');
},
});
}());