Antonio Espinosa
10 years ago
committed by
Jairo Llopis
14 changed files with 876 additions and 0 deletions
-
91base_location_nuts/README.rst
-
7base_location_nuts/__init__.py
-
42base_location_nuts/__openerp__.py
-
220base_location_nuts/i18n/es.po
-
7base_location_nuts/models/__init__.py
-
61base_location_nuts/models/res_partner.py
-
30base_location_nuts/models/res_partner_nuts.py
-
2base_location_nuts/security/ir.model.access.csv
-
BINbase_location_nuts/static/description/icon.png
-
76base_location_nuts/views/res_partner_nuts_view.xml
-
78base_location_nuts/views/res_partner_view.xml
-
6base_location_nuts/wizard/__init__.py
-
205base_location_nuts/wizard/nuts_import.py
-
51base_location_nuts/wizard/nuts_import_view.xml
@ -0,0 +1,91 @@ |
|||||
|
.. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg |
||||
|
:alt: License: AGPL-3 |
||||
|
|
||||
|
============ |
||||
|
NUTS Regions |
||||
|
============ |
||||
|
|
||||
|
This module allows to import NUTS locations. |
||||
|
|
||||
|
Creates two new fields in Partner object: |
||||
|
|
||||
|
* Region (res.partner.region): Classification over state, automatically |
||||
|
calculated when state is selected |
||||
|
* Substate (res.partner.substate): Classification above state, user must select |
||||
|
one from available for selected state |
||||
|
|
||||
|
|
||||
|
Installation |
||||
|
============ |
||||
|
|
||||
|
You need to install another addon (one for each country) in order to use |
||||
|
these NUTS, for example: |
||||
|
|
||||
|
* l10n_es_location_nuts : |
||||
|
* Spanish Provinces (NUTS level 4) as Partner State |
||||
|
* Spanish Autonomous communities (NUTS level 3) as Partner Substate |
||||
|
* Spanish Regions (NUTS level 2) as Partner Region |
||||
|
* l10n_de_location_nuts : |
||||
|
* German states (NUTS level 2) as Partner State |
||||
|
* German districts (NUTS level 3) as Partner Substate |
||||
|
* German regions (NUTS level 4) as Partner Region |
||||
|
|
||||
|
|
||||
|
Configuration |
||||
|
============= |
||||
|
|
||||
|
After installation, you must click at import wizard to populate NUTS items |
||||
|
in Odoo database in: |
||||
|
Sales > Configuration > Address Book > Import NUTS 2013 |
||||
|
|
||||
|
This wizard will download from Europe RAMON service the metadata to |
||||
|
build NUTS in Odoo. Each localization addon (l10n_es_location_nuts, |
||||
|
l10n_de_location_nuts, ...) will inherit this wizard and |
||||
|
relate each NUTS item with states. So if you install a new localization addon |
||||
|
you must re-build NUTS clicking this wizard again. |
||||
|
|
||||
|
|
||||
|
Usage |
||||
|
===== |
||||
|
|
||||
|
Only Administrator can manage NUTS list (it is not neccesary because |
||||
|
it is an European convention) but any registered user can read them, |
||||
|
in order to allow to assign them to partner object. |
||||
|
|
||||
|
.. 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/{branch} |
||||
|
|
||||
|
|
||||
|
Bug Tracker |
||||
|
=========== |
||||
|
|
||||
|
Bugs are tracked on `GitHub Issues <https://github.com/OCA/partner-contact/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 |
||||
|
`here <https://github.com/OCA/partner-contact/issues/new?body=module:%20base_location_nuts%0Aversion:%208.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**>`_. |
||||
|
|
||||
|
|
||||
|
Credits |
||||
|
======= |
||||
|
|
||||
|
Contributors |
||||
|
------------ |
||||
|
|
||||
|
* Rafael Blasco <rafabn@antiun.com> |
||||
|
* Antonio Espinosa <antonioea@antiun.com> |
||||
|
|
||||
|
Maintainer |
||||
|
---------- |
||||
|
|
||||
|
.. image:: https://odoo-community.org/logo.png |
||||
|
:alt: Odoo Community Association |
||||
|
:target: https://odoo-community.org |
||||
|
|
||||
|
This module is maintained by the OCA. |
||||
|
|
||||
|
OCA, or the Odoo Community Association, is a nonprofit organization whose |
||||
|
mission is to support the collaborative development of Odoo features and |
||||
|
promote its widespread use. |
||||
|
|
||||
|
To contribute to this module, please visit http://odoo-community.org. |
@ -0,0 +1,7 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# For copyright and license notices, see __openerp__.py file in root directory |
||||
|
############################################################################## |
||||
|
|
||||
|
from . import models |
||||
|
from . import wizard |
@ -0,0 +1,42 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Python source code encoding : https://www.python.org/dev/peps/pep-0263/ |
||||
|
############################################################################## |
||||
|
# |
||||
|
# OpenERP, Odoo Source Management Solution |
||||
|
# Copyright (c) 2015 Antiun Ingeniería S.L. (http://www.antiun.com) |
||||
|
# Antonio Espinosa <antonioea@antiun.com> |
||||
|
# |
||||
|
# This program is free software: you can redistribute it and/or modify |
||||
|
# it under the terms of the GNU Affero General Public License as published |
||||
|
# by the Free Software Foundation, either version 3 of the License, or |
||||
|
# (at your option) any later version. |
||||
|
# |
||||
|
# This program is distributed in the hope that it will be useful, |
||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
|
# GNU Affero General Public License for more details. |
||||
|
# |
||||
|
# You should have received a copy of the GNU Affero General Public License |
||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
|
# |
||||
|
############################################################################## |
||||
|
|
||||
|
{ |
||||
|
'name': 'NUTS Regions', |
||||
|
'category': 'Localisation/Europe', |
||||
|
'version': '8.0.1.0.0', |
||||
|
'depends': [ |
||||
|
'base', |
||||
|
], |
||||
|
'data': [ |
||||
|
'views/res_partner_nuts_view.xml', |
||||
|
'views/res_partner_view.xml', |
||||
|
'wizard/nuts_import_view.xml', |
||||
|
'security/ir.model.access.csv', |
||||
|
], |
||||
|
'author': 'Antiun Ingeniería S.L., ' |
||||
|
'Odoo Community Association (OCA)', |
||||
|
'website': 'http://www.antiun.com', |
||||
|
'license': 'AGPL-3', |
||||
|
'installable': True, |
||||
|
} |
@ -0,0 +1,220 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * base_location_nuts |
||||
|
# |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 8.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2015-06-11 15:26+0000\n" |
||||
|
"PO-Revision-Date: 2015-06-11 15:26+0000\n" |
||||
|
"Last-Translator: <>\n" |
||||
|
"Language-Team: \n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: \n" |
||||
|
"Plural-Forms: \n" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:nuts.import:base_location_nuts.nuts_import_form |
||||
|
msgid "Cancel" |
||||
|
msgstr "Cancelar" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,children:0 |
||||
|
msgid "Children" |
||||
|
msgstr "Hijos" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,code:0 |
||||
|
msgid "Code" |
||||
|
msgstr "Código" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:res.partner.nuts:base_location_nuts.view_res_partner_nuts_filter |
||||
|
#: field:res.partner.nuts,country_id:0 |
||||
|
msgid "Country" |
||||
|
msgstr "País" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:nuts.import,create_uid:0 |
||||
|
#: field:res.partner.nuts,create_uid:0 |
||||
|
msgid "Created by" |
||||
|
msgstr "Creado por" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:nuts.import,create_date:0 |
||||
|
#: field:res.partner.nuts,create_date:0 |
||||
|
msgid "Created on" |
||||
|
msgstr "Creado en" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: code:addons/base_location_nuts/wizard/nuts_import.py:149 |
||||
|
#, python-format |
||||
|
msgid "Downloaded file is not a valid XML file" |
||||
|
msgstr "El fichero descargado no es un fichero XML válido" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: code:addons/base_location_nuts/wizard/nuts_import.py:141 |
||||
|
#, python-format |
||||
|
msgid "Got an error %d when trying to download the file %s." |
||||
|
msgstr "Error %d al intentar descargar el fichero %s." |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: code:addons/base_location_nuts/wizard/nuts_import.py:137 |
||||
|
#, python-format |
||||
|
msgid "Got an error when trying to download the file: %s." |
||||
|
msgstr "Error al intentar descargar el fichero: %s." |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:res.partner.nuts:base_location_nuts.view_res_partner_nuts_filter |
||||
|
msgid "Group By" |
||||
|
msgstr "Agrupar por" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:nuts.import,id:0 |
||||
|
#: field:res.partner.nuts,id:0 |
||||
|
msgid "ID" |
||||
|
msgstr "ID" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:nuts.import:base_location_nuts.nuts_import_form |
||||
|
msgid "Import" |
||||
|
msgstr "Importar" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.ui.menu,name:base_location_nuts.nuts_import_menu |
||||
|
msgid "Import NUTS 2013" |
||||
|
msgstr "Importar NUTS 2013" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.actions.act_window,name:base_location_nuts.nuts_import_action |
||||
|
#: view:nuts.import:base_location_nuts.nuts_import_form |
||||
|
msgid "Import NUTS 2013 from RAMON" |
||||
|
msgstr "Importar NUTS 2013 desde RAMON" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.model,name:base_location_nuts.model_nuts_import |
||||
|
msgid "Import NUTS items from European RAMON service" |
||||
|
msgstr "Importar regiones NUTS desde el servicio europeo RAMON" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:nuts.import,write_uid:0 |
||||
|
#: field:res.partner.nuts,write_uid:0 |
||||
|
msgid "Last Updated by" |
||||
|
msgstr "Última actualización por" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:nuts.import,write_date:0 |
||||
|
#: field:res.partner.nuts,write_date:0 |
||||
|
msgid "Last Updated on" |
||||
|
msgstr "Última actualización en" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:res.partner.nuts:base_location_nuts.view_res_partner_nuts_filter |
||||
|
#: field:res.partner.nuts,level:0 |
||||
|
msgid "Level" |
||||
|
msgstr "Nivel" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.model,name:base_location_nuts.model_res_partner_nuts |
||||
|
#: view:res.partner.nuts:base_location_nuts.res_partner_nuts_form |
||||
|
msgid "NUTS Item" |
||||
|
msgstr "Región NUTS" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.actions.act_window,name:base_location_nuts.res_partner_nuts_action |
||||
|
#: model:ir.ui.menu,name:base_location_nuts.res_partner_nuts_menu |
||||
|
#: view:res.partner.nuts:base_location_nuts.res_partner_nuts_tree |
||||
|
msgid "NUTS Items" |
||||
|
msgstr "Regiones NUTS" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,name:0 |
||||
|
msgid "Name" |
||||
|
msgstr "Nombre" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,parent_left:0 |
||||
|
msgid "Parent Left" |
||||
|
msgstr "Padre izquierda" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,parent_right:0 |
||||
|
msgid "Parent Right" |
||||
|
msgstr "Padre derecha" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,parent_id:0 |
||||
|
msgid "Parent id" |
||||
|
msgstr "ID del padre" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.model,name:base_location_nuts.model_res_partner |
||||
|
msgid "Partner" |
||||
|
msgstr "Empresa" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: code:addons/base_location_nuts/models/res_partner.py:53 |
||||
|
#: view:res.partner:base_location_nuts.view_res_partner_filter_nuts |
||||
|
#: field:res.partner,region:0 |
||||
|
#, python-format |
||||
|
msgid "Region" |
||||
|
msgstr "Región" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:res.partner:base_location_nuts.view_res_partner_filter_nuts |
||||
|
msgid "Salesperson" |
||||
|
msgstr "Comercial" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:res.partner.nuts:base_location_nuts.view_res_partner_nuts_filter |
||||
|
msgid "Search NUTS" |
||||
|
msgstr "Buscar NUTS" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: field:res.partner.nuts,state_id:0 |
||||
|
msgid "State" |
||||
|
msgstr "Provincia" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: code:addons/base_location_nuts/models/res_partner.py:54 |
||||
|
#: view:res.partner:base_location_nuts.view_res_partner_filter_nuts |
||||
|
#: field:res.partner,substate:0 |
||||
|
#, python-format |
||||
|
msgid "Substate" |
||||
|
msgstr "Estado" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: view:nuts.import:base_location_nuts.nuts_import_form |
||||
|
msgid "This wizard will download the lastest version of\n" |
||||
|
" NUTS 2013 from Europe RAMON metadata service.\n" |
||||
|
" Updating or creating new NUTS entries if not\n" |
||||
|
" found already in the system, and DELETING MISSING\n" |
||||
|
" ENTRIES from new downloaded file." |
||||
|
msgstr "Este asistente descargará la última version de\n" |
||||
|
" NUTS 2013 desde el servicio de matadatos europeo RAMON.\n" |
||||
|
" Actualizando o creando nuevas regiones NUTS si no\n" |
||||
|
" las encuentra en el sistemma, y BORRANDO LAS QUE NO ENCUENTRE\n" |
||||
|
" en el nuevo fichero descargado." |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: code:addons/base_location_nuts/wizard/nuts_import.py:116 |
||||
|
#, python-format |
||||
|
msgid "Value not found for mandatory field %s" |
||||
|
msgstr "El valor no se ha encontrado para el campo obligatorio %s" |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.actions.todo,note:base_location_nuts.config_wizard_nuts |
||||
|
msgid "You can import NUTS from RAMON european service." |
||||
|
msgstr "Usted puede importar NUTS desde el servicion europeo RAMON." |
||||
|
|
||||
|
#. module: base_location_nuts |
||||
|
#: model:ir.actions.act_window,help:base_location_nuts.res_partner_nuts_action |
||||
|
msgid "You must click at import wizard to populate NUTS items\n" |
||||
|
" in Odoo database in:\n" |
||||
|
" Sales > Configuration > Address Book > Localization > Import NUTS 2013" |
||||
|
msgstr "Debes clicar en el asistente de importación para crear las\n" |
||||
|
" las regiones NUTS en la base de datos de Odoo, en el menú:\n" |
||||
|
" Ventas > Configuración > Libreta de direcciones > Localización > Importar NUTS 2013" |
||||
|
|
@ -0,0 +1,7 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# For copyright and license notices, see __openerp__.py file in root directory |
||||
|
############################################################################## |
||||
|
|
||||
|
from . import res_partner_nuts |
||||
|
from . import res_partner |
@ -0,0 +1,61 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# For copyright and license notices, see __openerp__.py file in root directory |
||||
|
############################################################################## |
||||
|
|
||||
|
from openerp import models, fields, api |
||||
|
from openerp.tools.translate import _ |
||||
|
import collections |
||||
|
|
||||
|
|
||||
|
def dict_recursive_update(d, u): |
||||
|
for k, v in u.iteritems(): |
||||
|
if isinstance(v, collections.Mapping): |
||||
|
r = dict_recursive_update(d.get(k, {}), v) |
||||
|
d[k] = r |
||||
|
else: |
||||
|
d[k] = u[k] |
||||
|
return d |
||||
|
|
||||
|
|
||||
|
class ResPartner(models.Model): |
||||
|
_inherit = 'res.partner' |
||||
|
|
||||
|
region = fields.Many2one(comodel_name='res.partner.nuts', |
||||
|
string="Region") |
||||
|
substate = fields.Many2one(comodel_name='res.partner.nuts', |
||||
|
string="Substate") |
||||
|
lbl_region = fields.Char(compute='_labels_get') |
||||
|
lbl_substate = fields.Char(compute='_labels_get') |
||||
|
|
||||
|
@api.one |
||||
|
@api.depends('country_id') |
||||
|
def _labels_get(self): |
||||
|
self.lbl_region = _('Region') |
||||
|
self.lbl_substate = _('Substate') |
||||
|
|
||||
|
@api.multi |
||||
|
def onchange_state(self, state_id): |
||||
|
result = super(ResPartner, self).onchange_state(state_id) |
||||
|
if not state_id: |
||||
|
changes = { |
||||
|
'domain': { |
||||
|
'substate': [], |
||||
|
'region': [], |
||||
|
}, |
||||
|
'value': { |
||||
|
'substate': False, |
||||
|
'region': False, |
||||
|
} |
||||
|
} |
||||
|
dict_recursive_update(result, changes) |
||||
|
return result |
||||
|
|
||||
|
@api.onchange('substate', 'region') |
||||
|
def onchange_substate_or_region(self): |
||||
|
result = {'domain': {}} |
||||
|
if not self.substate: |
||||
|
result['domain']['substate'] = [] |
||||
|
if not self.region: |
||||
|
result['domain']['region'] = [] |
||||
|
return result |
@ -0,0 +1,30 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# For copyright and license notices, see __openerp__.py file in root directory |
||||
|
############################################################################## |
||||
|
|
||||
|
from openerp import models, fields |
||||
|
|
||||
|
|
||||
|
class ResPartnerNuts(models.Model): |
||||
|
_name = 'res.partner.nuts' |
||||
|
_order = "parent_left" |
||||
|
_parent_order = "name" |
||||
|
_parent_store = True |
||||
|
_description = "NUTS Item" |
||||
|
|
||||
|
# NUTS fields |
||||
|
level = fields.Integer(required=True) |
||||
|
code = fields.Char(required=True) |
||||
|
name = fields.Char(required=True, translate=True) |
||||
|
country_id = fields.Many2one(comodel_name='res.country', string="Country", |
||||
|
required=True) |
||||
|
state_id = fields.Many2one(comodel_name='res.country.state', |
||||
|
string='State') |
||||
|
# Parent hierarchy |
||||
|
parent_id = fields.Many2one(comodel_name='res.partner.nuts', |
||||
|
ondelete='restrict') |
||||
|
children = fields.One2many(comodel_name='res.partner.nuts', |
||||
|
inverse_name='parent_id') |
||||
|
parent_left = fields.Integer('Parent Left', select=True) |
||||
|
parent_right = fields.Integer('Parent Right', select=True) |
@ -0,0 +1,2 @@ |
|||||
|
"id","name","model_id:id","group_id:id","perm_read","perm_write","perm_create","perm_unlink" |
||||
|
"access_res_partner_nuts_user","res_partner_nuts group_user","model_res_partner_nuts","base.group_user",1,0,0,0 |
After Width: 128 | Height: 128 | Size: 7.7 KiB |
@ -0,0 +1,76 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<openerp> |
||||
|
<data> |
||||
|
|
||||
|
<record id="res_partner_nuts_tree" model="ir.ui.view"> |
||||
|
<field name="name">NUTS Items tree</field> |
||||
|
<field name="model">res.partner.nuts</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<tree string="NUTS Items"> |
||||
|
<field name="level"/> |
||||
|
<field name="code"/> |
||||
|
<field name="name"/> |
||||
|
</tree> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="res_partner_nuts_form" model="ir.ui.view"> |
||||
|
<field name="name">NUTS Items tree</field> |
||||
|
<field name="model">res.partner.nuts</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form string="NUTS Item"> |
||||
|
<group> |
||||
|
<group> |
||||
|
<field name="level"/> |
||||
|
<field name="code"/> |
||||
|
<field name="parent_id"/> |
||||
|
<field name="name"/> |
||||
|
</group> |
||||
|
<group> |
||||
|
<field name="country_id"/> |
||||
|
<field name="state_id"/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="res_partner_nuts_action" model="ir.actions.act_window"> |
||||
|
<field name="name">NUTS Items</field> |
||||
|
<field name="type">ir.actions.act_window</field> |
||||
|
<field name="res_model">res.partner.nuts</field> |
||||
|
<field name="view_type">form</field> |
||||
|
<field name="view_mode">tree,form</field> |
||||
|
<field name="help">You must click at import wizard to populate NUTS items |
||||
|
in Odoo database in: |
||||
|
Sales > Configuration > Address Book > Localization > Import NUTS 2013</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="view_res_partner_nuts_filter"> |
||||
|
<field name="name">NUTS search filters</field> |
||||
|
<field name="model">res.partner.nuts</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<search string="Search NUTS"> |
||||
|
<field name="name"/> |
||||
|
<field name="country_id"/> |
||||
|
<field name="state_id"/> |
||||
|
<group expand="0" string="Group By"> |
||||
|
<filter string="Country" |
||||
|
domain="[]" |
||||
|
context="{'group_by': 'country_id'}"/> |
||||
|
<filter string="Level" |
||||
|
domain="[]" |
||||
|
context="{'group_by': 'level'}"/> |
||||
|
</group> |
||||
|
</search> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<menuitem action="res_partner_nuts_action" |
||||
|
id="res_partner_nuts_menu" |
||||
|
name="NUTS Items" |
||||
|
parent="base.menu_localisation" |
||||
|
sequence="40"/> |
||||
|
|
||||
|
</data> |
||||
|
</openerp> |
@ -0,0 +1,78 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<openerp> |
||||
|
<data> |
||||
|
|
||||
|
<record model="ir.ui.view" id="view_partner_form_nuts"> |
||||
|
<field name="name">Partner form with NUTS</field> |
||||
|
<field name="model">res.partner</field> |
||||
|
<field name="inherit_id" ref="base.view_partner_form"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<xpath expr="//sheet/group//field[@name='country_id']/.." position="after"> |
||||
|
<div width="1%" cell-class="oe_form_group_cell_label" |
||||
|
attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="lbl_substate" class="oe_inline"/> |
||||
|
</div> |
||||
|
<div attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="substate" |
||||
|
class="oe_no_button" |
||||
|
options="{'no_open': True, 'no_create': True}" |
||||
|
attrs="{'readonly': [('use_parent_address','=',True)]}"/> |
||||
|
</div> |
||||
|
<div width="1%" cell-class="oe_form_group_cell_label" |
||||
|
attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="lbl_region" class="oe_inline"/> |
||||
|
</div> |
||||
|
<div attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="region" |
||||
|
class="oe_no_button" |
||||
|
options="{'no_open': True, 'no_create': True}" |
||||
|
attrs="{'readonly': [('use_parent_address','=',True)]}"/> |
||||
|
</div> |
||||
|
</xpath> |
||||
|
<xpath expr="//field[@name='child_ids']/form//field[@name='country_id']/.." position="after"> |
||||
|
<div width="1%" cell-class="oe_form_group_cell_label" |
||||
|
attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="lbl_substate" class="oe_inline"/> |
||||
|
</div> |
||||
|
<div attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="substate" |
||||
|
class="oe_no_button" |
||||
|
options="{'no_open': True, 'no_create': True}" |
||||
|
attrs="{'readonly': [('use_parent_address','=',True)]}"/> |
||||
|
</div> |
||||
|
<div width="1%" cell-class="oe_form_group_cell_label" |
||||
|
attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="lbl_region" class="oe_inline"/> |
||||
|
</div> |
||||
|
<div attrs="{'invisible': [('use_parent_address','=',True)]}"> |
||||
|
<field name="region" |
||||
|
class="oe_no_button" |
||||
|
options="{'no_open': True, 'no_create': True}" |
||||
|
attrs="{'readonly': [('use_parent_address','=',True)]}"/> |
||||
|
</div> |
||||
|
</xpath> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="view_res_partner_filter_nuts"> |
||||
|
<field name="name">Partner search with NUTS</field> |
||||
|
<field name="model">res.partner</field> |
||||
|
<field name="inherit_id" ref="base.view_res_partner_filter"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<field name="category_id" position="after"> |
||||
|
<field name="region"/> |
||||
|
<field name="substate"/> |
||||
|
</field> |
||||
|
<filter string="Salesperson" position="after"> |
||||
|
<filter string="Region" |
||||
|
domain="[]" |
||||
|
context="{'group_by': 'region'}"/> |
||||
|
<filter string="Substate" |
||||
|
domain="[]" |
||||
|
context="{'group_by': 'substate'}"/> |
||||
|
</filter> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
||||
|
</openerp> |
@ -0,0 +1,6 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# For copyright and license notices, see __openerp__.py file in root directory |
||||
|
############################################################################## |
||||
|
|
||||
|
from . import nuts_import |
@ -0,0 +1,205 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
############################################################################## |
||||
|
# For copyright and license notices, see __openerp__.py file in root directory |
||||
|
############################################################################## |
||||
|
|
||||
|
from openerp import models, api, _ |
||||
|
from openerp.exceptions import Warning |
||||
|
import requests |
||||
|
import re |
||||
|
import logging |
||||
|
from lxml import etree |
||||
|
from collections import OrderedDict |
||||
|
|
||||
|
from pprint import pformat |
||||
|
|
||||
|
logger = logging.getLogger(__name__) |
||||
|
|
||||
|
|
||||
|
class NutsImport(models.TransientModel): |
||||
|
_name = 'nuts.import' |
||||
|
_description = 'Import NUTS items from European RAMON service' |
||||
|
_parents = [False, False, False, False] |
||||
|
_countries = { |
||||
|
"BE": False, |
||||
|
"BG": False, |
||||
|
"CZ": False, |
||||
|
"DK": False, |
||||
|
"DE": False, |
||||
|
"EE": False, |
||||
|
"IE": False, |
||||
|
"GR": False, # EL |
||||
|
"ES": False, |
||||
|
"FR": False, |
||||
|
"HR": False, |
||||
|
"IT": False, |
||||
|
"CY": False, |
||||
|
"LV": False, |
||||
|
"LT": False, |
||||
|
"LU": False, |
||||
|
"HU": False, |
||||
|
"MT": False, |
||||
|
"NL": False, |
||||
|
"AT": False, |
||||
|
"PL": False, |
||||
|
"PT": False, |
||||
|
"RO": False, |
||||
|
"SI": False, |
||||
|
"SK": False, |
||||
|
"FI": False, |
||||
|
"SE": False, |
||||
|
"GB": False, # UK |
||||
|
} |
||||
|
_current_country = False |
||||
|
_map = OrderedDict([ |
||||
|
('level', { |
||||
|
'xpath': '', 'attrib': 'idLevel', |
||||
|
'type': 'integer', 'required': True}), |
||||
|
('code', { |
||||
|
'xpath': './Label/LabelText[@language="ALL"]', |
||||
|
'type': 'string', 'required': True}), |
||||
|
('name', { |
||||
|
'xpath': './Label/LabelText[@language="EN"]', |
||||
|
'type': 'string', 'required': True}), |
||||
|
]) |
||||
|
|
||||
|
def _check_node(self, node): |
||||
|
if node.get('id') and node.get('idLevel'): |
||||
|
return True |
||||
|
return False |
||||
|
|
||||
|
def _mapping(self, node): |
||||
|
item = {} |
||||
|
for k, v in self._map.iteritems(): |
||||
|
field_xpath = v.get('xpath', '') |
||||
|
field_attrib = v.get('attrib', False) |
||||
|
field_type = v.get('type', 'string') |
||||
|
field_required = v.get('required', False) |
||||
|
value = '' |
||||
|
if field_xpath: |
||||
|
n = node.find(field_xpath) |
||||
|
else: |
||||
|
n = node |
||||
|
if n is not None: |
||||
|
if field_attrib: |
||||
|
value = n.get(field_attrib, '') |
||||
|
else: |
||||
|
value = n.text |
||||
|
if field_type == 'integer': |
||||
|
try: |
||||
|
value = int(value) |
||||
|
except: |
||||
|
value = 0 |
||||
|
else: |
||||
|
logger.debug("xpath = '%s', not found" % field_xpath) |
||||
|
if field_required and not value: |
||||
|
raise Warning( |
||||
|
_('Value not found for mandatory field %s' % k)) |
||||
|
item[k] = value |
||||
|
return item |
||||
|
|
||||
|
def _download_nuts(self): |
||||
|
url_base = 'http://ec.europa.eu' |
||||
|
url_path = '/eurostat/ramon/nomenclatures/index.cfm' |
||||
|
url_params = { |
||||
|
'TargetUrl': 'ACT_OTH_CLS_DLD', |
||||
|
'StrNom': 'NUTS_2013', |
||||
|
'StrFormat': 'XML', |
||||
|
'StrLanguageCode': 'EN', |
||||
|
'StrLayoutCode': 'HIERARCHIC' |
||||
|
} |
||||
|
url = url_base + url_path + '?' |
||||
|
url += '&'.join([k + '=' + v for k, v in url_params.iteritems()]) |
||||
|
logger.info('Starting to download %s' % url) |
||||
|
try: |
||||
|
res_request = requests.get(url) |
||||
|
except Exception, e: |
||||
|
raise Warning( |
||||
|
_('Got an error when trying to download the file: %s.') % |
||||
|
str(e)) |
||||
|
if res_request.status_code != requests.codes.ok: |
||||
|
raise Warning( |
||||
|
_('Got an error %d when trying to download the file %s.') |
||||
|
% (res_request.status_code, url)) |
||||
|
logger.info('Download successfully %d bytes' % |
||||
|
len(res_request.content)) |
||||
|
# Workaround XML: Remove all characters before <?xml |
||||
|
pattern = re.compile(r'^.*<\?xml', re.DOTALL) |
||||
|
content_fixed = re.sub(pattern, '<?xml', res_request.content) |
||||
|
if not re.match(r'<\?xml', content_fixed): |
||||
|
raise Warning(_('Downloaded file is not a valid XML file')) |
||||
|
return content_fixed |
||||
|
|
||||
|
@api.model |
||||
|
def _load_countries(self): |
||||
|
for k in self._countries.keys(): |
||||
|
self._countries[k] = self.env['res.country'].search( |
||||
|
[('code', '=', k)]) |
||||
|
# Workaround to translate some country codes: |
||||
|
# EL => GR (Greece) |
||||
|
# UK => GB (United Kingdom) |
||||
|
self._countries['EL'] = self._countries['GR'] |
||||
|
self._countries['UK'] = self._countries['GB'] |
||||
|
logger.info('_load_countries = %s' % pformat(self._countries)) |
||||
|
|
||||
|
@api.model |
||||
|
def state_mapping(self, data, node): |
||||
|
# Method to inherit and add state_id relation depending on country |
||||
|
level = data.get('level', 0) |
||||
|
code = data.get('code', '') |
||||
|
if level == 1: |
||||
|
self._current_country = self._countries[code] |
||||
|
return { |
||||
|
'country_id': self._current_country.id, |
||||
|
} |
||||
|
|
||||
|
@api.model |
||||
|
def create_or_update_nuts(self, node): |
||||
|
if not self._check_node(node): |
||||
|
return False |
||||
|
|
||||
|
nuts_model = self.env['res.partner.nuts'] |
||||
|
data = self._mapping(node) |
||||
|
data.update(self.state_mapping(data, node)) |
||||
|
level = data.get('level', 0) |
||||
|
if level >= 2 and level <= 5: |
||||
|
data['parent_id'] = self._parents[level - 2] |
||||
|
nuts = nuts_model.search([('level', '=', data['level']), |
||||
|
('code', '=', data['code'])]) |
||||
|
if nuts: |
||||
|
nuts.write(data) |
||||
|
else: |
||||
|
nuts = nuts_model.create(data) |
||||
|
if level >= 1 and level <= 4: |
||||
|
self._parents[level - 1] = nuts.id |
||||
|
return nuts |
||||
|
|
||||
|
@api.one |
||||
|
def run_import(self): |
||||
|
nuts_model = self.env['res.partner.nuts'].\ |
||||
|
with_context(defer_parent_store_computation=True) |
||||
|
self._load_countries() |
||||
|
# All current NUTS (for available countries), |
||||
|
# delete if not found above |
||||
|
nuts_to_delete = nuts_model.search( |
||||
|
[('country_id', 'in', [x.id for x in self._countries.values()])]) |
||||
|
# Download NUTS in english, create or update |
||||
|
logger.info('Import NUTS 2013 English') |
||||
|
xmlcontent = self._download_nuts() |
||||
|
dom = etree.fromstring(xmlcontent) |
||||
|
for node in dom.iter('Item'): |
||||
|
logger.info('Reading level=%s, id=%s' % |
||||
|
(node.get('idLevel', 'N/A'), |
||||
|
node.get('id', 'N/A'))) |
||||
|
nuts = self.create_or_update_nuts(node) |
||||
|
if nuts and nuts in nuts_to_delete: |
||||
|
nuts_to_delete -= nuts |
||||
|
# Delete obsolete NUTS |
||||
|
if nuts_to_delete: |
||||
|
logger.info('%d NUTS entries deleted' % len(nuts_to_delete)) |
||||
|
nuts_to_delete.unlink() |
||||
|
logger.info( |
||||
|
'The wizard to create NUTS entries from RAMON ' |
||||
|
'has been successfully completed.') |
||||
|
|
||||
|
return True |
@ -0,0 +1,51 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<openerp> |
||||
|
<data> |
||||
|
|
||||
|
<record id="nuts_import_form" model="ir.ui.view"> |
||||
|
<field name="name">NUTS import</field> |
||||
|
<field name="model">nuts.import</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form string="Import NUTS 2013 from RAMON"> |
||||
|
<div> |
||||
|
This wizard will download the lastest version of |
||||
|
NUTS 2013 from Europe RAMON metadata service. |
||||
|
Updating or creating new NUTS entries if not |
||||
|
found already in the system, and DELETING MISSING |
||||
|
ENTRIES from new downloaded file. |
||||
|
</div> |
||||
|
<footer> |
||||
|
<button name="run_import" type="object" |
||||
|
class="oe_highlight" string="Import"/> |
||||
|
<button special="cancel" string="Cancel" class="oe_link"/> |
||||
|
</footer> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="nuts_import_action" model="ir.actions.act_window"> |
||||
|
<field name="name">Import NUTS 2013 from RAMON</field> |
||||
|
<field name="type">ir.actions.act_window</field> |
||||
|
<field name="res_model">nuts.import</field> |
||||
|
<field name="view_id" ref="nuts_import_form"/> |
||||
|
<field name="view_type">form</field> |
||||
|
<field name="view_mode">form</field> |
||||
|
<field name="target">new</field> |
||||
|
</record> |
||||
|
|
||||
|
<menuitem action="nuts_import_action" |
||||
|
id="nuts_import_menu" |
||||
|
name="Import NUTS 2013" |
||||
|
parent="base.menu_localisation" |
||||
|
sequence="45"/> |
||||
|
|
||||
|
<record id="config_wizard_nuts" model="ir.actions.todo"> |
||||
|
<field name="name">Import NUTS 2013 from RAMON</field> |
||||
|
<field name="note">You can import NUTS from RAMON european service.</field> |
||||
|
<field name="action_id" ref="nuts_import_action"/> |
||||
|
<field name="sequence">20</field> |
||||
|
<field name="type">automatic</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
||||
|
</openerp> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue