membership : bug fix
[odoo/odoo.git] / addons / membership / membership.py
1 # -*- encoding: utf-8 -*-
2 ##############################################################################
3 #
4 #    OpenERP, Open Source Management Solution
5 #    Copyright (C) 2004-2009 Tiny SPRL (<http://tiny.be>). All Rights Reserved
6 #    $Id$
7 #
8 #    This program is free software: you can redistribute it and/or modify
9 #    it under the terms of the GNU General Public License as published by
10 #    the Free Software Foundation, either version 3 of the License, or
11 #    (at your option) any later version.
12 #
13 #    This program is distributed in the hope that it will be useful,
14 #    but WITHOUT ANY WARRANTY; without even the implied warranty of
15 #    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 #    GNU General Public License for more details.
17 #
18 #    You should have received a copy of the GNU General Public License
19 #    along with this program.  If not, see <http://www.gnu.org/licenses/>.
20 #
21 ##############################################################################
22
23 from osv import fields, osv
24 from tools import config
25 import time
26
27 STATE = [
28     ('none', 'Non Member'),
29     ('canceled', 'Canceled Member'),
30     ('old', 'Old Member'),
31     ('waiting', 'Waiting Member'),
32     ('invoiced', 'Invoiced Member'),
33     ('free', 'Free Member'),
34     ('paid', 'Paid Member'),
35 ]
36
37 STATE_PRIOR = {
38         'none' : 0,
39         'canceled' : 1,
40         'old' : 2,
41         'waiting' : 3,
42         'invoiced' : 4,
43         'free' : 6,
44         'paid' : 7
45         }
46
47
48
49 #~ REQUETE = '''SELECT partner, state FROM (
50 #~ SELECT members.partner AS partner,
51 #~ CASE WHEN MAX(members.state) = 0 THEN 'none'
52 #~ ELSE CASE WHEN MAX(members.state) = 1 THEN 'canceled'
53 #~ ELSE CASE WHEN MAX(members.state) = 2 THEN 'old'
54 #~ ELSE CASE WHEN MAX(members.state) = 3 THEN 'waiting'
55 #~ ELSE CASE WHEN MAX(members.state) = 4 THEN 'invoiced'
56 #~ ELSE CASE WHEN MAX(members.state) = 6 THEN 'free'
57 #~ ELSE CASE WHEN MAX(members.state) = 7 THEN 'paid'
58 #~ END END END END END END END END
59     #~ AS state FROM (
60 #~ SELECT partner,
61     #~ CASE WHEN MAX(inv_digit.state) = 4 THEN 7
62     #~ ELSE CASE WHEN MAX(inv_digit.state) = 3 THEN 4
63     #~ ELSE CASE WHEN MAX(inv_digit.state) = 2 THEN 3
64     #~ ELSE CASE WHEN MAX(inv_digit.state) = 1 THEN 1
65 #~ END END END END
66 #~ AS state
67 #~ FROM (
68     #~ SELECT p.id as partner,
69     #~ CASE WHEN ai.state = 'paid' THEN 4
70     #~ ELSE CASE WHEN ai.state = 'open' THEN 3
71     #~ ELSE CASE WHEN ai.state = 'proforma' THEN 2
72     #~ ELSE CASE WHEN ai.state = 'draft' THEN 2
73     #~ ELSE CASE WHEN ai.state = 'cancel' THEN 1
74 #~ END END END END END
75 #~ AS state
76 #~ FROM res_partner p
77 #~ JOIN account_invoice ai ON (
78     #~ p.id = ai.partner_id
79 #~ )
80 #~ JOIN account_invoice_line ail ON (
81     #~ ail.invoice_id = ai.id
82 #~ )
83 #~ JOIN membership_membership_line ml ON (
84     #~ ml.account_invoice_line  = ail.id
85 #~ )
86 #~ WHERE ml.date_from <= '%s'
87 #~ AND ml.date_to >= '%s'
88 #~ GROUP BY
89 #~ p.id,
90 #~ ai.state
91     #~ )
92     #~ AS inv_digit
93     #~ GROUP by partner
94 #~ UNION
95 #~ SELECT p.id AS partner,
96     #~ CASE WHEN  p.free_member THEN 6
97     #~ ELSE CASE WHEN p.associate_member IN (
98         #~ SELECT ai.partner_id FROM account_invoice ai JOIN
99         #~ account_invoice_line ail ON (ail.invoice_id = ai.id AND ai.state = 'paid')
100         #~ JOIN membership_membership_line ml ON (ml.account_invoice_line = ail.id)
101         #~ WHERE ml.date_from <= '%s'
102         #~ AND ml.date_to >= '%s'
103     #~ )
104     #~ THEN 5
105 #~ END END
106 #~ AS state
107 #~ FROM res_partner p
108 #~ WHERE p.free_member
109 #~ OR p.associate_member > 0
110 #~ UNION
111 #~ SELECT p.id as partner,
112     #~ MAX(CASE WHEN ai.state = 'paid' THEN 2
113     #~ ELSE 0
114     #~ END)
115 #~ AS state
116 #~ FROM res_partner p
117 #~ JOIN account_invoice ai ON (
118     #~ p.id = ai.partner_id
119 #~ )
120 #~ JOIN account_invoice_line ail ON (
121     #~ ail.invoice_id = ai.id
122 #~ )
123 #~ JOIN membership_membership_line ml ON (
124     #~ ml.account_invoice_line  = ail.id
125 #~ )
126 #~ WHERE ml.date_from < '%s'
127 #~ AND ml.date_to < '%s'
128 #~ AND ml.date_from <= ml.date_to
129 #~ GROUP BY
130 #~ p.id
131 #~ )
132 #~ AS members
133 #~ GROUP BY members.partner
134 #~ )
135 #~ AS final
136 #~ %s
137 #~ '''
138
139
140 class membership_line(osv.osv):
141     '''Member line'''
142
143     def _check_membership_date(self, cr, uid, ids, context=None):
144         '''Check if membership product is not in the past'''
145
146         cr.execute('''
147          SELECT MIN(ml.date_to - ai.date_invoice)
148          FROM membership_membership_line ml
149          JOIN account_invoice_line ail ON (
150             ml.account_invoice_line = ail.id
151             )
152         JOIN account_invoice ai ON (
153             ai.id = ail.invoice_id)
154         WHERE ml.id in (%s)
155         ''' % ','.join([str(id) for id in ids]))
156
157         res = cr.fetchall()
158         for r in res:
159             if r[0] and r[0] < 0:
160                 return False
161         return True
162
163     def _state(self, cr, uid, ids, name, args, context=None):
164         '''Compute the state lines'''
165         res = {}
166         for line in self.browse(cr, uid, ids):
167             cr.execute('''
168             SELECT i.state FROM
169             account_invoice i WHERE
170             i.id = (
171                 SELECT l.invoice_id FROM
172                 account_invoice_line l WHERE
173                 l.id = (
174                     SELECT  ml.account_invoice_line FROM
175                     membership_membership_line ml WHERE
176                     ml.id = %s
177                     )
178                 )
179             ''', (line.id,))
180             fetched = cr.fetchone()
181             if not fetched :
182                 res[line.id] = 'canceled'
183                 continue
184             istate = fetched[0]
185             state = 'none'
186             if (istate == 'draft') | (istate == 'proforma'):
187                 state = 'waiting'
188             elif istate == 'open':
189                 state = 'invoiced'
190             elif istate == 'paid':
191                 state = 'paid'
192             elif istate == 'cancel':
193                 state = 'canceled'
194             res[line.id] = state
195         return res
196
197
198     _description = __doc__
199     _name = 'membership.membership_line'
200     _columns = {
201             'partner': fields.many2one('res.partner', 'Partner', ondelete='cascade', select=1),
202             'date_from': fields.date('From'),
203             'date_to': fields.date('To'),
204             'date_cancel' : fields.date('Cancel date'),
205             'account_invoice_line': fields.many2one('account.invoice.line', 'Account Invoice line'),
206             'state': fields.function(_state, method=True, string='State', type='selection', selection=STATE),
207             }
208     _rec_name = 'partner'
209     _order = 'id desc'
210     _constraints = [
211             (_check_membership_date, 'Error, this membership product is out of date', [])
212     ]
213
214 membership_line()
215
216
217 class Partner(osv.osv):
218     '''Partner'''
219     _inherit = 'res.partner'
220
221     def _get_partner_id(self, cr, uid, ids, context=None):
222         data_inv = self.pool.get('membership.membership_line').browse(cr, uid, ids, context)
223         list_partner = []
224         for data in data_inv:
225             list_partner.append(data.partner.id)
226         ids2 = list_partner
227         while ids2:
228             ids2 = self.pool.get('res.partner').search(cr, uid, [('associate_member','in',ids2)], context=context)
229             list_partner += ids2
230         return list_partner
231
232     def _get_invoice_partner(self, cr, uid, ids, context=None):
233         data_inv = self.pool.get('account.invoice').browse(cr, uid, ids, context)
234         list_partner = []
235         for data in data_inv:
236             list_partner.append(data.partner_id.id)
237         ids2 = list_partner
238         while ids2:
239             ids2 = self.pool.get('res.partner').search(cr, uid, [('associate_member','in',ids2)], context=context)
240             list_partner += ids2
241         return list_partner
242
243     def _membership_state(self, cr, uid, ids, name, args, context=None):
244         res = {}
245         for id in ids:
246             res[id] = 'none'
247         today = time.strftime('%Y-%m-%d')
248         for id in ids:
249             partner_data = self.browse(cr,uid,id)
250             if partner_data.membership_cancel and today > partner_data.membership_cancel:
251                 res[id] = 'canceled'
252                 continue
253             if partner_data.membership_stop and today > partner_data.membership_stop:
254                 res[id] = 'old'
255                 continue
256             s = 4
257             if partner_data.member_lines:
258                 for mline in partner_data.member_lines:
259                     if mline.date_from <= today and mline.date_to >= today:
260                         if mline.account_invoice_line and mline.account_invoice_line.invoice_id:
261                             mstate = mline.account_invoice_line.invoice_id.state
262                             if mstate == 'paid':
263                                 s = 0
264                                 break
265                             elif mstate == 'open' and s!=0:
266                                 s = 1
267                             elif mstate == 'cancel' and s!=0 and s!=1:
268                                 s = 2
269                             elif  (mstate == 'draft' or mstate == 'proforma') and s!=0 and s!=1:
270                                 s = 3
271                 if s==4:
272                     for mline in partner_data.member_lines:
273                         if mline.date_from < today and mline.date_to < today and mline.date_from<=mline.date_to and mline.account_invoice_line.invoice_id.state == 'paid':
274                             s = 5
275                         else:
276                             s = 6
277                 if s==0:
278                     res[id] = 'paid'
279                 elif s==1:
280                     res[id] = 'invoiced'
281                 elif s==2:
282                     res[id] = 'canceled'
283                 elif s==3:
284                     res[id] = 'waiting'
285                 elif s==5:
286                     res[id] = 'old'
287                 elif s==6:
288                     res[id] = 'none'
289             if partner_data.free_member and s!=0:
290                 res[id] = 'free'
291             if partner_data.associate_member:
292                 associate_partners_list = []
293                 query="SELECT DISTINCT associate_member FROM res_partner"
294                 cr.execute(query)
295                 for p in cr.fetchall():
296                     if p[0] and p[0] != partner_data.id:
297                         associate_partners_list.append(p[0])
298                 if associate_partners_list != []:
299                     res_state = self._membership_state(cr, uid, associate_partners_list, name, args, context)
300                 res[id] = res_state[partner_data.associate_member.id] #partner_data.associate_member.membership_state
301         return res
302
303     def _membership_start(self, cr, uid, ids, name, args, context=None):
304         '''Return the start date of membership'''
305         res = {}
306         member_line_obj = self.pool.get('membership.membership_line')
307         for partner in self.browse(cr, uid, ids):
308             if partner.associate_member:
309                 partner_id = partner.associate_member.id
310             else:
311                 partner_id = partner.id
312             line_id = member_line_obj.search(cr, uid, [('partner', '=', partner_id)],
313                     limit=1, order='date_from')
314             if line_id:
315                 res[partner.id] = member_line_obj.read(cr, uid, line_id[0],
316                         ['date_from'])['date_from']
317             else:
318                 res[partner.id] = False
319         return res
320
321     def _membership_stop(self, cr, uid, ids, name, args, context=None):
322         '''Return the stop date of membership'''
323         res = {}
324         member_line_obj = self.pool.get('membership.membership_line')
325         for partner in self.browse(cr, uid, ids):
326             cr.execute('select membership_state from res_partner where id=%s', (partner.id,))
327             data_state = cr.fetchall()
328             if partner.associate_member:
329                 partner_id = partner.associate_member.id
330             else:
331                 partner_id = partner.id
332             line_id = member_line_obj.search(cr, uid, [('partner', '=', partner_id)],
333                     limit=1, order='date_to desc')
334             if line_id:
335                 res[partner.id] = member_line_obj.read(cr, uid, line_id[0],
336                         ['date_to'])['date_to']
337             else:
338                 res[partner.id] = False
339         return res
340
341     def _membership_cancel(self, cr, uid, ids, name, args, context=None):
342         '''Return the cancel date of membership'''
343         res = {}
344         member_line_obj = self.pool.get('membership.membership_line')
345         for partner_id in ids:
346             line_id = member_line_obj.search(cr, uid, [('partner', '=', partner_id)],
347                     limit=1, order='date_cancel')
348             if line_id:
349                 res[partner_id] = member_line_obj.read(cr, uid, line_id[0],
350                         ['date_cancel'])['date_cancel']
351             else:
352                 res[partner_id] = False
353         return res
354
355     def _get_partners(self, cr, uid, ids, context={}):
356         ids2 = ids
357         while ids2:
358             ids2 = self.search(cr, uid, [('associate_member','in',ids2)], context=context)
359             ids+=ids2
360         return ids
361
362     _columns = {
363         'associate_member': fields.many2one('res.partner', 'Associate member'),
364         'member_lines': fields.one2many('membership.membership_line', 'partner', 'Membership'),
365         'free_member': fields.boolean('Free member'),
366         'membership_amount': fields.float(
367                     'Membership amount', digites=(16, 2),
368                     help='The price negociated by the partner'),
369         'membership_state': fields.function(
370                     _membership_state, method = True,
371                     string = 'Current membership state', type = 'selection',
372                     selection = STATE ,store = {
373                         'account.invoice':(_get_invoice_partner,['state'], 10),
374                         'membership.membership_line':(_get_partner_id,['state'], 10),
375                         'res.partner':(_get_partners, ['free_member','membership_state'], 10)
376                         }
377                     ),
378         'membership_start': fields.function(
379                     _membership_start, method=True,
380                     string = 'Start membership date', type = 'date',
381                     store = {
382                         'account.invoice':(_get_invoice_partner,['state'], 10),
383                         'membership.membership_line':(_get_partner_id,['state'], 10),
384                         'res.partner':(lambda self,cr,uid,ids,c={}:ids, ['free_member'], 10)
385                         }
386                     ),
387         'membership_stop': fields.function(
388                     _membership_stop, method = True,
389                     string = 'Stop membership date', type = 'date',
390                     store = {
391                         'account.invoice':(_get_invoice_partner,['state'], 10),
392                         'membership.membership_line':(_get_partner_id,['state'], 10),
393                         'res.partner':(lambda self,cr,uid,ids,c={}:ids, ['free_member'], 10)
394                         }
395                     ),
396
397         'membership_cancel': fields.function(
398                     _membership_cancel, method = True,
399                     string = 'Cancel membership date', type='date',
400                     store = {
401                         'account.invoice':(_get_invoice_partner,['state'], 10),
402                         'membership.membership_line':(_get_partner_id,['state'], 10),
403                         'res.partner':(lambda self,cr,uid,ids,c={}:ids, ['free_member'], 10)
404                         }
405                     ),
406     }
407     _defaults = {
408         'free_member': lambda *a: False,
409         'membership_cancel' : lambda *d : False,
410     }
411
412     def _check_recursion(self, cr, uid, ids):
413         level = 100
414         while len(ids):
415             cr.execute('select distinct associate_member from res_partner where id in ('+','.join(map(str,ids))+')')
416             ids = filter(None, map(lambda x:x[0], cr.fetchall()))
417             if not level:
418                 return False
419             level -= 1
420         return True
421
422     _constraints = [
423         (_check_recursion, 'Error ! You can not create recursive associated members.', ['associate_member'])
424     ]
425
426 Partner()
427
428 class product_template(osv.osv):
429     _inherit = 'product.template'
430     _columns = {
431             'member_price':fields.float('Member Price', digits=(16, int(config['price_accuracy']))),
432             }
433 product_template()
434
435 class Product(osv.osv):
436
437     def fields_view_get(self, cr, user, view_id=None, view_type='form', context=None, toolbar=False):
438         if ('product' in context) and (context['product']=='membership_product'):
439             model_data_ids_form = self.pool.get('ir.model.data').search(cr,user,[('model','=','ir.ui.view'),('name','in',['membership_products_form','membership_products_tree'])])
440             resource_id_form = self.pool.get('ir.model.data').read(cr,user,model_data_ids_form,fields=['res_id','name'])
441             dict_model={}
442             for i in resource_id_form:
443                 dict_model[i['name']]=i['res_id']
444             if view_type=='form':
445                 view_id = dict_model['membership_products_form']
446             else:
447                 view_id = dict_model['membership_products_tree']
448         return super(Product,self).fields_view_get(cr, user, view_id, view_type, context, toolbar)
449
450     '''Product'''
451     _inherit = 'product.product'
452     _description = 'product.product'
453
454     _columns = {
455             'membership': fields.boolean('Membership', help='Specify if this product is a membership product'),
456             'membership_date_from': fields.date('Date from'),
457             'membership_date_to': fields.date('Date to'),
458 #           'member_price':fields.float('Member Price'),
459             }
460
461     _defaults = {
462             'membership': lambda *args: False
463             }
464 Product()
465
466
467 class Invoice(osv.osv):
468     '''Invoice'''
469
470     _inherit = 'account.invoice'
471
472     def action_cancel(self, cr, uid, ids, context=None):
473         '''Create a 'date_cancel' on the membership_line object'''
474         if context is None:
475             context = {}
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])], context)
482             member_line_obj.write(cr,uid,mlines, {'date_cancel':today}, context)
483         return super(Invoice, self).action_cancel(cr, uid, ids, context)
484 Invoice()
485
486
487 class ReportPartnerMemberYear(osv.osv):
488     '''Membership by Years'''
489
490     _name = 'report.partner_member.year'
491     _description = __doc__
492     _auto = False
493     _rec_name = 'year'
494     _columns = {
495         'year': fields.char('Year', size='4', readonly=True, select=1),
496         'canceled_number': fields.integer('Canceled', readonly=True),
497         'waiting_number': fields.integer('Waiting', readonly=True),
498         'invoiced_number': fields.integer('Invoiced', readonly=True),
499         'paid_number': fields.integer('Paid', readonly=True),
500         'canceled_amount': fields.float('Canceled', digits=(16, 2), readonly=True),
501         'waiting_amount': fields.float('Waiting', digits=(16, 2), readonly=True),
502         'invoiced_amount': fields.float('Invoiced', digits=(16, 2), readonly=True),
503         'paid_amount': fields.float('Paid', digits=(16, 2), readonly=True),
504         'currency': fields.many2one('res.currency', 'Currency', readonly=True,
505             select=2),
506     }
507
508     def init(self, cr):
509         '''Create the view'''
510         cr.execute("""
511     CREATE OR REPLACE VIEW report_partner_member_year AS (
512         SELECT
513         MIN(id) AS id,
514         COUNT(ncanceled) as canceled_number,
515         COUNT(npaid) as paid_number,
516         COUNT(ninvoiced) as invoiced_number,
517         COUNT(nwaiting) as waiting_number,
518         SUM(acanceled) as canceled_amount,
519         SUM(apaid) as paid_amount,
520         SUM(ainvoiced) as invoiced_amount,
521         SUM(awaiting) as waiting_amount,
522         year,
523         currency
524         FROM (SELECT
525             CASE WHEN ai.state = 'cancel' THEN ml.id END AS ncanceled,
526             CASE WHEN ai.state = 'paid' THEN ml.id END AS npaid,
527             CASE WHEN ai.state = 'open' THEN ml.id END AS ninvoiced,
528             CASE WHEN (ai.state = 'draft' OR ai.state = 'proforma')
529                 THEN ml.id END AS nwaiting,
530             CASE WHEN ai.state = 'cancel'
531                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
532             ELSE 0 END AS acanceled,
533             CASE WHEN ai.state = 'paid'
534                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
535             ELSE 0 END AS apaid,
536             CASE WHEN ai.state = 'open'
537                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
538             ELSE 0 END AS ainvoiced,
539             CASE WHEN (ai.state = 'draft' OR ai.state = 'proforma')
540                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
541             ELSE 0 END AS awaiting,
542             TO_CHAR(ml.date_from, 'YYYY') AS year,
543             ai.currency_id AS currency,
544             MIN(ml.id) AS id
545             FROM membership_membership_line ml
546             JOIN (account_invoice_line ail
547                 LEFT JOIN account_invoice ai
548                 ON (ail.invoice_id = ai.id))
549             ON (ml.account_invoice_line = ail.id)
550             JOIN res_partner p
551             ON (ml.partner = p.id)
552             GROUP BY TO_CHAR(ml.date_from, 'YYYY'), ai.state,
553             ai.currency_id, ml.id) AS foo
554         GROUP BY year, currency)
555                 """)
556
557 ReportPartnerMemberYear()
558
559
560 class ReportPartnerMemberYearNew(osv.osv):
561     '''New Membership by Years'''
562
563     _name = 'report.partner_member.year_new'
564     _description = __doc__
565     _auto = False
566     _rec_name = 'year'
567
568     _columns = {
569         'year': fields.char('Year', size='4', readonly=True, select=1),
570         'canceled_number': fields.integer('Canceled', readonly=True),
571         'waiting_number': fields.integer('Waiting', readonly=True),
572         'invoiced_number': fields.integer('Invoiced', readonly=True),
573         'paid_number': fields.integer('Paid', readonly=True),
574         'canceled_amount': fields.float('Canceled', digits=(16, 2), readonly=True),
575         'waiting_amount': fields.float('Waiting', digits=(16, 2), readonly=True),
576         'invoiced_amount': fields.float('Invoiced', digits=(16, 2), readonly=True),
577         'paid_amount': fields.float('Paid', digits=(16, 2), readonly=True),
578         'currency': fields.many2one('res.currency', 'Currency', readonly=True,
579             select=2),
580     }
581
582     def init(self, cr):
583         '''Create the view'''
584         cr.execute("""
585     CREATE OR REPLACE VIEW report_partner_member_year AS (
586         SELECT
587         MIN(id) AS id,
588         COUNT(ncanceled) as canceled_number,
589         COUNT(npaid) as paid_number,
590         COUNT(ninvoiced) as invoiced_number,
591         COUNT(nwaiting) as waiting_number,
592         SUM(acanceled) as canceled_amount,
593         SUM(apaid) as paid_amount,
594         SUM(ainvoiced) as invoiced_amount,
595         SUM(awaiting) as waiting_amount,
596         year,
597         currency
598         FROM (SELECT
599             CASE WHEN ai.state = 'cancel' THEN ml.id END AS ncanceled,
600             CASE WHEN ai.state = 'paid' THEN ml.id END AS npaid,
601             CASE WHEN ai.state = 'open' THEN ml.id END AS ninvoiced,
602             CASE WHEN (ai.state = 'draft' OR ai.state = 'proforma')
603                 THEN ml.id END AS nwaiting,
604             CASE WHEN ai.state = 'cancel'
605                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
606             ELSE 0 END AS acanceled,
607             CASE WHEN ai.state = 'paid'
608                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
609             ELSE 0 END AS apaid,
610             CASE WHEN ai.state = 'open'
611                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
612             ELSE 0 END AS ainvoiced,
613             CASE WHEN (ai.state = 'draft' OR ai.state = 'proforma')
614                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
615             ELSE 0 END AS awaiting,
616             TO_CHAR(ml.date_from, 'YYYY') AS year,
617             ai.currency_id AS currency,
618             MIN(ml.id) AS id
619             FROM membership_membership_line ml
620             JOIN (account_invoice_line ail
621                 LEFT JOIN account_invoice ai
622                 ON (ail.invoice_id = ai.id))
623             ON (ml.account_invoice_line = ail.id)
624             JOIN res_partner p
625             ON (ml.partner = p.id)
626             GROUP BY TO_CHAR(ml.date_from, 'YYYY'), ai.state,
627             ai.currency_id, ml.id) AS foo
628         GROUP BY year, currency)
629                 """)
630
631 ReportPartnerMemberYear()
632
633
634 class ReportPartnerMemberYearNew(osv.osv):
635     '''New Membership by Years'''
636
637     _name = 'report.partner_member.year_new'
638     _description = __doc__
639     _auto = False
640     _rec_name = 'year'
641     _columns = {
642         'year': fields.char('Year', size='4', readonly=True, select=1),
643         'canceled_number': fields.integer('Canceled', readonly=True),
644         'waiting_number': fields.integer('Waiting', readonly=True),
645         'invoiced_number': fields.integer('Invoiced', readonly=True),
646         'paid_number': fields.integer('Paid', readonly=True),
647         'canceled_amount': fields.float('Canceled', digits=(16, 2), readonly=True),
648         'waiting_amount': fields.float('Waiting', digits=(16, 2), readonly=True),
649         'invoiced_amount': fields.float('Invoiced', digits=(16, 2), readonly=True),
650         'paid_amount': fields.float('Paid', digits=(16, 2), readonly=True),
651         'currency': fields.many2one('res.currency', 'Currency', readonly=True,
652             select=2),
653     }
654
655     def init(self, cursor):
656         '''Create the view'''
657         cursor.execute("""
658         CREATE OR REPLACE VIEW report_partner_member_year_new AS (
659         SELECT
660         MIN(id) AS id,
661         COUNT(ncanceled) AS canceled_number,
662         COUNT(npaid) AS paid_number,
663         COUNT(ninvoiced) AS invoiced_number,
664         COUNT(nwaiting) AS waiting_number,
665         SUM(acanceled) AS canceled_amount,
666         SUM(apaid) AS paid_amount,
667         SUM(ainvoiced) AS invoiced_amount,
668         SUM(awaiting) AS waiting_amount,
669         year,
670         currency
671         FROM (SELECT
672             CASE WHEN ai.state = 'cancel' THEN ml2.id END AS ncanceled,
673             CASE WHEN ai.state = 'paid' THEN ml2.id END AS npaid,
674             CASE WHEN ai.state = 'open' THEN ml2.id END AS ninvoiced,
675             CASE WHEN (ai.state = 'draft' OR ai.state = 'proforma')
676                 THEN ml2.id END AS nwaiting,
677             CASE WHEN ai.state = 'cancel'
678                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
679             ELSE 0 END AS acanceled,
680             CASE WHEN ai.state = 'paid'
681                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
682             ELSE 0 END AS apaid,
683             CASE WHEN ai.state = 'open'
684                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
685             ELSE 0 END AS ainvoiced,
686             CASE WHEN (ai.state = 'draft' OR ai.state = 'proforma')
687                 THEN SUM(ail.price_unit * ail.quantity * (1 - ail.discount / 100))
688             ELSE 0 END AS awaiting,
689             TO_CHAR(ml2.date_from, 'YYYY') AS year,
690             ai.currency_id AS currency,
691             MIN(ml2.id) AS id
692             FROM (SELECT
693                     partner AS id,
694                     MIN(date_from) AS date_from
695                     FROM membership_membership_line
696                     GROUP BY partner
697                 ) AS ml1
698                 JOIN membership_membership_line ml2
699                 JOIN (account_invoice_line ail
700                     LEFT JOIN account_invoice ai
701                     ON (ail.invoice_id = ai.id))
702                 ON (ml2.account_invoice_line = ail.id)
703                 ON (ml1.id = ml2.partner AND ml1.date_from = ml2.date_from)
704             JOIN res_partner p
705             ON (ml2.partner = p.id)
706             GROUP BY TO_CHAR(ml2.date_from, 'YYYY'), ai.state,
707             ai.currency_id, ml2.id) AS foo
708         GROUP BY year, currency
709         )
710     """)
711
712 ReportPartnerMemberYearNew()
713 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
714
715
716 class account_invoice_line(osv.osv):
717     _inherit='account.invoice.line'
718     def write(self, cr, uid, ids, vals, context=None):
719         if not context:
720             context={}
721         res = super(account_invoice_line, self).write(cr, uid, ids, vals, context=context)
722         member_line_obj = self.pool.get('membership.membership_line')
723         for line in self.browse(cr, uid, ids):
724             ml_ids = member_line_obj.search(cr, uid, [('account_invoice_line','=',line.id)])
725             if line.product_id and line.product_id.membership and not ml_ids:
726                 # Product line has changed to a membership product
727                 date_from = line.product_id.membership_date_from
728                 date_to = line.product_id.membership_date_to
729                 if line.invoice_id.date_invoice > date_from and line.invoice_id.date_invoice < date_to:
730                     date_from = line.invoice_id.date_invoice
731                 line_id = member_line_obj.create(cr, uid, {
732                     'partner': line.invoice_id.partner_id.id,
733                     'date_from': date_from,
734                     'date_to': date_to,
735                     'account_invoice_line': line.id,
736                     })
737             if line.product_id and not line.product_id.membership and ml_ids:
738                 # Product line has changed to a non membership product
739                 member_line_obj.unlink(cr, uid, ml_ids, context=context)
740         return res
741
742     def unlink(self, cr, uid, ids, context=None):
743         if not context:
744             context={}
745         member_line_obj = self.pool.get('membership.membership_line')
746         for id in ids:
747             ml_ids = member_line_obj.search(cr, uid, [('account_invoice_line','=',id)])
748             member_line_obj.unlink(cr, uid, ml_ids, context=context)
749         return super(account_invoice_line, self).unlink(cr, uid, ids, context=context)
750
751     def create(self, cr, uid, vals, context={}):
752         result = super(account_invoice_line, self).create(cr, uid, vals, context)
753         line = self.browse(cr, uid, result)
754         member_line_obj = self.pool.get('membership.membership_line')
755         ml_ids = member_line_obj.search(cr, uid, [('account_invoice_line','=',line.id)])
756         if line.product_id and line.product_id.membership and not ml_ids:
757             # Product line is a membership product
758             date_from = line.product_id.membership_date_from
759             date_to = line.product_id.membership_date_to
760             if line.invoice_id.date_invoice > date_from and line.invoice_id.date_invoice < date_to:
761                 date_from = line.invoice_id.date_invoice
762             line_id = member_line_obj.create(cr, uid, {
763                 'partner': line.invoice_id.partner_id and line.invoice_id.partner_id.id or False,
764                 'date_from': date_from,
765                 'date_to': date_to,
766                 'account_invoice_line': line.id,
767                 })
768         return result
769 account_invoice_line()