# This file is part of Tryton. The COPYRIGHT file at the top level of # this repository contains the full copyright notices and license terms. import datetime from sql import Literal from sql.aggregate import Max, Sum from sql.functions import Extract from trytond.i18n import gettext from trytond.model import Index, ModelSQL, ModelView, Unique, fields from trytond.pool import Pool from trytond.pyson import Date, Eval, PYSONEncoder, TimeDelta from trytond.transaction import Transaction from trytond.wizard import Button, StateAction, StateView, Wizard from .exceptions import DurationValidationError class Line(ModelSQL, ModelView): 'Timesheet Line' __name__ = 'timesheet.line' company = fields.Many2One('company.company', 'Company', required=True, help="The company on which the time is spent.") employee = fields.Many2One( 'company.employee', "Employee", required=True, search_context={ 'active_test': False, }, domain=[ ('company', '=', Eval('company', -1)), ['OR', ('start_date', '=', None), ('start_date', '<=', Eval('date', None)), ], ['OR', ('end_date', '=', None), ('end_date', '>=', Eval('date', None)), ], ], help="The employee who spends the time.") date = fields.Date( "Date", required=True, help="When the time is spent.") duration = fields.TimeDelta( 'Duration', 'company_work_time', required=True, domain=[ ('duration', '>=', TimeDelta()), ]) work = fields.Many2One( 'timesheet.work', "Work", required=True, domain=[ ('company', '=', Eval('company', -1)), ['OR', ('timesheet_start_date', '=', None), ('timesheet_start_date', '<=', Eval('date', None)), ], ['OR', ('timesheet_end_date', '=', None), ('timesheet_end_date', '>=', Eval('date', None)), ], ], help="The work on which the time is spent.") description = fields.Char('Description', help="Additional description of the work done.") uuid = fields.Char("UUID", readonly=True, strip=False) @classmethod def __setup__(cls): super(Line, cls).__setup__() cls._order = [ ('date', 'DESC'), ('id', 'DESC'), ] t = cls.__table__() cls._sql_constraints = [ ('uuid_unique', Unique(t, t.uuid), 'timesheet.msg_line_uuid_unique'), ] cls._sql_indexes.update({ Index( t, (t.date, Index.Equality()), (t.company, Index.Equality()), (t.employee, Index.Equality())), Index( t, (t.date, Index.Range()), (t.employee, Index.Equality())), Index( t, (Extract('YEAR', t.date), Index.Equality()), (Extract('WEEK', t.date), Index.Equality()), (t.employee, Index.Equality())), Index( t, (Extract('YEAR', t.date), Index.Equality()), (Extract('MONTH', t.date), Index.Equality()), (t.employee, Index.Equality())), }) @staticmethod def default_company(): return Transaction().context.get('company') @staticmethod def default_employee(): User = Pool().get('res.user') employee_id = None if Transaction().context.get('employee'): employee_id = Transaction().context['employee'] else: user = User(Transaction().user) if user.employee: employee_id = user.employee.id if employee_id: return employee_id @staticmethod def default_date(): Date_ = Pool().get('ir.date') return Transaction().context.get('date') or Date_.today() @classmethod def validate_fields(cls, lines, field_names): super().validate_fields(lines, field_names) cls.check_duration(lines, field_names) @classmethod def check_duration(cls, lines, field_names=None): if field_names and 'duration' not in field_names: return for line in lines: if line.duration < datetime.timedelta(): raise DurationValidationError( gettext('timesheet.msg_line_duration_positive', line=line.rec_name)) @classmethod def copy(cls, lines, default=None): if default is not None: default = default.copy() else: default = {} default.setdefault('uuid') return super().copy(lines, default=default) @property def hours(self): return self.duration.total_seconds() / 60 / 60 def to_json(self): return { 'id': self.id, 'work': self.work.id, 'work.name': self.work.rec_name, 'duration': self.duration.total_seconds(), 'description': self.description, 'uuid': self.uuid, } class EnterLinesStart(ModelView): 'Enter Lines' __name__ = 'timesheet.line.enter.start' employee = fields.Many2One('company.employee', 'Employee', required=True, domain=[ ('company', '=', Eval('context', {}).get('company', -1)), ['OR', ('start_date', '=', None), ('start_date', '<=', Eval('date', None)), ], ['OR', ('end_date', '=', None), ('end_date', '>=', Eval('date', None)), ], ], help="The employee who spends the time.") date = fields.Date('Date', required=True, help="When the time is spent.") @staticmethod def default_employee(): Line = Pool().get('timesheet.line') return Line.default_employee() @staticmethod def default_date(): Line = Pool().get('timesheet.line') return Line.default_date() class EnterLines(Wizard): 'Enter Lines' __name__ = 'timesheet.line.enter' start = StateView('timesheet.line.enter.start', 'timesheet.line_enter_start_view_form', [ Button('Cancel', 'end', 'tryton-cancel'), Button('Enter', 'enter', 'tryton-ok', default=True), ]) enter = StateAction('timesheet.act_line_form') def do_enter(self, action): pool = Pool() Lang = pool.get('ir.lang') date = self.start.date date = Date(date.year, date.month, date.day) action['pyson_domain'] = PYSONEncoder().encode([ ('employee', '=', self.start.employee.id), ('company', '=', self.start.employee.company.id), ('date', '=', date), ]) action['pyson_context'] = PYSONEncoder().encode({ 'employee': self.start.employee.id, 'company': self.start.employee.company.id, 'date': date, }) action['name'] += ' @ %(date)s - %(employee)s' % { 'date': Lang.get().strftime(self.start.date), 'employee': self.start.employee.rec_name, } return action, {} def transition_enter(self): return 'end' class HoursEmployee(ModelSQL, ModelView): 'Hours per Employee' __name__ = 'timesheet.hours_employee' employee = fields.Many2One('company.employee', 'Employee') duration = fields.TimeDelta('Duration', 'company_work_time') @staticmethod def table_query(): pool = Pool() Line = pool.get('timesheet.line') line = Line.__table__() where = Literal(True) if Transaction().context.get('start_date'): where &= line.date >= Transaction().context['start_date'] if Transaction().context.get('end_date'): where &= line.date <= Transaction().context['end_date'] return line.select( line.employee.as_('id'), Max(line.create_uid).as_('create_uid'), Max(line.create_date).as_('create_date'), Max(line.write_uid).as_('write_uid'), Max(line.write_date).as_('write_date'), line.employee, Sum(line.duration).as_('duration'), where=where, group_by=line.employee) class HoursEmployeeContext(ModelView): 'Hours per Employee Context' __name__ = 'timesheet.hours_employee.context' start_date = fields.Date('Start Date') end_date = fields.Date('End Date') class HoursEmployeeWeekly(ModelSQL, ModelView): 'Hours per Employee per Week' __name__ = 'timesheet.hours_employee_weekly' year = fields.Integer("Year") week = fields.Integer("Week") employee = fields.Many2One('company.employee', 'Employee') duration = fields.TimeDelta('Duration', 'company_work_time') @classmethod def __setup__(cls): super(HoursEmployeeWeekly, cls).__setup__() cls._order.insert(0, ('year', 'DESC')) cls._order.insert(1, ('week', 'DESC')) cls._order.insert(2, ('employee', 'ASC')) @classmethod def table_query(cls): pool = Pool() Line = pool.get('timesheet.line') line = Line.__table__() year_column = Extract('YEAR', line.date).as_('year') week_column = Extract('WEEK', line.date).as_('week') return line.select( Max(Extract('WEEK', line.date) + Extract('YEAR', line.date) * 100 + line.employee * 1000000).as_('id'), Max(line.create_uid).as_('create_uid'), Max(line.create_date).as_('create_date'), Max(line.write_uid).as_('write_uid'), Max(line.write_date).as_('write_date'), year_column, week_column, line.employee, Sum(line.duration).as_('duration'), group_by=(year_column, week_column, line.employee)) class HoursEmployeeMonthly(ModelSQL, ModelView): 'Hours per Employee per Month' __name__ = 'timesheet.hours_employee_monthly' year = fields.Integer("Year") month = fields.Many2One('ir.calendar.month', "Month") employee = fields.Many2One('company.employee', 'Employee') duration = fields.TimeDelta('Duration', 'company_work_time') @classmethod def __setup__(cls): super(HoursEmployeeMonthly, cls).__setup__() cls._order.insert(0, ('year', 'DESC')) cls._order.insert(1, ('month.index', 'DESC')) cls._order.insert(2, ('employee', 'ASC')) @classmethod def table_query(cls): pool = Pool() Line = pool.get('timesheet.line') Month = pool.get('ir.calendar.month') line = Line.__table__() month = Month.__table__() year_column = Extract('YEAR', line.date).as_('year') month_index = Extract('MONTH', line.date) return line.join(month, condition=month_index == month.id).select( Max(Extract('MONTH', line.date) + Extract('YEAR', line.date) * 100 + line.employee * 1000000).as_('id'), Max(line.create_uid).as_('create_uid'), Max(line.create_date).as_('create_date'), Max(line.write_uid).as_('write_uid'), Max(line.write_date).as_('write_date'), year_column, month.id.as_('month'), line.employee, Sum(line.duration).as_('duration'), group_by=(year_column, month.id, line.employee))