[ADD] upstream patching till 88ff6beb016702cbda710aa31bbf10d51ab68fab
This commit is contained in:
parent
efb6c34f02
commit
973d6f5003
@ -1,3 +1,4 @@
|
|||||||
|
from flectra import api
|
||||||
from flectra.addons.account.tests.account_test_classes import AccountingTestCase
|
from flectra.addons.account.tests.account_test_classes import AccountingTestCase
|
||||||
import time
|
import time
|
||||||
import unittest
|
import unittest
|
||||||
|
@ -137,6 +137,12 @@ class Partner(models.Model):
|
|||||||
for k, v in vals.items():
|
for k, v in vals.items():
|
||||||
partner[k] = v
|
partner[k] = v
|
||||||
|
|
||||||
|
def write(self, vals):
|
||||||
|
res = super(Partner, self).write(vals)
|
||||||
|
if 'country_id' in vals:
|
||||||
|
self._set_street()
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
class Company(models.Model):
|
class Company(models.Model):
|
||||||
_inherit = 'res.company'
|
_inherit = 'res.company'
|
||||||
|
@ -51,5 +51,5 @@ class TestStreetFields(TransactionCase):
|
|||||||
self.write_and_assert(p1, {'street': 'Chaussee de Namur'}, 'Chaussee de Namur', 'Chaussee de Namur', '', '')
|
self.write_and_assert(p1, {'street': 'Chaussee de Namur'}, 'Chaussee de Namur', 'Chaussee de Namur', '', '')
|
||||||
self.write_and_assert(p1, {'street_name': 'Chee de Namur', 'street_number': '40'}, 'Chee de Namur, 40', 'Chee de Namur', '40', '')
|
self.write_and_assert(p1, {'street_name': 'Chee de Namur', 'street_number': '40'}, 'Chee de Namur, 40', 'Chee de Namur', '40', '')
|
||||||
self.write_and_assert(p1, {'street_number2': '4'}, 'Chee de Namur, 40/4', 'Chee de Namur', '40', '4')
|
self.write_and_assert(p1, {'street_number2': '4'}, 'Chee de Namur, 40/4', 'Chee de Namur', '40', '4')
|
||||||
#we don't recompute the street fields when we change the country
|
#we do recompute the street fields when we change the country
|
||||||
self.write_and_assert(p1, {'country_id': self.env.ref('base.us').id}, 'Chee de Namur, 40/4', 'Chee de Namur', '40', '4')
|
self.write_and_assert(p1, {'country_id': self.env.ref('base.us').id}, '40/4 Chee de Namur', 'Chee de Namur', '40', '4')
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
from flectra import models, fields, api
|
from flectra import models, fields, api, _
|
||||||
|
from flectra.exceptions import UserError
|
||||||
|
|
||||||
|
|
||||||
class IrModelFields(models.Model):
|
class IrModelFields(models.Model):
|
||||||
|
@ -4,7 +4,7 @@ import base64
|
|||||||
from email.utils import formataddr
|
from email.utils import formataddr
|
||||||
|
|
||||||
import re
|
import re
|
||||||
import uuid
|
from uuid import uuid4
|
||||||
|
|
||||||
from flectra import _, api, fields, models, modules, tools
|
from flectra import _, api, fields, models, modules, tools
|
||||||
from flectra.exceptions import UserError
|
from flectra.exceptions import UserError
|
||||||
@ -56,7 +56,7 @@ class Channel(models.Model):
|
|||||||
('channel', 'Channel')],
|
('channel', 'Channel')],
|
||||||
'Channel Type', default='channel')
|
'Channel Type', default='channel')
|
||||||
description = fields.Text('Description')
|
description = fields.Text('Description')
|
||||||
uuid = fields.Char('UUID', size=50, index=True, default=lambda self: '%s' % uuid.uuid4(), copy=False)
|
uuid = fields.Char('UUID', size=50, index=True, default=lambda self: str(uuid4()), copy=False)
|
||||||
email_send = fields.Boolean('Send messages by email', default=False)
|
email_send = fields.Boolean('Send messages by email', default=False)
|
||||||
# multi users channel
|
# multi users channel
|
||||||
channel_last_seen_partner_ids = fields.One2many('mail.channel.partner', 'channel_id', string='Last Seen')
|
channel_last_seen_partner_ids = fields.One2many('mail.channel.partner', 'channel_id', string='Last Seen')
|
||||||
|
@ -7,7 +7,7 @@ from dateutil.relativedelta import relativedelta
|
|||||||
from flectra import api, fields, models, SUPERUSER_ID, _
|
from flectra import api, fields, models, SUPERUSER_ID, _
|
||||||
from flectra.tools import DEFAULT_SERVER_DATETIME_FORMAT
|
from flectra.tools import DEFAULT_SERVER_DATETIME_FORMAT
|
||||||
from flectra.tools.float_utils import float_is_zero, float_compare
|
from flectra.tools.float_utils import float_is_zero, float_compare
|
||||||
from flectra.exceptions import UserError, AccessError, ValidationError
|
from flectra.exceptions import UserError, AccessError
|
||||||
from flectra.tools.misc import formatLang
|
from flectra.tools.misc import formatLang
|
||||||
from flectra.addons.base.res.res_partner import WARNING_MESSAGE, WARNING_HELP
|
from flectra.addons.base.res.res_partner import WARNING_MESSAGE, WARNING_HELP
|
||||||
from flectra.addons import decimal_precision as dp
|
from flectra.addons import decimal_precision as dp
|
||||||
@ -167,8 +167,8 @@ class PurchaseOrder(models.Model):
|
|||||||
('invoiced', 'No Bill to Receive'),
|
('invoiced', 'No Bill to Receive'),
|
||||||
], string='Billing Status', compute='_get_invoiced', store=True, readonly=True, copy=False, default='no')
|
], string='Billing Status', compute='_get_invoiced', store=True, readonly=True, copy=False, default='no')
|
||||||
|
|
||||||
picking_count = fields.Integer(compute='_compute_picking', string='Receptions', default=0, store=True)
|
picking_count = fields.Integer(compute='_compute_picking', string='Receptions', default=0, store=True, compute_sudo=True)
|
||||||
picking_ids = fields.Many2many('stock.picking', compute='_compute_picking', string='Receptions', copy=False, store=True)
|
picking_ids = fields.Many2many('stock.picking', compute='_compute_picking', string='Receptions', copy=False, store=True, compute_sudo=True)
|
||||||
|
|
||||||
# There is no inverse function on purpose since the date may be different on each line
|
# There is no inverse function on purpose since the date may be different on each line
|
||||||
date_planned = fields.Datetime(string='Scheduled Date', compute='_compute_date_planned', store=True, index=True)
|
date_planned = fields.Datetime(string='Scheduled Date', compute='_compute_date_planned', store=True, index=True)
|
||||||
@ -669,7 +669,7 @@ class PurchaseOrderLine(models.Model):
|
|||||||
|
|
||||||
# Replace by invoiced Qty
|
# Replace by invoiced Qty
|
||||||
qty_invoiced = fields.Float(compute='_compute_qty_invoiced', string="Billed Qty", digits=dp.get_precision('Product Unit of Measure'), store=True)
|
qty_invoiced = fields.Float(compute='_compute_qty_invoiced', string="Billed Qty", digits=dp.get_precision('Product Unit of Measure'), store=True)
|
||||||
qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", digits=dp.get_precision('Product Unit of Measure'), store=True)
|
qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", digits=dp.get_precision('Product Unit of Measure'), store=True, compute_sudo=True)
|
||||||
|
|
||||||
partner_id = fields.Many2one('res.partner', related='order_id.partner_id', string='Partner', readonly=True, store=True)
|
partner_id = fields.Many2one('res.partner', related='order_id.partner_id', string='Partner', readonly=True, store=True)
|
||||||
currency_id = fields.Many2one(related='order_id.currency_id', store=True, string='Currency', readonly=True)
|
currency_id = fields.Many2one(related='order_id.currency_id', store=True, string='Currency', readonly=True)
|
||||||
|
@ -8,7 +8,7 @@ from flectra.tools import float_compare
|
|||||||
class PurchaseOrderLine(models.Model):
|
class PurchaseOrderLine(models.Model):
|
||||||
_inherit = 'purchase.order.line'
|
_inherit = 'purchase.order.line'
|
||||||
|
|
||||||
qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", store=True)
|
qty_received = fields.Float(compute='_compute_qty_received', string="Received Qty", store=True, compute_sudo=True)
|
||||||
|
|
||||||
def _compute_qty_received(self):
|
def _compute_qty_received(self):
|
||||||
super(PurchaseOrderLine, self)._compute_qty_received()
|
super(PurchaseOrderLine, self)._compute_qty_received()
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
# Part of Odoo, Flectra. See LICENSE file for full copyright and licensing details.
|
# Part of Odoo, Flectra. See LICENSE file for full copyright and licensing details.
|
||||||
|
|
||||||
from flectra import api, models
|
from flectra import api, models, fields
|
||||||
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
|
@ -107,7 +107,7 @@ class LandedCost(models.Model):
|
|||||||
'landed_cost_value': new_landed_cost_value,
|
'landed_cost_value': new_landed_cost_value,
|
||||||
'value': line.move_id.value + cost_to_add,
|
'value': line.move_id.value + cost_to_add,
|
||||||
'remaining_value': line.move_id.remaining_value + cost_to_add,
|
'remaining_value': line.move_id.remaining_value + cost_to_add,
|
||||||
'price_unit': (line.move_id.value + new_landed_cost_value) / line.move_id.product_qty,
|
'price_unit': (line.move_id.value + cost_to_add) / line.move_id.product_qty,
|
||||||
})
|
})
|
||||||
# `remaining_qty` is negative if the move is out and delivered proudcts that were not
|
# `remaining_qty` is negative if the move is out and delivered proudcts that were not
|
||||||
# in stock.
|
# in stock.
|
||||||
|
@ -1758,7 +1758,7 @@ var StatInfo = AbstractField.extend({
|
|||||||
*/
|
*/
|
||||||
_render: function () {
|
_render: function () {
|
||||||
var options = {
|
var options = {
|
||||||
value: this.value || 0,
|
value: this._formatValue(this.value || 0),
|
||||||
};
|
};
|
||||||
if (! this.attrs.nolabel) {
|
if (! this.attrs.nolabel) {
|
||||||
if (this.nodeOptions.label_field && this.recordData[this.nodeOptions.label_field]) {
|
if (this.nodeOptions.label_field && this.recordData[this.nodeOptions.label_field]) {
|
||||||
|
@ -4000,6 +4000,40 @@ QUnit.module('basic_fields', {
|
|||||||
|
|
||||||
QUnit.module('StatInfo');
|
QUnit.module('StatInfo');
|
||||||
|
|
||||||
|
QUnit.test('statinfo widget formats decimal precision', function (assert) {
|
||||||
|
// sometimes the round method can return numbers such as 14.000001
|
||||||
|
// when asked to round a number to 2 decimals, as such is the behaviour of floats.
|
||||||
|
// we check that even in that eventuality, only two decimals are displayed
|
||||||
|
assert.expect(2);
|
||||||
|
|
||||||
|
this.data.partner.fields.monetary = {string: "Monetary", type: 'monetary'};
|
||||||
|
this.data.partner.records[0].monetary = 9.999999;
|
||||||
|
this.data.partner.records[0].currency_id = 1;
|
||||||
|
|
||||||
|
var form = createView({
|
||||||
|
View: FormView,
|
||||||
|
model: 'partner',
|
||||||
|
data: this.data,
|
||||||
|
arch: '<form string="Partners">' +
|
||||||
|
'<button class="oe_stat_button" name="items" icon="fa-gear">' +
|
||||||
|
'<field name="qux" widget="statinfo"/>' +
|
||||||
|
'</button>' +
|
||||||
|
'<button class="oe_stat_button" name="money" icon="fa-money">' +
|
||||||
|
'<field name="monetary" widget="statinfo"/>' +
|
||||||
|
'</button>' +
|
||||||
|
'</form>',
|
||||||
|
res_id: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
// formatFloat renders according to this.field.digits
|
||||||
|
assert.strictEqual(form.$('.oe_stat_button .o_field_widget.o_stat_info .o_stat_value').eq(0).text(),
|
||||||
|
'0.4', "Default precision should be [16,1]");
|
||||||
|
assert.strictEqual(form.$('.oe_stat_button .o_field_widget.o_stat_info .o_stat_value').eq(1).text(),
|
||||||
|
'10.00', "Currency decimal precision should be 2");
|
||||||
|
|
||||||
|
form.destroy();
|
||||||
|
});
|
||||||
|
|
||||||
QUnit.test('statinfo widget in form view', function (assert) {
|
QUnit.test('statinfo widget in form view', function (assert) {
|
||||||
assert.expect(9);
|
assert.expect(9);
|
||||||
|
|
||||||
|
@ -8,6 +8,7 @@ var Widget = require('web.Widget');
|
|||||||
var weContext = require('web_editor.context');
|
var weContext = require('web_editor.context');
|
||||||
var rte = require('web_editor.rte');
|
var rte = require('web_editor.rte');
|
||||||
var weWidgets = require('web_editor.widget');
|
var weWidgets = require('web_editor.widget');
|
||||||
|
var Dialog = require('web.Dialog');
|
||||||
|
|
||||||
var _t = core._t;
|
var _t = core._t;
|
||||||
|
|
||||||
@ -26,6 +27,7 @@ var RTETranslatorWidget = rte.Class.extend({
|
|||||||
* @override
|
* @override
|
||||||
*/
|
*/
|
||||||
_saveElement: function ($el, context, withLang) {
|
_saveElement: function ($el, context, withLang) {
|
||||||
|
var self = this;
|
||||||
if ($el.data('oe-translation-id')) {
|
if ($el.data('oe-translation-id')) {
|
||||||
return this._rpc({
|
return this._rpc({
|
||||||
model: 'ir.translation',
|
model: 'ir.translation',
|
||||||
@ -35,7 +37,9 @@ var RTETranslatorWidget = rte.Class.extend({
|
|||||||
this._getEscapedElement($el).html(),
|
this._getEscapedElement($el).html(),
|
||||||
context || weContext.get()
|
context || weContext.get()
|
||||||
],
|
],
|
||||||
});
|
}).fail(function (error) {
|
||||||
|
Dialog.alert(null, error.data.message);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
return this._super($el, context, withLang === undefined ? true : withLang);
|
return this._super($el, context, withLang === undefined ? true : withLang);
|
||||||
},
|
},
|
||||||
|
@ -546,23 +546,29 @@ class IrModelFields(models.Model):
|
|||||||
This method prevents the modification/deletion of many2one fields
|
This method prevents the modification/deletion of many2one fields
|
||||||
that have an inverse one2many, for instance.
|
that have an inverse one2many, for instance.
|
||||||
"""
|
"""
|
||||||
|
failed_dependencies = []
|
||||||
|
for rec in self:
|
||||||
|
model = self.env[rec.model]
|
||||||
|
field = model._fields[rec.name]
|
||||||
|
for dependant, path in model._field_triggers.get(field, ()):
|
||||||
|
if dependant.manual:
|
||||||
|
failed_dependencies.append((field, dependant))
|
||||||
|
for inverse in model._field_inverses.get(field, ()):
|
||||||
|
if inverse.manual and inverse.type == 'one2many':
|
||||||
|
failed_dependencies.append((field, inverse))
|
||||||
|
|
||||||
|
if not self._context.get(MODULE_UNINSTALL_FLAG) and failed_dependencies:
|
||||||
|
msg = _("The field '%s' cannot be removed because the field '%s' depends on it.")
|
||||||
|
raise UserError(msg % failed_dependencies[0])
|
||||||
|
elif failed_dependencies:
|
||||||
|
dependants = {rel[1] for rel in failed_dependencies}
|
||||||
|
to_unlink = [self._get(field.model_name, field.name) for field in dependants]
|
||||||
|
self.browse().union(*to_unlink).unlink()
|
||||||
|
|
||||||
self = self.filtered(lambda record: record.state == 'manual')
|
self = self.filtered(lambda record: record.state == 'manual')
|
||||||
if not self:
|
if not self:
|
||||||
return
|
return
|
||||||
|
|
||||||
for record in self:
|
|
||||||
model = self.env[record.model]
|
|
||||||
field = model._fields[record.name]
|
|
||||||
if field.type == 'many2one' and model._field_inverses.get(field):
|
|
||||||
if self._context.get(MODULE_UNINSTALL_FLAG):
|
|
||||||
# automatically unlink the corresponding one2many field(s)
|
|
||||||
inverses = self.search([('relation', '=', field.model_name),
|
|
||||||
('relation_field', '=', field.name)])
|
|
||||||
inverses.unlink()
|
|
||||||
continue
|
|
||||||
msg = _("The field '%s' cannot be removed because the field '%s' depends on it.")
|
|
||||||
raise UserError(msg % (field, model._field_inverses[field][0]))
|
|
||||||
|
|
||||||
# remove fields from registry, and check that views are not broken
|
# remove fields from registry, and check that views are not broken
|
||||||
fields = [self.env[record.model]._pop_field(record.name) for record in self]
|
fields = [self.env[record.model]._pop_field(record.name) for record in self]
|
||||||
domain = expression.OR([('arch_db', 'like', record.name)] for record in self)
|
domain = expression.OR([('arch_db', 'like', record.name)] for record in self)
|
||||||
|
@ -24,9 +24,10 @@ _logger = logging.getLogger(__name__)
|
|||||||
class TestPyLint(TransactionCase):
|
class TestPyLint(TransactionCase):
|
||||||
|
|
||||||
ENABLED_CODES = [
|
ENABLED_CODES = [
|
||||||
'E0601', # using variable before assignment
|
'used-before-assignment',
|
||||||
'W0123', # eval used
|
'undefined-variable',
|
||||||
'W0101', # unreachable code
|
'eval-used',
|
||||||
|
'unreachable',
|
||||||
|
|
||||||
'mixed-indentation',
|
'mixed-indentation',
|
||||||
|
|
||||||
|
@ -1630,7 +1630,8 @@ class Datetime(Field):
|
|||||||
# Received data is returned as buffer (in Python 2) or memoryview (in Python 3).
|
# Received data is returned as buffer (in Python 2) or memoryview (in Python 3).
|
||||||
_BINARY = memoryview
|
_BINARY = memoryview
|
||||||
if pycompat.PY2:
|
if pycompat.PY2:
|
||||||
_BINARY = buffer #pylint: disable=buffer-builtin
|
#pylint: disable=buffer-builtin,undefined-variable
|
||||||
|
_BINARY = buffer
|
||||||
|
|
||||||
class Binary(Field):
|
class Binary(Field):
|
||||||
type = 'binary'
|
type = 'binary'
|
||||||
|
@ -21,7 +21,8 @@ if pycompat.PY2:
|
|||||||
fp, fname = tools.file_open(path, pathinfo=True)
|
fp, fname = tools.file_open(path, pathinfo=True)
|
||||||
fp2 = None
|
fp2 = None
|
||||||
|
|
||||||
if not isinstance(fp, file): # pylint: disable=file-builtin
|
# pylint: disable=file-builtin,undefined-variable
|
||||||
|
if not isinstance(fp, file):
|
||||||
# imp.load_source need a real file object, so we create
|
# imp.load_source need a real file object, so we create
|
||||||
# one from the file-like object we get from file_open
|
# one from the file-like object we get from file_open
|
||||||
fp2 = os.tmpfile()
|
fp2 = os.tmpfile()
|
||||||
|
@ -1069,7 +1069,7 @@ def formatLang(env, value, digits=None, grouping=True, monetary=False, dp=False,
|
|||||||
digits = decimal_precision_obj.precision_get(dp)
|
digits = decimal_precision_obj.precision_get(dp)
|
||||||
elif currency_obj:
|
elif currency_obj:
|
||||||
digits = currency_obj.decimal_places
|
digits = currency_obj.decimal_places
|
||||||
elif (hasattr(value, '_field') and isinstance(value._field, (float_field, function_field)) and value._field.digits):
|
elif (hasattr(value, '_field') and getattr(value._field, 'digits', None)):
|
||||||
digits = value._field.digits[1]
|
digits = value._field.digits[1]
|
||||||
if not digits and digits is not 0:
|
if not digits and digits is not 0:
|
||||||
digits = DEFAULT_DIGITS
|
digits = DEFAULT_DIGITS
|
||||||
|
@ -13,7 +13,7 @@ PY2 = sys.version_info[0] == 2
|
|||||||
|
|
||||||
_Writer = collections.namedtuple('_Writer', 'writerow writerows')
|
_Writer = collections.namedtuple('_Writer', 'writerow writerows')
|
||||||
if PY2:
|
if PY2:
|
||||||
# pylint: disable=long-builtin,unichr-builtin,unicode-builtin
|
# pylint: disable=long-builtin,unichr-builtin,unicode-builtin,undefined-variable
|
||||||
unichr = unichr
|
unichr = unichr
|
||||||
text_type = unicode
|
text_type = unicode
|
||||||
string_types = (str, unicode)
|
string_types = (str, unicode)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
from lxml import etree
|
from lxml import etree
|
||||||
from flectra.tools.misc import file_open
|
from flectra.tools.misc import file_open
|
||||||
|
from flectra.exceptions import UserError
|
||||||
|
|
||||||
def check_with_xsd(tree_or_str, stream):
|
def check_with_xsd(tree_or_str, stream):
|
||||||
raise UserError("Method 'check_with_xsd' deprecated ")
|
raise UserError("Method 'check_with_xsd' deprecated ")
|
||||||
|
@ -3,11 +3,14 @@ decorator==4.0.10
|
|||||||
docutils==0.12
|
docutils==0.12
|
||||||
ebaysdk==2.1.5
|
ebaysdk==2.1.5
|
||||||
feedparser==5.2.1
|
feedparser==5.2.1
|
||||||
gevent==1.1.2 ; sys_platform != 'win32'
|
gevent==1.1.2 ; sys_platform != 'win32' and python_version < '3.7'
|
||||||
greenlet==0.4.10
|
gevent==1.3.4 ; sys_platform != 'win32' and python_version >= '3.7'
|
||||||
|
greenlet==0.4.10 ; python_version < '3.7'
|
||||||
|
greenlet==0.4.13 ; python_version >= '3.7'
|
||||||
html2text==2016.9.19
|
html2text==2016.9.19
|
||||||
Jinja2==2.8
|
Jinja2==2.8
|
||||||
lxml==3.7.1 ; sys_platform != 'win32'
|
lxml==3.7.1 ; sys_platform != 'win32' and python_version < '3.7'
|
||||||
|
lxml==4.2.3 ; sys_platform != 'win32' and python_version >= '3.7'
|
||||||
lxml ; sys_platform == 'win32'
|
lxml ; sys_platform == 'win32'
|
||||||
Mako==1.0.4
|
Mako==1.0.4
|
||||||
MarkupSafe==0.23
|
MarkupSafe==0.23
|
||||||
|
Loading…
Reference in New Issue
Block a user