diff --git a/mute_notification_user_autosubscribe/README.rst b/mute_notification_user_autosubscribe/README.rst new file mode 100644 index 0000000000..43739e2fd1 --- /dev/null +++ b/mute_notification_user_autosubscribe/README.rst @@ -0,0 +1,125 @@ +==================================== +Mute Notification User Autosubscribe +==================================== + +.. + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! This file is generated by oca-gen-addon-readme !! + !! changes will be overwritten. !! + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + !! source digest: sha256:a4066de365398f1629027be5bf5dd6fcc6fdc886eb81465f8f8224f603dd1333 + !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! + +.. |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%2Fsocial-lightgray.png?logo=github + :target: https://github.com/OCA/social/tree/17.0/mute_notification_user_autosubscribe + :alt: OCA/social +.. |badge4| image:: https://img.shields.io/badge/weblate-Translate%20me-F47D42.png + :target: https://translation.odoo-community.org/projects/social-17-0/social-17-0-mute_notification_user_autosubscribe + :alt: Translate me on Weblate +.. |badge5| image:: https://img.shields.io/badge/runboat-Try%20me-875A7B.png + :target: https://runboat.odoo-community.org/builds?repo=OCA/social&target_branch=17.0 + :alt: Try me on Runboat + +|badge1| |badge2| |badge3| |badge4| |badge5| + +This module allows to select users and/or groups whose members do not +have to be sent notifications through the chatter. The models in which +these rules have to be applied can also be selected. + +**Table of contents** + +.. contents:: + :local: + +Configuration +============= + +To configure this module you need to: + +- Go to Settings > Technical > Auto Subscribe Mute +- Create a new instance or edit an exiting one. The following fields + are available: + + - **Name**. The name of the rule. + - **Model**. Model in which the rule is applied. Only models with a + user_id field can be selected. + - **Users**. List of users which the rule will be applied to. The + users in this field will be autosubscribed to the document when + they are set in the user_id field but only with the "Muted" + subtype, which notifies nothing. Note that this rule will not be + applied when the user in the list set itself in the user_id field, + as the subscription in this case depends on the creation of the + document, not the autosubscription. + - **Groups**. List of groups which the rule will be applied to. The + users that belong to the groups in this field will be + autosubscribed to the document when they are set in the user_id + field but only with the "Muted" subtype, which notifies nothing. + Note that this rule will not be applied when a user in any of the + groups in the list set itself in the user_id field, as the + subscription in this case depends on the creation of the document, + not the autosubscription. + - **Notes**. Text field. + +Usage +===== + +In case a user that has been automatically applied the "Mute" message +subtype in a subscription through one of the rules provided in this +module needs to be sent notifications in a specific document, just edit +the subscription and select the subtypes for which the user has to be +sent notifications. + +Known issues / Roadmap +====================== + +- If a and "Auto Subscribe Mute" rule is modified, created or deleted, + changes will not affect existing documents, so their users subtypes + will keep the current configuration. + +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 to smash it by providing a detailed and welcomed +`feedback `_. + +Do not contact contributors directly about support or help with technical issues. + +Credits +======= + +Authors +------- + +* Sygel + +Contributors +------------ + +- Manuel Regidor +- Valentín Vinagre +- Harald Panten + +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/social `_ project on GitHub. + +You are welcome to contribute. To learn how please visit https://odoo-community.org/page/Contribute. diff --git a/mute_notification_user_autosubscribe/__init__.py b/mute_notification_user_autosubscribe/__init__.py new file mode 100644 index 0000000000..a6aeb406c0 --- /dev/null +++ b/mute_notification_user_autosubscribe/__init__.py @@ -0,0 +1,4 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from . import models diff --git a/mute_notification_user_autosubscribe/__manifest__.py b/mute_notification_user_autosubscribe/__manifest__.py new file mode 100644 index 0000000000..1c7a1da6fc --- /dev/null +++ b/mute_notification_user_autosubscribe/__manifest__.py @@ -0,0 +1,21 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +{ + "name": "Mute Notification User Autosubscribe", + "summary": "Do not send notifications to users autosubcribed through user_id field", + "version": "17.0.1.0.0", + "category": "Social", + "website": "https://github.com/OCA/social", + "author": "Sygel,Odoo Community Association (OCA)", + "license": "AGPL-3", + "installable": True, + "depends": [ + "mail", + ], + "data": [ + "data/mail_message_subtype_data.xml", + "security/ir.model.access.csv", + "views/user_autosubscribe_mute_views.xml", + ], +} diff --git a/mute_notification_user_autosubscribe/data/mail_message_subtype_data.xml b/mute_notification_user_autosubscribe/data/mail_message_subtype_data.xml new file mode 100644 index 0000000000..693792823f --- /dev/null +++ b/mute_notification_user_autosubscribe/data/mail_message_subtype_data.xml @@ -0,0 +1,8 @@ + + + + Mute + + 999 + + diff --git a/mute_notification_user_autosubscribe/models/__init__.py b/mute_notification_user_autosubscribe/models/__init__.py new file mode 100644 index 0000000000..5ca893076f --- /dev/null +++ b/mute_notification_user_autosubscribe/models/__init__.py @@ -0,0 +1,6 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from . import user_autosubscribe_mute +from . import mail_thread +from . import res_users diff --git a/mute_notification_user_autosubscribe/models/mail_thread.py b/mute_notification_user_autosubscribe/models/mail_thread.py new file mode 100644 index 0000000000..efc24eefbd --- /dev/null +++ b/mute_notification_user_autosubscribe/models/mail_thread.py @@ -0,0 +1,31 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from odoo import models + + +class MailThread(models.AbstractModel): + _inherit = "mail.thread" + + def _message_auto_subscribe_followers(self, updated_values, default_subtype_ids): + user_id = updated_values.get("user_id") + if user_id: + model = self.env["ir.model"]._get(self._name) + user = self.env["res.users"].browse(user_id) + partner_id = user.partner_id.id + # Do nothing in case the user is already subscribe. + # It happes, for example, when it is the user_id who creates + # the document + if ( + partner_id not in self.message_partner_ids.ids + and user + and model + and user._is_muted(model) + ): + default_subtype_ids = [ + self.env.ref("mute_notification_user_autosubscribe.muted").id + ] + vals = super()._message_auto_subscribe_followers( + updated_values, default_subtype_ids + ) + return vals diff --git a/mute_notification_user_autosubscribe/models/res_users.py b/mute_notification_user_autosubscribe/models/res_users.py new file mode 100644 index 0000000000..f956a0a50a --- /dev/null +++ b/mute_notification_user_autosubscribe/models/res_users.py @@ -0,0 +1,25 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from odoo import models + + +class ResUsers(models.Model): + _inherit = "res.users" + + def _is_muted(self, model): + self.ensure_one() + muted = False + user_autosubscribe_mute = self.env["user.autosubscribe.mute"].search( + [("model_id", "=", model.id)], limit=1 + ) + if user_autosubscribe_mute: + groups = ( + ",".join(user_autosubscribe_mute.group_ids.get_external_id().values()) + or "" + ) + if self.id in user_autosubscribe_mute.user_ids.ids or ( + groups and self.user_has_groups(groups) + ): + muted = True + return muted diff --git a/mute_notification_user_autosubscribe/models/user_autosubscribe_mute.py b/mute_notification_user_autosubscribe/models/user_autosubscribe_mute.py new file mode 100644 index 0000000000..f7b0451848 --- /dev/null +++ b/mute_notification_user_autosubscribe/models/user_autosubscribe_mute.py @@ -0,0 +1,37 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from odoo import _, fields, models + + +class UserAutosubscribeMute(models.Model): + _name = "user.autosubscribe.mute" + _description = "User Autosubscribe Mute" + + def _get_user_models_domain(self): + models = ( + self.env["ir.model.fields"] + .search([("name", "=", "user_id"), ("relation", "=", "res.users")]) + .mapped("model_id") + ) + return f"[('id', 'in', {models.ids})]" + + name = fields.Char(required=True) + active = fields.Boolean(default=True) + model_id = fields.Many2one( + comodel_name="ir.model", + domain=_get_user_models_domain, + ondelete="cascade", + required=True, + ) + user_ids = fields.Many2many(comodel_name="res.users", string="Users") + group_ids = fields.Many2many(comodel_name="res.groups", string="Groups") + notes = fields.Text() + + _sql_constraints = [ + ( + "unique_model_id", + "UNIQUE(model_id)", + _("Model must be unique in User Autosubscribe Mute instances."), + ) + ] diff --git a/mute_notification_user_autosubscribe/pyproject.toml b/mute_notification_user_autosubscribe/pyproject.toml new file mode 100644 index 0000000000..4231d0cccb --- /dev/null +++ b/mute_notification_user_autosubscribe/pyproject.toml @@ -0,0 +1,3 @@ +[build-system] +requires = ["whool"] +build-backend = "whool.buildapi" diff --git a/mute_notification_user_autosubscribe/readme/CONFIGURE.md b/mute_notification_user_autosubscribe/readme/CONFIGURE.md new file mode 100644 index 0000000000..f10cf827d7 --- /dev/null +++ b/mute_notification_user_autosubscribe/readme/CONFIGURE.md @@ -0,0 +1,9 @@ +To configure this module you need to: + +- Go to Settings > Technical > Auto Subscribe Mute +- Create a new instance or edit an exiting one. The following fields are available: + - **Name**. The name of the rule. + - **Model**. Model in which the rule is applied. Only models with a user_id field can be selected. + - **Users**. List of users which the rule will be applied to. The users in this field will be autosubscribed to the document when they are set in the user_id field but only with the "Muted" subtype, which notifies nothing. Note that this rule will not be applied when the user in the list set itself in the user_id field, as the subscription in this case depends on the creation of the document, not the autosubscription. + - **Groups**. List of groups which the rule will be applied to. The users that belong to the groups in this field will be autosubscribed to the document when they are set in the user_id field but only with the "Muted" subtype, which notifies nothing. Note that this rule will not be applied when a user in any of the groups in the list set itself in the user_id field, as the subscription in this case depends on the creation of the document, not the autosubscription. + - **Notes**. Text field. diff --git a/mute_notification_user_autosubscribe/readme/CONTRIBUTORS.md b/mute_notification_user_autosubscribe/readme/CONTRIBUTORS.md new file mode 100644 index 0000000000..6e25ae445d --- /dev/null +++ b/mute_notification_user_autosubscribe/readme/CONTRIBUTORS.md @@ -0,0 +1,3 @@ +- Manuel Regidor \<\> +- Valentín Vinagre \<\> +- Harald Panten \<\> diff --git a/mute_notification_user_autosubscribe/readme/DESCRIPTION.md b/mute_notification_user_autosubscribe/readme/DESCRIPTION.md new file mode 100644 index 0000000000..e452723ac8 --- /dev/null +++ b/mute_notification_user_autosubscribe/readme/DESCRIPTION.md @@ -0,0 +1 @@ +This module allows to select users and/or groups whose members do not have to be sent notifications through the chatter. The models in which these rules have to be applied can also be selected. diff --git a/mute_notification_user_autosubscribe/readme/ROADMAP.md b/mute_notification_user_autosubscribe/readme/ROADMAP.md new file mode 100644 index 0000000000..d6f2aba903 --- /dev/null +++ b/mute_notification_user_autosubscribe/readme/ROADMAP.md @@ -0,0 +1 @@ +- If a and "Auto Subscribe Mute" rule is modified, created or deleted, changes will not affect existing documents, so their users subtypes will keep the current configuration. diff --git a/mute_notification_user_autosubscribe/readme/USAGE.md b/mute_notification_user_autosubscribe/readme/USAGE.md new file mode 100644 index 0000000000..4356a5ef01 --- /dev/null +++ b/mute_notification_user_autosubscribe/readme/USAGE.md @@ -0,0 +1 @@ +In case a user that has been automatically applied the "Mute" message subtype in a subscription through one of the rules provided in this module needs to be sent notifications in a specific document, just edit the subscription and select the subtypes for which the user has to be sent notifications. diff --git a/mute_notification_user_autosubscribe/security/ir.model.access.csv b/mute_notification_user_autosubscribe/security/ir.model.access.csv new file mode 100644 index 0000000000..53269bf430 --- /dev/null +++ b/mute_notification_user_autosubscribe/security/ir.model.access.csv @@ -0,0 +1,3 @@ +"id","name","model_id:id","group_id:id","perm_read","perm_write","perm_create","perm_unlink" +"user_autosubscribe_mute_access","user.autosubscribe.mute.access","model_user_autosubscribe_mute",base.group_user,1,0,0,0 +"user_autosubscribe_mute_manage","user.autosubscribe.mute.manage","model_user_autosubscribe_mute",base.group_system,1,1,1,1 diff --git a/mute_notification_user_autosubscribe/static/description/icon.png b/mute_notification_user_autosubscribe/static/description/icon.png new file mode 100644 index 0000000000..3a0328b516 Binary files /dev/null and b/mute_notification_user_autosubscribe/static/description/icon.png differ diff --git a/mute_notification_user_autosubscribe/static/description/index.html b/mute_notification_user_autosubscribe/static/description/index.html new file mode 100644 index 0000000000..a8d5fb2438 --- /dev/null +++ b/mute_notification_user_autosubscribe/static/description/index.html @@ -0,0 +1,473 @@ + + + + + +Mute Notification User Autosubscribe + + + +
+

Mute Notification User Autosubscribe

+ + +

Beta License: AGPL-3 OCA/social Translate me on Weblate Try me on Runboat

+

This module allows to select users and/or groups whose members do not +have to be sent notifications through the chatter. The models in which +these rules have to be applied can also be selected.

+

Table of contents

+ +
+

Configuration

+

To configure this module you need to:

+
    +
  • Go to Settings > Technical > Auto Subscribe Mute
  • +
  • Create a new instance or edit an exiting one. The following fields +are available:
      +
    • Name. The name of the rule.
    • +
    • Model. Model in which the rule is applied. Only models with a +user_id field can be selected.
    • +
    • Users. List of users which the rule will be applied to. The +users in this field will be autosubscribed to the document when +they are set in the user_id field but only with the “Muted” +subtype, which notifies nothing. Note that this rule will not be +applied when the user in the list set itself in the user_id field, +as the subscription in this case depends on the creation of the +document, not the autosubscription.
    • +
    • Groups. List of groups which the rule will be applied to. The +users that belong to the groups in this field will be +autosubscribed to the document when they are set in the user_id +field but only with the “Muted” subtype, which notifies nothing. +Note that this rule will not be applied when a user in any of the +groups in the list set itself in the user_id field, as the +subscription in this case depends on the creation of the document, +not the autosubscription.
    • +
    • Notes. Text field.
    • +
    +
  • +
+
+
+

Usage

+

In case a user that has been automatically applied the “Mute” message +subtype in a subscription through one of the rules provided in this +module needs to be sent notifications in a specific document, just edit +the subscription and select the subtypes for which the user has to be +sent notifications.

+
+
+

Known issues / Roadmap

+
    +
  • If a and “Auto Subscribe Mute” rule is modified, created or deleted, +changes will not affect existing documents, so their users subtypes +will keep the current configuration.
  • +
+
+
+

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 to smash it by providing a detailed and welcomed +feedback.

+

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

+
+
+

Credits

+
+

Authors

+
    +
  • Sygel
  • +
+
+
+

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/social project on GitHub.

+

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

+
+
+
+ + diff --git a/mute_notification_user_autosubscribe/tests/__init__.py b/mute_notification_user_autosubscribe/tests/__init__.py new file mode 100644 index 0000000000..19ad0c5b2b --- /dev/null +++ b/mute_notification_user_autosubscribe/tests/__init__.py @@ -0,0 +1,4 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from . import test_mute_user_sutosubscribe diff --git a/mute_notification_user_autosubscribe/tests/test_mute_user_sutosubscribe.py b/mute_notification_user_autosubscribe/tests/test_mute_user_sutosubscribe.py new file mode 100644 index 0000000000..1290e32cd2 --- /dev/null +++ b/mute_notification_user_autosubscribe/tests/test_mute_user_sutosubscribe.py @@ -0,0 +1,119 @@ +# Copyright 2024 Manuel Regidor +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). + +from odoo.tests import new_test_user + +from odoo.addons.mail.tests.common import MailCommon + + +class TestMuteUserAutosubscribe(MailCommon): + @classmethod + def setUpClass(cls): + super().setUpClass() + cls.user_1 = new_test_user( + cls.env, login="user_1", groups="base.group_partner_manager,base.group_user" + ) + cls.user_2 = new_test_user( + cls.env, login="user_2", groups="base.group_partner_manager,base.group_user" + ) + cls.user_autosubscribe_mute = cls.env["user.autosubscribe.mute"].create( + { + "name": "Mute Contact Notification", + "model_id": cls.env["ir.model"]._get("res.partner").id, + } + ) + + def send_email_mute_autosubscribe(cls, doc, user_id, body): + with cls.mock_mail_gateway(): + return doc.with_user(user_id.id).message_post( + body=body, subtype_xmlid="mail.mt_comment" + ) + + def test_do_not_mute_user_field(self): + user_2_partner = self.user_2.partner_id + # Create new contact with user_1. + contact = ( + self.env["res.partner"] + .with_user(self.user_1.id) + .create({"name": "Contact"}) + ) + # Check user_2 is not subscribed + self.assertFalse(user_2_partner.id in contact.message_partner_ids.ids) + # Set user_2 as the Salesperson + # Check user_2 is now subscribed + contact.write({"user_id": self.user_2.id}) + self.assertTrue(user_2_partner.id in contact.message_partner_ids.ids) + # Post message with user_1 + message = self.send_email_mute_autosubscribe(contact, self.user_1, "Test-1") + # user_2 has been sent a notification + receivers = message.notification_ids.mapped("res_partner_id") + self.assertTrue(user_2_partner.id in receivers.ids) + + def test_mute_users_field(self): + user_2_partner = self.user_2.partner_id + # Mute user_2 from salesperson autosubscription in res.partner + self.user_autosubscribe_mute.write({"user_ids": [self.user_2.id]}) + # Create new contact with user_1 and set user_2 and set user_2 as the + # salesperson + contact = ( + self.env["res.partner"] + .with_user(self.user_1.id) + .create({"name": "Contact", "user_id": self.user_2.id}) + ) + # Check user_2 is subscribed + self.assertTrue(user_2_partner.id in contact.message_partner_ids.ids) + # Post message with user_1 + message = self.send_email_mute_autosubscribe(contact, self.user_1, "Test-2") + # user_2 has NOT been sent a notification + receivers = message.notification_ids.mapped("res_partner_id") + self.assertFalse(user_2_partner.id in receivers.ids) + # user_2 subscription only contains "Mute" subtype + follower = self.env["mail.followers"].search( + [ + ("res_model", "=", "res.partner"), + ("res_id", "=", contact.id), + ("partner_id", "=", user_2_partner.id), + ], + limit=1, + ) + self.assertEqual(len(follower.subtype_ids), 1) + self.assertEqual( + follower.subtype_ids[0], + self.env.ref("mute_notification_user_autosubscribe.muted"), + ) + + def test_mute_groups_field(self): + user_2_partner = self.user_2.partner_id + # Mute group base.group_partner_manager from salesperson + # autosubscription in res.partner + self.user_autosubscribe_mute.write( + {"group_ids": [self.env.ref("base.group_partner_manager").id]} + ) + # Create new contact with user_1 and set user_2 and set user_2 as the + # salesperson + contact = ( + self.env["res.partner"] + .with_user(self.user_1.id) + .create({"name": "Contact", "user_id": self.user_2.id}) + ) + # Check user_2 is subscribed + self.assertTrue(user_2_partner.id in contact.message_partner_ids.ids) + # Post message with user_1 + message = self.send_email_mute_autosubscribe(contact, self.user_1, "Test-2") + # user_2 has NOT been sent a notification + receivers = message.notification_ids.mapped("res_partner_id") + self.assertFalse(user_2_partner.id in receivers.ids) + # user_2 subscription only contains "Mute" subtype + follower = self.env["mail.followers"].search( + [ + ("res_model", "=", "res.partner"), + ("res_id", "=", contact.id), + ("partner_id", "=", user_2_partner.id), + ], + limit=1, + ) + self.assertEqual(len(follower.subtype_ids), 1) + self.assertEqual( + follower.subtype_ids[0], + self.env.ref("mute_notification_user_autosubscribe.muted"), + ) diff --git a/mute_notification_user_autosubscribe/views/user_autosubscribe_mute_views.xml b/mute_notification_user_autosubscribe/views/user_autosubscribe_mute_views.xml new file mode 100644 index 0000000000..d4df67c278 --- /dev/null +++ b/mute_notification_user_autosubscribe/views/user_autosubscribe_mute_views.xml @@ -0,0 +1,75 @@ + + + + + user.autosubscribe.mute.form.view + user.autosubscribe.mute + +
+ + +
+
+ + + + + + + + + + + + + +
+
+
+
+ + user.autosubscribe.mute.tree.view + user.autosubscribe.mute + + + + + + + + + + + User Autosubscribe Mute + user.autosubscribe.mute + tree,form + + +