Projet

Général

Profil

Télécharger (9,27 ko) Statistiques
| Branche: | Tag: | Révision:

calebasse / calebasse / agenda / appointments.py @ 8c0910ce

1
# -*- coding: utf-8 -*-
2

    
3
from datetime import datetime, time
4
from datetime import time as datetime_time
5

    
6
from interval import Interval, IntervalSet
7

    
8
from calebasse.actes.validation_states import VALIDATION_STATES
9
from .models import EventWithAct
10

    
11
class Appointment(object):
12

    
13
    def __init__(self, title=None, begin_time=None, type=None,
14
            length=None, description=None, ressource=None):
15
        self.title = title
16
        self.type = type
17
        self.length = length
18
        self.description = description
19
        self.ressource = ressource
20
        self.is_recurrent = False
21
        self.is_billed = False
22
        self.convocation_sent = None
23
        self.other_services_names = []
24
        self.patient = None
25
        self.patient_record_id = None
26
        self.patient_record_paper_id = None
27
        self.event_id = None
28
        self.event_type = None
29
        self.workers = None
30
        self.workers_initial = None
31
        self.workers_codes = None
32
        self.act_id = None
33
        self.act_state = None
34
        self.act_absence = None
35
        self.weight = 0
36
        self.act_type = None
37
        self.validation = None
38
        self.holiday = False
39
        self.services_names = []
40
        self.event = False
41
        self.__set_time(begin_time)
42

    
43
    def __set_time(self, time):
44
        self.begin_time = time
45
        if time:
46
            self.begin_hour = time.strftime("%H:%M")
47
        else:
48
            self.begin_hour = None
49

    
50
    def init_from_event(self, event, service, validation_states=None):
51
        delta = event.end_datetime - event.start_datetime
52
        self.event = isinstance(event, EventWithAct)
53
        self.event_id = event.id
54
        self.length = delta.seconds / 60
55
        self.title = event.title
56
        if (hasattr(event, 'parent') and event.parent.recurrence_periodicity) or \
57
                event.exception_to:
58
            self.is_recurrent = True
59
        services = event.services.all()
60
        self.date = event.start_datetime.date()
61
        self.__set_time(time(event.start_datetime.hour, event.start_datetime.minute))
62
        for e_service in services:
63
            name = e_service.name.lower().replace(' ', '-')
64
            if e_service != service:
65
                self.other_services_names.append(name)
66
            self.services_names.append(name)
67
        if service in services:
68
            self.type = "busy-here"
69
        else:
70
            self.type = "busy-elsewhere"
71
        self.event_id = event.id
72
        if event.ressource:
73
            self.ressource = event.ressource.name
74
        self.description = event.description
75
        self.workers_initial = ""
76
        self.workers_code = []
77
        self.workers = event.participants.all()
78
        self.len_workers = event.participants.count()
79
        self.workers_absent = event.get_missing_participants()
80
        if event.event_type.id == 1:
81
            self.act_id = event.act.id
82
            self.convocation_sent = event.convocation_sent
83
            self.patient = event.patient
84
            self.patient_record_id = event.patient.id
85
            self.patient_record_paper_id = event.patient.paper_id
86
            self.act_type = event.act_type.name
87
            if self.act_id:
88
                self.description = event.act.comment
89
            self.is_billed = event.act.is_billed
90
            state = event.get_state()
91
            state_name = state.state_name if state else 'NON_VALIDE'
92
            display_name = VALIDATION_STATES[state_name]
93
            if event.is_absent():
94
                self.act_absence = VALIDATION_STATES.get(state_name)
95
            if state and not state.previous_state and state.state_name == 'NON_VALIDE':
96
                state = None
97
            if not service in services:
98
                validation_states = None
99
            self.validation = (event.act, state, display_name, validation_states)
100
            self.title = event.patient.display_name
101
        else:
102
            if event.event_type.label == 'Autre' and event.title:
103
                self.title = event.title
104
            else:
105
                self.title = '%s' % event.event_type.label
106
                if event.title:
107
                    self.title += ' - %s' % event.title
108
            self.event_type = event.event_type
109
        for worker in self.workers:
110
            self.workers_code.append("%s-%s" % (worker.id, worker.last_name.upper()))
111
        if not self.description:
112
            self.description = ''
113

    
114
    def init_free_time(self, length, begin_time):
115
        self.type = "free"
116
        self.length = length
117
        self.__set_time(begin_time)
118

    
119
    def init_busy_time(self, title, length, begin_time, description=None):
120
        self.title = title
121
        self.length = length
122
        self.__set_time(begin_time)
123
        self.description = description
124

    
125
    def init_holiday_time(self, title, length, begin_time, description=None):
126
        self.init_busy_time(title, length, begin_time, description)
127
        self.holiday = True
128

    
129
    def init_start_stop(self, title, time):
130
        """
131
        title: Arrivee ou Depart
132
        """
133
        self.type = "info"
134
        self.title = title
135
        self.__set_time(time)
136

    
137
def get_daily_appointments(date, worker, service, time_tables, events, holidays):
138
    """
139
    """
140
    appointments = []
141

    
142
    timetables_set = IntervalSet((t.to_interval(date) for t in time_tables))
143
    holidays_set = IntervalSet((h.to_interval(date) for h in holidays))
144
    busy_occurrences_set = IntervalSet((o.to_interval() for o in events if not o.is_event_absence()))
145
    for free_time in timetables_set - (busy_occurrences_set+holidays_set):
146
        if free_time:
147
            delta = free_time.upper_bound - free_time.lower_bound
148
            delta_minutes = delta.seconds / 60
149
            appointment = Appointment()
150
            appointment.init_free_time(delta_minutes,
151
                    time(free_time.lower_bound.hour, free_time.lower_bound.minute))
152
            appointments.append(appointment)
153
    validation_states = dict(VALIDATION_STATES)
154
    if service.name != 'CMPP' and \
155
            'ACT_DOUBLE' in validation_states:
156
        validation_states.pop('ACT_DOUBLE')
157
    vs = [('VALIDE', 'Présent')]
158
    validation_states.pop('VALIDE')
159
    validation_states.pop('ACT_LOST')
160
    validation_states = vs + sorted(validation_states.items(), key=lambda tup: tup[0])
161
    for event in events:
162
        appointment = Appointment()
163
        appointment.init_from_event(event, service, validation_states)
164
        appointments.append(appointment)
165
    for holiday in holidays:
166
        interval = holiday.to_interval(date)
167
        delta = interval.upper_bound - interval.lower_bound
168
        delta_minutes = delta.seconds / 60
169
        appointment = Appointment()
170
        appointment.type = 'busy-here'
171
        label = None
172
        if not holiday.worker:
173
            label = u"Absence de groupe : %s" % holiday.holiday_type.name
174
        else:
175
            label = u"Absence indiv. : %s" % holiday.holiday_type.name
176
        appointment.init_holiday_time(label,
177
                    delta_minutes,
178
                    time(interval.lower_bound.hour, interval.lower_bound.minute),
179
                    description=holiday.comment)
180
        services = holiday.services.all()
181
        if service not in services:
182
            appointment.type = 'busy-elsewhere'
183
        appointment.other_services_names = [s.slug for s in services if s != service]
184
        appointments.append(appointment)
185
    for time_table in time_tables:
186
        interval_set = IntervalSet.between(time_table.to_interval(date).lower_bound.time(),
187
                                   time_table.to_interval(date).upper_bound.time())
188
        for holiday in holidays:
189
            holiday_interval_set = IntervalSet.between(holiday.to_interval(date).lower_bound.time(),
190
                                   holiday.to_interval(date).upper_bound.time())
191
            interval_set = interval_set - holiday_interval_set
192
        if not interval_set:
193
            continue
194
        start_time = interval_set.lower_bound()
195
        end_time = interval_set.upper_bound()
196
        appointment = Appointment()
197
        appointment.init_start_stop(u"Arrivée", start_time)
198
        appointment.weight = -1
199
        appointments.append(appointment)
200
        appointment = Appointment()
201
        appointment.init_start_stop(u"Départ", end_time)
202
        appointment.weight = 1
203
        appointments.append(appointment)
204

    
205
    return sorted(appointments, key=lambda app: (app.begin_time, app.weight, app.event_id))
206

    
207
def get_daily_usage(date, ressource, service, occurrences):
208
    """
209
    """
210
    appointments = []
211

    
212
    start_time = datetime_time(8, 0)
213
    end_time = datetime_time(20, 0)
214
    all_day = Interval(datetime.combine(date, start_time), datetime.combine(date, end_time))
215
    timetables_set = IntervalSet([all_day])
216
    occurrences_set = IntervalSet((o.to_interval() for o in occurrences))
217
    for free_time in timetables_set - occurrences_set:
218
        if free_time:
219
            delta = free_time.upper_bound - free_time.lower_bound
220
            delta_minutes = delta.seconds / 60
221
            appointment = Appointment()
222
            appointment.init_free_time(delta_minutes,
223
                    time(free_time.lower_bound.hour, free_time.lower_bound.minute))
224
            appointments.append(appointment)
225
    for occurrence in occurrences:
226
        appointment = Appointment()
227
        appointment.init_from_event(occurrence, service)
228
        appointments.append(appointment)
229

    
230
    return sorted(appointments, key=lambda app: (app.begin_time, app.weight))
(4-4/10)