You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

273 lines
9.5 KiB

4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
[ADD] Improve Shift management après avoir généré les shifts pour une semaine, rediriger vers les shifts générés : ok Ajouter un supercoopérateur au niveau du template de shift. Pas de champs related car il faut pouvoir gérer les remplacements entre supercoopérateurs. : Rajouter un boolean sur res.users pour dire qu'ils sont super coopérateur du coup on peut choisir qu'un utilisateur super coopérateur : Ok Sur la vue des shifts, permettre de sélectionner un ensemble de shifts pour pouvoir changer son supercoopérateur en une action : Ok Avoir une vue des shifts par défaut plus lisible que le calendrier : KANBAN avec TODAY par défaut et les colonnes selon le type de shift : ok Dans la vue KANBAN du planning des semaines, faire un tri sur la start date pour que ça ressemble à une vue calendrier :ok Droits d'accès : 3 groupes : modifier un shift existant et confirmer son statut : absence/présence générer et supprimer des shifts à partir du template modifier le template des semaines et gérer la config En réfléchissant à ceux qui vont gérer les présences : il faudrait avoir une vue facile sur les shifts du jour et une manière simple de dire présent/absent/remplacé pour l'ensemble des coopérateurs présents (éviter de devoir chaque fois sélectionner le shift, cliquer sur le statut, aller au shift suivant, cliquer sur le statut...). J'ai fait un nouveau point de menu avec une vue Kanban grouper par status, il suffit de faire du drag and drop pour changer le status, les shifts sont aussi filtré sur aujourd'hui et seulement ceux ayant un worker assigné.
7 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
[ADD] Improve Shift management après avoir généré les shifts pour une semaine, rediriger vers les shifts générés : ok Ajouter un supercoopérateur au niveau du template de shift. Pas de champs related car il faut pouvoir gérer les remplacements entre supercoopérateurs. : Rajouter un boolean sur res.users pour dire qu'ils sont super coopérateur du coup on peut choisir qu'un utilisateur super coopérateur : Ok Sur la vue des shifts, permettre de sélectionner un ensemble de shifts pour pouvoir changer son supercoopérateur en une action : Ok Avoir une vue des shifts par défaut plus lisible que le calendrier : KANBAN avec TODAY par défaut et les colonnes selon le type de shift : ok Dans la vue KANBAN du planning des semaines, faire un tri sur la start date pour que ça ressemble à une vue calendrier :ok Droits d'accès : 3 groupes : modifier un shift existant et confirmer son statut : absence/présence générer et supprimer des shifts à partir du template modifier le template des semaines et gérer la config En réfléchissant à ceux qui vont gérer les présences : il faudrait avoir une vue facile sur les shifts du jour et une manière simple de dire présent/absent/remplacé pour l'ensemble des coopérateurs présents (éviter de devoir chaque fois sélectionner le shift, cliquer sur le statut, aller au shift suivant, cliquer sur le statut...). J'ai fait un nouveau point de menu avec une vue Kanban grouper par status, il suffit de faire du drag and drop pour changer le status, les shifts sont aussi filtré sur aujourd'hui et seulement ceux ayant un worker assigné.
7 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
[ADD] Improve Shift management après avoir généré les shifts pour une semaine, rediriger vers les shifts générés : ok Ajouter un supercoopérateur au niveau du template de shift. Pas de champs related car il faut pouvoir gérer les remplacements entre supercoopérateurs. : Rajouter un boolean sur res.users pour dire qu'ils sont super coopérateur du coup on peut choisir qu'un utilisateur super coopérateur : Ok Sur la vue des shifts, permettre de sélectionner un ensemble de shifts pour pouvoir changer son supercoopérateur en une action : Ok Avoir une vue des shifts par défaut plus lisible que le calendrier : KANBAN avec TODAY par défaut et les colonnes selon le type de shift : ok Dans la vue KANBAN du planning des semaines, faire un tri sur la start date pour que ça ressemble à une vue calendrier :ok Droits d'accès : 3 groupes : modifier un shift existant et confirmer son statut : absence/présence générer et supprimer des shifts à partir du template modifier le template des semaines et gérer la config En réfléchissant à ceux qui vont gérer les présences : il faudrait avoir une vue facile sur les shifts du jour et une manière simple de dire présent/absent/remplacé pour l'ensemble des coopérateurs présents (éviter de devoir chaque fois sélectionner le shift, cliquer sur le statut, aller au shift suivant, cliquer sur le statut...). J'ai fait un nouveau point de menu avec une vue Kanban grouper par status, il suffit de faire du drag and drop pour changer le status, les shifts sont aussi filtré sur aujourd'hui et seulement ceux ayant un worker assigné.
7 years ago
4 years ago
4 years ago
4 years ago
[ADD] Improve Shift management après avoir généré les shifts pour une semaine, rediriger vers les shifts générés : ok Ajouter un supercoopérateur au niveau du template de shift. Pas de champs related car il faut pouvoir gérer les remplacements entre supercoopérateurs. : Rajouter un boolean sur res.users pour dire qu'ils sont super coopérateur du coup on peut choisir qu'un utilisateur super coopérateur : Ok Sur la vue des shifts, permettre de sélectionner un ensemble de shifts pour pouvoir changer son supercoopérateur en une action : Ok Avoir une vue des shifts par défaut plus lisible que le calendrier : KANBAN avec TODAY par défaut et les colonnes selon le type de shift : ok Dans la vue KANBAN du planning des semaines, faire un tri sur la start date pour que ça ressemble à une vue calendrier :ok Droits d'accès : 3 groupes : modifier un shift existant et confirmer son statut : absence/présence générer et supprimer des shifts à partir du template modifier le template des semaines et gérer la config En réfléchissant à ceux qui vont gérer les présences : il faudrait avoir une vue facile sur les shifts du jour et une manière simple de dire présent/absent/remplacé pour l'ensemble des coopérateurs présents (éviter de devoir chaque fois sélectionner le shift, cliquer sur le statut, aller au shift suivant, cliquer sur le statut...). J'ai fait un nouveau point de menu avec une vue Kanban grouper par status, il suffit de faire du drag and drop pour changer le status, les shifts sont aussi filtré sur aujourd'hui et seulement ceux ayant un worker assigné.
7 years ago
4 years ago
4 years ago
4 years ago
4 years ago
[ADD] Improve Shift management après avoir généré les shifts pour une semaine, rediriger vers les shifts générés : ok Ajouter un supercoopérateur au niveau du template de shift. Pas de champs related car il faut pouvoir gérer les remplacements entre supercoopérateurs. : Rajouter un boolean sur res.users pour dire qu'ils sont super coopérateur du coup on peut choisir qu'un utilisateur super coopérateur : Ok Sur la vue des shifts, permettre de sélectionner un ensemble de shifts pour pouvoir changer son supercoopérateur en une action : Ok Avoir une vue des shifts par défaut plus lisible que le calendrier : KANBAN avec TODAY par défaut et les colonnes selon le type de shift : ok Dans la vue KANBAN du planning des semaines, faire un tri sur la start date pour que ça ressemble à une vue calendrier :ok Droits d'accès : 3 groupes : modifier un shift existant et confirmer son statut : absence/présence générer et supprimer des shifts à partir du template modifier le template des semaines et gérer la config En réfléchissant à ceux qui vont gérer les présences : il faudrait avoir une vue facile sur les shifts du jour et une manière simple de dire présent/absent/remplacé pour l'ensemble des coopérateurs présents (éviter de devoir chaque fois sélectionner le shift, cliquer sur le statut, aller au shift suivant, cliquer sur le statut...). J'ai fait un nouveau point de menu avec une vue Kanban grouper par status, il suffit de faire du drag and drop pour changer le status, les shifts sont aussi filtré sur aujourd'hui et seulement ceux ayant un worker assigné.
7 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
4 years ago
  1. import math
  2. from datetime import datetime, time, timedelta
  3. from pytz import UTC, timezone
  4. from odoo import _, api, fields, models
  5. from odoo.exceptions import UserError
  6. def float_to_time(f):
  7. decimal, integer = math.modf(f)
  8. return "{}:{}".format(
  9. str(int(integer)).zfill(2), str(int(round(decimal * 60))).zfill(2)
  10. )
  11. def floatime_to_hour_minute(f):
  12. decimal, integer = math.modf(f)
  13. return int(integer), int(round(decimal * 60))
  14. def get_first_day_of_week():
  15. today = datetime.now()
  16. return (datetime.now() - timedelta(days=today.weekday())).date()
  17. class TaskType(models.Model):
  18. _name = "beesdoo.shift.type"
  19. _description = "beesdoo.shift.type"
  20. name = fields.Char()
  21. description = fields.Text()
  22. active = fields.Boolean(default=True)
  23. class DayNumber(models.Model):
  24. _name = "beesdoo.shift.daynumber"
  25. _description = "beesdoo.shift.daynumber"
  26. _order = "number asc"
  27. name = fields.Char()
  28. number = fields.Integer(
  29. "Day Number",
  30. help="From 1 to N, When you will instanciate your planning, Day 1 "
  31. "will be the start date of the instance, Day 2 the second, "
  32. "etc...",
  33. )
  34. active = fields.Boolean(default=True)
  35. class Planning(models.Model):
  36. _name = "beesdoo.shift.planning"
  37. _description = "beesdoo.shift.planning"
  38. _order = "sequence asc"
  39. sequence = fields.Integer()
  40. name = fields.Char()
  41. task_template_ids = fields.One2many(
  42. "beesdoo.shift.template", "planning_id"
  43. )
  44. @api.model
  45. def _get_next_planning(self, sequence):
  46. next_planning = self.search([("sequence", ">", sequence)])
  47. if not next_planning:
  48. return self.search([])[0]
  49. return next_planning[0]
  50. @api.multi
  51. def _get_next_planning_date(self, date):
  52. self.ensure_one()
  53. nb_of_day = max(self.task_template_ids.mapped("day_nb_id.number"))
  54. return date + timedelta(days=nb_of_day)
  55. @api.model
  56. def _generate_next_planning(self):
  57. config = self.env["ir.config_parameter"].sudo()
  58. last_seq = int(config.get_param("last_planning_seq", 0))
  59. date = fields.Date.from_string(
  60. config.get_param("next_planning_date", 0)
  61. )
  62. planning = self._get_next_planning(last_seq)
  63. planning = planning.with_context(visualize_date=date)
  64. planning.task_template_ids._generate_task_day()
  65. next_date = planning._get_next_planning_date(date)
  66. config.set_param("last_planning_seq", planning.sequence)
  67. config.set_param("next_planning_date", next_date)
  68. class TaskTemplate(models.Model):
  69. _name = "beesdoo.shift.template"
  70. _description = "beesdoo.shift.template"
  71. _order = "start_time"
  72. name = fields.Char(required=True)
  73. planning_id = fields.Many2one("beesdoo.shift.planning", required=True)
  74. day_nb_id = fields.Many2one(
  75. "beesdoo.shift.daynumber", string="Day", required=True
  76. )
  77. task_type_id = fields.Many2one("beesdoo.shift.type", string="Type")
  78. # FIXME removed because beesdoo.shift.sheet is from another module.
  79. # attendance_sheet_id = fields.Many2one('beesdoo.shift.sheet', string="Attendance Sheet") # noqa
  80. start_time = fields.Float(required=True)
  81. end_time = fields.Float(required=True)
  82. super_coop_id = fields.Many2one(
  83. "res.users",
  84. string="Super Cooperative",
  85. domain=[("partner_id.super", "=", True)],
  86. )
  87. duration = fields.Float(help="Duration in Hour")
  88. worker_nb = fields.Integer(
  89. string="Number of worker",
  90. help="Max number of worker for this task",
  91. default=1,
  92. )
  93. worker_ids = fields.Many2many(
  94. "res.partner",
  95. string="Recurrent worker assigned",
  96. domain=[("is_worker", "=", True)],
  97. )
  98. remaining_worker = fields.Integer(
  99. compute="_compute_remaining", store=True, string="Remaining Place"
  100. )
  101. active = fields.Boolean(default=True)
  102. # For Kanban View Only
  103. color = fields.Integer("Color Index")
  104. worker_name = fields.Char(compute="_compute_worker_name")
  105. # For calendar View
  106. start_date = fields.Datetime(
  107. compute="_compute_fake_date", search="_search_dummy"
  108. )
  109. end_date = fields.Datetime(
  110. compute="_compute_fake_date", search="_search_dummy"
  111. )
  112. def _get_utc_date(self, day, hour, minute):
  113. """Combine day number, hours and minutes to save
  114. corresponding UTC datetime in database.
  115. """
  116. context_tz = timezone(self._context.get("tz") or self.env.user.tz)
  117. day_local_time = datetime.combine(day, time(hour=hour, minute=minute))
  118. day_local_time = context_tz.localize(day_local_time)
  119. day_utc_time = day_local_time.astimezone(UTC)
  120. # Return naïve datetime so as to be saved in database
  121. return day_utc_time.replace(tzinfo=None)
  122. @api.depends("start_time", "end_time")
  123. def _compute_fake_date(self):
  124. today = self._context.get("visualize_date", get_first_day_of_week())
  125. for rec in self:
  126. # Find the day of this task template 'rec'.
  127. day = today + timedelta(days=rec.day_nb_id.number - 1)
  128. # Compute the beginning and ending time according to the
  129. # context timezone.
  130. h_begin, m_begin = floatime_to_hour_minute(rec.start_time)
  131. h_end, m_end = floatime_to_hour_minute(rec.end_time)
  132. rec.start_date = self._get_utc_date(day, h_begin, m_begin)
  133. rec.end_date = self._get_utc_date(day, h_end, m_end)
  134. def _search_dummy(self, operator, value):
  135. return []
  136. @api.depends("worker_ids", "worker_nb")
  137. def _compute_remaining(self):
  138. for rec in self:
  139. rec.remaining_worker = rec.worker_nb - len(rec.worker_ids)
  140. @api.depends("worker_ids")
  141. def _compute_worker_name(self):
  142. for rec in self:
  143. rec.worker_name = ",".join(rec.worker_ids.mapped("display_name"))
  144. @api.constrains("worker_nb", "worker_ids")
  145. def _nb_worker_max(self):
  146. for rec in self:
  147. if len(rec.worker_ids) > rec.worker_nb:
  148. raise UserError(
  149. _(
  150. "You cannot assign more workers than the maximal "
  151. "number defined on template. "
  152. )
  153. )
  154. @api.onchange("start_time", "end_time")
  155. def _get_duration(self):
  156. if self.start_time and self.end_time:
  157. self.duration = self.end_time - self.start_time
  158. @api.onchange("duration")
  159. def _set_duration(self):
  160. if self.start_time:
  161. self.end_time = self.start_time + self.duration
  162. def _generate_task_day(self):
  163. tasks = self.env["beesdoo.shift.shift"]
  164. for rec in self:
  165. for i in range(0, rec.worker_nb):
  166. worker_id = (
  167. rec.worker_ids[i] if len(rec.worker_ids) > i else False
  168. )
  169. # remove worker in holiday and temporary exempted
  170. if worker_id and worker_id.cooperative_status_ids:
  171. status = worker_id.cooperative_status_ids[0]
  172. if (
  173. status.holiday_start_time
  174. and status.holiday_end_time
  175. and status.holiday_start_time <= rec.start_date.date()
  176. and status.holiday_end_time >= rec.end_date.date()
  177. ):
  178. worker_id = False
  179. if (
  180. status.temporary_exempt_start_date
  181. and status.temporary_exempt_end_date
  182. and status.temporary_exempt_start_date
  183. <= rec.start_date.date()
  184. and status.temporary_exempt_end_date
  185. >= rec.end_date.date()
  186. ):
  187. worker_id = False
  188. tasks |= tasks.create(
  189. {
  190. "name": "[%s] %s %s (%s - %s) [%s]"
  191. % (
  192. rec.start_date.date(),
  193. rec.planning_id.name,
  194. rec.day_nb_id.name,
  195. float_to_time(rec.start_time),
  196. float_to_time(rec.end_time),
  197. i,
  198. ),
  199. "task_template_id": rec.id,
  200. "task_type_id": rec.task_type_id.id,
  201. "super_coop_id": rec.super_coop_id.id,
  202. "worker_id": worker_id and worker_id.id or False,
  203. "is_regular": True if worker_id else False,
  204. "start_time": rec.start_date,
  205. "end_time": rec.end_date,
  206. "state": "open",
  207. }
  208. )
  209. return tasks
  210. @api.onchange("worker_ids")
  211. def check_for_multiple_shifts(self):
  212. original_ids = {worker.id for worker in self._origin.worker_ids}
  213. warnings = []
  214. for worker in self.worker_ids:
  215. if worker.id not in original_ids:
  216. shifts = [
  217. shift.name
  218. for shift in worker.subscribed_shift_ids
  219. if shift.id != self.id
  220. ]
  221. if shifts:
  222. warnings.append(
  223. worker.name
  224. + _(" is already assigned to ")
  225. + ", ".join(shifts)
  226. )
  227. if warnings:
  228. return {
  229. "warning": {
  230. "title": _("Warning"),
  231. "message": "\n".join(warnings),
  232. }
  233. }