forked from OCA/account-invoicing
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathsale.py
234 lines (210 loc) · 10.3 KB
/
sale.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
# -*- coding: utf-8 -*-
##############################################################################
#
# Author: Alexandre Fayolle
# Copyright 2013 Camptocamp SA
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
##############################################################################
"""
* Adding qty_invoiced field on SO lines, computed based on invoice lines
linked to it that has the same product. So this way, advance invoice will
still work !
* Adding qty_delivered field in SO Lines, computed from move lines linked to
it. For services, the quantity delivered is a problem, the MRP will
automatically run the procurement linked to this line and pass it to done. I
suggest that in that case, delivered qty = invoiced_qty as the procurement
is for the whole qty, it'll be a good alternative to follow what has been
done and not.
* Add in the "Order Line to invoice" view those fields
* Change the behavior of the "invoiced" field of the SO line to be true when
all is invoiced
* Adapt the "_make_invoice" method in SO to deal with qty_invoiced
* Adapt the sale_line_invoice.py wizard to deal with qty_invoiced, asking the
user how much he want to invoice.
By having the delivered quantity, we can imagine in the future to provide an
invoicing "based on delivery" that will look at those values instead of looking
in picking.
"""
import logging
_logger = logging.getLogger(__name__)
from openerp.osv import orm, fields
from openerp import netsvc
class SaleOrderLine(orm.Model):
_inherit = 'sale.order.line'
def field_qty_invoiced(self, cr, uid, ids, fields, arg, context):
res = dict.fromkeys(ids, 0)
for line in self.browse(cr, uid, ids, context=context):
for invoice_line in line.invoice_lines:
if invoice_line.invoice_id.state != 'cancel':
res[line.id] += invoice_line.quantity # XXX uom !
return res
def field_qty_delivered(self, cr, uid, ids, fields, arg, context):
res = dict.fromkeys(ids, 0)
for line in self.browse(cr, uid, ids, context=context):
if not line.move_ids:
# consumable or service: assume delivered == invoiced
res[line.id] = line.qty_invoiced
else:
for move in line.move_ids:
if (move.state == 'done' and
move.picking_id and
move.picking_id.type == 'out'):
res[line.id] += move.product_qty
return res
def _prepare_order_line_invoice_line(self, cr, uid, line, account_id=False,
context=None):
if context is None:
context = {}
res = super(SaleOrderLine,
self)._prepare_order_line_invoice_line(
cr, uid, line, account_id, context=context)
if '_partial_invoice' in context:
# we are making a partial invoice for the line
to_invoice_qty = context['_partial_invoice'][line.id]
else:
# we are invoicing the yet uninvoiced part of the line
to_invoice_qty = line.product_uom_qty - line.qty_invoiced
res['quantity'] = to_invoice_qty
return res
def _fnct_line_invoiced(self, cr, uid, ids, field_name, args, context=None):
res = dict.fromkeys(ids, False)
for this in self.browse(cr, uid, ids, context=context):
res[this.id] = (this.qty_invoiced == this.product_uom_qty)
return res
def _order_lines_from_invoice2(self, cr, uid, ids, context=None):
# overridden with different name because called by framework with
# 'self' an instance of another class
return self.pool['sale.order.line']._order_lines_from_invoice(
cr, uid, ids, context)
_columns = {
'qty_invoiced': fields.function(
field_qty_invoiced, string='Invoiced Quantity', type='float',
help="the quantity of product from this line already invoiced"),
'qty_delivered': fields.function(
field_qty_delivered, string='Invoiced Quantity', type='float',
help="the quantity of product from this line already invoiced"),
'invoiced': fields.function(
_fnct_line_invoiced, string='Invoiced', type='boolean',
store={
'account.invoice': (_order_lines_from_invoice2, ['state'], 10),
'sale.order.line': (lambda self, cr, uid, ids, ctx=None: ids,
['invoice_lines'], 10)
}),
}
class sale_advance_payment_inv(orm.TransientModel):
_inherit = "sale.advance.payment.inv"
def create_invoices(self, cr, uid, ids, context=None):
"""override standard behavior if payment method is set to 'lines':
"""
res = super(sale_advance_payment_inv, self).create_invoices(
cr, uid, ids, context)
wizard = self.browse(cr, uid, ids[0], context)
if wizard.advance_payment_method != 'lines':
return res
sale_ids = context.get('active_ids', [])
if not sale_ids:
return res
wizard_obj = self.pool['sale.order.line.invoice.partially']
order_line_obj = self.pool['sale.order.line']
so_domain = [('order_id', 'in', sale_ids), ]
so_line_ids = order_line_obj.search(
cr, uid, so_domain, context=context)
line_values = []
for so_line in order_line_obj.browse(cr, uid, so_line_ids,
context=context):
if so_line.state in ('confirmed', 'done') and not so_line.invoiced:
val = {'sale_order_line_id': so_line.id, }
if so_line.product_id and so_line.product_id.type == 'product':
val['quantity'] = so_line.qty_delivered - \
so_line.qty_invoiced
else:
# service or consumable
val['quantity'] = so_line.product_uom_qty - \
so_line.qty_invoiced
line_values.append((0, 0, val))
val = {'line_ids': line_values, }
wizard_id = wizard_obj.create(cr, uid, val, context=context)
wiz = wizard_obj.browse(cr, uid, wizard_id, context=context)
print wiz.line_ids
res = {'view_type': 'form',
'view_mode': 'form',
'res_model': 'sale.order.line.invoice.partially',
'res_id': wizard_id,
'type': 'ir.actions.act_window',
'target': 'new',
'context': context,
}
return res
class SaleOrderLineInvoicePartiallyLine(orm.TransientModel):
_name = "sale.order.line.invoice.partially.line"
_columns = {
'wizard_id': fields.many2one('sale.order.line.invoice.partially',
string='Wizard'),
'sale_order_line_id': fields.many2one('sale.order.line',
string='sale.order.line'),
'name': fields.related('sale_order_line_id', 'name',
type='char', string="Line"),
'order_qty': fields.related('sale_order_line_id', 'product_uom_qty',
type='float', string="Sold"),
'qty_invoiced': fields.related('sale_order_line_id', 'qty_invoiced',
type='float', string="Invoiced"),
'qty_delivered': fields.related('sale_order_line_id', 'qty_delivered',
type='float', string="Shipped"),
'quantity': fields.float('To invoice'),
}
class SaleOrderLineInvoicePartially(orm.TransientModel):
_name = "sale.order.line.invoice.partially"
_columns = {
'name': fields.char('Name'),
'line_ids': fields.one2many('sale.order.line.invoice.partially.line',
'wizard_id', string="Lines"),
}
def create_invoice(self, cr, uid, ids, context=None):
wf_service = netsvc.LocalService('workflow')
if context is None:
context = {}
ctx = context.copy()
ctx['_partial_invoice'] = {}
so_line_obj = self.pool['sale.order.line']
so_obj = self.pool['sale.order']
order_lines = {}
for wiz in self.browse(cr, uid, ids, context=context):
for line in wiz.line_ids:
if line.quantity == 0:
continue
sale_order = line.sale_order_line_id.order_id
if sale_order.id not in order_lines:
order_lines[sale_order.id] = []
order_lines[sale_order.id].append(line.sale_order_line_id.id)
ctx['_partial_invoice'][
line.sale_order_line_id.id] = line.quantity
for order_id in order_lines:
line_ids = order_lines[order_id]
invoice_line_ids = so_line_obj.invoice_line_create(
cr, uid, line_ids, context=ctx)
order = so_obj.browse(cr, uid, order_id, context=context)
invoice_id = so_obj._make_invoice(
cr, uid, order, invoice_line_ids, context=ctx)
_logger.info('created invoice %d', invoice_id)
# the following is copied from many places around
# (actually sale_line_invoice.py)
cr.execute('INSERT INTO sale_order_invoice_rel (order_id, '
' invoice_id) '
'VALUES (%s,%s)', (order_id, invoice_id))
if all(line.invoiced for line in order.order_line):
wf_service.trg_validate(
uid, 'sale.order', order.id, 'manual_invoice', cr)
return {'type': 'ir.actions.act_window_close'}