# HG changeset patch
# User Vincent Hatakeyama <vincent.hatakeyama@xcg-consulting.fr>
# Date 1741707019 -3600
#      Tue Mar 11 16:30:19 2025 +0100
# Branch 17.0
# Node ID 831e1897b5e189d35fca03f0d26d0075738af08a
# Parent  12af9d60c4793f77e1ba0e6393044317d7626165
✨ block more discuss features

diff --git a/.badges/code_style-black-000000.svg b/.badges/code_style-black-000000.svg
deleted file mode 100644
--- a/.badges/code_style-black-000000.svg
+++ /dev/null
@@ -1,33 +0,0 @@
-<?xml version="1.0" encoding="UTF-8" ?>
-<svg xmlns="http://www.w3.org/2000/svg" width="114" height="20">
-  <linearGradient id="b" x2="0" y2="100%">
-    <stop offset="0" stop-color="#bbb" stop-opacity=".1" />
-    <stop offset="1" stop-opacity=".1" />
-  </linearGradient>
-  <mask id="anybadge_1">
-    <rect width="114" height="20" rx="3" fill="#fff" />
-  </mask>
-  <g mask="url(#anybadge_1)">
-    <path fill="#555" d="M0 0h72v20H0z" />
-    <path fill="#000000" d="M72 0h42v20H72z" />
-    <path fill="url(#b)" d="M0 0h114v20H0z" />
-  </g>
-  <g
-    fill="#fff"
-    text-anchor="middle"
-    font-family="DejaVu Sans,Verdana,Geneva,sans-serif"
-    font-size="11"
-  >
-    <text x="37.0" y="15" fill="#010101" fill-opacity=".3">code style</text>
-    <text x="36.0" y="14">code style</text>
-  </g>
-  <g
-    fill="#fff"
-    text-anchor="middle"
-    font-family="DejaVu Sans,Verdana,Geneva,sans-serif"
-    font-size="11"
-  >
-    <text x="94.0" y="15" fill="#010101" fill-opacity=".3">black</text>
-    <text x="93.0" y="14">black</text>
-  </g>
-</svg>
diff --git a/.badges/code_style-ruff.svg b/.badges/code_style-ruff.svg
new file mode 100644
--- /dev/null
+++ b/.badges/code_style-ruff.svg
@@ -0,0 +1,49 @@
+<svg
+  xmlns="http://www.w3.org/2000/svg"
+  xmlns:xlink="http://www.w3.org/1999/xlink"
+  width="53"
+  height="20"
+  role="img"
+  aria-label="Ruff"
+>
+  <title>Ruff</title>
+  <linearGradient id="s" x2="0" y2="100%">
+    <stop offset="0" stop-color="#bbb" stop-opacity=".1" />
+    <stop offset="1" stop-opacity=".1" />
+  </linearGradient>
+  <clipPath id="r">
+    <rect width="53" height="20" rx="3" fill="#fff" />
+  </clipPath>
+  <g clip-path="url(#r)">
+    <rect width="20" height="20" fill="#555" />
+    <rect x="20" width="33" height="20" fill="#261230" />
+    <rect width="53" height="20" fill="url(#s)" />
+  </g>
+  <g
+    fill="#fff"
+    text-anchor="middle"
+    font-family="Verdana,Geneva,DejaVu Sans,sans-serif"
+    text-rendering="geometricPrecision"
+    font-size="110"
+  >
+    <image
+      x="5"
+      y="3"
+      width="10"
+      height="14"
+      xlink:href=""
+    />
+    <text
+      aria-hidden="true"
+      x="355"
+      y="150"
+      fill="#010101"
+      fill-opacity=".3"
+      transform="scale(.1)"
+      textLength="230"
+    >
+      Ruff
+    </text>
+    <text x="355" y="140" transform="scale(.1)" fill="#fff" textLength="230">Ruff</text>
+  </g>
+</svg>
diff --git a/NEWS.rst b/NEWS.rst
--- a/NEWS.rst
+++ b/NEWS.rst
@@ -2,6 +2,18 @@
 Changelog
 =========
 
+17.0.1.1.0
+----------
+
+User not in the message group, can not:
+
+- access messages related to discuss.channel
+- access discuss.channel.
+- see the button to display channels/messages.
+
+If a user with discuss send a message to a user without it, the message is blocked.
+The UI for that is not very nice, the message appears to be sent but is not.
+
 17.0.1.0.0
 ----------
 
diff --git a/README.rst b/README.rst
--- a/README.rst
+++ b/README.rst
@@ -17,5 +17,7 @@
 
 |maturity| |license| |ruff| |prettier|
 
-Introduce a new group, messaging, and only member of this group can see the messaging menu.
+Introduce a new group, messaging, and only member of this group can see the discuss menu.
 
+Discussion of internal users is also blocked if they do not belong to this group.
+
diff --git a/__init__.py b/__init__.py
--- a/__init__.py
+++ b/__init__.py
@@ -0,0 +1,2 @@
+from ._hook import _uninstall_hook, _post_init_hook
+from . import models
diff --git a/__manifest__.py b/__manifest__.py
--- a/__manifest__.py
+++ b/__manifest__.py
@@ -1,7 +1,7 @@
 ###############################################################################
 #
 #    Messaging Group a module for Odoo
-#    Copyright © 2014, 2018, 2022 XCG Consulting (https://xcg-consulting.fr/)
+#    Copyright © 2014, 2018, 2022, 2025 XCG Consulting (https://xcg-consulting.fr/)
 #
 #    This program is free software: you can redistribute it and/or modify
 #    it under the terms of the GNU Affero General Public License as
@@ -21,11 +21,18 @@
     "name": "Messaging Group",
     "license": "AGPL-3",
     "summary": "New group to access discussions",
-    "version": "17.0.1.0.0",
+    "version": "17.0.1.1.0",
     "category": "Extra Rights",
     "author": "XCG Consulting",
     "website": "https://orbeet.io/",
     "depends": ["mail"],
-    "data": ["security/security.xml"],
+    "data": ["security/security.xml", "security/ir.model.access.csv"],
+    "assets": {
+        "web.assets_backend": [
+            "mail_messaging_group/static/src/discuss/**",
+        ],
+    },
     "auto_install": True,
+    "uninstall_hook": "_uninstall_hook",
+    "post_init_hook": "_post_init_hook",
 }
diff --git a/_hook.py b/_hook.py
new file mode 100644
--- /dev/null
+++ b/_hook.py
@@ -0,0 +1,37 @@
+###############################################################################
+#
+#    Messaging Group a module for Odoo
+#    Copyright © 2025 XCG Consulting (https://xcg-consulting.fr/)
+#
+#    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/>.
+#
+###############################################################################
+from odoo import Command  # type: ignore[import-untyped]
+from odoo.api import Environment  # type: ignore[import-untyped]
+
+
+def _uninstall_hook(env: Environment):
+    """Revert changes made on other module data"""
+    env.ref("mail.menu_root_discuss").groups_id = [Command.clear()]
+    env.ref("mail.access_discuss_channel_user").active = True
+    env.ref("mail.channel_all_employees").group_ids = [
+        Command.set([env.ref("base.group_user").id])
+    ]
+
+
+def _post_init_hook(env: Environment):
+    """Change some noupdate data"""
+    env.ref("mail.channel_all_employees").group_ids = [
+        Command.set([env.ref("mail_messaging_group.group_messaging").id])
+    ]
diff --git a/models/__init__.py b/models/__init__.py
new file mode 100644
--- /dev/null
+++ b/models/__init__.py
@@ -0,0 +1,1 @@
+from . import mail_message
diff --git a/models/mail_message.py b/models/mail_message.py
new file mode 100644
--- /dev/null
+++ b/models/mail_message.py
@@ -0,0 +1,64 @@
+###############################################################################
+#
+#    Messaging Group a module for Odoo
+#    Copyright © 2025 XCG Consulting (https://xcg-consulting.fr/)
+#
+#    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/>.
+#
+###############################################################################
+from odoo import _, models  # type: ignore[import-untyped]
+from odoo.exceptions import AccessError  # type: ignore[import-untyped]
+
+
+class Message(models.Model):
+    _inherit = "mail.message"
+
+    def create(self, vals_list):
+        created = super().create(vals_list)
+        if not self.env.su:
+            for record in created:
+                if record.model == "discuss.channel":
+                    channel = self.env["discuss.channel"].browse(record.res_id)
+                    if channel.channel_type == "chat":
+                        # check that all members have messaging group
+                        users = (
+                            self.sudo()
+                            .env["res.users"]
+                            .search(
+                                [
+                                    (
+                                        "partner_id",
+                                        "in",
+                                        channel.mapped(
+                                            "channel_member_ids.partner_id.id"
+                                        ),
+                                    )
+                                ]
+                            )
+                        )
+                        for user in users:
+                            if (
+                                not user.has_group(
+                                    "mail_messaging_group.group_messaging"
+                                )
+                                and not user.has_group("base.group_public")
+                                and not user.has_group("base.group_portal")
+                            ):
+                                raise AccessError(
+                                    _(
+                                        "Creating messages to user that can not read "
+                                        "them is not allowed."
+                                    )
+                                )
+        return created
diff --git a/py.typed b/py.typed
new file mode 100644
diff --git a/security/ir.model.access.csv b/security/ir.model.access.csv
new file mode 100644
--- /dev/null
+++ b/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
+access_discuss_channel_user,discuss.channel ,mail.model_discuss_channel,base.group_user,1,0,0,0
+access_discuss_channel_messaging,discuss.channel messaging,mail.model_discuss_channel,group_messaging,1,1,1,0
diff --git a/security/security.xml b/security/security.xml
--- a/security/security.xml
+++ b/security/security.xml
@@ -14,4 +14,21 @@
     <record id="mail.menu_root_discuss" model="ir.ui.menu">
         <field name="groups_id" eval="[(6, 0, [ref('group_messaging')])]" />
     </record>
+
+    <record id="mail.access_discuss_channel_user" model="ir.model.access">
+        <field name="active" eval="False" />
+    </record>
+
+    <record id="message_no_discuss_channel_access" model="ir.rule">
+        <field name="name">No access to discuss messages to internal users</field>
+        <field name="model_id" ref="mail.model_mail_message" />
+        <field name="domain_force">[('model', '!=', 'discuss.channel')]</field>
+        <field name="groups" eval="[(4, ref('base.group_user'))]" />
+    </record>
+    <record id="message" model="ir.rule">
+        <field name="name">Only messaging users have access</field>
+        <field name="model_id" ref="mail.model_mail_message" />
+        <field name="domain_force">[(1, '=', 1)]</field>
+        <field name="groups" eval="[(4, ref('group_messaging'))]" />
+    </record>
 </odoo>
diff --git a/static/src/discuss/messaging_menu.js b/static/src/discuss/messaging_menu.js
new file mode 100644
--- /dev/null
+++ b/static/src/discuss/messaging_menu.js
@@ -0,0 +1,22 @@
+/** @odoo-module **/
+// Adapted from https://www.odoo.com/fr_FR/forum/aide-1/groups-option-on-owl-template-263872
+import {patch} from "@web/core/utils/patch";
+import {useService} from "@web/core/utils/hooks";
+import {useState, onWillStart} from "@odoo/owl";
+import {MessagingMenu} from "@mail/core/web/messaging_menu";
+
+patch(MessagingMenu.prototype, {
+    setup() {
+        super.setup();
+        this.user = useService("user");
+        this.state = useState({
+            canAccessDiscuss: false, // Initial visibility state
+        });
+        // Check if the user belongs to the "mail_messaging_group.group_messaging" group
+        onWillStart(async () => {
+            this.state.canAccessDiscuss = await this.user.hasGroup(
+                "mail_messaging_group.group_messaging"
+            );
+        });
+    },
+});
diff --git a/static/src/discuss/messaging_menu.xml b/static/src/discuss/messaging_menu.xml
new file mode 100644
--- /dev/null
+++ b/static/src/discuss/messaging_menu.xml
@@ -0,0 +1,13 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<templates xml:space="preserve">
+    <t t-inherit="mail.MessagingMenu" t-inherit-mode="extension">
+        <xpath expr="//Dropdown" position="attributes">
+            <attribute
+                name="t-if"
+            >!env.inDiscussApp &amp; state.canAccessDiscuss</attribute>
+        </xpath>
+        <xpath expr="//t[@t-else='']" position="attributes">
+            <attribute name="t-if">state.canAccessDiscuss</attribute>
+        </xpath>
+    </t>
+</templates>
diff --git a/upgrades/17.0.1.1.0/end-update-general-channel.py b/upgrades/17.0.1.1.0/end-update-general-channel.py
new file mode 100644
--- /dev/null
+++ b/upgrades/17.0.1.1.0/end-update-general-channel.py
@@ -0,0 +1,16 @@
+from odoo import SUPERUSER_ID, api  # type: ignore[import-untyped]
+
+
+def migrate(cr, _installed_version):
+    """Remove user without access to messages from general channel"""
+    env = api.Environment(cr, SUPERUSER_ID, {})
+    general_channel = env.ref("mail.channel_all_employees")
+    if general_channel:
+        members_to_remove = env["discuss.channel.member"]
+        for member in env.ref("mail.channel_all_employees").channel_member_ids:
+            users = env["res.users"].search([("partner_id", "=", member.partner_id.id)])
+            for user in users:
+                if not user.has_group("mail_messaging_group.group_messaging"):
+                    members_to_remove |= member
+        if members_to_remove:
+            members_to_remove.unlink()