From c1d7eab2dc2f5954de853044c092ca628f415e82 Mon Sep 17 00:00:00 2001
From: "Julien (jula)" <jula@odoo.com>
Date: Tue, 18 Apr 2023 12:47:20 +0000
Subject: [PATCH] [FIX] tools: image fields size guess (backport)
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Backport: https://github.com/odoo/odoo/pull/117343

__Current behavior before PR:__
The size of an image field is guessed using the field name. For instance, an image field with `field_name = "XXXX_123"` is resized to 123 pixels when fetched. This is can be an issue if a user creates an image field using studio in a form view. If the user sets the label of the field as "Image 1", the technical name will become `x_studio_image_1`. Therefore, the image field will be resized to 1 pixel width.

Note that in version anterior to 16, only the placeholder is resized since `image_guess_size_from_field_name` is never called when there is an actual image.

__Description of the fix:__
Refactor the `image_guess_size_from_field_name` method to return `(0, 0)` when the field name starts with `x_`.

closes odoo/odoo#118932

Signed-off-by: Sébastien Theys (seb) <seb@odoo.com>
---
 odoo/addons/base/tests/test_image.py | 25 +++++++++++++++++++++++++
 odoo/tools/image.py                  | 19 +++++++++++++------
 2 files changed, 38 insertions(+), 6 deletions(-)

diff --git a/odoo/addons/base/tests/test_image.py b/odoo/addons/base/tests/test_image.py
index c5d55b4eee47..1b92a829e358 100644
--- a/odoo/addons/base/tests/test_image.py
+++ b/odoo/addons/base/tests/test_image.py
@@ -285,6 +285,31 @@ class TestImage(TransactionCase):
         """Test that image_data_uri is working as expected."""
         self.assertEqual(tools.image_data_uri(self.base64_1x1_png), 'data:image/png;base64,' + self.base64_1x1_png.decode('ascii'))
 
+    def test_21_image_guess_size_from_field_name(self):
+        f = tools.image_guess_size_from_field_name
+        # Test case: empty field_name input
+        self.assertEqual(f(''), (0, 0))
+        # Test case: custom field_name input
+        self.assertEqual(f('custom_field'), (0, 0))
+        # Test case: field_name input that starts with 'x_'
+        self.assertEqual(f('x_field'), (0, 0))
+        # Test case: field_name input that starts with 'x_' and ends with a number less than 16
+        self.assertEqual(f('x_studio_image_1'), (0, 0))
+        # Test case: field_name input that starts with 'x_' and ends with a number greater than 16
+        self.assertEqual(f('x_studio_image_32'), (0, 0))
+        # Test case: field_name input that has a suffix less than 16
+        self.assertEqual(f('image_15'), (0, 0))
+        # Test case: field_name input that has a suffix equal to 16
+        self.assertEqual(f('image_16'), (16, 16))
+        # Test case: field_name input that has a suffix greater than 16
+        self.assertEqual(f('image_32'), (32, 32))
+        # Test case: field_name input that has a suffix with 2 numbers
+        self.assertEqual(f('image_1920_1080'), (1080, 1080))
+        # Test case: field_name input that has a float as suffix
+        self.assertEqual(f('image_32.5'), (0, 0))
+        # Test case: field_name input that has a suffix greater than 16 but no underscore
+        self.assertEqual(f('image32'), (0, 0))
+
     def _assertAlmostEqualSequence(self, rgb1, rgb2, delta=10):
         self.assertEqual(len(rgb1), len(rgb2))
         for index, t in enumerate(zip(rgb1, rgb2)):
diff --git a/odoo/tools/image.py b/odoo/tools/image.py
index 18380d1202d0..e70303bba1ea 100644
--- a/odoo/tools/image.py
+++ b/odoo/tools/image.py
@@ -498,20 +498,27 @@ def is_image_size_above(base64_source_1, base64_source_2):
 def image_guess_size_from_field_name(field_name):
     """Attempt to guess the image size based on `field_name`.
 
-    If it can't be guessed, return (0, 0) instead.
-
-    :param field_name: the name of a field
-    :type field_name: string
+    If it can't be guessed or if it is a custom field: return (0, 0) instead.
 
+    :param str field_name: the name of a field
     :return: the guessed size
     :rtype: tuple (width, height)
     """
-    suffix = '1024' if field_name == 'image' else field_name.split('_')[-1]
+    if field_name == 'image':
+        return (1024, 1024)
+    if field_name.startswith('x_'):
+        return (0, 0)
     try:
-        return (int(suffix), int(suffix))
+        suffix = int(field_name.split('_')[-1])
     except ValueError:
         return (0, 0)
 
+    if suffix < 16:
+        # If the suffix is less than 16, it's probably not the size
+        return (0, 0)
+
+    return (suffix, suffix)
+
 
 def image_data_uri(base64_source):
     """This returns data URL scheme according RFC 2397
-- 
GitLab