1 # -*- coding: utf-8 -*-
2 ##############################################################################
4 # OpenERP, Open Source Management Solution
5 # Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
7 # This program is free software: you can redistribute it and/or modify
8 # it under the terms of the GNU Affero General Public License as
9 # published by the Free Software Foundation, either version 3 of the
10 # License, or (at your option) any later version.
12 # This program is distributed in the hope that it will be useful,
13 # but WITHOUT ANY WARRANTY; without even the implied warranty of
14 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 # GNU Affero General Public License for more details.
17 # You should have received a copy of the GNU Affero General Public License
18 # along with this program. If not, see <http://www.gnu.org/licenses/>.
20 ##############################################################################
24 from osv import fields, osv
25 from tools import config
26 import decimal_precision as dp
27 from tools.translate import _
30 ('none', 'Non Member'),
31 ('canceled', 'Cancelled Member'),
32 ('old', 'Old Member'),
33 ('waiting', 'Waiting Member'),
34 ('invoiced', 'Invoiced Member'),
35 ('free', 'Free Member'),
36 ('paid', 'Paid Member'),
49 class membership_line(osv.osv):
52 def _get_membership_lines(self, cr, uid, ids, context=None):
53 list_membership_line = []
54 member_line_obj = self.pool.get('membership.membership_line')
55 for invoice in self.pool.get('account.invoice').browse(cr, uid, ids, context=context):
56 if invoice.invoice_line:
57 list_membership_line += member_line_obj.search(cr, uid, [('account_invoice_line', 'in', [ l.id for l in invoice.invoice_line])], context=context)
58 return list_membership_line
60 def _check_membership_date(self, cr, uid, ids, context=None):
61 """Check if membership product is not in the past
62 @param self: The object pointer
63 @param cr: the current row, from the database cursor,
64 @param uid: the current user’s ID for security checks,
65 @param ids: List of Membership Line IDs
66 @param context: A standard dictionary for contextual values
70 SELECT MIN(ml.date_to - ai.date_invoice)
71 FROM membership_membership_line ml
72 JOIN account_invoice_line ail ON (
73 ml.account_invoice_line = ail.id
75 JOIN account_invoice ai ON (
76 ai.id = ail.invoice_id)
77 WHERE ml.id IN %s''',(tuple(ids),))
84 def _state(self, cr, uid, ids, name, args, context=None):
85 """Compute the state lines
86 @param self: The object pointer
87 @param cr: the current row, from the database cursor,
88 @param uid: the current user’s ID for security checks,
89 @param ids: List of Membership Line IDs
90 @param name: Field Name
91 @param context: A standard dictionary for contextual values
92 @param return: Dictionary of state Value
95 inv_obj = self.pool.get('account.invoice')
96 for line in self.browse(cr, uid, ids, context=context):
98 SELECT i.state, i.id FROM
102 SELECT l.invoice_id FROM
103 account_invoice_line l WHERE
105 SELECT ml.account_invoice_line FROM
106 membership_membership_line ml WHERE
111 fetched = cr.fetchone()
113 res[line.id] = 'canceled'
117 if (istate == 'draft') | (istate == 'proforma'):
119 elif istate == 'open':
121 elif istate == 'paid':
123 inv = inv_obj.browse(cr, uid, fetched[1], context=context)
124 for payment in inv.payment_ids:
125 if payment.invoice and payment.invoice.type == 'out_refund':
127 elif istate == 'cancel':
133 _description = __doc__
134 _name = 'membership.membership_line'
136 'partner': fields.many2one('res.partner', 'Partner', ondelete='cascade', select=1),
137 'membership_id': fields.many2one('product.product', string="Membership Product", required=True),
138 'date_from': fields.date('From', readonly=True),
139 'date_to': fields.date('To', readonly=True),
140 'date_cancel': fields.date('Cancel date'),
141 'date': fields.date('Join Date'),
142 'member_price':fields.float('Member Price', digits_compute= dp.get_precision('Sale Price'), required=True),
143 'account_invoice_line': fields.many2one('account.invoice.line', 'Account Invoice line', readonly=True),
144 'account_invoice_id': fields.related('account_invoice_line', 'invoice_id', type='many2one', relation='account.invoice', string='Invoice', readonly=True),
145 'state': fields.function(
147 string='Membership State', type='selection',
148 selection=STATE, store = {
149 'account.invoice':(_get_membership_lines, ['state'], 10),
151 'company_id': fields.related('account_invoice_line', 'invoice_id', 'company_id', type="many2one", relation="res.company", string="Company", readonly=True, store=True)
153 _rec_name = 'partner'
156 (_check_membership_date, 'Error, this membership product is out of date', [])
162 class Partner(osv.osv):
164 _inherit = 'res.partner'
166 def _get_partner_id(self, cr, uid, ids, context=None):
167 member_line_obj = self.pool.get('membership.membership_line')
168 res_obj = self.pool.get('res.partner')
169 data_inv = member_line_obj.browse(cr, uid, ids, context=context)
171 for data in data_inv:
172 list_partner.append(data.partner.id)
175 ids2 = res_obj.search(cr, uid, [('associate_member','in',ids2)], context=context)
179 def _get_invoice_partner(self, cr, uid, ids, context=None):
180 inv_obj = self.pool.get('account.invoice')
181 res_obj = self.pool.get('res.partner')
182 data_inv = inv_obj.browse(cr, uid, ids, context=context)
184 for data in data_inv:
185 list_partner.append(data.partner_id.id)
188 ids2 = res_obj.search(cr, uid, [('associate_member','in',ids2)], context=context)
192 def _membership_state(self, cr, uid, ids, name, args, context=None):
193 """This Function return Membership State For Given Partner.
194 @param self: The object pointer
195 @param cr: the current row, from the database cursor,
196 @param uid: the current user’s ID for security checks,
197 @param ids: List of Partner IDs
198 @param name: Field Name
199 @param context: A standard dictionary for contextual values
200 @param return: Dictionary of Membership state Value
205 today = time.strftime('%Y-%m-%d')
207 partner_data = self.browse(cr, uid, id, context=context)
208 if partner_data.membership_cancel and today > partner_data.membership_cancel:
211 if partner_data.membership_stop and today > partner_data.membership_stop:
215 if partner_data.member_lines:
216 for mline in partner_data.member_lines:
217 if mline.date_to >= today:
218 if mline.account_invoice_line and mline.account_invoice_line.invoice_id:
219 mstate = mline.account_invoice_line.invoice_id.state
222 inv = mline.account_invoice_line.invoice_id
223 for payment in inv.payment_ids:
224 if payment.invoice.type == 'out_refund':
227 elif mstate == 'open' and s!=0:
229 elif mstate == 'cancel' and s!=0 and s!=1:
231 elif (mstate == 'draft' or mstate == 'proforma') and s!=0 and s!=1:
234 for mline in partner_data.member_lines:
235 if mline.date_from < today and mline.date_to < today and mline.date_from<=mline.date_to and (mline.account_invoice_line and mline.account_invoice_line.invoice_id.state) == 'paid':
251 if partner_data.free_member and s!=0:
253 if partner_data.associate_member:
254 res_state = self._membership_state(cr, uid, [partner_data.associate_member.id], name, args, context=context)
255 res[id] = res_state[partner_data.associate_member.id]
258 def _membership_date(self, cr, uid, ids, name, args, context=None):
259 """Return date of membership"""
262 member_line_obj = self.pool.get('membership.membership_line')
263 for partner in self.browse(cr, uid, ids, context=context):
264 if partner.associate_member:
265 partner_id = partner.associate_member.id
267 partner_id = partner.id
269 'membership_start': False,
270 'membership_stop': False,
271 'membership_cancel': False
273 if name == 'membership_start':
274 line_id = member_line_obj.search(cr, uid, [('partner', '=', partner_id)],
275 limit=1, order='date_from', context=context)
277 res[partner.id]['membership_start'] = member_line_obj.read(cr, uid, line_id[0],
278 ['date_from'], context=context)['date_from']
280 if name == 'membership_stop':
281 line_id1 = member_line_obj.search(cr, uid, [('partner', '=', partner_id)],
282 limit=1, order='date_to desc', context=context)
284 res[partner.id]['membership_stop'] = member_line_obj.read(cr, uid, line_id1[0],
285 ['date_to'], context=context)['date_to']
287 if name == 'membership_cancel':
288 if partner.membership_state == 'canceled':
289 line_id2 = member_line_obj.search(cr, uid, [('partner', '=', partner.id)], limit=1, order='date_cancel', context=context)
291 res[partner.id]['membership_cancel'] = member_line_obj.read(cr, uid, line_id2[0], ['date_cancel'], context=context)['date_cancel']
294 def _get_partners(self, cr, uid, ids, context=None):
297 ids2 = self.search(cr, uid, [('associate_member','in',ids2)], context=context)
301 def __get_membership_state(self, *args, **kwargs):
302 return self._membership_state(*args, **kwargs)
305 'associate_member': fields.many2one('res.partner', 'Associate member'),
306 'member_lines': fields.one2many('membership.membership_line', 'partner', 'Membership'),
307 'free_member': fields.boolean('Free member'),
308 'membership_amount': fields.float(
309 'Membership amount', digits=(16, 2),
310 help = 'The price negociated by the partner'),
311 'membership_state': fields.function(
312 __get_membership_state, method = True,
313 string = 'Current Membership State', type = 'selection',
316 'account.invoice': (_get_invoice_partner, ['state'], 10),
317 'membership.membership_line': (_get_partner_id, ['state'], 10),
318 'res.partner': (_get_partners, ['free_member', 'membership_state', 'associate_member'], 10)
320 'membership_start': fields.function(
321 _membership_date, method = True, multi = 'membeship_start',
322 string = 'Start membership date', type = 'date',
324 'account.invoice': (_get_invoice_partner, ['state'], 10),
325 'membership.membership_line': (_get_partner_id, ['state'], 10, ),
326 'res.partner': (lambda self, cr, uid, ids, c={}: ids, ['free_member'], 10)
328 'membership_stop': fields.function(
329 _membership_date, method = True,
330 string = 'Stop membership date', type = 'date', multi='membership_stop',
332 'account.invoice': (_get_invoice_partner, ['state'], 10),
333 'membership.membership_line': (_get_partner_id, ['state'], 10),
334 'res.partner': (lambda self,cr,uid,ids,c={}:ids, ['free_member'], 10)
336 'membership_cancel': fields.function(
337 _membership_date, method = True,
338 string = 'Cancel membership date', type='date', multi='membership_cancel',
340 'account.invoice': (_get_invoice_partner, ['state'], 11),
341 'membership.membership_line': (_get_partner_id, ['state'], 10),
342 'res.partner': (lambda self,cr,uid,ids,c={}:ids, ['free_member'], 10)
346 'free_member': False,
347 'membership_cancel': False,
350 def _check_recursion(self, cr, uid, ids):
351 """Check Recursive for Associated Members.
355 cr.execute('SELECT DISTINCT associate_member FROM res_partner WHERE id IN %s', (tuple(ids),))
356 ids = filter(None, map(lambda x:x[0], cr.fetchall()))
363 (_check_recursion, 'Error ! You can not create recursive associated members.', ['associate_member'])
366 def copy(self, cr, uid, id, default=None, context=None):
371 default = default.copy()
372 default['member_lines'] = []
373 return super(Partner, self).copy(cr, uid, id, default, context=context)
375 def create_membership_invoice(self, cr, uid, ids, product_id=None, datas=None, context=None):
376 """ Create Customer Invoice of Membership for partners.
377 @param datas: datas has dictionary value which consist Id of Membership product and Cost Amount of Membership.
378 datas = {'membership_product_id': None, 'amount':None}
380 invoice_obj = self.pool.get('account.invoice')
381 invoice_line_obj = self.pool.get('account.invoice.line')
382 invoice_tax_obj = self.pool.get('account.invoice.tax')
383 product_id = product_id or datas.get('membership_product_id',False)
384 amount = datas.get('amount', 0.0)
388 if type(ids) in (int,long,):
390 for partner in self.browse(cr, uid, ids, context=context):
391 account_id = partner.property_account_receivable and partner.property_account_receivable.id or False
392 fpos_id = partner.property_account_position and partner.property_account_position.id or False
393 addr = self.address_get(cr, uid, [partner.id], ['invoice'])
394 if partner.free_member:
395 raise osv.except_osv(_('Error !'),
396 _("Partner is a free Member."))
397 if not addr.get('invoice', False):
398 raise osv.except_osv(_('Error !'),
399 _("Partner doesn't have an address to make the invoice."))
402 'product_id': product_id,
405 line_dict = invoice_line_obj.product_id_change(cr, uid, {},
406 product_id, False, quantity, '', 'out_invoice', partner.id, fpos_id, price_unit=amount, context=context)
407 line_value.update(line_dict['value'])
408 if line_value.get('invoice_line_tax_id', False):
409 tax_tab = [(6, 0, line_value['invoice_line_tax_id'])]
410 line_value['invoice_line_tax_id'] = tax_tab
412 invoice_id = invoice_obj.create(cr, uid, {
413 'partner_id': partner.id,
414 'address_invoice_id': addr.get('invoice', False),
415 'account_id': account_id,
416 'fiscal_position': fpos_id or False
418 line_value['invoice_id'] = invoice_id
419 invoice_line_id = invoice_line_obj.create(cr, uid, line_value, context=context)
420 invoice_obj.write(cr, uid, invoice_id, {'invoice_line': [(6, 0, [invoice_line_id])]}, context=context)
421 invoice_list.append(invoice_id)
422 if line_value['invoice_line_tax_id']:
423 tax_value = invoice_tax_obj.compute(cr, uid, invoice_id).values()
424 for tax in tax_value:
425 invoice_tax_obj.create(cr, uid, tax, context=context)
426 #recompute the membership_state of those partners
427 self.pool.get('res.partner').write(cr, uid, ids, {})
432 class product_template(osv.osv):
433 _inherit = 'product.template'
435 'member_price':fields.float('Member Price', digits_compute= dp.get_precision('Sale Price')),
439 class Product(osv.osv):
441 def fields_view_get(self, cr, user, view_id=None, view_type='form', context=None, toolbar=False, submenu=False):
442 model_obj = self.pool.get('ir.model.data')
444 if ('product' in context) and (context['product']=='membership_product'):
445 model_data_ids_form = model_obj.search(cr, user, [('model','=','ir.ui.view'), ('name','in',['membership_products_form', 'membership_products_tree'])], context=context)
446 resource_id_form = model_obj.read(cr, user, model_data_ids_form, fields=['res_id', 'name'], context=context)
448 for i in resource_id_form:
449 dict_model[i['name']] = i['res_id']
450 if view_type == 'form':
451 view_id = dict_model['membership_products_form']
453 view_id = dict_model['membership_products_tree']
454 return super(Product,self).fields_view_get(cr, user, view_id, view_type, context, toolbar, submenu)
457 _inherit = 'product.product'
459 'membership': fields.boolean('Membership', help='Specify if this product is a membership product'),
460 'membership_date_from': fields.date('Date from', help='Active Membership since this date'),
461 'membership_date_to': fields.date('Date to', help='Expired date of Membership'),
470 class Invoice(osv.osv):
472 _inherit = 'account.invoice'
474 def action_cancel(self, cr, uid, ids, *args):
475 '''Create a 'date_cancel' on the membership_line object'''
476 member_line_obj = self.pool.get('membership.membership_line')
477 today = time.strftime('%Y-%m-%d')
478 for invoice in self.browse(cr, uid, ids):
479 mlines = member_line_obj.search(cr, uid,
480 [('account_invoice_line','in',
481 [ l.id for l in invoice.invoice_line])])
482 member_line_obj.write(cr, uid, mlines, {'date_cancel': today})
483 return super(Invoice, self).action_cancel(cr, uid, ids)
486 class account_invoice_line(osv.osv):
487 _inherit='account.invoice.line'
489 def write(self, cr, uid, ids, vals, context=None):
490 """Overrides orm write method
494 res = super(account_invoice_line, self).write(cr, uid, ids, vals, context=context)
495 member_line_obj = self.pool.get('membership.membership_line')
496 for line in self.browse(cr, uid, ids, context=context):
497 if line.invoice_id.type == 'out_invoice':
498 ml_ids = member_line_obj.search(cr, uid, [('account_invoice_line','=',line.id)], context=context)
499 if line.product_id and line.product_id.membership and not ml_ids:
500 # Product line has changed to a membership product
501 date_from = line.product_id.membership_date_from
502 date_to = line.product_id.membership_date_to
503 if line.invoice_id.date_invoice > date_from and line.invoice_id.date_invoice < date_to:
504 date_from = line.invoice_id.date_invoice
505 line_id = member_line_obj.create(cr, uid, {
506 'partner': line.invoice_id.partner_id.id,
507 'membership_id': line.product_id.id,
508 'member_price': line.price_unit,
509 'date': time.strftime('%Y-%m-%d'),
510 'date_from': date_from,
512 'account_invoice_line': line.id,
514 if line.product_id and not line.product_id.membership and ml_ids:
515 # Product line has changed to a non membership product
516 member_line_obj.unlink(cr, uid, ml_ids, context=context)
519 def unlink(self, cr, uid, ids, context=None):
520 """Remove Membership Line Record for Account Invoice Line
524 member_line_obj = self.pool.get('membership.membership_line')
526 ml_ids = member_line_obj.search(cr, uid, [('account_invoice_line','=',id)], context=context)
527 member_line_obj.unlink(cr, uid, ml_ids, context=context)
528 return super(account_invoice_line, self).unlink(cr, uid, ids, context=context)
530 def create(self, cr, uid, vals, context=None):
531 """Overrides orm create method
533 result = super(account_invoice_line, self).create(cr, uid, vals, context=context)
534 line = self.browse(cr, uid, result, context=context)
535 member_line_obj = self.pool.get('membership.membership_line')
536 if line.invoice_id.type == 'out_invoice':
537 ml_ids = member_line_obj.search(cr, uid, [('account_invoice_line','=',line.id)], context=context)
538 if line.product_id and line.product_id.membership and not ml_ids:
539 # Product line is a membership product
540 date_from = line.product_id.membership_date_from
541 date_to = line.product_id.membership_date_to
542 if line.invoice_id.date_invoice > date_from and line.invoice_id.date_invoice < date_to:
543 date_from = line.invoice_id.date_invoice
544 line_id = member_line_obj.create(cr, uid, {
545 'partner': line.invoice_id.partner_id and line.invoice_id.partner_id.id or False,
546 'membership_id': line.product_id.id,
547 'member_price': line.price_unit,
548 'date': time.strftime('%Y-%m-%d'),
549 'date_from': date_from,
551 'account_invoice_line': line.id,
555 account_invoice_line()
556 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: