2883 lines
115 KiB
JavaScript
2883 lines
115 KiB
JavaScript
/* 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('×'))
|
||
).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(' '))
|
||
.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');
|
||
},
|
||
});
|
||
|
||
}());
|