odoo/addons/product/pricelist.py

246 lines
9.5 KiB
Python

# -*- encoding: utf-8 -*-
##############################################################################
#
# Copyright (c) 2004-2006 TINY SPRL. (http://tiny.be) All Rights Reserved.
#
# WARNING: This program as such is intended to be used by professional
# programmers who take the whole responsability of assessing all potential
# consequences resulting from its eventual inadequacies and bugs
# End users who are looking for a ready-to-use solution with commercial
# garantees and support are strongly adviced to contract a Free Software
# Service Company
#
# This program is Free Software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
#
##############################################################################
from osv import fields, osv
#from tools.misc import currency
from _common import rounding
class price_type(osv.osv):
"""
The price type is used to points which field in the product form
is a price and in which currency is this price expressed.
When a field is a price, you can use it in pricelists to base
sale and purchase prices based on some fields of the product.
"""
def _price_field_get(self, cr, uid, context={}):
cr.execute('select name, field_description from ir_model_fields where model in (%s,%s) and ttype=%s', ('product.product', 'product.template', 'float'))
return cr.fetchall()
_name = "product.price.type"
_description = "Price type"
_columns = {
"name" : fields.char("Price Name", size=32, required=True, translate=True) ,
"active" : fields.boolean("Active"),
"field" : fields.selection(_price_field_get, "Product Field", required=True),
"currency_id" : fields.many2one('res.currency', "Currency", required=True),
}
_defaults = {
"active": lambda *args: True ,
}
price_type()
#----------------------------------------------------------
# Price lists
#----------------------------------------------------------
class product_pricelist_type(osv.osv):
_name = "product.pricelist.type"
_description = "Pricelist Type"
_columns = {
'name': fields.char('Name',size=64, required=True),
'key': fields.char('Key', size=64, required=True),
}
product_pricelist_type()
class product_pricelist(osv.osv):
def _pricelist_type_get(self, cr, uid, context={}):
cr.execute('select key,name from product_pricelist_type order by name')
return cr.fetchall()
_name = "product.pricelist"
_description = "Pricelist"
_columns = {
'name': fields.char('Name',size=64, required=True),
'active': fields.boolean('Active'),
'type': fields.selection(_pricelist_type_get, 'Pricelist Type', required=True),
'version_id': fields.one2many('product.pricelist.version', 'pricelist_id', 'Pricelist Versions'),
'currency_id': fields.many2one('res.currency', 'Currency', required=True),
}
_defaults = {
'active': lambda *a: 1,
}
#
# IN:
# Context {
# 'uom': Unit of Measure (Int)
# 'partner': Partner ID (int)
# }
#
def price_get(self, cr, uid, ids, prod_id, qty, partner=None, context={}):
if context and ('partner_id' in context):
partner = context['partner_id']
result = {}
for id in ids:
cr.execute('select * from product_pricelist_version where pricelist_id=%d and active=True order by id limit 1', (id,))
#
# Ajouter le test de la date du jour
#
plversion = False
# Ahahahahaha
for plversion in cr.dictfetchall():
break
if not plversion:
raise osv.except_osv('Warning !', 'No active version for the selected pricelist !\nPlease create or activate one.')
cr.execute('select id,categ_id from product_template where id=(select product_tmpl_id from product_product where id=%d)', (prod_id,))
tmpl_id,categ = cr.fetchone()
categ_ids = []
while categ:
categ_ids.append(str(categ))
cr.execute('select parent_id from product_category where id=%d', (categ,))
categ = cr.fetchone()[0]
if str(categ) in categ_ids:
raise osv.except_osv('Warning !', 'Could not resolve product category, you have defined cyclic categories of products !')
if categ_ids:
categ_where = '(categ_id in ('+','.join(categ_ids)+'))'
else:
categ_where = '(categ_id is null)'
cr.execute(
'select i.*, pl.currency_id '
'from product_pricelist_item as i, product_pricelist_version as v, product_pricelist as pl '
'where (product_tmpl_id is null or product_tmpl_id=%d) '
'and (product_id is null or product_id=%d) '
'and ('+categ_where+' or (categ_id is null)) '
'and price_version_id=%d '
'and (min_quantity is null or min_quantity<=%f) '
'and i.price_version_id=v.id and v.pricelist_id=pl.id '
'order by sequence limit 1', (tmpl_id, prod_id, plversion['id'], qty))
res = cr.dictfetchone()
if res:
if res['base'] == -1:
if not res['base_pricelist_id']:
price = 0.0
else:
price_tmp = self.price_get(cr, uid, [res['base_pricelist_id']], prod_id, qty)[res['base_pricelist_id']]
ptype_src = self.pool.get('product.pricelist').browse(cr, uid, res['base_pricelist_id']).currency_id.id
price = self.pool.get('res.currency').compute(cr, uid, ptype_src, res['currency_id'], price_tmp)
elif res['base'] == -2:
where = []
if partner:
where = [('name', '=', partner) ]
sinfo = self.pool.get('product.supplierinfo').search(cr, uid, [('product_id', '=', prod_id)]+where)
if not sinfo:
result[id] = 0
continue
cr.execute('select * from pricelist_partnerinfo where suppinfo_id in (' + ','.join(map(str, sinfo)) + ') and min_quantity<=%f order by min_quantity desc limit 1', (qty,))
res = cr.dictfetchone()
if res:
result[id] = res['price']
else:
result[id] = 0
continue
else:
price_type_o=self.pool.get('product.price.type').read(cr, uid, [ res['base'] ])[0]
price = self.pool.get('res.currency').compute(cr, uid, price_type_o['currency_id'][0], res['currency_id'], self.pool.get('product.product').price_get(cr, uid, [prod_id], price_type_o['field'])[prod_id])
price_limit = price
price = price * (1.0-(res['price_discount'] or 0.0))
price = rounding(price, res['price_round'])
price += (res['price_surcharge'] or 0.0)
if res['price_min_margin']:
price = max(price, price_limit+res['price_min_margin'])
if res['price_max_margin']:
price = min(price, price_limit+res['price_max_margin'])
else:
# False means no valid line found ! But we may not raise an
# exception here because it breaks the search
price = False
result[id] = price
if 'uom' in context:
result[id] = self.pool.get('product.uom')._compute_price(cr, uid, context['uom'], result[id])
return result
product_pricelist()
class product_pricelist_version(osv.osv):
_name = "product.pricelist.version"
_description = "Pricelist Version"
_columns = {
'pricelist_id': fields.many2one('product.pricelist', 'Price List', required=True, select=True, relate=True),
'name': fields.char('Name', size=64, required=True),
'active': fields.boolean('Active'),
'items_id': fields.one2many('product.pricelist.item', 'price_version_id', 'Price List Items', required=True),
'date_start': fields.date('Start Date'),
'date_end': fields.date('End Date')
}
_defaults = {
'active': lambda *a: 1,
}
product_pricelist_version()
class product_pricelist_item(osv.osv):
def _price_field_get(self, cr, uid, context={}):
cr.execute('select id,name from product_price_type where active')
result = cr.fetchall()
result.append((-1,'Other Pricelist'))
result.append((-2,'Partner section of the product form'))
return result
_name = "product.pricelist.item"
_description = "Pricelist item"
_order = "sequence"
_defaults = {
'base': lambda *a: -1,
'min_quantity': lambda *a: 1,
'sequence': lambda *a: 5,
'price_discount': lambda *a: 0,
}
_columns = {
'name': fields.char('Name', size=64),
'price_version_id': fields.many2one('product.pricelist.version', 'Price List Version', required=True, select=True),
'product_tmpl_id': fields.many2one('product.template', 'Product Template'),
'product_id': fields.many2one('product.product', 'Product'),
'categ_id': fields.many2one('product.category', 'Product Category'),
'min_quantity': fields.integer('Min. Quantity', required=True),
'sequence': fields.integer('Priority', required=True),
'base': fields.selection(_price_field_get, 'Based on', required=True, size=-1),
#'base': fields.many2one('product.price.type', 'Based on', required=True),
'base_pricelist_id': fields.many2one('product.pricelist', 'If Other Pricelist'),
'price_surcharge': fields.float('Price Surcharge'),
'price_discount': fields.float('Price Discount'),
'price_round': fields.float('Price Rounding'),
'price_min_margin': fields.float('Price Min. Margin'),
'price_max_margin': fields.float('Price Max. Margin'),
}
def product_id_change(self, cr, uid, ids, product_id, context={}):
if not product_id:
return {}
prod = self.pool.get('product.product').read(cr, uid, [product_id], ['code','name'])
if prod[0]['code']:
return {'value': {'name': prod[0]['code']}}
return {}
product_pricelist_item()