stock_valuation_xlsx: Improve perfs
Add ability to force cost price to current Improve headers in XLSX
This commit is contained in:
@@ -56,6 +56,11 @@ class StockValuationXlsx(models.TransientModel):
|
|||||||
string='Subtotals per Categories', default=True,
|
string='Subtotals per Categories', default=True,
|
||||||
states={'done': [('readonly', True)]},
|
states={'done': [('readonly', True)]},
|
||||||
help="Show a subtotal per product category")
|
help="Show a subtotal per product category")
|
||||||
|
standard_price_date = fields.Selection([
|
||||||
|
('past', 'Past Date or Inventory Date'),
|
||||||
|
('present', 'Current'),
|
||||||
|
], default='past', string='Cost Price Date',
|
||||||
|
states={'done': [('readonly', True)]})
|
||||||
split_by_lot = fields.Boolean(
|
split_by_lot = fields.Boolean(
|
||||||
string='Display Lots', states={'done': [('readonly', True)]})
|
string='Display Lots', states={'done': [('readonly', True)]})
|
||||||
split_by_location = fields.Boolean(
|
split_by_location = fields.Boolean(
|
||||||
@@ -115,26 +120,25 @@ class StockValuationXlsx(models.TransientModel):
|
|||||||
return ['uom_id', 'name', 'default_code', 'categ_id']
|
return ['uom_id', 'name', 'default_code', 'categ_id']
|
||||||
|
|
||||||
def compute_product_data(
|
def compute_product_data(
|
||||||
self, company_id, in_stock_product_ids, past_date=False):
|
self, company_id, in_stock_product_ids, standard_price_past_date=False):
|
||||||
self.ensure_one()
|
self.ensure_one()
|
||||||
logger.debug('Start compute_product_data')
|
logger.debug('Start compute_product_data')
|
||||||
ppo = self.env['product.product']
|
ppo = self.env['product.product']
|
||||||
ppho = self.env['product.price.history']
|
ppho = self.env['product.price.history']
|
||||||
fields_list = self._prepare_product_fields()
|
fields_list = self._prepare_product_fields()
|
||||||
if not past_date:
|
if not standard_price_past_date:
|
||||||
fields_list.append('standard_price')
|
fields_list.append('standard_price')
|
||||||
products = ppo.search_read([('id', 'in', in_stock_product_ids)], fields_list)
|
products = ppo.search_read([('id', 'in', in_stock_product_ids)], fields_list)
|
||||||
product_id2data = {}
|
product_id2data = {}
|
||||||
now = fields.Datetime.now()
|
|
||||||
for p in products:
|
for p in products:
|
||||||
logger.debug('p=%d', p['id'])
|
logger.debug('p=%d', p['id'])
|
||||||
# I don't call the native method get_history_price()
|
# I don't call the native method get_history_price()
|
||||||
# because it requires a browse record and it is too slow
|
# because it requires a browse record and it is too slow
|
||||||
if past_date:
|
if standard_price_past_date:
|
||||||
history = ppho.search_read([
|
history = ppho.search_read([
|
||||||
('company_id', '=', company_id),
|
('company_id', '=', company_id),
|
||||||
('product_id', '=', p['id']),
|
('product_id', '=', p['id']),
|
||||||
('datetime', '<=', past_date or now)],
|
('datetime', '<=', standard_price_past_date)],
|
||||||
['cost'], order='datetime desc, id desc', limit=1)
|
['cost'], order='datetime desc, id desc', limit=1)
|
||||||
standard_price = history and history[0]['cost'] or 0.0
|
standard_price = history and history[0]['cost'] or 0.0
|
||||||
else:
|
else:
|
||||||
@@ -305,32 +309,34 @@ class StockValuationXlsx(models.TransientModel):
|
|||||||
prec_cur_rounding = company.currency_id.rounding
|
prec_cur_rounding = company.currency_id.rounding
|
||||||
self._check_config(company_id)
|
self._check_config(company_id)
|
||||||
|
|
||||||
split_by_lot = self.split_by_lot
|
|
||||||
split_by_location = self.split_by_location
|
|
||||||
past_date = False
|
|
||||||
if self.source == 'stock' and self.stock_date_type == 'past':
|
|
||||||
split_by_lot = False
|
|
||||||
split_by_location = False
|
|
||||||
past_date = self.past_date
|
|
||||||
elif self.source == 'inventory':
|
|
||||||
past_date = self.inventory_id.date
|
|
||||||
product_ids = self.get_product_ids()
|
product_ids = self.get_product_ids()
|
||||||
if not product_ids:
|
if not product_ids:
|
||||||
raise UserError(_("There are no products to analyse."))
|
raise UserError(_("There are no products to analyse."))
|
||||||
|
split_by_lot = self.split_by_lot
|
||||||
|
split_by_location = self.split_by_location
|
||||||
if self.source == 'stock':
|
if self.source == 'stock':
|
||||||
if self.stock_date_type == 'present':
|
if self.stock_date_type == 'present':
|
||||||
|
past_date = False
|
||||||
data, in_stock_products = self.compute_data_from_present_stock(
|
data, in_stock_products = self.compute_data_from_present_stock(
|
||||||
company_id, product_ids, prec_qty)
|
company_id, product_ids, prec_qty)
|
||||||
elif self.stock_date_type == 'past':
|
elif self.stock_date_type == 'past':
|
||||||
|
split_by_lot = False
|
||||||
|
split_by_location = False
|
||||||
|
past_date = self.past_date
|
||||||
data, in_stock_products = self.compute_data_from_past_stock(
|
data, in_stock_products = self.compute_data_from_past_stock(
|
||||||
product_ids, prec_qty, past_date)
|
product_ids, prec_qty, past_date)
|
||||||
elif self.source == 'inventory':
|
elif self.source == 'inventory':
|
||||||
|
past_date = self.inventory_id.date
|
||||||
data, in_stock_products = self.compute_data_from_inventory(product_ids, prec_qty)
|
data, in_stock_products = self.compute_data_from_inventory(product_ids, prec_qty)
|
||||||
|
standard_price_past_date = past_date
|
||||||
|
if not (self.source == 'stock' and self.stock_date_type == 'present') and self.standard_price_date == 'present':
|
||||||
|
standard_price_past_date = False
|
||||||
in_stock_product_ids = in_stock_products.keys()
|
in_stock_product_ids = in_stock_products.keys()
|
||||||
product_id2data = self.compute_product_data(
|
product_id2data = self.compute_product_data(
|
||||||
company_id, in_stock_product_ids, past_date=past_date)
|
company_id, in_stock_product_ids,
|
||||||
|
standard_price_past_date=standard_price_past_date)
|
||||||
data_res = self.group_result(data, split_by_lot, split_by_location)
|
data_res = self.group_result(data, split_by_lot, split_by_location)
|
||||||
categ_id2name, uom_id2name, lot_id2data, loc_id2name = self.id2name(product_ids)
|
categ_id2name, uom_id2name, lot_id2data, loc_id2name = self.id2name(in_stock_product_ids)
|
||||||
res = self.stringify_and_sort_result(
|
res = self.stringify_and_sort_result(
|
||||||
product_ids, product_id2data, data_res, prec_qty, prec_price, prec_cur_rounding,
|
product_ids, product_id2data, data_res, prec_qty, prec_price, prec_cur_rounding,
|
||||||
categ_id2name, uom_id2name, lot_id2data, loc_id2name)
|
categ_id2name, uom_id2name, lot_id2data, loc_id2name)
|
||||||
@@ -361,22 +367,33 @@ class StockValuationXlsx(models.TransientModel):
|
|||||||
j += 1
|
j += 1
|
||||||
|
|
||||||
# HEADER
|
# HEADER
|
||||||
|
now_dt = fields.Datetime.context_timestamp(self, datetime.now())
|
||||||
|
now_str = fields.Datetime.to_string(now_dt)
|
||||||
if past_date:
|
if past_date:
|
||||||
# TODO take TZ into account
|
stock_time_utc_str = past_date
|
||||||
stock_time_str = self.past_date
|
stock_time_utc_dt = fields.Datetime.from_string(stock_time_utc_str)
|
||||||
else:
|
stock_time_dt = fields.Datetime.context_timestamp(self, stock_time_utc_dt)
|
||||||
stock_time_dt = fields.Datetime.context_timestamp(self, datetime.now())
|
|
||||||
stock_time_str = fields.Datetime.to_string(stock_time_dt)
|
stock_time_str = fields.Datetime.to_string(stock_time_dt)
|
||||||
|
else:
|
||||||
|
stock_time_str = now_str
|
||||||
|
if standard_price_past_date:
|
||||||
|
standard_price_date_str = stock_time_str
|
||||||
|
else:
|
||||||
|
standard_price_date_str = now_str
|
||||||
i = 0
|
i = 0
|
||||||
sheet.write(i, 0, 'Odoo - Stock Valuation', styles['doc_title'])
|
sheet.write(i, 0, 'Odoo - Stock Valuation', styles['doc_title'])
|
||||||
sheet.set_row(0, 26)
|
sheet.set_row(0, 26)
|
||||||
i += 1
|
i += 1
|
||||||
sheet.write(i, 0, 'Valuation Date: %s' % stock_time_str, styles['doc_subtitle'])
|
sheet.write(i, 0, 'Inventory Date: %s' % stock_time_str, styles['doc_subtitle'])
|
||||||
|
i += 1
|
||||||
|
sheet.write(i, 0, 'Cost Price Date: %s' % standard_price_date_str, styles['doc_subtitle'])
|
||||||
i += 1
|
i += 1
|
||||||
sheet.write(i, 0, 'Stock location (children included): %s' % self.location_id.complete_name, styles['doc_subtitle'])
|
sheet.write(i, 0, 'Stock location (children included): %s' % self.location_id.complete_name, styles['doc_subtitle'])
|
||||||
if self.categ_ids:
|
if self.categ_ids:
|
||||||
i += 1
|
i += 1
|
||||||
sheet.write(i, 0, 'Product Categories: %s' % ', '.join([categ.display_name for categ in self.categ_ids]), styles['doc_subtitle'])
|
sheet.write(i, 0, 'Product Categories: %s' % ', '.join([categ.display_name for categ in self.categ_ids]), styles['doc_subtitle'])
|
||||||
|
i += 1
|
||||||
|
sheet.write(i, 0, 'Generated on %s by %s' % (now_str, self.env.user.name), styles['regular_small'])
|
||||||
|
|
||||||
# TITLE of COLS
|
# TITLE of COLS
|
||||||
i += 2
|
i += 2
|
||||||
@@ -498,7 +515,7 @@ class StockValuationXlsx(models.TransientModel):
|
|||||||
'expiry_date': {'width': 11, 'style': 'regular_date', 'sequence': 50, 'title': _('Expiry Date'), 'type': 'date'},
|
'expiry_date': {'width': 11, 'style': 'regular_date', 'sequence': 50, 'title': _('Expiry Date'), 'type': 'date'},
|
||||||
'qty': {'width': 8, 'style': 'regular', 'sequence': 60, 'title': _('Qty')},
|
'qty': {'width': 8, 'style': 'regular', 'sequence': 60, 'title': _('Qty')},
|
||||||
'uom_name': {'width': 5, 'style': 'regular_small', 'sequence': 70, 'title': _('UoM')},
|
'uom_name': {'width': 5, 'style': 'regular_small', 'sequence': 70, 'title': _('UoM')},
|
||||||
'standard_price': {'width': 12, 'style': 'regular_price_currency', 'sequence': 80, 'title': _('Price')},
|
'standard_price': {'width': 14, 'style': 'regular_price_currency', 'sequence': 80, 'title': _('Cost Price')},
|
||||||
'subtotal': {'width': 16, 'style': 'regular_currency', 'sequence': 90, 'title': _('Sub-total'), 'formula': True},
|
'subtotal': {'width': 16, 'style': 'regular_currency', 'sequence': 90, 'title': _('Sub-total'), 'formula': True},
|
||||||
'categ_subtotal': {'width': 16, 'style': 'regular_currency', 'sequence': 100, 'title': _('Categ Sub-total'), 'formula': True},
|
'categ_subtotal': {'width': 16, 'style': 'regular_currency', 'sequence': 100, 'title': _('Categ Sub-total'), 'formula': True},
|
||||||
'categ_name': {'width': 40, 'style': 'regular_small', 'sequence': 110, 'title': _('Product Category')},
|
'categ_name': {'width': 40, 'style': 'regular_small', 'sequence': 110, 'title': _('Product Category')},
|
||||||
|
|||||||
@@ -16,10 +16,8 @@
|
|||||||
<div name="help">
|
<div name="help">
|
||||||
<p>The generated XLSX report has the valuation of stockable products located on the selected stock locations (and their childrens).</p>
|
<p>The generated XLSX report has the valuation of stockable products located on the selected stock locations (and their childrens).</p>
|
||||||
</div>
|
</div>
|
||||||
<group name="invisible">
|
<group name="setup">
|
||||||
<field name="state" invisible="1"/>
|
<field name="state" invisible="1"/>
|
||||||
</group>
|
|
||||||
<group name="setup" states="setup">
|
|
||||||
<field name="categ_ids" widget="many2many_tags"/>
|
<field name="categ_ids" widget="many2many_tags"/>
|
||||||
<field name="warehouse_id"/>
|
<field name="warehouse_id"/>
|
||||||
<field name="location_id"/>
|
<field name="location_id"/>
|
||||||
@@ -27,11 +25,12 @@
|
|||||||
<field name="inventory_id" attrs="{'invisible': [('source', '!=', 'inventory')], 'required': [('source', '=', 'inventory')]}"/>
|
<field name="inventory_id" attrs="{'invisible': [('source', '!=', 'inventory')], 'required': [('source', '=', 'inventory')]}"/>
|
||||||
<field name="stock_date_type" attrs="{'invisible': [('source', '!=', 'stock')], 'required': [('source', '=', 'stock')]}" widget="radio"/>
|
<field name="stock_date_type" attrs="{'invisible': [('source', '!=', 'stock')], 'required': [('source', '=', 'stock')]}" widget="radio"/>
|
||||||
<field name="past_date" attrs="{'invisible': ['|', ('source', '!=', 'stock'), ('stock_date_type', '!=', 'past')], 'required': [('source', '=', 'stock'), ('stock_date_type', '=', 'past')]}"/>
|
<field name="past_date" attrs="{'invisible': ['|', ('source', '!=', 'stock'), ('stock_date_type', '!=', 'past')], 'required': [('source', '=', 'stock'), ('stock_date_type', '=', 'past')]}"/>
|
||||||
|
<field name="standard_price_date" attrs="{'invisible': [('source', '=', 'stock'), ('stock_date_type', '=', 'present')]}" widget="radio"/>
|
||||||
<field name="categ_subtotal" />
|
<field name="categ_subtotal" />
|
||||||
<field name="split_by_lot" attrs="{'invisible': [('source', '=', 'stock'), ('stock_date_type', '=', 'past')]}" groups="stock.group_production_lot"/>
|
<field name="split_by_lot" attrs="{'invisible': [('source', '=', 'stock'), ('stock_date_type', '=', 'past')]}" groups="stock.group_production_lot"/>
|
||||||
<field name="split_by_location" attrs="{'invisible': [('source', '=', 'stock'), ('stock_date_type', '=', 'past')]}"/>
|
<field name="split_by_location" attrs="{'invisible': [('source', '=', 'stock'), ('stock_date_type', '=', 'past')]}"/>
|
||||||
</group>
|
</group>
|
||||||
<group name="done" states="done">
|
<group name="done" states="done" string="Result">
|
||||||
<field name="export_file" filename="export_filename"/>
|
<field name="export_file" filename="export_filename"/>
|
||||||
<field name="export_filename" invisible="1"/>
|
<field name="export_filename" invisible="1"/>
|
||||||
</group>
|
</group>
|
||||||
|
|||||||
Reference in New Issue
Block a user