diff --git a/partner_firstname/README.rst b/partner_firstname/README.rst index 7c9bf10e9..2b045f04f 100644 --- a/partner_firstname/README.rst +++ b/partner_firstname/README.rst @@ -1,12 +1,33 @@ .. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg + :target: http://www.gnu.org/licenses/agpl-3.0-standalone.html :alt: License: AGPL-3 +================================ Partner first name and last name ================================ This module was written to extend the functionality of contacts to support having separate last name and first name. +Configuration +============= + +You can configure some common name patterns for the inverse function +in Settings > Configuration > General settings: + +* Lastname Firstname: For example 'Anderson Robert' +* Lastname, Firstname: For example 'Anderson, Robert' +* Firstname Lastname: For example 'Robert Anderson' + +After applying the changes, you can recalculate all partners name clicking +"Recalculate names" button. Note: This process could take so much time depending +how many partners there are in database. + +You can use *_get_inverse_name* method to get lastname and firstname from a simple string +and also *_get_computed_name* to get a name form the lastname and firstname. +These methods can be overridden to change the format specified above. + + Usage ===== @@ -16,22 +37,39 @@ and the *first name*. This avoids breaking compatibility with other modules. Users should fulfill manually the separate fields for *last name* and *first name*, but in case you edit just the *name* field in some unexpected module, there is an inverse function that tries to split that automatically. It assumes -that you write the *name* in format *"Lastname Firstname"*, but it could lead to -wrong splitting (because it's just blindly trying to guess what you meant), so -you better specify it manually. +that you write the *name* in format configured (*"Lastname Firstname"*, by default), +but it could lead to wrong splitting (because it's just blindly trying to +guess what you meant), so you better specify it manually. For the same reason, after installing, previous names for contacts will stay in the *name* field, and the first time you edit any of them you will be asked to supply the *last name* and *first name* (just once per contact). -You can use *_get_inverse_name* method to get lastname and firstname from a simple string -and also *_get_computed_name* to get a name form the lastname and firstname. -These methods can be overridden to change the format specified above +.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas + :alt: Try me on Runbot + :target: https://runbot.odoo-community.org/runbot/134/10.0 For further information, please visit: * https://www.odoo.com/forum/help-1 + +Known issues / Roadmap +====================== + +Patterns for the inverse function are configurable only at system level. Maybe +this configuration could depend on partner language, country or company, +as discussed at `this OCA issue `_ + + +Bug Tracker +=========== + +Bugs are tracked on `GitHub Issues +`_. In case of trouble, please +check there if your issue has already been reported. If you spotted it first, +help us smashing it by providing a detailed and welcomed feedback. + Credits ======= @@ -54,6 +92,7 @@ Contributors * Ronald Portier * Sylvain Van Hoof * Pedro Baeza +* Dave Lasley Translations ------------ @@ -61,6 +100,7 @@ Translations * Danish: Hans Henrik Gabelgaard * Italian: Leonardo Donelli * Spanish: Antonio Espinosa +* Antonio Espinosa Maintainer ---------- diff --git a/partner_firstname/__manifest__.py b/partner_firstname/__manifest__.py index 086f313b0..da5c33ebe 100644 --- a/partner_firstname/__manifest__.py +++ b/partner_firstname/__manifest__.py @@ -7,17 +7,21 @@ { 'name': 'Partner first name and last name', 'summary': "Split first name and last name for non company partners", - 'version': '10.0.1.0.0', + 'version': '10.0.2.0.0', 'author': "Camptocamp, " "Grupo ESOC Ingeniería de Servicios, " + "Tecnativa, " + "LasLabs, " "Odoo Community Association (OCA)", 'license': "AGPL-3", 'maintainer': 'Camptocamp, Acsone', 'category': 'Extra Tools', - 'website': - 'http://www.camptocamp.com, http://www.acsone.eu, http://grupoesoc.es', - 'depends': ['base'], + 'website': 'http://www.camptocamp.com, ' + 'http://www.acsone.eu, ' + 'http://grupoesoc.es', + 'depends': ['base_setup'], 'data': [ + 'views/base_config_view.xml', 'views/res_partner.xml', 'views/res_user.xml', 'data/res_partner.yml', diff --git a/partner_firstname/models/__init__.py b/partner_firstname/models/__init__.py index 95688f355..05e809bb7 100644 --- a/partner_firstname/models/__init__.py +++ b/partner_firstname/models/__init__.py @@ -1,5 +1,7 @@ # -*- coding: utf-8 -*- -# © 2013 Nicolas Bessi (Camptocamp SA) +# © 2015 Antiun Ingenieria S.L. - Antonio Espinosa # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + +from . import base_config_settings from . import res_partner -from . import res_user +from . import res_users diff --git a/partner_firstname/models/base_config_settings.py b/partner_firstname/models/base_config_settings.py new file mode 100644 index 000000000..35d2b52fe --- /dev/null +++ b/partner_firstname/models/base_config_settings.py @@ -0,0 +1,76 @@ +# -*- coding: utf-8 -*- +# © 2015 Antiun Ingenieria S.L. - Antonio Espinosa +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + +import logging +from openerp import api, fields, models +_logger = logging.getLogger(__name__) + + +class BaseConfigSettings(models.TransientModel): + _inherit = 'base.config.settings' + + partner_names_order = fields.Selection( + string="Partner names order", + selection="_partner_names_order_selection", + help="Order to compose partner fullname", + required=True, + ) + partner_names_order_changed = fields.Boolean( + compute="_compute_names_order_changed", + ) + + def _partner_names_order_selection(self): + return [ + ('last_first', 'Lastname Firstname'), + ('last_first_comma', 'Lastname, Firstname'), + ('first_last', 'Firstname Lastname'), + ] + + @api.multi + def _partner_names_order_default(self): + return self.env['res.partner']._names_order_default() + + @api.model + def get_default_partner_names_order(self, fields): + return { + 'partner_names_order': self.env['ir.config_parameter'].get_param( + 'partner_names_order', self._partner_names_order_default(), + ), + } + + @api.multi + @api.depends('partner_names_order') + def _compute_names_order_changed(self): + current = self.env['ir.config_parameter'].get_param( + 'partner_names_order', self._partner_names_order_default(), + ) + for record in self: + record.partner_names_order_changed = bool( + record.partner_names_order != current + ) + + @api.multi + @api.onchange('partner_names_order') + def _onchange_partner_names_order(self): + self.partner_names_order_changed = self._compute_names_order_changed() + + @api.multi + def set_partner_names_order(self): + self.env['ir.config_parameter'].set_param( + 'partner_names_order', self.partner_names_order) + + @api.multi + def _partners_for_recalculating(self): + return self.env['res.partner'].search([ + ('is_company', '=', False), + ('firstname', '!=', False), ('lastname', '!=', False), + ]) + + @api.multi + def action_recalculate_partners_name(self): + partners = self._partners_for_recalculating() + _logger.info("Recalculating names for %d partners.", len(partners)) + partners._compute_name() + _logger.info("%d partners updated.", len(partners)) + return True diff --git a/partner_firstname/models/res_partner.py b/partner_firstname/models/res_partner.py index 8beed1c87..d576e6729 100644 --- a/partner_firstname/models/res_partner.py +++ b/partner_firstname/models/res_partner.py @@ -3,6 +3,7 @@ # © 2014 Agile Business Group () # © 2015 Grupo ESOC () # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + import logging from odoo import api, fields, models from .. import exceptions @@ -73,44 +74,71 @@ class ResPartner(models.Model): return result + @api.model + def _names_order_default(self): + return 'last_first' + + @api.model + def _get_names_order(self): + """Get names order configuration from system parameters. + You can override this method to read configuration from language, + country, company or other""" + return self.env['ir.config_parameter'].get_param( + 'partner_names_order', self._names_order_default()) + @api.model def _get_computed_name(self, lastname, firstname): """Compute the 'name' field according to splitted data. You can override this method to change the order of lastname and firstname the computed name""" - return u" ".join((p for p in (lastname, firstname) if p)) + order = self._get_names_order() + if order == 'last_first_comma': + return u", ".join((p for p in (lastname, firstname) if p)) + elif order == 'first_last': + return u" ".join((p for p in (firstname, lastname) if p)) + else: + return u" ".join((p for p in (lastname, firstname) if p)) - @api.one + @api.multi @api.depends("firstname", "lastname") def _compute_name(self): """Write the 'name' field according to splitted data.""" - self.name = self._get_computed_name(self.lastname, self.firstname) + for record in self: + record.name = record._get_computed_name( + record.lastname, record.firstname, + ) - @api.one + @api.multi def _inverse_name_after_cleaning_whitespace(self): """Clean whitespace in :attr:`~.name` and split it. The splitting logic is stored separately in :meth:`~._inverse_name`, so submodules can extend that method and get whitespace cleaning for free. """ - # Remove unneeded whitespace - clean = self._get_whitespace_cleaned_name(self.name) + for record in self: + # Remove unneeded whitespace + clean = record._get_whitespace_cleaned_name(record.name) - # Clean name avoiding infinite recursion - if self.name != clean: - self.name = clean + # Clean name avoiding infinite recursion + if record.name != clean: + record.name = clean - # Save name in the real fields - else: - self._inverse_name() + # Save name in the real fields + else: + record._inverse_name() @api.model - def _get_whitespace_cleaned_name(self, name): + def _get_whitespace_cleaned_name(self, name, comma=False): """Remove redundant whitespace from :param:`name`. Removes leading, trailing and duplicated whitespace. """ - return u" ".join(name.split(None)) if name else name + if name: + name = u" ".join(name.split(None)) + if comma: + name = name.replace(" ,", ",") + name = name.replace(", ", ",") + return name @api.model def _get_inverse_name(self, name, is_company=False): @@ -131,24 +159,39 @@ class ResPartner(models.Model): parts = [name or False, False] # Guess name splitting else: - parts = name.strip().split(" ", 1) - while len(parts) < 2: - parts.append(False) + order = self._get_names_order() + # Remove redundant spaces + name = self._get_whitespace_cleaned_name( + name, comma=(order == 'last_first_comma')) + parts = name.split("," if order == 'last_first_comma' else " ", 1) + if len(parts) > 1: + if order == 'first_last': + parts = [u" ".join(parts[1:]), parts[0]] + else: + parts = [parts[0], u" ".join(parts[1:])] + else: + while len(parts) < 2: + parts.append(False) return {"lastname": parts[0], "firstname": parts[1]} - @api.one + @api.multi def _inverse_name(self): """Try to revert the effect of :meth:`._compute_name`.""" - parts = self._get_inverse_name(self.name, self.is_company) - self.lastname, self.firstname = parts["lastname"], parts["firstname"] + for record in self: + parts = record._get_inverse_name(record.name, record.is_company) + record.lastname = parts['lastname'] + record.firstname = parts['firstname'] - @api.one + @api.multi @api.constrains("firstname", "lastname") def _check_name(self): """Ensure at least one name is set.""" - if ((self.type == 'contact' or self.is_company) and - not (self.firstname or self.lastname)): - raise exceptions.EmptyNamesError(self) + for record in self: + if all(( + record.type == 'contact' or record.is_company, + not (record.firstname or record.lastname) + )): + raise exceptions.EmptyNamesError(record) @api.onchange("firstname", "lastname") def _onchange_subnames(self): diff --git a/partner_firstname/models/res_user.py b/partner_firstname/models/res_users.py similarity index 100% rename from partner_firstname/models/res_user.py rename to partner_firstname/models/res_users.py diff --git a/partner_firstname/tests/__init__.py b/partner_firstname/tests/__init__.py index 1fed38100..e85089b63 100644 --- a/partner_firstname/tests/__init__.py +++ b/partner_firstname/tests/__init__.py @@ -9,5 +9,6 @@ from . import ( test_empty, test_name, test_onchange, - test_user_onchange + test_user_onchange, + test_order, ) diff --git a/partner_firstname/tests/test_create.py b/partner_firstname/tests/test_create.py index 810383808..6d80abdad 100644 --- a/partner_firstname/tests/test_create.py +++ b/partner_firstname/tests/test_create.py @@ -29,7 +29,6 @@ class PersonCase(TransactionCase): self.record = (self.env[self.model] .with_context(self.context) .create(self.values)) - for key, value in self.good_values.iteritems(): self.assertEqual( self.record[key], diff --git a/partner_firstname/tests/test_order.py b/partner_firstname/tests/test_order.py new file mode 100644 index 000000000..afc978b05 --- /dev/null +++ b/partner_firstname/tests/test_order.py @@ -0,0 +1,40 @@ +# -*- coding: utf-8 -*- +# © 2015 Antiun Ingenieria S.L. - Antonio Espinosa +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). + +from openerp.tests.common import TransactionCase + + +class PartnerNamesOrder(TransactionCase): + def order_set(self, order): + return self.env['ir.config_parameter'].set_param( + 'partner_names_order', order) + + def test_get_computed_name(self): + lastname = u"García Lorca" + firstname = u"Federico" + cases = ( + ('last_first', u"García Lorca Federico"), + ('last_first_comma', u"García Lorca, Federico"), + ('first_last', u"Federico García Lorca"), + ) + + for order, name in cases: + self.order_set(order) + result = self.env['res.partner']._get_computed_name( + lastname, firstname) + self.assertEqual(result, name) + + def test_get_inverse_name(self): + lastname = u"Flanker" + firstname = u"Petër" + cases = ( + ('last_first', u"Flanker Petër"), + ('last_first_comma', u"Flanker, Petër"), + ('first_last', u"Petër Flanker"), + ) + for order, name in cases: + self.order_set(order) + result = self.env['res.partner']._get_inverse_name(name) + self.assertEqual(result['lastname'], lastname) + self.assertEqual(result['firstname'], firstname) diff --git a/partner_firstname/views/base_config_view.xml b/partner_firstname/views/base_config_view.xml new file mode 100644 index 000000000..424b5d28f --- /dev/null +++ b/partner_firstname/views/base_config_view.xml @@ -0,0 +1,31 @@ + + + + + + Add partner_names_order config parameter + base.config.settings + + + + + + + + + +