from datetime import datetime, timedelta, date from odoo import models, fields, api import re def time_H_selection(): return [('00', '00'), ('01', '01'), ('02', '02'), ('03', '03'), ('04', '04'), ('05', '05'), ('06', '06'), ('07', '07'), ('08', '08'), ('09', '09'), ('10', '10'), ('11', '11'), ('12', '12'), ('13', '13'), ('14', '14'), ('15', '15'), ('16', '16'), ('17', '17'), ('18', '18'), ('19', '19'), ('20', '20'), ('21', '21'), ('22', '22'), ('23', '23')] def time_M_or_S_selection(): return [('00', '00'), ('05', '05'), ('10', '10'), ('15', '15'), ('20', '20'), ('25', '25'), ('30', '30'), ('35', '35'), ('40', '40'), ('45', '45'), ('50', '50'), ('55', '55')] class WorkLogSetting(models.Model): _name = 'sf.work.log.setting' _description = '工作日历设置' def _get_code(self): """ 自动生成编码 :return: """ fixture_material = self.env['sf.work.log.setting'].sudo().search( [('code', '!=', '')], limit=1, order="id desc") if not fixture_material: num = "%03d" % 1 else: m = int(fixture_material.code) + 1 num = "%03d" % m return num code = fields.Char(string='序号', default=_get_code, readonly=True) name = fields.Char(string='工作日历名称', required=True) start_time = fields.Char(string='日开始时间', readonly=True, compute='_compute_start_time') start_time_H = fields.Selection(time_H_selection(), '时', required=True) start_time_M = fields.Selection(time_M_or_S_selection(), '分', required=True) end_time = fields.Char(string='日结束时间', readonly=True, compute='_compute_end_time') end_time_H = fields.Selection(time_H_selection(), '时', required=True) end_time_M = fields.Selection(time_M_or_S_selection(), '分', required=True) duration = fields.Char(string='时长', readonly=True, compute='_compute_duration') day_off_id = fields.Many2many('sf.day.off', string='休息日', required=True) working_shift_id = fields.Many2many('sf.working.shift', string='班次') status = fields.Boolean(string='状态', default=True) update_person = fields.Char(string='更新人', default=lambda self: self.env.user.name) update_time = fields.Datetime(string='更新时间', default=lambda self: fields.Datetime.now()) setting_to_calendar_ids = fields.One2many('sf.work.schedule.calendar', 'name_id', '工作日历') # @api.model # def create(self, vals): # vals['setting_to_calendar_ids'] = [(4, child.id)] # return super(WorkLogSetting, self).create(vals) @api.depends('start_time_H', 'start_time_M') def _compute_start_time(self): """ 设置输入日开始时间 :return: """ for record in self: record.start_time = f"{record.start_time_H}:{record.start_time_M}:00" @api.depends('end_time_H', 'end_time_M') def _compute_end_time(self): """ 设置输入日结束时间 :return: """ for record in self: record.end_time = f"{record.end_time_H}:{record.end_time_M}:00" @api.depends('start_time_H', 'start_time_M', 'end_time_H', 'end_time_M') def _compute_duration(self): """ 根据日开始时间和日结束时间计算每日工作时长 :return: """ for record in self: st_h = float(record.start_time_H) st_m = float(record.start_time_M) end_h = float(record.end_time_H) end_m = float(record.end_time_M) # 日开始时间小于日结束时间 if st_h < end_h: record.duration = str(round(end_h - st_h + (end_m - st_m) / 60, 2)) else: record.duration = str(round(end_h - st_h + (end_m - st_m) / 60 + 24, 2)) # @api.onchange('day_off_id') # def _onchange_day_off_id(self): # # 先删除之前创建的工作日历事件记录 # self.env['sf.work.schedule.calendar'].search([('calendar_code', '=', self.code)]).unlink() # # # 获取当年的一月一号的日期 # year = fields.Datetime.now().year # 2023 # first_day = datetime(year, 1, 1).date() # 2023-01-01 # day_of_week = first_day.strftime("%A") # 星期日 # # # 根据day_of_week,将其设置为起始0,循环周一到周日,按循环顺序设置为0-6 # # 列:{'星期日': 0, '星期一': 1, '星期二': 2, '星期三': 3, '星期四': 4, '星期五': 5, '星期六': 6} # desc = {} # desc_weekdays = ['星期一', '星期二', '星期三', '星期四', '星期五', '星期六', '星期日'] # status = False # ints = 0 # while len(desc) < 7: # if len(desc) != 0: # for week in desc_weekdays: # if len(desc) < 7: # desc.update({week: ints}) # ints += 1 # else: # for week in desc_weekdays: # if status: # desc.update({week: ints}) # ints += 1 # if day_of_week == week: # desc.update({week: ints}) # status = True # ints += 1 # # # 创建工作日历的休息日事件 # day_off_ids = self.day_off_id.mapped('name') # record = self.env['sf.work.log.setting'].search([('code', '=', self.code)]) # # record.setting_to_calendar_ids.unlink() # # self.env.cr.commit() # for week in day_off_ids: # num = desc[week] # while num <= 30: # target_date = first_day + timedelta(days=num) # num += 7 # self.env['sf.work.schedule.calendar'].create({'name': '休息日', # 'name_id': record.id, # 'calendar_code': self.code, # 'date_time': target_date}) @api.model def create(self, vals): """ 创建记录时,生成工作日历 """ record = super(WorkLogSetting, self).create(vals) record.generate_work_calendar() return record def write(self, vals): """ 更新记录时,生成工作日历 """ # 先删除之前创建的工作日历事件记录 self.env['sf.work.schedule.calendar'].search([('calendar_code', '=', self.code)]).unlink() result = super(WorkLogSetting, self).write(vals) self.generate_work_calendar() return result def generate_work_calendar(self): """ 生成工作日历 """ self.ensure_one() # start_date = date.today() # 开始日期 # end_date = start_date + timedelta(days=365) # 结束日期 # 获取本年第一天和最后一天 start_date = datetime.now().replace(month=1, day=1).date() end_date = datetime.now().replace(month=12, day=31).date() # 休息日列表 rest_days = self.day_off_id.mapped('name') for single_date in self.daterange(start_date, end_date): is_workday = single_date.strftime("%A") if is_workday in rest_days: print('is_workday in rest_days', is_workday) self.env['sf.work.schedule.calendar'].sudo().create({ 'name': '休息日', 'name_id': self.id, 'calendar_code': self.code, 'date_time': single_date}) @staticmethod def daterange(start_date, end_date): """ 生成日期范围 """ # 生成日期范围 for n in range(int((end_date - start_date).days)): yield start_date + timedelta(n) def open_work_schedule_calendar(self): action = self.env.ref('sf_plan_management.sf_work_schedule_calendar_act') result = action.read()[0] result['domain'] = [('name_id', '=', self.id)] return result class WorkingShift(models.Model): _name = 'sf.working.shift' _description = '班次' def _get_code(self): """ 自动生成编码 :return: """ fixture_material = self.env['sf.working.shift'].sudo().search( [('code', '!=', '')], limit=1, order="id desc") if not fixture_material: num = "%03d" % 1 else: m = int(fixture_material.code) + 1 num = "%03d" % m return num code = fields.Char('编码', default=_get_code, readonly=True) name = fields.Char('名称', required=True) start_time = fields.Datetime('班次开始时间') end_time = fields.Datetime('班次结束时间') remark = fields.Char('备注') class DayOff(models.Model): _name = 'sf.day.off' _description = '休息日' name = fields.Selection([ ('星期一', '星期一'), ('星期二', '星期二'), ('星期三', '星期三'), ('星期四', '星期四'), ('星期五', '星期五'), ('星期六', '星期六'), ('星期日', '星期日')], '休息日名称') class WorkScheduleCalendar(models.Model): _name = 'sf.work.schedule.calendar' _description = '工作日历' name = fields.Selection([('休息日', '休息日'), ('计划停机', '计划停机'), ('工作日', '工作日')], '日历事件名称') date_time = fields.Date('休息时间') name_id = fields.Many2one('sf.work.log.setting', '工作日历名称') calendar_code = fields.Char('工作日历编码') day_off_id = fields.Many2many('sf.day.off', string='休息日') scheduled_outage = fields.Char('计划停机') monthly_rest_days = fields.Char('月休息天数', readonly=True) annual_rest_days = fields.Char('年休息天数', readonly=True) monthly_planned_downtime = fields.Char('月计划停机时长', readonly=True) annual_planned_downtime = fields.Char('年计划停机时长', readonly=True)