From 53410da351cc14a152f181d00b9abea4151b836f Mon Sep 17 00:00:00 2001
From: MerlinGuillaume <megu@odoo.com>
Date: Fri, 26 Aug 2022 08:26:48 +0000
Subject: [PATCH] [REV] web: do not use number format in exported group header
 for floats

Reverts [this commit] because it modified the format of float numbers in
exported xlsx files and we want to avoid introducing an error or
unwanted behavior in version 13, for which support will soon come to an
end.

[this commit]:https://github.com/odoo/odoo/commit/27ab24214dabfafc652280f680f7a650e7d98d2b

closes odoo/odoo#98975

Signed-off-by: Martin Trigaux (mat) <mat@odoo.com>
---
 addons/test_xlsx_export/tests/test_export.py | 38 +++++++++-----------
 addons/web/controllers/main.py               | 34 ++++++++++--------
 2 files changed, 35 insertions(+), 37 deletions(-)

diff --git a/addons/test_xlsx_export/tests/test_export.py b/addons/test_xlsx_export/tests/test_export.py
index 6722e7a23e62..deea378b6982 100644
--- a/addons/test_xlsx_export/tests/test_export.py
+++ b/addons/test_xlsx_export/tests/test_export.py
@@ -37,12 +37,7 @@ class XlsxCreatorCase(common.HttpCase):
         }
 
     def _mock_write(self, row, column, value, style=None):
-        if isinstance(value, float):
-            decimal_places = style.num_format[::-1].find('.')
-            style_format = "{:." + str(decimal_places) + "f}"
-            self.worksheet[row, column] = style_format.format(value)
-        else:
-            self.worksheet[row, column] = str(value)
+        self.worksheet[row, column] = str(value)
 
     def make(self, values, context=None):
         return self.model.with_context(**(context or {})).create(values)
@@ -77,7 +72,6 @@ class XlsxCreatorCase(common.HttpCase):
 
 class TestGroupedExport(XlsxCreatorCase):
     model_name = 'export.group_operator'
-    # pylint: disable=bad-whitespace
 
     def test_int_sum_max(self):
         values = [
@@ -124,12 +118,12 @@ class TestGroupedExport(XlsxCreatorCase):
             ['Int Sum'      ,'Float Min'],
             ['10 (2)'       ,'111.00'],
             ['    111.0 (1)','111.00'],
-            ['10'           ,'111.00'],
+            ['10'           ,'111.0'],
             ['    222.0 (1)','222.00'],
-            ['10'           ,'222.00'],
+            ['10'           ,'222.0'],
             ['20 (1)'       ,'333.00'],
             ['    333.0 (1)','333.00'],
-            ['20'           ,'333.00'],
+            ['20'           ,'333.0'],
         ])
 
     def test_float_avg(self):
@@ -144,12 +138,12 @@ class TestGroupedExport(XlsxCreatorCase):
             ['Int Sum'      ,'Float Avg'],
             ['10 (2)'       ,'150.00'],
             ['    100.0 (1)','100.00'],
-            ['10'           ,'100.00'],
+            ['10'           ,'100.0'],
             ['    200.0 (1)','200.00'],
-            ['10'           ,'200.00'],
+            ['10'           ,'200.0'],
             ['20 (1)'       ,'300.00'],
             ['    300.0 (1)','300.00'],
-            ['20'           ,'300.00'],
+            ['20'           ,'300.0'],
         ])
 
     def test_float_avg_nested(self):
@@ -166,12 +160,12 @@ class TestGroupedExport(XlsxCreatorCase):
             ['10 (3)'           ,'300.00'],
             ['    20 (1)'       ,'600.00'],
             ['        600.0 (1)','600.00'],
-            ['10'               ,'600.00'],
+            ['10'               ,'600.0'],
             ['    30 (2)'       ,'150.00'],
             ['        100.0 (1)','100.00'],
-            ['10'               ,'100.00'],
+            ['10'               ,'100.0'],
             ['        200.0 (1)','200.00'],
-            ['10'               ,'200.00'],
+            ['10'               ,'200.0'],
         ])
 
     def test_float_avg_nested_no_value(self):
@@ -188,11 +182,11 @@ class TestGroupedExport(XlsxCreatorCase):
             ['10 (3)'               ,'0.00'],
             ['    20 (1)'           ,'0.00'],
             ['        Undefined (1)','0.00'],
-            ['10'                   ,'0.00'],
+            ['10'                   ,'0.0'],
             ['    30 (2)'           ,'0.00'],
             ['        Undefined (2)','0.00'],
-            ['10'                   ,'0.00'],
-            ['10'                   ,'0.00'],
+            ['10'                   ,'0.0'],
+            ['10'                   ,'0.0'],
         ])
 
     def test_date_max(self):
@@ -366,11 +360,11 @@ class TestGroupedExport(XlsxCreatorCase):
             ['Int Sum', 'Float Monetary'],
             ['1 (1)','60739.200'],
             ['    60739.2 (1)','60739.200'],
-            ['1','60739.20'],
+            ['1','60739.2'],
             ['2 (1)','2.000'],
             ['    2.0 (1)','2.000'],
-            ['2','2.00'],
+            ['2','2.0'],
             ['3 (1)','1000.000'],
             ['    1000.0 (1)','1000.000'],
-            ['3','1000.00'],
+            ['3','1000.0'],
         ])
diff --git a/addons/web/controllers/main.py b/addons/web/controllers/main.py
index dc36ed510be8..b6e5cf3f1fac 100644
--- a/addons/web/controllers/main.py
+++ b/addons/web/controllers/main.py
@@ -35,10 +35,10 @@ import odoo
 import odoo.modules.registry
 from odoo.api import call_kw, Environment
 from odoo.modules import get_module_path, get_resource_path
-from odoo.tools import image_process, topological_sort, html_escape, pycompat, ustr, apply_inheritance_specs, lazy_property
+from odoo.tools import image_process, topological_sort, html_escape, pycompat, ustr, apply_inheritance_specs, lazy_property, float_repr
 from odoo.tools.mimetypes import guess_mimetype
 from odoo.tools.translate import _
-from odoo.tools.misc import str2bool, xlsxwriter, file_open, get_lang
+from odoo.tools.misc import str2bool, xlsxwriter, file_open
 from odoo.tools.safe_eval import safe_eval
 from odoo import http, tools
 from odoo.http import content_disposition, dispatch_rpc, request, serialize_exception as _serialize_exception, Response
@@ -738,10 +738,6 @@ class ExportXlsxWriter:
         self.datetime_style = self.workbook.add_format({'text_wrap': True, 'num_format': 'yyyy-mm-dd hh:mm:ss'})
         self.worksheet = self.workbook.add_worksheet()
         self.value = False
-        decimal_separator = get_lang(request.env).decimal_point
-        self.float_format = f'0{decimal_separator}00'
-        decimal_places = [res['decimal_places'] for res in request.env['res.currency'].search_read([], ['decimal_places'])]
-        self.monetary_format = f'0{decimal_separator}{max(decimal_places or [2]) * "0"}'
 
         if row_count > self.worksheet.xls_rowmax:
             raise UserError(_('There are too many rows (%s rows, limit: %s) to export as Excel 2007-2013 (.xlsx) format. Consider splitting the export.') % (row_count, self.worksheet.xls_rowmax))
@@ -789,8 +785,6 @@ class ExportXlsxWriter:
             cell_style = self.datetime_style
         elif isinstance(cell_value, datetime.date):
             cell_style = self.date_style
-        elif isinstance(cell_value, float):
-            cell_style.set_num_format(self.float_format)
         self.write(row, column, cell_value, cell_style)
 
 class GroupExportXlsxWriter(ExportXlsxWriter):
@@ -824,16 +818,26 @@ class GroupExportXlsxWriter(ExportXlsxWriter):
 
         label = '%s%s (%s)' % ('    ' * group_depth, label, group.count)
         self.write(row, column, label, self.header_bold_style)
+        if any(f.get('type') == 'monetary' for f in self.fields[1:]):
+
+            decimal_places = [res['decimal_places'] for res in group._model.env['res.currency'].search_read([], ['decimal_places'])]
+            decimal_places = max(decimal_places) if decimal_places else 2
         for field in self.fields[1:]: # No aggregates allowed in the first column because of the group title
             column += 1
             aggregated_value = aggregates.get(field['name'])
-            if field.get('type') == 'monetary':
-                self.header_bold_style.set_num_format(self.monetary_format)
-            elif field.get('type') == 'float':
-                self.header_bold_style.set_num_format(self.float_format)
-            else:
-                aggregated_value = str(aggregated_value if aggregated_value is not None else '')
-            self.write(row, column, aggregated_value, self.header_bold_style)
+            # Float fields may not be displayed properly because of float
+            # representation issue with non stored fields or with values
+            # that, even stored, cannot be rounded properly and it is not
+            # acceptable to display useless digits (i.e. monetary)
+            #
+            # non stored field ->  we force 2 digits
+            # stored monetary -> we force max digits of installed currencies
+            if isinstance(aggregated_value, float):
+                if field.get('type') == 'monetary':
+                    aggregated_value = float_repr(aggregated_value, decimal_places)
+                elif not field.get('store'):
+                    aggregated_value = float_repr(aggregated_value, 2)
+            self.write(row, column, str(aggregated_value if aggregated_value is not None else ''), self.header_bold_style)
         return row + 1, 0
 
 
-- 
GitLab