diff --git a/setup/web_company_color/odoo/addons/web_company_color b/setup/web_company_color/odoo/addons/web_company_color new file mode 120000 index 000000000..ba46d5db5 --- /dev/null +++ b/setup/web_company_color/odoo/addons/web_company_color @@ -0,0 +1 @@ +../../../../web_company_color \ No newline at end of file diff --git a/setup/web_company_color/setup.py b/setup/web_company_color/setup.py new file mode 100644 index 000000000..28c57bb64 --- /dev/null +++ b/setup/web_company_color/setup.py @@ -0,0 +1,6 @@ +import setuptools + +setuptools.setup( + setup_requires=['setuptools-odoo'], + odoo_addon=True, +) diff --git a/web_company_color/README.rst b/web_company_color/README.rst new file mode 100644 index 000000000..68e1d45e7 --- /dev/null +++ b/web_company_color/README.rst @@ -0,0 +1,89 @@ +================= +Web Company Color +================= + +.. !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |badge1| image:: https://img.shields.io/badge/maturity-Beta-yellow.png + :target: https://odoo-community.org/page/development-status + :alt: Beta +.. |badge2| image:: https://img.shields.io/badge/licence-AGPL--3-blue.png + :target: http://www.gnu.org/licenses/agpl-3.0-standalone.html + :alt: License: AGPL-3 +.. |badge3| image:: https://img.shields.io/badge/github-OCA%2Fweb-lightgray.png?logo=github + :target: https://github.com/OCA/web/tree/13.0/web_company_color + :alt: OCA/web +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/web-13-0/web-13-0-web_company_color + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runbot-Try%20me-875A7B.png + :target: https://runbot.odoo-community.org/runbot/162/13.0 + :alt: Try me on Runbot + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module change navbar colors based in the company logo colors. + +**Table of contents** + +.. contents:: + :local: + +Usage +===== + +Go to company record and set a logo. Can see/modify applied colors on the "Navbar" section. + +For optimal results use images with alpha channel. + +Known issues / Roadmap +====================== + +White color is omitted in the addition operation to support images without alpha channel. + +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 `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +~~~~~~~ + +* Alexandre Díaz + +Contributors +~~~~~~~~~~~~ + +* Jordi Ballester Alomar (ForgeFlow) +* Lois Rilo (ForgeFlow) +* Simone Orsi +* Jairo Llopis +* Iván Antón + +Maintainers +~~~~~~~~~~~ + +This module is maintained by the OCA. + +.. image:: https://odoo-community.org/logo.png + :alt: Odoo Community Association + :target: https://odoo-community.org + +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. + +This module is part of the `OCA/web `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/web_company_color/__init__.py b/web_company_color/__init__.py new file mode 100644 index 000000000..62048b1df --- /dev/null +++ b/web_company_color/__init__.py @@ -0,0 +1,5 @@ +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import models +from .hooks import post_init_hook, uninstall_hook +from . import utils diff --git a/web_company_color/__manifest__.py b/web_company_color/__manifest__.py new file mode 100644 index 000000000..e73cf17e5 --- /dev/null +++ b/web_company_color/__manifest__.py @@ -0,0 +1,18 @@ +# Odoo, Open Source Web Company Color +# Copyright (C) 2019 Alexandre Díaz +# +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl).# +{ + "name": "Web Company Color", + "category": "web", + "version": "14.0.1.0.0", + "author": "Alexandre Díaz, Odoo Community Association (OCA)", + "website": "https://github.com/OCA/web", + "depends": ["web", "base_sparse_field"], + "data": ["view/assets.xml", "view/res_company.xml"], + "uninstall_hook": "uninstall_hook", + "post_init_hook": "post_init_hook", + "license": "AGPL-3", + "auto_install": False, + "installable": True, +} diff --git a/web_company_color/hooks.py b/web_company_color/hooks.py new file mode 100644 index 000000000..ca23677bf --- /dev/null +++ b/web_company_color/hooks.py @@ -0,0 +1,15 @@ +# Copyright 2019 Alexandre Díaz +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). +from odoo import SUPERUSER_ID, api + +from .models.res_company import URL_BASE + + +def uninstall_hook(cr, registry): + env = api.Environment(cr, SUPERUSER_ID, {}) + env["ir.attachment"].search([("url", "=like", "%s%%" % URL_BASE)]).unlink() + + +def post_init_hook(cr, registry): + env = api.Environment(cr, SUPERUSER_ID, {}) + env["res.company"].search([]).scss_create_or_update_attachment() diff --git a/web_company_color/i18n/hr.po b/web_company_color/i18n/hr.po new file mode 100644 index 000000000..52117b0cf --- /dev/null +++ b/web_company_color/i18n/hr.po @@ -0,0 +1,73 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * web_company_color +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 12.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2019-11-18 14:34+0000\n" +"Last-Translator: Bole \n" +"Language-Team: none\n" +"Language: hr\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=3; plural=n%10==1 && n%100!=11 ? 0 : n%10>=2 && n" +"%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;\n" +"X-Generator: Weblate 3.8\n" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "" +"\n" +" In order for the changes to take effect, please " +"refresh\n" +" the page." +msgstr "" +"\n" +" Kako bi promjene postale vidljive, molimo \n" +" učitajte ponovo ovu stranicu." + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_res_company +msgid "Companies" +msgstr "Tvrtke" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__company_colors +msgid "Company Colors" +msgstr "Boje tvrtke" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Company Styles" +msgstr "Stilovi tvrtke" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg +msgid "Navbar Background Color" +msgstr "Boja pozadine Navigacijske trake" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg_hover +msgid "Navbar Background Color Hover" +msgstr "Poja pozadine navigacijske trake (hoover)" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Navbar Colors" +msgstr "Boje navigacijske trake" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_text +msgid "Navbar Text Color" +msgstr "Boja teksta navigacijske trake" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_ir_qweb +msgid "Qweb" +msgstr "" + +#~ msgid "SCSS Modif. Timestamp" +#~ msgstr "SCSS Modif. Vremenski žig" diff --git a/web_company_color/i18n/it.po b/web_company_color/i18n/it.po new file mode 100644 index 000000000..b55ba1a43 --- /dev/null +++ b/web_company_color/i18n/it.po @@ -0,0 +1,68 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * web_company_color +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 13.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2020-05-16 03:19+0000\n" +"Last-Translator: Alessandro Fiorino \n" +"Language-Team: none\n" +"Language: it\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=2; plural=n != 1;\n" +"X-Generator: Weblate 3.10\n" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "" +"\n" +" In order for the changes to take effect, please refresh\n" +" the page." +msgstr "" +"\n" +" Perché i cambiamenti abbiano effetto, prego " +"aggiornare la pagina." + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_res_company +msgid "Companies" +msgstr "Aziende" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__company_colors +msgid "Company Colors" +msgstr "Colori Aziendali" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Company Styles" +msgstr "Stili Aziendali" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg +msgid "Navbar Background Color" +msgstr "Sfondo Barra Navigazione" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg_hover +msgid "Navbar Background Color Hover" +msgstr "Colore Hover Sfondo Barra Navigazione" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Navbar Colors" +msgstr "Colori Barra Navigazione" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_text +msgid "Navbar Text Color" +msgstr "Colori Testo Barra Navigazione" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_ir_qweb +msgid "Qweb" +msgstr "Qweb" diff --git a/web_company_color/i18n/nl.po b/web_company_color/i18n/nl.po new file mode 100644 index 000000000..ba37a460a --- /dev/null +++ b/web_company_color/i18n/nl.po @@ -0,0 +1,67 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * web_company_color +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 13.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2020-11-11 13:08+0000\n" +"Last-Translator: Dennis Sluijk \n" +"Language-Team: none\n" +"Language: nl\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=2; plural=n != 1;\n" +"X-Generator: Weblate 3.10\n" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "" +"\n" +" In order for the changes to take effect, please refresh\n" +" the page." +msgstr "" +"\n" +" Ververs de pagina om de wijzigingen door te voeren." + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_res_company +msgid "Companies" +msgstr "Bedrijven" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__company_colors +msgid "Company Colors" +msgstr "Bedrijfskleuren" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Company Styles" +msgstr "Bedrijfsstijlen" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg +msgid "Navbar Background Color" +msgstr "Navigatiebalk Achtergrondkleur" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg_hover +msgid "Navbar Background Color Hover" +msgstr "Navigatiebalk Achtergrondkleur (als je de muis er overheen beweegt)" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Navbar Colors" +msgstr "Navigatiebalk Kleuren" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_text +msgid "Navbar Text Color" +msgstr "Navigatiebalk Tekstkleuren" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_ir_qweb +msgid "Qweb" +msgstr "Qweb" diff --git a/web_company_color/i18n/web_company_color.pot b/web_company_color/i18n/web_company_color.pot new file mode 100644 index 000000000..368b414e3 --- /dev/null +++ b/web_company_color/i18n/web_company_color.pot @@ -0,0 +1,62 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * web_company_color +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 13.0\n" +"Report-Msgid-Bugs-To: \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: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "" +"\n" +" In order for the changes to take effect, please refresh\n" +" the page." +msgstr "" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_res_company +msgid "Companies" +msgstr "" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__company_colors +msgid "Company Colors" +msgstr "" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Company Styles" +msgstr "" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg +msgid "Navbar Background Color" +msgstr "" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg_hover +msgid "Navbar Background Color Hover" +msgstr "" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Navbar Colors" +msgstr "" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_text +msgid "Navbar Text Color" +msgstr "" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_ir_qweb +msgid "Qweb" +msgstr "" diff --git a/web_company_color/i18n/zh_CN.po b/web_company_color/i18n/zh_CN.po new file mode 100644 index 000000000..a44a740e6 --- /dev/null +++ b/web_company_color/i18n/zh_CN.po @@ -0,0 +1,72 @@ +# Translation of Odoo Server. +# This file contains the translation of the following modules: +# * web_company_color +# +msgid "" +msgstr "" +"Project-Id-Version: Odoo Server 12.0\n" +"Report-Msgid-Bugs-To: \n" +"PO-Revision-Date: 2020-07-08 05:19+0000\n" +"Last-Translator: 黎伟杰 <674416404@qq.com>\n" +"Language-Team: none\n" +"Language: zh_CN\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: \n" +"Plural-Forms: nplurals=1; plural=0;\n" +"X-Generator: Weblate 3.10\n" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "" +"\n" +" In order for the changes to take effect, please " +"refresh\n" +" the page." +msgstr "" +"\n" +" 为了使更改生效,请刷新\n" +" 这个页面。" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_res_company +msgid "Companies" +msgstr "公司" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__company_colors +msgid "Company Colors" +msgstr "公司颜色" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Company Styles" +msgstr "公司风格" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg +msgid "Navbar Background Color" +msgstr "导航栏背景颜色" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_bg_hover +msgid "Navbar Background Color Hover" +msgstr "导航栏鼠标悬停背景颜色" + +#. module: web_company_color +#: model_terms:ir.ui.view,arch_db:web_company_color.view_company_form +msgid "Navbar Colors" +msgstr "导航栏颜色" + +#. module: web_company_color +#: model:ir.model.fields,field_description:web_company_color.field_res_company__color_navbar_text +msgid "Navbar Text Color" +msgstr "导航栏文字颜色" + +#. module: web_company_color +#: model:ir.model,name:web_company_color.model_ir_qweb +msgid "Qweb" +msgstr "Qweb" + +#~ msgid "SCSS Modif. Timestamp" +#~ msgstr "SCSS修改时间戳" diff --git a/web_company_color/models/__init__.py b/web_company_color/models/__init__.py new file mode 100644 index 000000000..8677b09f5 --- /dev/null +++ b/web_company_color/models/__init__.py @@ -0,0 +1,5 @@ +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). + +from . import assetsbundle +from . import res_company +from . import ir_qweb diff --git a/web_company_color/models/assetsbundle.py b/web_company_color/models/assetsbundle.py new file mode 100644 index 000000000..d6c31c2be --- /dev/null +++ b/web_company_color/models/assetsbundle.py @@ -0,0 +1,14 @@ +# Copyright 2020 Alexandre Díaz +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). +from odoo.addons.base.models.assetsbundle import AssetsBundle, ScssStylesheetAsset + + +class AssetsBundleCompanyColor(AssetsBundle): + def get_company_color_asset_node(self): + """ Process the user active company scss and returns the node to inject """ + company_id = self.env["res.company"].browse( + self.env.context.get("active_company_id", 0) + ) + asset = ScssStylesheetAsset(self, url=company_id.scss_get_url()) + compiled = self.compile_css(asset.compile, asset.get_source()) + return ("style", {}, compiled) diff --git a/web_company_color/models/ir_qweb.py b/web_company_color/models/ir_qweb.py new file mode 100644 index 000000000..956f9964f --- /dev/null +++ b/web_company_color/models/ir_qweb.py @@ -0,0 +1,86 @@ +# Copyright 2020 Alexandre Díaz +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). +from inspect import unwrap + +from odoo import api, models, tools +from odoo.http import request + +from odoo.addons.base.models.ir_qweb import IrQWeb + +from .assetsbundle import AssetsBundleCompanyColor + +# Monkey Patch to change the ormcache_context decorator of '_get_asset_nodes' to +# add 'active_company_id' context key. This is done to avoid "clear_caches" usage +# that works in a more aggressive way to the LRU cache. + +_orig_get_asset_nodes = unwrap(IrQWeb._get_asset_nodes) + + +@tools.conditional( + "xml" not in tools.config["dev_mode"], + tools.ormcache_context( + "xmlid", + 'options.get("lang", "en_US")', + "css", + "js", + "debug", + "async_load", + "defer_load", + "lazy_load", + keys=("website_id", "active_company_id"), + ), +) +def _get_asset_nodes__mp( + self, + xmlid, + options, + css=True, + js=True, + debug=False, + async_load=False, + defer_load=False, + lazy_load=False, + values=None, +): + return _orig_get_asset_nodes( + self, + xmlid, + options, + css=css, + js=js, + debug=debug, + async_load=async_load, + defer_load=defer_load, + lazy_load=lazy_load, + values=values, + ) + + +IrQWeb._get_asset_nodes = _get_asset_nodes__mp + + +class QWeb(models.AbstractModel): + _inherit = "ir.qweb" + + @api.model + def _render(self, id_or_xml_id, values=None, **options): + """ Adds the active company to the context """ + try: + active_company_id = int( + request.httprequest.cookies.get("cids", "").split(",")[0] + ) + except Exception: + active_company_id = False + company_id = ( + self.env["res.company"].browse(active_company_id) + or self.env.user.company_id + ) + self = self.with_context(active_company_id=company_id.id) + return super()._render(id_or_xml_id, values=values, **options) + + def _get_asset_content(self, xmlid, options): + """ Handle 'special' web_company_color xmlid """ + if xmlid == "web_company_color.company_color_assets": + asset = AssetsBundleCompanyColor(xmlid, [], env=self.env) + return ([], [asset.get_company_color_asset_node()]) + return super()._get_asset_content(xmlid, options) diff --git a/web_company_color/models/res_company.py b/web_company_color/models/res_company.py new file mode 100644 index 000000000..a458b25f1 --- /dev/null +++ b/web_company_color/models/res_company.py @@ -0,0 +1,154 @@ +# Copyright 2019 Alexandre Díaz +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). +import base64 +from colorsys import hls_to_rgb, rgb_to_hls + +from odoo import api, fields, models + +from ..utils import convert_to_image, image_to_rgb, n_rgb_to_hex + +URL_BASE = "/web_company_color/static/src/scss/" +URL_SCSS_GEN_TEMPLATE = URL_BASE + "custom_colors.%d.gen.scss" + + +class ResCompany(models.Model): + _inherit = "res.company" + + SCSS_TEMPLATE = """ + .o_main_navbar { + background-color: %(color_navbar_bg)s !important; + color: %(color_navbar_text)s !important; + + > .o_menu_brand { + color: %(color_navbar_text)s !important; + &:hover, &:focus, &:active, &:focus:active { + background-color: %(color_navbar_bg_hover)s !important; + } + } + + .show { + .dropdown-toggle { + background-color: %(color_navbar_bg_hover)s !important; + } + } + + > ul { + > li { + > a, > label { + color: %(color_navbar_text)s !important; + + &:hover, &:focus, &:active, &:focus:active { + background-color: %(color_navbar_bg_hover)s !important; + } + } + } + } + } + """ + + company_colors = fields.Serialized() + color_navbar_bg = fields.Char("Navbar Background Color", sparse="company_colors") + color_navbar_bg_hover = fields.Char( + "Navbar Background Color Hover", sparse="company_colors" + ) + color_navbar_text = fields.Char("Navbar Text Color", sparse="company_colors") + + @api.model_create_multi + def create(self, vals_list): + records = super().create(vals_list) + records.scss_create_or_update_attachment() + return records + + def unlink(self): + IrAttachmentObj = self.env["ir.attachment"] + for record in self: + IrAttachmentObj.sudo().search( + [("url", "=", record.scss_get_url()), ("company_id", "=", record.id)] + ).sudo().unlink() + return super().unlink() + + def write(self, values): + if not self.env.context.get("ignore_company_color", False): + fields_to_check = ( + "color_navbar_bg", + "color_navbar_bg_hover", + "color_navbar_text", + ) + if "logo" in values: + if values["logo"]: + _r, _g, _b = image_to_rgb(convert_to_image(values["logo"])) + # Make color 10% darker + _h, _l, _s = rgb_to_hls(_r, _g, _b) + _l = max(0, _l - 0.1) + _rd, _gd, _bd = hls_to_rgb(_h, _l, _s) + # Calc. optimal text color (b/w) + # Grayscale human vision perception (Rec. 709 values) + _a = 1 - (0.2126 * _r + 0.7152 * _g + 0.0722 * _b) + values.update( + { + "color_navbar_bg": n_rgb_to_hex(_r, _g, _b), + "color_navbar_bg_hover": n_rgb_to_hex(_rd, _gd, _bd), + "color_navbar_text": "#000" if _a < 0.5 else "#fff", + } + ) + else: + values.update(self.default_get(fields_to_check)) + + result = super().write(values) + + if any([field in values for field in fields_to_check]): + self.scss_create_or_update_attachment() + else: + result = super().write(values) + return result + + def _scss_get_sanitized_values(self): + self.ensure_one() + # Clone company_color as dictionary to avoid ORM operations + # This allow extend company_colors and only sanitize selected fields + # or add custom values + values = dict(self.company_colors or {}) + values.update( + { + "color_navbar_bg": (values.get("color_navbar_bg") or "$o-brand-odoo"), + "color_navbar_bg_hover": ( + values.get("color_navbar_bg_hover") + or "$o-navbar-inverse-link-hover-bg" + ), + "color_navbar_text": (values.get("color_navbar_text") or "#FFF"), + } + ) + return values + + def _scss_generate_content(self): + self.ensure_one() + # ir.attachment need files with content to work + if not self.company_colors: + return "// No Web Company Color SCSS Content\n" + return self.SCSS_TEMPLATE % self._scss_get_sanitized_values() + + def scss_get_url(self): + self.ensure_one() + return URL_SCSS_GEN_TEMPLATE % self.id + + def scss_create_or_update_attachment(self): + IrAttachmentObj = self.env["ir.attachment"] + for record in self: + datas = base64.b64encode(record._scss_generate_content().encode("utf-8")) + custom_url = record.scss_get_url() + custom_attachment = IrAttachmentObj.sudo().search( + [("url", "=", custom_url), ("company_id", "=", record.id)] + ) + values = { + "datas": datas, + "db_datas": datas, + "url": custom_url, + "name": custom_url, + "company_id": record.id, + } + if custom_attachment: + custom_attachment.sudo().write(values) + else: + values.update({"type": "binary", "mimetype": "text/scss"}) + IrAttachmentObj.sudo().create(values) + self.env["ir.qweb"].sudo().clear_caches() diff --git a/web_company_color/readme/CONTRIBUTORS.rst b/web_company_color/readme/CONTRIBUTORS.rst new file mode 100644 index 000000000..71df9a76b --- /dev/null +++ b/web_company_color/readme/CONTRIBUTORS.rst @@ -0,0 +1,5 @@ +* Jordi Ballester Alomar (ForgeFlow) +* Lois Rilo (ForgeFlow) +* Simone Orsi +* Jairo Llopis +* Iván Antón diff --git a/web_company_color/readme/DESCRIPTION.rst b/web_company_color/readme/DESCRIPTION.rst new file mode 100644 index 000000000..b164fc0f5 --- /dev/null +++ b/web_company_color/readme/DESCRIPTION.rst @@ -0,0 +1 @@ +This module change navbar colors based in the company logo colors. diff --git a/web_company_color/readme/ROADMAP.rst b/web_company_color/readme/ROADMAP.rst new file mode 100644 index 000000000..2b669379a --- /dev/null +++ b/web_company_color/readme/ROADMAP.rst @@ -0,0 +1 @@ +White color is omitted in the addition operation to support images without alpha channel. diff --git a/web_company_color/readme/USAGE.rst b/web_company_color/readme/USAGE.rst new file mode 100644 index 000000000..0143fee7d --- /dev/null +++ b/web_company_color/readme/USAGE.rst @@ -0,0 +1,3 @@ +Go to company record and set a logo. Can see/modify applied colors on the "Navbar" section. + +For optimal results use images with alpha channel. diff --git a/web_company_color/static/description/icon.png b/web_company_color/static/description/icon.png new file mode 100644 index 000000000..3a0328b51 Binary files /dev/null and b/web_company_color/static/description/icon.png differ diff --git a/web_company_color/static/description/index.html b/web_company_color/static/description/index.html new file mode 100644 index 000000000..50ab87995 --- /dev/null +++ b/web_company_color/static/description/index.html @@ -0,0 +1,433 @@ + + + + + + +Web Company Color + + + +
+

Web Company Color

+ + +

Beta License: AGPL-3 OCA/web Translate me on Weblate Try me on Runbot

+

This module change navbar colors based in the company logo colors.

+

Table of contents

+ +
+

Usage

+

Go to company record and set a logo. Can see/modify applied colors on the “Navbar” section.

+

For optimal results use images with alpha channel.

+
+
+

Known issues / Roadmap

+

White color is omitted in the addition operation to support images without alpha channel.

+
+
+

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.

+

Do not contact contributors directly about support or help with technical issues.

+
+
+

Credits

+
+

Authors

+
    +
  • Alexandre Díaz
  • +
+
+
+

Contributors

+ +
+
+

Maintainers

+

This module is maintained by the OCA.

+Odoo Community Association +

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.

+

This module is part of the OCA/web project on GitHub.

+

You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute.

+
+
+
+ + diff --git a/web_company_color/tests/__init__.py b/web_company_color/tests/__init__.py new file mode 100644 index 000000000..2c9568e64 --- /dev/null +++ b/web_company_color/tests/__init__.py @@ -0,0 +1 @@ +from . import test_res_company diff --git a/web_company_color/tests/test_res_company.py b/web_company_color/tests/test_res_company.py new file mode 100644 index 000000000..8234b6f44 --- /dev/null +++ b/web_company_color/tests/test_res_company.py @@ -0,0 +1,75 @@ +# Copyright 2019 Alexandre Díaz +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). +from odoo.tests import common + +from ..models.res_company import URL_BASE + + +class TestResCompany(common.TransactionCase): + IMG_GREEN = ( + "iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUl" + + "EQVR42mNk+M/wHwAEBgIApD5fRAAAAABJRU5ErkJggg==" + ) + + def _test_scss_attachment(self): + num_scss = self.env["ir.attachment"].search_count( + [("url", "ilike", "%s%%" % URL_BASE)] + ) + num_companies = self.env["res.company"].search_count([]) + self.assertEqual(num_scss, num_companies, "Invalid scss attachments") + + def test_create_unlink_company(self): + company_id = self.env["res.company"].create({"name": "Company Test"}) + self.assertEqual( + company_id.color_navbar_bg, False, "Invalid Navbar Background Color" + ) + self._test_scss_attachment() + company_id.sudo().write({"logo": self.IMG_GREEN}) + self.assertEqual( + company_id.color_navbar_bg, "#00ff00", "Invalid Navbar Background Color" + ) + # TODO: We can't remove companies if they have attached data, like + # warehouse when we have stock module installed + # company_id.sudo().unlink() + # self._test_scss_attachment() + + def test_change_logo(self): + company_id = self.env["res.company"].search([], limit=1) + company_id.sudo().write({"logo": self.IMG_GREEN}) + self.assertEqual( + company_id.color_navbar_bg, "#00ff00", "Invalid Navbar Background Color" + ) + + def test_scss_sanitized_values(self): + company_id = self.env["res.company"].search([], limit=1) + company_id.sudo().write({"color_navbar_bg": False}) + values = company_id.sudo()._scss_get_sanitized_values() + self.assertEqual( + values["color_navbar_bg"], + "$o-brand-odoo", + "Invalid Navbar Background Color", + ) + company_id.sudo().write({"color_navbar_bg": "#DEAD00"}) + values = company_id.sudo()._scss_get_sanitized_values() + self.assertEqual( + values["color_navbar_bg"], "#DEAD00", "Invalid Navbar Background Color" + ) + + def test_change_color(self): + company_id = self.env["res.company"].search([], limit=1) + company_id.sudo().write({"color_navbar_bg": "#DEAD00"}) + self.assertEqual( + company_id.color_navbar_bg, "#DEAD00", "Invalid Navbar Background Color" + ) + self.assertEqual( + company_id.company_colors["color_navbar_bg"], + "#DEAD00", + "Invalid Navbar Background Color", + ) + company_id.sudo().write({"color_navbar_bg": False}) + self.assertFalse(company_id.color_navbar_bg, "Invalid Navbar Background Color") + self.assertNotIn( + "color_navbar_bg", + company_id.company_colors, + "Invalid Navbar Background Color", + ) diff --git a/web_company_color/utils.py b/web_company_color/utils.py new file mode 100644 index 000000000..118ceb26a --- /dev/null +++ b/web_company_color/utils.py @@ -0,0 +1,40 @@ +# Copyright 2019 Alexandre Díaz +# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). +import base64 +import math +from io import BytesIO + +from PIL import Image + + +def n_rgb_to_hex(_r, _g, _b): + return "#{:02x}{:02x}{:02x}".format(int(255 * _r), int(255 * _g), int(255 * _b)) + + +def convert_to_image(field_binary): + return Image.open(BytesIO(base64.b64decode(field_binary))) + + +def image_to_rgb(img): + def normalize_vec3(vec3): + _l = 1.0 / math.sqrt(vec3[0] * vec3[0] + vec3[1] * vec3[1] + vec3[2] * vec3[2]) + return (vec3[0] * _l, vec3[1] * _l, vec3[2] * _l) + + # Force Alpha Channel + if img.mode != "RGBA": + img = img.convert("RGBA") + width, height = img.size + # Reduce pixels + width, height = (max(1, int(width / 4)), max(1, int(height / 4))) + img = img.resize((width, height)) + rgb_sum = [0, 0, 0] + # Mix. image colors using addition method + RGBA_WHITE = (255, 255, 255, 255) + for i in range(0, height * width): + rgba = img.getpixel((i % width, i / width)) + if rgba[3] > 128 and rgba != RGBA_WHITE: + rgb_sum[0] += rgba[0] + rgb_sum[1] += rgba[1] + rgb_sum[2] += rgba[2] + _r, _g, _b = normalize_vec3(rgb_sum) + return (_r, _g, _b) diff --git a/web_company_color/view/assets.xml b/web_company_color/view/assets.xml new file mode 100644 index 000000000..94b3bafc2 --- /dev/null +++ b/web_company_color/view/assets.xml @@ -0,0 +1,19 @@ + + + + + + diff --git a/web_company_color/view/res_company.xml b/web_company_color/view/res_company.xml new file mode 100644 index 000000000..3f3bcc49f --- /dev/null +++ b/web_company_color/view/res_company.xml @@ -0,0 +1,32 @@ + + + + + res.company + + + + + + + + + + + + + + +