Add attachment on mas mailing - odoo

So this is my mass mailing code for sale orders. But I just missing here one thing. it's that it should add a sale order report that I created to each of SO I'm generating e-mail for. any suggestions how can I make this work?
<data>
<act_window name="Mass Sale Order Email"
res_model="mail.compose.message"
src_model="sale.order"
view_mode="form"
multi="True"
target="new"
key2="client_action_multi"
id="action_send_mass_sale_order"
context="{
'default_composition_mode': 'mass_mail',
'default_email_to': '{($object.email or \'\')|safe}',
'mass_mark_sale_order_as_sent': True,
'default_model': 'sale.order',
}"
/>
</data>
class MailComposeMessage(models.TransientModel):
_inherit = 'mail.compose.message'
#api.multi
def send_mail(self, auto_commit=False):
context = self._context
sale_order = self.env['sale.order']
sale_order_ids = context.get('active_ids')
if context.get('mass_mark_sale_order_as_sent') and \
context.get('default_model') == 'sale.order':
for order in sale_order.browse(sale_order_ids):
order.sent = True
return super(MailComposeMessage, self).send_mail(auto_commit=auto_commit)
<report
id="report_htp_sale_order"
string="Htp Sale Order"
model="sale.order"
name="config_htp.htp_sale_order"
rml="config_htp/report/htp_sale_order.ods"
menu="True"
/>
<record model="ir.actions.report.xml" id="report_htp_sale_order">
<field name="report_type">aeroo</field>
<field name="parser_loc">config_htp/report/htp_sale_order.pyc</field>
<field name="tml_Source">file</field>
<field name="parser_state">loc</field>
<field name="out_format" ref="report_aeroo.report_mimetypes_ods_ods"/>
<field name="in_format">oo-ods</field>
<field name="auto" eval="True" />
</record>

In context, we need to add a default template and in that template, we can choose what report we want to attach to it.
'default_use_template': True,
'default_template_id': ref('sale.email_template_edi_sale'),

Related

Odoo 13 - multiple model and menus in same module

I'm new to odoo and learning developing custom module. Trying to develop an contact management app for company and person. Following are the files and code structure:
#models.py
from odoo import models, fields, api
class company(models.Model):
_name = 'cs_contact.company'
_description = 'Model for create company profile.'
name = fields.Char('Company name', required=True)
country_id = fields.Many2one('res.country', string='Country', help='Select Country', ondelete='restrict', required=True)
ho_address = fields.Text('HO address')
website = fields.Char('Website')
courier_account = fields.Char('Courier Account')
email = fields.Char('Email')
class person(models.Model):
_name = 'cs_contact.person'
_description = 'Model for create person contact.'
name = fields.Char('Full Name', required=True)
country_id = fields.Many2one('res.country', string='Country', help='Select Country', ondelete='restrict', required=True)
email = fields.Char('Email')
im_id = name = fields.Char('Instant messaging ID (Skype/line)')
worked_before = fields.Selection([
('Yes', 'Yes'),
('No', 'No'),
], string="Worked Before?")
how_we_meet = fields.Selection([
('Fair', 'Fair'),
('Email', 'Email'),
('Agent', 'Agent'),
], string="How we meet?")
quantity = fields.Integer(string='Quantity')
note = fields.Text('Note')
Views looke like:
#views.xml
<odoo>
<data>
<!-- explicit list view definition -->
<record model="ir.ui.view" id="cs_contact.list">
<field name="name">cs_contact list</field>
<field name="model">cs_contact.person</field>
<field name="arch" type="xml">
<tree>
<field name="name"/>
<field name="country_id" />
<field name="email"/>
</tree>
</field>
</record>
<record id="view_cs_contactsearch" model="ir.ui.view">
<field name="name">cs_contact list</field>
<field name="model">cs_contact.person</field>
<field name="arch" type="xml">
<search string="Search contacts">
<field name="name"></field>
<field name="country_id"></field>
<field name="email"></field>
</search>
</field>
</record>
</data>
</odoo>
Menu looks like:
#menu.xml
<odoo>
<act_window id="action_company" name=" Company Contacts" res_model="cs_contact.company" view_mode="tree,form" />
<menuitem id="contact_root" name="Contacts" sequence='-1' />
<menuitem id="contact_company" name="Company" parent="contact_root" action="action_company" sequence="-1" />
</odoo>
It's working fine for company contact. Now I'm not getting how to create top menu for person and define view. This is the the design I want. I tried various method from blogs but didn't work. Please help me out.
You have added country_id in cs_contact.person view definition but the field does not exist. Remove it from the view definition or declare it in the corresponding model.
To add the Person menu item next to the Company menu item, you just need to use the same parent and higher sequence number then connect it to the corresponding action.
Example:
<act_window id="action_person" name="Persons" res_model="cs_contact.person" view_mode="tree,form"/>
<menuitem id="contact_person" name="Person" parent="contact_root" action="action_person" sequence="2"/>

Filter stock moves and return order's

I want to filter in Tree view SO's that have stock.pickings and some stock.moves in those pickings are in a state "assigned".
I created selection field that is computed and fnct_search method but it's not even triggered. what is wrong in my code? and logic.
picking_id_states = fields.Selection([
('draft', 'Draft'), ('cancel', 'Cancelled'),
('waiting', 'Waiting Another Operation'),
('confirmed', 'Waiting Availability'),
('partially_available', 'Partially Available'),
('assigned', 'Available'), ('done', 'Done')], compute='compute_picking_state', string='Picking State',
fnct_search='_move_search', store=False, copy=False, index=True, readonly=True)
def _move_search(self, operator, value):
print "test"
for so in self:
moves = so.picking_ids.mapped('move_lines')
filtered_moves = moves.filtered(lambda l: l.state == 'assigned')
if filtered_moves:
so_ids = filtered_moves.mapped('picking_id.sale_id')
return [('id', 'in', so_ids)]
#api.depends('order_line.move_ids', 'order_line.move_ids.state')
def compute_picking_state(self):
for order in self:
for picking in order.picking_ids:
order.picking_id_state = picking.state
XML code is
<record id="view_sale_order_search_picking_state" model="ir.ui.view">
<field name="name">Config Hetlta picking state</field>
<field name="model">sale.order</field>
<field name="inherit_id" ref="sale.view_sales_order_filter" />
<field name="arch" type="xml">
<field name="user_id" position="after">
<field name="picking_id_state"/>
<field name="note"/>
</field>
<filter name="sales" position="after">
<filter string="Picking state" name="picking_id_states" />
</filter>
</field>
</record>
The self in computed fields search methods is always empty, there are no recordsets. You have to take the operator and value (search term) and create your own "indirect" search for order ids.
But a maybe easier way is to just store the computed field. Odoo can only search by database persistent fields, which would be given when stored.

Adding a field to sale order lines in sale orders

I want to add a computed field "markup" to sale order lines in sale orders (quotes/sales orders).
I have created the model:
class SaleOrderLine(models.Model):
_inherit = "sale.order.line"
markup = fields.Float(compute='_compute_markup', digits=dp.get_precision('.2f%'), store=True)
def _compute_markup(self, order_id, product_id, product_uom_id):
frm_cur = self.env.user.company_id.currency_id
to_cur = order_id.pricelist_id.currency_id
purchase_price = product_id.standard_price
if product_uom_id != product_id.uom_id:
purchase_price = product_id.uom_id._compute_price(purchase_price, product_uom_id)
ctx = self.env.context.copy()
ctx['date'] = order_id.date_order
price = frm_cur.with_context(ctx).compute(purchase_price, to_cur, round=False)
return price
And a new view which inherits sale.view_order_form:
<?xml version="1.0"?>
<odoo>
<record id="view_order_form_margin" model="ir.ui.view">
<field name="name">sale.order.form.margin</field>
<field name="model">sale.order</field>
<field name="inherit_id" ref="sale.view_order_form"/>
<field name="arch" type="xml">
<xpath expr='//field[#name="order_line"]/form/group/group/field[#name="price_unit"]' position="before">
<field name="markup"/>
</xpath>
</field>
</record>
</odoo>
But the field is not shown (the view appears when you check views that inherit current view). I have reloaded everything, restarted the server and cleared the browser cache.
Any tip on why the field is not being shown is welcomed. Maybe the Xpath expression?
Thanks.
may be in sale.order view price_unit getting 2 times so it's confusion where to add and sale order view consist as formview and tree view for the sale orderline.here is the code you can get in view.
formview:
<xpath expr="//notebook//page//field//form//field[#name='price_unit']" position="before">
<field name="markup"/>
</xpath>
in Tree View:
<xpath expr="//notebook//page//field//tree//field[#name='price_unit']" position="before">
<field name="markup"/>
</xpath>

Odoo. Tree/form display field data

I have some problem with tree/form view in Odoo.
My model have such classes: https://yadi.sk/d/sCLVo3gHtbVEu
class URLList(models.Model):
_name = 'webvisitorcalc.url_list'
url = fields.Char(string="URL", required=True)
url_parametes = fields.Char(string="URL parameters") #пераметры URLб всё что идёт после ?
target_session_id = fields.One2many('webvisitorcalc.session_visitor', 'target_url_ids', string='Target URL')
site_trip_prevouse_id = fields.One2many('webvisitorcalc.site_trip', 'url_prevouse_ids', string='Prevouse URL')
site_trip_current_id = fields.One2many('webvisitorcalc.site_trip', 'url_current_ids', string='Current URL')
remote_sites_id = fields.One2many('webvisitorcalc.remote_sites', 'site_url_ids', string='Remote site page with URL')
remote_sites_target_url_id = fields.One2many('webvisitorcalc.remote_sites', 'target_url_ids', string='URL on remote site page')
#api.multi
def url_exist(self, cr, SUPERUSER_ID, urlForCheck):
_logger.error("Check URL exist in DB ")
result = False
if (self.search_count(cr, SUPERUSER_ID, [('url', '=', urlForCheck)])>0):
result = True
return result
class SiteTrip(models.Model):
_name = 'webvisitorcalc.site_trip'
session_ids = fields.Many2one('webvisitorcalc.session_visitor', string='Session ID', index=True)
url_prevouse_ids = fields.Many2one('webvisitorcalc.url_list', string='Prevouse URL', index=True)
url_current_ids = fields.Many2one('webvisitorcalc.url_list', string='Current URL', index=True)
Template for this model: https://yadi.sk/d/Ob0o65PutbVFA
<record model="ir.actions.act_window" id="site_trip_list_action">
<field name="name">Site trip</field>
<field name="res_model">webvisitorcalc.site_trip</field>
<field name="view_type">form</field>
<field name="view_mode">tree,form</field>
<field name="help" type="html">
<p class="oe_view_nocontent_create">Create the first record for site trip
</p>
</field>
</record>
<record model="ir.actions.act_window" id="url_list_list_action">
<field name="name">URL list</field>
<field name="res_model">webvisitorcalc.url_list</field>
<field name="view_type">form</field>
<field name="view_mode">tree,form</field>
<field name="help" type="html">
<p class="oe_view_nocontent_create">Create the first url
</p>
</field>
</record>
<record model="ir.ui.view" id="site_trip_tree_view">
<field name="name">site_trip.tree</field>
<field name="model">webvisitorcalc.site_trip</field>
<field name="arch" type="xml">
<tree string="URL list tree">
<field name="session_ids"/>
<field name="url_prevouse_ids" string="PrevURL">
</field>
<!--<field name="url_prevouse_ids"/>-->
<field name="url_current_ids"/>
</tree>
</field>
</record>
<record model="ir.ui.view" id="url_list_tree_view">
<field name="name">url_list.tree</field>
<field name="model">webvisitorcalc.url_list</field>
<field name="arch" type="xml">
<tree string="URL list tree">
<field name="url"/>
<field name="url_parametes"/>
</tree>
</field>
</record>
<menuitem id="site_trip_menu" name="Site trip" parent="webvisitorcalc_menu"
action="site_trip_list_action"/>
<menuitem id="url_list_menu" name="URL list" parent="webvisitorcalc_menu"
action="url_list_list_action"/>
Screenshots are here:
Tree view for class SiteTrip
http://i.stack.imgur.com/FjRDK.png
Form view for class SiteTrip
http://i.stack.imgur.com/uDbOp.png
Tree view for class URLList
http://i.stack.imgur.com/tXzqL.png
Form view for class URLList
http://i.stack.imgur.com/oVnqg.png
As you see URLList displayed fine. For class SiteTrip present problem. Field is displaying not data from URLList. This is field stored element such webvisitorcalc.url_list.ID (array?). How I can display real data in this field (for example URL: http://some-site.com/page.html)?
URL in URLList must be uniq. SiteTrip must have stored only ID of URLList record.
UPD:
class RemoteSites(models.Model):
_name = 'webvisitorcalc.remote_sites'
site_id = advert_company_id = fields.One2many('webvisitorcalc.site_list', 'remote_sites_ids', string='Site')
site_url_ids = fields.Many2one('webvisitorcalc.url_list', string='URL page ')
target_url_ids = fields.Many2one('webvisitorcalc.url_list', string='URL target page')
You obviously have no name field on your webvisitorcalc.url_list model. Odoo needs this to use it as name in webclient wherever you use this model as e.g. many2one field or in the breadcrumb navigation.
So either you define a name field or you set _name on your class with another field identifier.
You can also (re-)define the method display_name on your model (enough examples in Odoo code) where you can do more cool stuff with the record display name :-)

OpenERP - Show child_ids invoices of a partner

I'd like to show invoices adressed to child partner in the parent partner form view.
I've already a inherited res_partner model as follow :
class res_partner(osv.osv):
_inherit = 'res.partner'
_columns = {
'invoice_ids': fields.one2many('account.invoice', 'partner_id', 'Invoices'),
}
And a view displaying invoices as follow :
<?xml version="1.0"?>
<openerp>
<data>
<!-- Partners inherited form -->
<record id="view_history_partner_info_form" model="ir.ui.view">
<field name="name">res.partner.cap_history.form.inherit</field>
<field name="inherit_id" ref="base.view_partner_form"/>
<field name="model">res.partner</field>
<field name="arch" type="xml">
<page string="Accounting" position="after" version="7.0">
<page string="History" name="cap_history_tab">
<group name="grp_invoice_history" string="Invoices History">
<field name="invoice_ids" colspan="4" nolabel="1">
<tree string="Partner Invoices" create="false" delete="false">
<field name="number" readonly="True"/>
<field name="origin" readonly="True"/>
<field name="name" string="Reference" readonly="True"/>
<field name="date_invoice" readonly="True"/>
<field name="x_category" readonly="True"/>
<field name="state" readonly="True"/>
<field name="payment_term" readonly="True"/>
<field name="amount_total" readonly="True"/>
</tree>
</field>
</group>
</page>
</page>
</field>
</record>
With this code I can see invoices that are directly adressed to a company or a person on their respective form view.
But if an invoice is adressed to person, and none is adressed to the parent company, when I am on the company form view, I won't see the invoice adressed to the child contact.
Is there a way to make visible the contact's invoice in the parent partner form view ?
Thank you for your help !
Cheers
One way to do this is to add a new field, all_invoice_ids, as a function field, and then have the function return both the contents of invoice_ids, plus the contents of any child's invoice_ids.
Something like this (untested):
'all_invoice_ids': fields.function(
_get_invoice_ids,
type='one2many',
obj='account.invoice',
method=True,
string='Invoices',
),
and _get_invoice_ids (which should be defined before columns) like this (also untested):
def _get_invoice_ids(self, cr, uid, ids, field_name, arg, context=None):
res = {}
if isinstance(ids, (int, long)):
ids = [ids] # in case an id was passed in directly
for main_partner in self.browse(cr, uid, ids, context=context):
main_invoices = main_partner.invoice_ids or [] # in case it was False
invoices = [inv.id for inv in main_invoices]
for child_partner in main_partner.child_ids:
child_invoices = child_partner.invoice_ids or []
invoices.extend([inv.id for inv in child_invoices])
# at this point we should have all the invoice ids
# use a set to get rid of duplicates
invoices = list(set(invoices))
# and store in res to be returned
res[main_partner.id] = invoices
return res