diff options
| author | stephanchrst <stephanchrst@gmail.com> | 2022-05-10 21:51:50 +0700 |
|---|---|---|
| committer | stephanchrst <stephanchrst@gmail.com> | 2022-05-10 21:51:50 +0700 |
| commit | 3751379f1e9a4c215fb6eb898b4ccc67659b9ace (patch) | |
| tree | a44932296ef4a9b71d5f010906253d8c53727726 /addons/mail/models/res_partner.py | |
| parent | 0a15094050bfde69a06d6eff798e9a8ddf2b8c21 (diff) | |
initial commit 2
Diffstat (limited to 'addons/mail/models/res_partner.py')
| -rw-r--r-- | addons/mail/models/res_partner.py | 184 |
1 files changed, 184 insertions, 0 deletions
diff --git a/addons/mail/models/res_partner.py b/addons/mail/models/res_partner.py new file mode 100644 index 00000000..08b94805 --- /dev/null +++ b/addons/mail/models/res_partner.py @@ -0,0 +1,184 @@ +# -*- coding: utf-8 -*- +# Part of Odoo. See LICENSE file for full copyright and licensing details. + +import logging + +from odoo import _, api, fields, models, tools +from odoo.addons.bus.models.bus_presence import AWAY_TIMER +from odoo.addons.bus.models.bus_presence import DISCONNECTION_TIMER +from odoo.exceptions import AccessError +from odoo.osv import expression + +_logger = logging.getLogger(__name__) + + +class Partner(models.Model): + """ Update partner to add a field about notification preferences. Add a generic opt-out field that can be used + to restrict usage of automatic email templates. """ + _name = "res.partner" + _inherit = ['res.partner', 'mail.activity.mixin', 'mail.thread.blacklist'] + _mail_flat_thread = False + + email = fields.Char(tracking=1) + phone = fields.Char(tracking=2) + + channel_ids = fields.Many2many('mail.channel', 'mail_channel_partner', 'partner_id', 'channel_id', string='Channels', copy=False) + # override the field to track the visibility of user + user_id = fields.Many2one(tracking=True) + + def _compute_im_status(self): + super()._compute_im_status() + odoobot_id = self.env['ir.model.data'].xmlid_to_res_id('base.partner_root') + odoobot = self.env['res.partner'].browse(odoobot_id) + if odoobot in self: + odoobot.im_status = 'bot' + + def _message_get_suggested_recipients(self): + recipients = super(Partner, self)._message_get_suggested_recipients() + for partner in self: + partner._message_add_suggested_recipient(recipients, partner=partner, reason=_('Partner Profile')) + return recipients + + def _message_get_default_recipients(self): + return {r.id: { + 'partner_ids': [r.id], + 'email_to': False, + 'email_cc': False} + for r in self} + + @api.model + @api.returns('self', lambda value: value.id) + def find_or_create(self, email, assert_valid_email=False): + """ Override to use the email_normalized field. """ + if not email: + raise ValueError(_('An email is required for find_or_create to work')) + + parsed_name, parsed_email = self._parse_partner_name(email) + if parsed_email: + email_normalized = tools.email_normalize(parsed_email) + if email_normalized: + partners = self.search([('email_normalized', '=', email_normalized)], limit=1) + if partners: + return partners + + return super(Partner, self).find_or_create(email, assert_valid_email=assert_valid_email) + + def mail_partner_format(self): + self.ensure_one() + internal_users = self.user_ids - self.user_ids.filtered('share') + main_user = internal_users[0] if len(internal_users) else self.user_ids[0] if len(self.user_ids) else self.env['res.users'] + res = { + "id": self.id, + "display_name": self.display_name, + "name": self.name, + "email": self.email, + "active": self.active, + "im_status": self.im_status, + "user_id": main_user.id, + } + if main_user: + res["is_internal_user"] = not main_user.share + return res + + @api.model + def get_needaction_count(self): + """ compute the number of needaction of the current user """ + if self.env.user.partner_id: + self.env['mail.notification'].flush(['is_read', 'res_partner_id']) + self.env.cr.execute(""" + SELECT count(*) as needaction_count + FROM mail_message_res_partner_needaction_rel R + WHERE R.res_partner_id = %s AND (R.is_read = false OR R.is_read IS NULL)""", (self.env.user.partner_id.id,)) + return self.env.cr.dictfetchall()[0].get('needaction_count') + _logger.error('Call to needaction_count without partner_id') + return 0 + + @api.model + def get_starred_count(self): + """ compute the number of starred of the current user """ + if self.env.user.partner_id: + self.env.cr.execute(""" + SELECT count(*) as starred_count + FROM mail_message_res_partner_starred_rel R + WHERE R.res_partner_id = %s """, (self.env.user.partner_id.id,)) + return self.env.cr.dictfetchall()[0].get('starred_count') + _logger.error('Call to starred_count without partner_id') + return 0 + + @api.model + def get_static_mention_suggestions(self): + """Returns static mention suggestions of partners, loaded once at + webclient initialization and stored client side. + By default all the internal users are returned. + + The return format is a list of lists. The first level of list is an + arbitrary split that allows overrides to return their own list. + The second level of list is a list of partner data (as per returned by + `mail_partner_format()`). + """ + suggestions = [] + try: + suggestions.append([partner.mail_partner_format() for partner in self.env.ref('base.group_user').users.partner_id]) + except AccessError: + pass + return suggestions + + @api.model + def get_mention_suggestions(self, search, limit=8, channel_id=None): + """ Return 'limit'-first partners' id, name and email such that the name or email matches a + 'search' string. Prioritize users, and then extend the research to all partners. + If channel_id is given, only members of this channel are returned. + """ + search_dom = expression.OR([[('name', 'ilike', search)], [('email', 'ilike', search)]]) + search_dom = expression.AND([[('active', '=', True), ('type', '!=', 'private')], search_dom]) + if channel_id: + search_dom = expression.AND([[('channel_ids', 'in', channel_id)], search_dom]) + + # Search users + domain = expression.AND([[('user_ids.id', '!=', False), ('user_ids.active', '=', True)], search_dom]) + users = self.search(domain, limit=limit) + + # Search partners if less than 'limit' users found + partners = self.env['res.partner'] + if len(users) < limit: + partners = self.search(expression.AND([[('id', 'not in', users.ids)], search_dom]), limit=limit) + + return [ + [partner.mail_partner_format() for partner in users], + [partner.mail_partner_format() for partner in partners], + ] + + @api.model + def im_search(self, name, limit=20): + """ Search partner with a name and return its id, name and im_status. + Note : the user must be logged + :param name : the partner name to search + :param limit : the limit of result to return + """ + # This method is supposed to be used only in the context of channel creation or + # extension via an invite. As both of these actions require the 'create' access + # right, we check this specific ACL. + if self.env['mail.channel'].check_access_rights('create', raise_exception=False): + name = '%' + name + '%' + excluded_partner_ids = [self.env.user.partner_id.id] + self.env.cr.execute(""" + SELECT + U.id as user_id, + P.id as id, + P.name as name, + CASE WHEN B.last_poll IS NULL THEN 'offline' + WHEN age(now() AT TIME ZONE 'UTC', B.last_poll) > interval %s THEN 'offline' + WHEN age(now() AT TIME ZONE 'UTC', B.last_presence) > interval %s THEN 'away' + ELSE 'online' + END as im_status + FROM res_users U + JOIN res_partner P ON P.id = U.partner_id + LEFT JOIN bus_presence B ON B.user_id = U.id + WHERE P.name ILIKE %s + AND P.id NOT IN %s + AND U.active = 't' + LIMIT %s + """, ("%s seconds" % DISCONNECTION_TIMER, "%s seconds" % AWAY_TIMER, name, tuple(excluded_partner_ids), limit)) + return self.env.cr.dictfetchall() + else: + return {} |
