oca-ocb-accounting/odoo-bringout-oca-ocb-account/account/wizard/account_move_send_wizard.py
Ernad Husremovic 768b70e05e 19.0 vanilla
2026-03-09 09:30:07 +01:00

405 lines
17 KiB
Python

from odoo import _, api, fields, models
from odoo.exceptions import UserError
from odoo.tools.misc import get_lang
from odoo.addons.mail.tools.parser import parse_res_ids
from odoo.addons.mail.wizard.mail_compose_message import _reopen
class AccountMoveSendWizard(models.TransientModel):
"""Wizard that handles the sending a single invoice."""
_name = 'account.move.send.wizard'
_inherit = ['account.move.send', 'mail.composer.mixin']
_description = "Account Move Send Wizard"
move_id = fields.Many2one(comodel_name='account.move', required=True)
company_id = fields.Many2one(comodel_name='res.company', related='move_id.company_id')
alerts = fields.Json(compute='_compute_alerts')
sending_methods = fields.Json(
compute='_compute_sending_methods',
inverse='_inverse_sending_methods',
)
sending_method_checkboxes = fields.Json(
compute='_compute_sending_method_checkboxes',
precompute=True,
readonly=False,
store=True,
)
# Technical field to display the attachments widget
display_attachments_widget = fields.Boolean(
compute='_compute_display_attachments_widget',
)
extra_edis = fields.Json(
compute='_compute_extra_edis',
inverse='_inverse_extra_edis',
)
extra_edi_checkboxes = fields.Json(
compute='_compute_extra_edi_checkboxes',
precompute=True,
readonly=False,
store=True,
)
invoice_edi_format = fields.Selection(
selection=lambda self: self.env['res.partner']._fields['invoice_edi_format'].selection,
compute='_compute_invoice_edi_format',
)
pdf_report_id = fields.Many2one(
comodel_name='ir.actions.report',
string="Invoice report",
domain="[('id', 'in', available_pdf_report_ids)]",
compute='_compute_pdf_report_id',
readonly=False,
store=True,
)
available_pdf_report_ids = fields.One2many(
comodel_name='ir.actions.report',
compute="_compute_available_pdf_report_ids",
)
display_pdf_report_id = fields.Boolean(compute='_compute_display_pdf_report_id')
# MAIL
# Template: override mail.composer.mixin field
template_id = fields.Many2one(
domain="[('model', '=', 'account.move')]",
compute='_compute_template_id',
compute_sudo=True,
readonly=False,
store=True,
)
# Language: override mail.composer.mixin field
lang = fields.Char(compute='_compute_lang', precompute=False, compute_sudo=True)
mail_partner_ids = fields.Many2many(
comodel_name='res.partner',
string="To",
compute='_compute_mail_partners',
store=True,
readonly=False,
)
mail_attachments_widget = fields.Json(
compute='_compute_mail_attachments_widget',
store=True,
readonly=False,
)
attachments_not_supported = fields.Json(compute='_compute_attachments_not_supported')
model = fields.Char('Related Document Model', compute='_compute_model', readonly=False, store=True)
res_ids = fields.Text('Related Document IDs', compute='_compute_res_ids', readonly=False, store=True)
template_name = fields.Char('Template Name') # used when saving a new mail template
# -------------------------------------------------------------------------
# DEFAULTS
# -------------------------------------------------------------------------
@api.model
def default_get(self, fields):
# EXTENDS 'base'
results = super().default_get(fields)
if 'move_id' in fields and 'move_id' not in results:
move_id = self.env.context.get('active_ids', [])[0]
results['move_id'] = move_id
return results
# -------------------------------------------------------------------------
# COMPUTE METHODS
# -------------------------------------------------------------------------
@api.depends('sending_methods', 'extra_edis', 'mail_partner_ids')
def _compute_alerts(self):
for wizard in self:
move_data = {
wizard.move_id: {
'sending_methods': wizard.sending_methods or {},
'invoice_edi_format': wizard.invoice_edi_format,
'extra_edis': wizard.extra_edis or {},
'mail_partner_ids': wizard.mail_partner_ids
}
}
wizard.alerts = self._get_alerts(wizard.move_id, move_data)
@api.depends('sending_method_checkboxes')
def _compute_sending_methods(self):
for wizard in self:
wizard.sending_methods = self._get_selected_checkboxes(wizard.sending_method_checkboxes)
def _inverse_sending_methods(self):
for wizard in self:
wizard.sending_method_checkboxes = {method_key: {'checked': True} for method_key in wizard.sending_methods or {}}
@api.depends('move_id')
def _compute_sending_method_checkboxes(self):
""" Select one applicable sending method given the following priority
1. preferred method set on partner,
2. email,
"""
methods = self.env['ir.model.fields'].get_field_selection('res.partner', 'invoice_sending_method')
# We never want to display the manual method.
methods = [method for method in methods if method[0] != 'manual']
for wizard in self:
preferred_methods = self._get_default_sending_methods(wizard.move_id)
wizard.sending_method_checkboxes = {
method_key: {
'checked': (
method_key in preferred_methods and (
method_key == 'email' or self._is_applicable_to_move(method_key, wizard.move_id, **self._get_default_sending_settings(wizard.move_id))
)), # email method is always ok in single mode since the email can be added if it's missing
'label': method_label,
}
for method_key, method_label in methods
if self._is_applicable_to_company(method_key, wizard.company_id)
}
@api.depends('invoice_edi_format')
def _compute_display_attachments_widget(self):
for wizard in self:
wizard.display_attachments_widget = wizard._display_attachments_widget(
edi_format=wizard.invoice_edi_format,
sending_methods=wizard.sending_methods or [],
)
@api.depends('extra_edi_checkboxes')
def _compute_extra_edis(self):
for wizard in self:
wizard.extra_edis = self._get_selected_checkboxes(wizard.extra_edi_checkboxes)
def _inverse_extra_edis(self):
for wizard in self:
wizard.extra_edi_checkboxes = {method_key: {'checked': True} for method_key in wizard.extra_edis or {}}
@api.depends('move_id')
def _compute_extra_edi_checkboxes(self):
all_extra_edis = self._get_all_extra_edis()
for wizard in self:
wizard.extra_edi_checkboxes = {
edi_key: {'checked': True, 'label': all_extra_edis[edi_key]['label'], 'help': all_extra_edis[edi_key].get('help')}
for edi_key in self._get_default_extra_edis(wizard.move_id)
}
@api.depends('move_id', 'sending_methods')
def _compute_invoice_edi_format(self):
for wizard in self:
wizard.invoice_edi_format = self._get_default_invoice_edi_format(wizard.move_id, sending_methods=wizard.sending_methods or {})
@api.depends('move_id')
def _compute_pdf_report_id(self):
for wizard in self:
wizard.pdf_report_id = self._get_default_pdf_report_id(wizard.move_id)
@api.depends('move_id')
def _compute_available_pdf_report_ids(self):
available_reports = self.move_id._get_available_action_reports()
for wizard in self:
wizard.available_pdf_report_ids = available_reports
@api.depends('move_id')
def _compute_display_pdf_report_id(self):
""" Show PDF template selection if there are more than 1 template available for invoices. """
for wizard in self:
wizard.display_pdf_report_id = len(wizard.available_pdf_report_ids) > 1 and not wizard.move_id.invoice_pdf_report_id
@api.depends('move_id')
def _compute_template_id(self):
for wizard in self:
wizard.template_id = self._get_default_mail_template_id(wizard.move_id)
@api.depends('template_id')
def _compute_lang(self):
# OVERRIDE 'mail.composer.mixin'
for wizard in self:
wizard.lang = self._get_default_mail_lang(wizard.move_id, wizard.template_id) if wizard.template_id else get_lang(self.env).code
@api.depends('template_id', 'lang')
def _compute_mail_partners(self):
for wizard in self:
wizard.mail_partner_ids = commercial_partner if (commercial_partner := wizard.move_id.commercial_partner_id).email else None
if wizard.template_id:
wizard.mail_partner_ids = self._get_default_mail_partner_ids(wizard.move_id, wizard.template_id, wizard.lang)
@api.depends('template_id', 'lang')
def _compute_subject(self):
# OVERRIDE 'mail.composer.mixin'
for wizard in self:
wizard.subject = None
if wizard.template_id:
wizard.subject = self._get_default_mail_subject(wizard.move_id, wizard.template_id, wizard.lang)
@api.depends('template_id', 'lang')
def _compute_body(self):
# OVERRIDE 'mail.composer.mixin'
for wizard in self:
wizard.body = None
if wizard.template_id:
wizard.body = self._get_default_mail_body(wizard.move_id, wizard.template_id, wizard.lang)
@api.depends('template_id', 'invoice_edi_format', 'extra_edis', 'pdf_report_id')
def _compute_mail_attachments_widget(self):
for wizard in self:
manual_attachments_data = [x for x in wizard.mail_attachments_widget or [] if x.get('manual')]
wizard.mail_attachments_widget = (
self._get_default_mail_attachments_widget(
wizard.move_id,
wizard.template_id,
invoice_edi_format=wizard.invoice_edi_format,
extra_edis=wizard.extra_edis or {},
pdf_report=wizard.pdf_report_id,
)
+ manual_attachments_data
)
# Similar of mail.compose.message
@api.depends('template_id')
def _compute_res_ids(self):
for wizard in self:
wizard.res_ids = wizard.move_id.ids
# Similar of mail.compose.message
@api.depends('template_id')
def _compute_model(self):
for wizard in self:
if wizard.model:
continue
wizard.model = self.env.context.get('active_model')
# Similar of mail.compose.message
@api.depends('sending_methods')
def _compute_can_edit_body(self):
for record in self:
record.can_edit_body = record.sending_methods and 'email' in record.sending_methods
@api.depends('model') # Fake trigger otherwise not computed in new mode
def _compute_render_model(self):
# OVERRIDE 'mail.composer.mixin'
self.render_model = 'account.move'
# Similar of mail.compose.message
def open_template_creation_wizard(self):
""" Hit save as template button: opens a wizard that prompts for the template's subject.
`create_mail_template` is called when saving the new wizard. """
self.ensure_one()
return {
'type': 'ir.actions.act_window',
'view_mode': 'form',
'view_id': self.env.ref('mail.mail_compose_message_view_form_template_save').id,
'name': _('Create a Mail Template'),
'res_model': 'account.move.send.wizard',
'context': {'dialog_size': 'medium'},
'target': 'new',
'res_id': self.id,
}
# Similar of mail.compose.message
def create_mail_template(self):
""" Creates a mail template with the current mail composer's fields. """
self.ensure_one()
if not self.model or not self.model in self.env:
raise UserError(_('Template creation from composer requires a valid model.'))
model_id = self.env['ir.model']._get_id(self.model)
values = {
'name': self.template_name or self.subject,
'subject': self.subject,
'body_html': self.body,
'model_id': model_id,
'use_default_to': True,
'user_id': self.env.uid,
}
template = self.env['mail.template'].create(values)
# generate the saved template
self.write({'template_id': template.id})
return _reopen(self, self.id, self.model, context={**self.env.context, 'dialog_size': 'large'})
# Similar of mail.compose.message
def cancel_save_template(self):
""" Restore old subject when canceling the 'save as template' action
as it was erased to let user give a more custom input. """
self.ensure_one()
return _reopen(self, self.id, self.model, context={**self.env.context, 'dialog_size': 'large'})
@api.depends('invoice_edi_format', 'mail_attachments_widget')
def _compute_attachments_not_supported(self):
for wizard in self:
wizard.attachments_not_supported = {}
# -------------------------------------------------------------------------
# CONSTRAINS
# -------------------------------------------------------------------------
@api.constrains('move_id')
def _check_move_id_constraints(self):
for wizard in self:
self._check_move_constraints(wizard.move_id)
# -------------------------------------------------------------------------
# HELPERS
# -------------------------------------------------------------------------
@api.model
def _get_selected_checkboxes(self, json_checkboxes):
if not json_checkboxes:
return {}
return [checkbox_key for checkbox_key, checkbox_vals in json_checkboxes.items() if checkbox_vals['checked']]
# -------------------------------------------------------------------------
# BUSINESS METHODS
# -------------------------------------------------------------------------
def _get_sending_settings(self):
self.ensure_one()
send_settings = {
'sending_methods': self.sending_methods or [],
'invoice_edi_format': self.invoice_edi_format,
'extra_edis': self.extra_edis or [],
'pdf_report': self.pdf_report_id,
'author_user_id': self.env.user.id,
'author_partner_id': self.env.user.partner_id.id,
}
if self.sending_methods and 'email' in self.sending_methods:
send_settings.update({
'mail_template': self.template_id,
'mail_lang': self.lang,
'mail_body': self.body,
'mail_subject': self.subject,
'mail_partner_ids': self.mail_partner_ids.ids,
})
if self.display_attachments_widget:
send_settings['mail_attachments_widget'] = self.mail_attachments_widget
return send_settings
def _update_preferred_settings(self):
"""If the partner's settings are not set, we use them as partner's default."""
self.ensure_one()
if not self.move_id.partner_id.invoice_template_pdf_report_id and self.pdf_report_id != self._get_default_pdf_report_id(self.move_id):
self.move_id.partner_id.sudo().invoice_template_pdf_report_id = self.pdf_report_id
# -------------------------------------------------------------------------
# BUSINESS ACTIONS
# -------------------------------------------------------------------------
@api.model
def _action_download(self, attachments):
""" Download the PDF attachment, or a zip of attachments if there are more than one. """
return {
'type': 'ir.actions.act_url',
'url': f'/account/download_invoice_attachments/{",".join(map(str, attachments.ids))}',
'close': True,
}
def action_send_and_print(self, allow_fallback_pdf=False):
""" Create invoice documents and send them."""
self.ensure_one()
if self.alerts:
self._raise_danger_alerts(self.alerts)
self._update_preferred_settings()
attachments = self._generate_and_send_invoices(
self.move_id,
**self._get_sending_settings(),
allow_fallback_pdf=allow_fallback_pdf,
)
if attachments and self.sending_methods and 'manual' in self.sending_methods:
return self._action_download(attachments)
else:
return {'type': 'ir.actions.act_window_close'}