2016-06-27 12:38:08 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2017-05-01 17:49:21 +02:00
|
|
|
# Copyright 2017 Fabien Bourgeois <fabien@yaltik.com>
|
2016-06-27 12:38:08 +02:00
|
|
|
#
|
|
|
|
# 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/>.
|
|
|
|
|
2017-05-01 17:49:21 +02:00
|
|
|
""" GOLEM activities related models """
|
|
|
|
|
2018-03-26 19:03:15 +02:00
|
|
|
import collections
|
|
|
|
from datetime import datetime, timedelta
|
|
|
|
from dateutil import parser
|
|
|
|
#from dateutil import rrule
|
|
|
|
from dateutil.rrule import WEEKLY, rrule
|
|
|
|
from dateutil.relativedelta import relativedelta
|
|
|
|
|
2017-04-29 09:16:41 +02:00
|
|
|
from odoo import models, fields, api, _
|
2018-03-26 01:48:28 +02:00
|
|
|
from datetime import timedelta
|
2018-03-26 19:03:15 +02:00
|
|
|
from odoo.tools import DEFAULT_SERVER_DATE_FORMAT, DEFAULT_SERVER_DATETIME_FORMAT
|
|
|
|
#from dateutil import rrule
|
|
|
|
#from dateutil.rrule import rrule
|
|
|
|
#from dateutil.rrule import rrule
|
2016-06-27 12:38:08 +02:00
|
|
|
|
2017-06-05 11:22:34 +02:00
|
|
|
class GolemActivityType(models.Model):
|
|
|
|
""" GOLEM Activity Type """
|
|
|
|
_name = 'golem.activity.type'
|
|
|
|
_description = 'GOLEM Activity Type'
|
|
|
|
|
|
|
|
_sql_constraints = [('golem_activity_type_name_uniq', 'UNIQUE (name)',
|
|
|
|
_('This activity type name has already been used.'))]
|
|
|
|
|
|
|
|
name = fields.Char('Activity type', required=True, translate=True)
|
|
|
|
is_recurrent = fields.Boolean('Is recurrent?')
|
2016-06-27 12:38:08 +02:00
|
|
|
|
2018-02-01 16:00:42 +01:00
|
|
|
|
2016-06-27 12:38:08 +02:00
|
|
|
class GolemActivity(models.Model):
|
2017-05-01 17:49:21 +02:00
|
|
|
""" GOLEM Activity """
|
2016-06-27 12:38:08 +02:00
|
|
|
_name = 'golem.activity'
|
|
|
|
_description = 'GOLEM Activity'
|
|
|
|
_inherit = 'mail.thread'
|
2016-10-10 16:25:55 +02:00
|
|
|
_inherits = {'product.template': 'product_id'}
|
|
|
|
_rec_name = 'full_name'
|
2016-07-08 21:49:21 +02:00
|
|
|
|
2016-10-10 16:25:55 +02:00
|
|
|
product_id = fields.Many2one('product.template', required=True,
|
|
|
|
ondelete='cascade')
|
|
|
|
|
|
|
|
image = fields.Binary(help='This field holds the image used as image for '
|
|
|
|
'the activity.')
|
|
|
|
|
|
|
|
full_name = fields.Char('Name', compute='_compute_full_name', store=True,
|
|
|
|
index=True)
|
2018-02-01 17:02:39 +01:00
|
|
|
is_fullseason = fields.Boolean('Is full season?',
|
2018-02-01 17:16:19 +01:00
|
|
|
compute='_compute_is_full_season')
|
2018-02-01 17:02:39 +01:00
|
|
|
|
2018-03-08 11:06:53 +01:00
|
|
|
@api.onchange('is_fullseason')
|
|
|
|
def onchange_fullseason(self):
|
|
|
|
""" Sets dates as season ones if needed """
|
|
|
|
for activity in self:
|
|
|
|
if activity.is_fullseason:
|
|
|
|
if activity.season_id.date_start:
|
|
|
|
activity.date_start = activity.season_id.date_start
|
|
|
|
if activity.season_id.date_end:
|
|
|
|
activity.date_stop = activity.season_id.date_end
|
|
|
|
|
2018-02-01 17:02:39 +01:00
|
|
|
@api.depends('date_start', 'date_stop')
|
|
|
|
def _compute_is_full_season(self):
|
|
|
|
""" Display date for is full season """
|
|
|
|
for activity in self:
|
|
|
|
if activity.date_start == activity.season_id.date_start and \
|
|
|
|
activity.date_stop == activity.season_id.date_end:
|
2018-02-01 17:16:19 +01:00
|
|
|
activity.is_fullseason = True
|
2016-10-10 16:25:55 +02:00
|
|
|
|
|
|
|
@api.depends('name', 'default_code')
|
|
|
|
def _compute_full_name(self):
|
|
|
|
""" Provide a better displayed name """
|
2017-05-01 17:49:21 +02:00
|
|
|
for activity in self:
|
|
|
|
full_name = unicode(activity.name)
|
|
|
|
if activity.default_code:
|
|
|
|
full_name = u'[{}] {}'.format(activity.default_code, full_name)
|
|
|
|
activity.full_name = full_name
|
2016-10-10 16:25:55 +02:00
|
|
|
|
2016-07-10 07:39:38 +02:00
|
|
|
@api.model
|
|
|
|
def _default_season(self):
|
|
|
|
""" Get default season """
|
|
|
|
domain = [('is_default', '=', True)]
|
|
|
|
return self.env['golem.season'].search(domain)
|
|
|
|
|
|
|
|
season_id = fields.Many2one('golem.season', string='Season', copy=False,
|
2016-07-11 07:25:09 +02:00
|
|
|
required=True, default=_default_season,
|
|
|
|
ondelete='restrict')
|
2016-10-10 16:25:55 +02:00
|
|
|
|
|
|
|
is_current = fields.Boolean('Current season?', store=True, default=False,
|
2017-05-02 01:31:45 +02:00
|
|
|
compute='compute_is_current')
|
2016-10-10 16:25:55 +02:00
|
|
|
|
|
|
|
@api.depends('season_id')
|
2017-05-02 01:31:45 +02:00
|
|
|
def compute_is_current(self):
|
2016-10-10 16:25:55 +02:00
|
|
|
""" Checks if activity is active for current season """
|
|
|
|
default_season = self._default_season()
|
2017-05-01 17:49:21 +02:00
|
|
|
for activity in self:
|
|
|
|
activity.is_current = (default_season == activity.season_id)
|
2016-10-10 16:25:55 +02:00
|
|
|
|
2016-06-28 19:56:27 +02:00
|
|
|
animator_id = fields.Many2one('res.partner', string='Animator',
|
2016-06-27 17:53:10 +02:00
|
|
|
domain=[('is_company', '=', False)])
|
2016-10-10 16:25:55 +02:00
|
|
|
categ_id = fields.Many2one('product.category',
|
2016-07-11 12:19:56 +02:00
|
|
|
help='Select category for the current activity')
|
2017-06-05 11:22:34 +02:00
|
|
|
type_id = fields.Many2one('golem.activity.type', required=True, index=True,
|
|
|
|
string='Type')
|
|
|
|
is_recurrent = fields.Boolean(related='type_id.is_recurrent')
|
2016-07-05 15:39:15 +02:00
|
|
|
date_start = fields.Date('Start date', copy=False)
|
2016-10-10 16:25:55 +02:00
|
|
|
date_stop = fields.Date('End date', copy=False)
|
|
|
|
|
|
|
|
@api.onchange('date_start')
|
2017-05-01 17:49:21 +02:00
|
|
|
def _onchange_date_start(self):
|
2016-10-10 16:25:55 +02:00
|
|
|
""" Sets end date to start date if no start date """
|
2017-05-01 17:49:21 +02:00
|
|
|
for activity in self:
|
|
|
|
if not activity.date_stop:
|
|
|
|
activity.date_stop = activity.date_start
|
2016-06-27 12:38:08 +02:00
|
|
|
|
2016-10-10 16:25:55 +02:00
|
|
|
@api.constrains('date_start', 'date_stop')
|
2016-06-27 12:38:08 +02:00
|
|
|
def _check_period(self):
|
2017-05-01 17:49:21 +02:00
|
|
|
""" Checks if end date if after start date """
|
|
|
|
for activity in self:
|
2018-02-27 18:49:08 +01:00
|
|
|
season = activity.season_id
|
2017-05-01 17:49:21 +02:00
|
|
|
if activity.date_start and activity.date_stop and \
|
|
|
|
activity.date_start > activity.date_stop:
|
2016-06-27 12:38:08 +02:00
|
|
|
raise models.ValidationError(_('Start of the period cannot be '
|
|
|
|
'after end of the period.'))
|
2018-02-27 18:49:08 +01:00
|
|
|
if season.date_start and season.date_start > activity.date_start:
|
2018-02-01 17:26:47 +01:00
|
|
|
msg = _(u'Activity start date can not be set before '
|
|
|
|
'linked season start.')
|
2018-01-29 22:55:36 +01:00
|
|
|
raise models.ValidationError(msg)
|
2018-02-27 18:49:08 +01:00
|
|
|
if season.date_end and season.date_end < activity.date_stop:
|
2018-02-01 17:26:47 +01:00
|
|
|
msg = _(u'Activity end date can not be set after '
|
2018-02-01 17:40:49 +01:00
|
|
|
'linked season end.')
|
2018-01-29 22:55:36 +01:00
|
|
|
raise models.ValidationError(msg)
|
2018-01-29 22:57:03 +01:00
|
|
|
|
2016-06-27 12:38:08 +02:00
|
|
|
@api.onchange('season_id')
|
2017-05-01 17:49:21 +02:00
|
|
|
def _onchange_season_dates(self):
|
2016-06-27 12:38:08 +02:00
|
|
|
""" Sets defaults dates according to season """
|
2017-05-01 17:49:21 +02:00
|
|
|
for activity in self:
|
|
|
|
if activity.season_id:
|
|
|
|
if not activity.date_start:
|
|
|
|
activity.date_start = activity.season_id.date_start
|
|
|
|
if not activity.date_stop:
|
|
|
|
activity.date_stop = activity.season_id.date_end
|
2016-06-27 12:38:08 +02:00
|
|
|
|
2018-03-26 19:03:15 +02:00
|
|
|
weekday = fields.Selection([(0, _('Monday')),
|
|
|
|
(1, _('Tuesday')),
|
|
|
|
(2, _('Wednesday')),
|
|
|
|
(3, _('Thursday')),
|
|
|
|
(4, _('Friday')),
|
|
|
|
(5, _('Saturday')),
|
|
|
|
(6, _('Sunday'))], copy=False)
|
2016-10-10 16:25:55 +02:00
|
|
|
hour_start = fields.Float('Start time', copy=False)
|
2016-10-26 09:39:59 +02:00
|
|
|
hour_stop = fields.Float('Stop time', copy=False)
|
2016-06-27 12:38:08 +02:00
|
|
|
|
2016-10-10 16:25:55 +02:00
|
|
|
@api.onchange('hour_start')
|
2017-05-01 17:49:21 +02:00
|
|
|
def _onchange_hour_start(self):
|
2016-10-10 16:25:55 +02:00
|
|
|
""" Sets end hour to start hour if no start hour """
|
2017-05-01 17:49:21 +02:00
|
|
|
for activity in self:
|
|
|
|
if activity.hour_start and not activity.hour_stop:
|
|
|
|
activity.hour_stop = activity.hour_start + 1
|
2016-10-10 16:25:55 +02:00
|
|
|
|
2016-10-26 09:39:59 +02:00
|
|
|
@api.constrains('hour_start', 'hour_stop')
|
2016-10-10 16:25:55 +02:00
|
|
|
def _check_hour_period(self):
|
|
|
|
""" Check if end hour if after start hour """
|
2017-05-01 17:49:21 +02:00
|
|
|
for activity in self:
|
|
|
|
if activity.hour_start > activity.hour_stop:
|
2016-10-10 16:25:55 +02:00
|
|
|
raise models.ValidationError(_('Start of the period cannot be '
|
|
|
|
'after end of the period.'))
|
|
|
|
|
|
|
|
|
2018-03-15 09:26:53 +01:00
|
|
|
###########################################
|
|
|
|
##########################################
|
|
|
|
@api.multi
|
|
|
|
def read(self, fields=None, load='_classic_read'):
|
2018-04-06 12:55:16 +02:00
|
|
|
print '_________read call _______________'
|
|
|
|
print 'load'
|
|
|
|
print load
|
|
|
|
print self.ids
|
2018-03-15 09:26:53 +01:00
|
|
|
fields2 = fields and fields[:] or None
|
|
|
|
select = map(lambda x: (x, self.calendar_id2real_id(x)), self.ids)
|
2018-04-06 12:55:16 +02:00
|
|
|
print '_____________________<<'
|
2018-03-15 09:26:53 +01:00
|
|
|
print select
|
2018-04-06 12:55:16 +02:00
|
|
|
print '>>_____________________'
|
2018-03-15 09:26:53 +01:00
|
|
|
real_events = self.browse([real_id for calendar_id, real_id in select])
|
2018-04-06 12:55:16 +02:00
|
|
|
#print real_events
|
|
|
|
#print " calendar_id n:1 : "+ str(calendar_id)
|
2018-03-26 01:48:28 +02:00
|
|
|
#verify that attribute of real id is working after next statement
|
2018-03-15 09:26:53 +01:00
|
|
|
real_data = super(GolemActivity, real_events).read(fields=fields2, load=load)
|
|
|
|
real_data = dict((d['id'], d) for d in real_data)
|
2018-04-06 12:55:16 +02:00
|
|
|
print real_data
|
2018-03-20 09:54:34 +01:00
|
|
|
result = []
|
2018-04-06 12:55:16 +02:00
|
|
|
#print " calendar_id n:3 : "+ str(calendar_id)
|
2018-03-15 09:26:53 +01:00
|
|
|
for calendar_id, real_id in select:
|
2018-04-06 12:55:16 +02:00
|
|
|
#print "cal"
|
|
|
|
#print calendar_id
|
2018-03-20 09:54:34 +01:00
|
|
|
res = real_data[real_id].copy()
|
2018-03-26 19:03:15 +02:00
|
|
|
date_start = str(calendar_id).split('_')
|
|
|
|
if len(date_start) == 2:
|
|
|
|
dt = datetime.strptime(date_start[1],DEFAULT_SERVER_DATETIME_FORMAT).date()# "%y-%m-%d %H:%M:%S")
|
|
|
|
res['date_start'] = str(dt)
|
2018-04-06 12:55:16 +02:00
|
|
|
res['date_stop'] = str(dt)
|
|
|
|
#print " calendar_id n:4 : "+ str(calendar_id)
|
2018-03-26 01:48:28 +02:00
|
|
|
res['id'] = calendar_id
|
2018-03-26 19:03:15 +02:00
|
|
|
|
|
|
|
|
|
|
|
#res['date_start'] = date_start[1]
|
2018-03-26 01:48:28 +02:00
|
|
|
result.append(res)
|
2018-04-06 12:55:16 +02:00
|
|
|
#print 'resuuuuuuuuuuuuuult'
|
|
|
|
#print result
|
2018-03-26 01:48:28 +02:00
|
|
|
return result
|
|
|
|
|
|
|
|
"""print res
|
2018-03-20 09:54:34 +01:00
|
|
|
print '_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_'
|
2018-03-15 09:26:53 +01:00
|
|
|
print calendar_id
|
2018-03-20 09:54:34 +01:00
|
|
|
print '_*_*_*_*_*_*_*_*_*_*_*_*_*_*_*_'
|
2018-03-26 01:48:28 +02:00
|
|
|
print real_id"""
|
|
|
|
|
2018-03-20 09:54:34 +01:00
|
|
|
#ls = self.calendar_id2real_id(calendar_id, with_date=res and res.get('duration', 0) > 0 and res.get('duration') or 1)
|
|
|
|
|
2018-03-26 01:48:28 +02:00
|
|
|
|
|
|
|
"""for calendar_id, real_id in select:
|
2018-03-20 09:54:34 +01:00
|
|
|
print calendar_id
|
|
|
|
print real_id"""
|
2018-03-15 09:26:53 +01:00
|
|
|
|
|
|
|
#for id in self.ids:
|
|
|
|
# print self.env['golem.activity'].browse(id)
|
|
|
|
#return super(GolemActivity, self).read(self, fields=None, load='_classic_read')
|
2018-04-06 12:55:16 +02:00
|
|
|
@api.multi
|
|
|
|
def get_metadata(self):
|
|
|
|
real = self.browse(set({x: self.calendar_id2real_id(x) for x in self.ids}.values()))
|
|
|
|
return super(GolemActivity, real).get_metadata()
|
2018-03-15 09:26:53 +01:00
|
|
|
@api.multi
|
|
|
|
def calendar_id2real_id(self, calendar_id=None):
|
|
|
|
if calendar_id and isinstance(calendar_id, (basestring)):
|
2018-03-20 09:54:34 +01:00
|
|
|
res = filter(None, calendar_id.split('_'))
|
2018-03-15 09:26:53 +01:00
|
|
|
if len(res) == 2:
|
|
|
|
real_id = res[0]
|
|
|
|
return int(real_id)
|
|
|
|
return calendar_id and int(calendar_id) or calendar_id
|
|
|
|
|
|
|
|
@api.multi
|
2018-03-26 01:48:28 +02:00
|
|
|
def real_id2calendar_id(self, record_id):
|
2018-03-26 19:03:15 +02:00
|
|
|
""" convert Real Id to Calendar ID """
|
2018-03-26 01:48:28 +02:00
|
|
|
activity = self.env['golem.activity'].browse(record_id)
|
2018-03-26 19:03:15 +02:00
|
|
|
date_start = fields.Date.from_string(activity.date_start)
|
|
|
|
date_stop = fields.Date.from_string(activity.date_stop)
|
|
|
|
weekday = 0#date_start.weekday()
|
|
|
|
if activity.weekday:
|
|
|
|
weekday = activity.weekday
|
2018-04-06 12:55:16 +02:00
|
|
|
#print '_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-_-'
|
|
|
|
#print weekday
|
2018-03-26 19:03:15 +02:00
|
|
|
list_date = list(rrule(freq=WEEKLY, dtstart=date_start, until=date_stop, byweekday=weekday))
|
|
|
|
calendar_ids = []
|
|
|
|
for date in list_date:
|
|
|
|
calendar_ids.append('%s_%s' % (record_id, date))
|
2018-04-06 12:55:16 +02:00
|
|
|
#print '_______________________________________________'
|
|
|
|
#print calendar_ids
|
2018-03-26 19:03:15 +02:00
|
|
|
return calendar_ids
|
|
|
|
|
2018-03-15 09:26:53 +01:00
|
|
|
|
|
|
|
@api.multi
|
|
|
|
def get_recurrent_ids(self):
|
|
|
|
result = []
|
|
|
|
for activity in self:
|
2018-04-06 12:55:16 +02:00
|
|
|
#print '___________________tktk______'
|
2018-03-15 09:26:53 +01:00
|
|
|
#if not meeting.recurrency or not meeting.rrule:
|
2018-03-26 01:48:28 +02:00
|
|
|
if activity.is_recurrent:
|
|
|
|
result += activity.real_id2calendar_id(activity.id)
|
|
|
|
else:
|
|
|
|
result.append(activity.id)
|
|
|
|
|
2018-03-15 09:26:53 +01:00
|
|
|
return result
|
|
|
|
|
|
|
|
@api.model
|
|
|
|
def search(self, args, offset=0, limit=0, order=None, count=False):
|
|
|
|
activities = super(GolemActivity, self).search(args, offset=0, limit=0, order=None, count=False)
|
|
|
|
activities = self.browse(activities.get_recurrent_ids())
|
2018-04-06 12:55:16 +02:00
|
|
|
#print "àààààààààààààààààààààààààààààààààààààààà"
|
|
|
|
#print activities.mapped('id')
|
2018-03-15 09:26:53 +01:00
|
|
|
return activities
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2016-10-10 16:25:55 +02:00
|
|
|
class ProductTemplate(models.Model):
|
2017-05-01 17:49:21 +02:00
|
|
|
""" GOLEM Activity Product adaptations """
|
2016-10-10 16:25:55 +02:00
|
|
|
_inherit = 'product.template'
|
|
|
|
|
|
|
|
type = fields.Selection(default='service')
|
|
|
|
default_code = fields.Char(copy=True)
|
2017-06-20 16:41:22 +02:00
|
|
|
categ_id = fields.Many2one(copy=True)
|