You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
449 lines
19 KiB
449 lines
19 KiB
# Copyright 2016 Ildar Nasyrov <https://it-projects.info/team/iledarn>
|
|
# Copyright 2016-2018 Ivan Yelizariev <https://it-projects.info/team/yelizariev>
|
|
# Copyright 2016 intero-chz <https://github.com/intero-chz>
|
|
# Copyright 2016 manawi <https://github.com/manawi>
|
|
# Copyright 2018 Kolushov Alexandr <https://it-projects.info/team/KolushovAlexandr>
|
|
# License LGPL-3.0 or later (http://www.gnu.org/licenses/lgpl.html).
|
|
|
|
from odoo import api
|
|
from odoo import fields
|
|
from odoo import models
|
|
from odoo.tools import email_split
|
|
from odoo.tools.translate import _
|
|
from odoo import exceptions
|
|
|
|
|
|
class Wizard(models.TransientModel):
|
|
_name = 'mail_move_message.wizard'
|
|
_description = 'Mail move message wizard'
|
|
|
|
@api.model
|
|
def _model_selection(self):
|
|
selection = []
|
|
config_parameters = self.env['ir.config_parameter']
|
|
model_names = config_parameters.sudo().get_param('mail_relocation_models')
|
|
model_names = model_names.split(',') if model_names else []
|
|
if 'default_message_id' in self.env.context:
|
|
message = self.env['mail.message'].browse(self.env.context['default_message_id'])
|
|
if message.model and message.model not in model_names:
|
|
model_names.append(message.model)
|
|
if message.moved_from_model and message.moved_from_model not in model_names:
|
|
model_names.append(message.moved_from_model)
|
|
if model_names:
|
|
selection = [(m.model, m.display_name) for m in self.env['ir.model'].search([('model', 'in', model_names)])]
|
|
return selection
|
|
|
|
@api.model
|
|
def default_get(self, fields_list):
|
|
res = super(Wizard, self).default_get(fields_list)
|
|
|
|
available_models = self._model_selection()
|
|
if len(available_models):
|
|
record = self.env[available_models[0][0]].search([], limit=1)
|
|
res['model_record'] = len(record) and (available_models[0][0] + ',' + str(record.id)) or False
|
|
|
|
if 'message_id' in res:
|
|
message = self.env['mail.message'].browse(res['message_id'])
|
|
email_from = message.email_from
|
|
parts = email_split(email_from.replace(' ', ','))
|
|
if parts:
|
|
email = parts[0]
|
|
name = email_from.find(email) != -1 and email_from[:email_from.index(email)].replace('"', '').replace('<', '').strip() or email_from
|
|
else:
|
|
name, email = email_from
|
|
res['message_name_from'] = name
|
|
res['message_email_from'] = email
|
|
|
|
res['partner_id'] = message.author_id.id
|
|
if message.author_id and self.env.uid not in [u.id for u in message.author_id.user_ids]:
|
|
res['filter_by_partner'] = True
|
|
if message.author_id and res.get('model'):
|
|
res_id = self.env[res['model']].search([], order='id desc', limit=1)
|
|
if res_id:
|
|
res['res_id'] = res_id[0].id
|
|
|
|
config_parameters = self.env['ir.config_parameter']
|
|
res['move_followers'] = config_parameters.sudo().get_param('mail_relocation_move_followers')
|
|
|
|
res['uid'] = self.env.uid
|
|
return res
|
|
|
|
message_id = fields.Many2one('mail.message', string='Message')
|
|
message_body = fields.Html(related='message_id.body', string='Message to move', readonly=True)
|
|
message_from = fields.Char(related='message_id.email_from', string='From', readonly=True)
|
|
message_subject = fields.Char(related='message_id.subject', string='Subject', readonly=True)
|
|
message_moved_by_message_id = fields.Many2one('mail.message', related='message_id.moved_by_message_id', string='Moved with', readonly=True)
|
|
message_moved_by_user_id = fields.Many2one('res.users', related='message_id.moved_by_user_id', string='Moved by', readonly=True)
|
|
message_is_moved = fields.Boolean(string='Is Moved', related='message_id.is_moved', readonly=True)
|
|
parent_id = fields.Many2one('mail.message', string='Search by name', )
|
|
model_record = fields.Reference(selection="_model_selection", string='Record')
|
|
model = fields.Char(compute="_compute_model_res_id", string='Model')
|
|
res_id = fields.Integer(compute="_compute_model_res_id", string='Record ID')
|
|
|
|
can_move = fields.Boolean('Can move', compute='_compute_get_can_move')
|
|
move_back = fields.Boolean('MOVE TO ORIGIN', help='Move message and submessages to original place')
|
|
partner_id = fields.Many2one('res.partner', string='Author')
|
|
filter_by_partner = fields.Boolean('Filter Records by partner')
|
|
message_email_from = fields.Char()
|
|
message_name_from = fields.Char()
|
|
# FIXME message_to_read should be True even if current message or any his childs are unread
|
|
message_to_read = fields.Boolean(compute='_compute_is_read', string="Unread message",
|
|
help="Service field shows that this message were unread when moved")
|
|
uid = fields.Integer()
|
|
move_followers = fields.Boolean(
|
|
'Move Followers',
|
|
help="Add followers of current record to a new record.\n"
|
|
"You must use this option, if new record has restricted access.\n"
|
|
"You can change default value for this option at Settings/System Parameters")
|
|
|
|
@api.depends('model_record')
|
|
def _compute_model_res_id(self):
|
|
for rec in self:
|
|
rec.model = rec.model_record and rec.model_record._name or False
|
|
rec.res_id = rec.model_record and rec.model_record.id or False
|
|
|
|
@api.depends('message_id')
|
|
def _compute_get_can_move(self):
|
|
for r in self:
|
|
r.get_can_move_one()
|
|
|
|
def _compute_is_read(self):
|
|
messages = self.env['mail.message'].sudo().browse(self.message_id.all_child_ids.ids + [self.message_id.id])
|
|
self.message_to_read = True in [m.needaction for m in messages]
|
|
|
|
def get_can_move_one(self):
|
|
self.ensure_one()
|
|
# message was not moved before OR message is a top message of previous move
|
|
self.can_move = not self.message_id.moved_by_message_id or self.message_id.moved_by_message_id.id == self.message_id.id
|
|
|
|
@api.onchange('move_back')
|
|
def on_change_move_back(self):
|
|
if not self.move_back:
|
|
return
|
|
self.parent_id = self.message_id.moved_from_parent_id
|
|
message = self.message_id
|
|
if message.is_moved:
|
|
self.model_record = self.env[message.moved_from_model].browse(message.moved_from_res_id)
|
|
|
|
@api.onchange('parent_id', 'model_record')
|
|
def update_move_back(self):
|
|
model = self.message_id.moved_from_model
|
|
self.move_back = self.parent_id == self.message_id.moved_from_parent_id \
|
|
and self.res_id == self.message_id.moved_from_res_id \
|
|
and (self.model == model or (not self.model and not model))
|
|
|
|
@api.onchange('parent_id')
|
|
def on_change_parent_id(self):
|
|
if self.parent_id and self.parent_id.model:
|
|
self.model = self.parent_id.model
|
|
self.res_id = self.parent_id.res_id
|
|
else:
|
|
self.model = None
|
|
self.res_id = None
|
|
|
|
@api.onchange('model', 'filter_by_partner', 'partner_id')
|
|
def on_change_partner(self):
|
|
domain = {'res_id': [('id', '!=', self.message_id.res_id)]}
|
|
if self.model and self.filter_by_partner and self.partner_id:
|
|
fields = self.env[self.model].fields_get(False)
|
|
contact_field = False
|
|
for n, f in fields.items():
|
|
if f['type'] == 'many2one' and f['relation'] == 'res.partner':
|
|
contact_field = n
|
|
break
|
|
if contact_field:
|
|
domain['res_id'].append((contact_field, '=', self.partner_id.id))
|
|
if self.model:
|
|
res_id = self.env[self.model].search(domain['res_id'], order='id desc', limit=1)
|
|
self.res_id = res_id and res_id[0].id
|
|
else:
|
|
self.res_id = None
|
|
return {'domain': domain}
|
|
|
|
def check_access(self):
|
|
for r in self:
|
|
r.check_access_one()
|
|
|
|
def check_access_one(self):
|
|
self.ensure_one()
|
|
operation = 'write'
|
|
|
|
if not (self.model and self.res_id):
|
|
return True
|
|
model_obj = self.env[self.model]
|
|
mids = model_obj.browse(self.res_id).exists()
|
|
if hasattr(model_obj, 'check_mail_message_access'):
|
|
model_obj.check_mail_message_access(mids.ids, operation)
|
|
else:
|
|
self.env['mail.thread'].check_mail_message_access(mids.ids, operation, model_name=self.model)
|
|
|
|
def open_moved_by_message_id(self):
|
|
message_id = None
|
|
for r in self:
|
|
message_id = r.message_moved_by_message_id.id
|
|
return {
|
|
'type': 'ir.actions.act_window',
|
|
'res_model': 'mail_move_message.wizard',
|
|
'view_mode': 'form',
|
|
'view_type': 'form',
|
|
'views': [[False, 'form']],
|
|
'target': 'new',
|
|
'context': {'default_message_id': message_id},
|
|
}
|
|
|
|
def move(self):
|
|
for r in self:
|
|
if not r.model:
|
|
raise exceptions.except_orm(_('Record field is empty!'), _('Select a record for relocation first'))
|
|
for r in self:
|
|
r.check_access()
|
|
if not r.parent_id or not (r.parent_id.model == r.model and
|
|
r.parent_id.res_id == r.res_id):
|
|
# link with the first message of record
|
|
parent = self.env['mail.message'].search([('model', '=', r.model), ('res_id', '=', r.res_id)], order='id', limit=1)
|
|
r.parent_id = parent.id or None
|
|
r.message_id.move(r.parent_id.id, r.res_id, r.model, r.move_back, r.move_followers, r.message_to_read, r.partner_id)
|
|
|
|
if r.model in ['mail.message', 'mail.channel', False]:
|
|
return {
|
|
'name': 'Chess game page',
|
|
'type': 'ir.actions.act_url',
|
|
'url': '/web',
|
|
'target': 'self',
|
|
}
|
|
return {
|
|
'name': _('Record'),
|
|
'view_type': 'form',
|
|
'view_mode': 'form',
|
|
'res_model': r.model,
|
|
'res_id': r.res_id,
|
|
'views': [(False, 'form')],
|
|
'type': 'ir.actions.act_window',
|
|
}
|
|
|
|
def delete(self):
|
|
for r in self:
|
|
r.delete_one()
|
|
|
|
def delete_one(self):
|
|
self.ensure_one()
|
|
msg_id = self.message_id.id
|
|
|
|
# Send notification
|
|
notification = {'id': msg_id}
|
|
self.env['bus.bus'].sendone((self._cr.dbname, 'mail_move_message.delete_message'), notification)
|
|
|
|
self.message_id.unlink()
|
|
return {}
|
|
|
|
def read_close(self):
|
|
for r in self:
|
|
r.read_close_one()
|
|
|
|
def read_close_one(self):
|
|
self.ensure_one()
|
|
self.message_id.set_message_done()
|
|
self.message_id.child_ids.set_message_done()
|
|
return {'type': 'ir.actions.act_window_close'}
|
|
|
|
|
|
class MailMessage(models.Model):
|
|
_inherit = 'mail.message'
|
|
|
|
is_moved = fields.Boolean('Is moved')
|
|
moved_from_res_id = fields.Integer('Related Document ID (Original)')
|
|
moved_from_model = fields.Char('Related Document Model (Original)')
|
|
moved_from_parent_id = fields.Many2one('mail.message', 'Parent Message (Original)', ondelete='set null')
|
|
moved_by_message_id = fields.Many2one('mail.message', 'Moved by message', ondelete='set null', help='Top message, that initate moving this message')
|
|
moved_by_user_id = fields.Many2one('res.users', 'Moved by user', ondelete='set null')
|
|
all_child_ids = fields.One2many('mail.message', string='All childs', compute='_compute_get_all_childs', help='all childs, including subchilds')
|
|
moved_as_unread = fields.Boolean('Was Unread', default=False)
|
|
|
|
def _compute_get_all_childs(self, include_myself=True):
|
|
for r in self:
|
|
r._get_all_childs_one(include_myself=include_myself)
|
|
|
|
def _get_all_childs_one(self, include_myself=True):
|
|
self.ensure_one()
|
|
ids = []
|
|
if include_myself:
|
|
ids.append(self.id)
|
|
while True:
|
|
new_ids = self.search([('parent_id', 'in', ids), ('id', 'not in', ids)]).ids
|
|
if new_ids:
|
|
ids = ids + new_ids
|
|
continue
|
|
break
|
|
moved_childs = self.search([('moved_by_message_id', '=', self.id)]).ids
|
|
self.all_child_ids = ids + moved_childs
|
|
|
|
def move_followers(self, model, ids):
|
|
fol_obj = self.env['mail.followers']
|
|
for message in self:
|
|
followers = fol_obj.sudo().search([('res_model', '=', message.model),
|
|
('res_id', '=', message.res_id)])
|
|
for f in followers:
|
|
self.env[model].browse(ids).message_subscribe([f.partner_id.id], [s.id for s in f.subtype_ids])
|
|
|
|
def move(self, parent_id, res_id, model, move_back, move_followers=False, message_to_read=False, author=False):
|
|
for r in self:
|
|
r.move_one(parent_id, res_id, model, move_back, move_followers=move_followers, message_to_read=message_to_read, author=author)
|
|
|
|
def move_one(self, parent_id, res_id, model, move_back, move_followers=False, message_to_read=False, author=False):
|
|
self.ensure_one()
|
|
if parent_id == self.id:
|
|
# if for any reason method is called to move message with parent
|
|
# equal to oneself, we need stop to prevent infinitive loop in
|
|
# building message tree
|
|
return
|
|
if not self.author_id:
|
|
self.write({
|
|
'author_id': author.id,
|
|
})
|
|
|
|
vals = {}
|
|
if move_back:
|
|
# clear variables if we move everything back
|
|
vals['is_moved'] = False
|
|
vals['moved_by_user_id'] = None
|
|
vals['moved_by_message_id'] = None
|
|
|
|
vals['moved_from_res_id'] = None
|
|
vals['moved_from_model'] = None
|
|
vals['moved_from_parent_id'] = None
|
|
vals['moved_as_unread'] = None
|
|
else:
|
|
vals['parent_id'] = parent_id
|
|
vals['res_id'] = res_id
|
|
vals['model'] = model
|
|
|
|
vals['is_moved'] = True
|
|
vals['moved_by_user_id'] = self.env.user.id
|
|
vals['moved_by_message_id'] = self.id
|
|
vals['moved_as_unread'] = message_to_read
|
|
# Update record_name in message
|
|
vals['record_name'] = self._get_record_name(vals)
|
|
|
|
# unread message remains unread after moving back to origin
|
|
if self.moved_as_unread and move_back:
|
|
notification = {
|
|
'mail_message_id': self.id,
|
|
'res_partner_id': self.env.user.partner_id.id,
|
|
'is_read': False,
|
|
}
|
|
self.write({
|
|
'notification_ids': [(0, 0, notification)],
|
|
})
|
|
|
|
for r in self.all_child_ids:
|
|
r_vals = vals.copy()
|
|
if not r.is_moved:
|
|
# moved_from_* variables contain not last, but original
|
|
# reference
|
|
r_vals['moved_from_parent_id'] = r.parent_id.id or r.env.context.get('uid')
|
|
r_vals['moved_from_res_id'] = r.res_id or r.id
|
|
r_vals['moved_from_model'] = r.model or r._name
|
|
elif move_back:
|
|
r_vals['parent_id'] = r.moved_from_parent_id.id
|
|
r_vals['res_id'] = r.moved_from_res_id
|
|
r_vals['model'] = (r.moved_from_model and r.moved_from_model not in ['mail.message', 'mail.channel', False]) and r.moved_from_model
|
|
r_vals['record_name'] = r_vals['model'] and self.env[r.moved_from_model].browse(r.moved_from_res_id).name
|
|
|
|
if move_followers:
|
|
r.sudo().move_followers(r_vals.get('model'), r_vals.get('res_id'))
|
|
r.sudo().write(r_vals)
|
|
r.attachment_ids.sudo().write({
|
|
'res_id': r_vals.get('res_id'),
|
|
'res_model': r_vals.get('model')
|
|
})
|
|
|
|
# Send notification
|
|
notification = {
|
|
'id': self.id,
|
|
'res_id': vals.get('res_id'),
|
|
'model': vals.get('model'),
|
|
'is_moved': vals['is_moved'],
|
|
'record_name': 'record_name' in vals and vals['record_name'],
|
|
}
|
|
self.env['bus.bus'].sendone((self._cr.dbname, 'mail_move_message'), notification)
|
|
|
|
def name_get(self):
|
|
context = self.env.context
|
|
if not (context or {}).get('extended_name'):
|
|
return super(MailMessage, self).name_get()
|
|
reads = self.read(['record_name', 'model', 'res_id'])
|
|
res = []
|
|
for record in reads:
|
|
name = record['record_name'] or ''
|
|
extended_name = ' [%s] ID %s' % (record.get('model', 'UNDEF'), record.get('res_id', 'UNDEF'))
|
|
res.append((record['id'], name + extended_name))
|
|
return res
|
|
|
|
def message_format(self):
|
|
message_values = super(MailMessage, self).message_format()
|
|
message_index = {message['id']: message for message in message_values}
|
|
for item in self:
|
|
msg = message_index.get(item.id)
|
|
if msg:
|
|
msg['is_moved'] = item.is_moved
|
|
return message_values
|
|
|
|
|
|
class MailMoveMessageConfiguration(models.TransientModel):
|
|
_inherit = 'res.config.settings'
|
|
|
|
model_ids = fields.Many2many(comodel_name='ir.model', string='Models')
|
|
move_followers = fields.Boolean('Move Followers')
|
|
|
|
@api.model
|
|
def get_values(self):
|
|
res = super(MailMoveMessageConfiguration, self).get_values()
|
|
config_parameters = self.env["ir.config_parameter"].sudo()
|
|
model_names = config_parameters.sudo().get_param('mail_relocation_models')
|
|
model_names = model_names.split(',')
|
|
model_ids = self.env['ir.model'].sudo().search([('model', 'in', model_names)])
|
|
res.update(
|
|
model_ids=[m.id for m in model_ids],
|
|
move_followers=config_parameters.sudo().get_param('mail_relocation_move_followers'),
|
|
)
|
|
return res
|
|
|
|
def set_values(self):
|
|
super(MailMoveMessageConfiguration, self).set_values()
|
|
config_parameters = self.env["ir.config_parameter"].sudo()
|
|
for record in self:
|
|
model_names = ','.join([x.model for x in record.model_ids])
|
|
config_parameters.set_param("mail_relocation_models", model_names or '')
|
|
config_parameters.set_param("mail_relocation_move_followers", record.move_followers or '')
|
|
|
|
|
|
class ResPartner(models.Model):
|
|
_inherit = 'res.partner'
|
|
|
|
@api.model
|
|
def create(self, vals):
|
|
res = super(ResPartner, self).create(vals)
|
|
if 'update_message_author' in self.env.context and 'email' in vals:
|
|
mail_message_obj = self.env['mail.message']
|
|
# Escape special SQL characters in email_address to avoid invalid matches
|
|
email_address = (vals['email'].replace('\\', '\\\\').replace('%', '\\%').replace('_', '\\_'))
|
|
email_brackets = "<%s>" % email_address
|
|
messages = mail_message_obj.search([
|
|
'|',
|
|
('email_from', '=ilike', email_address),
|
|
('email_from', 'ilike', email_brackets),
|
|
('author_id', '=', False)
|
|
])
|
|
if messages:
|
|
messages.sudo().write({'author_id': res.id})
|
|
return res
|
|
|
|
@api.model
|
|
def default_get(self, default_fields):
|
|
contextual_self = self
|
|
if 'mail_move_message' in self.env.context and self.env.context['mail_move_message']:
|
|
contextual_self = self.with_context(
|
|
default_name=self.env.context['message_name_from'] or '',
|
|
default_email=self.env.context['message_email_from'] or '',
|
|
)
|
|
return super(ResPartner, contextual_self).default_get(default_fields)
|