Projet

Général

Profil

Télécharger (10,3 ko) Statistiques
| Branche: | Tag: | Révision:

calebasse / calebasse / actes / models.py @ a9520794

1
# -*- coding: utf-8 -*-
2
from datetime import date, time
3

    
4
from django.db import models
5
from django.contrib.auth.models import User
6

    
7
from calebasse.actes.validation_states import VALIDATION_STATES
8
from calebasse.ressources.models import ServiceLinkedAbstractModel
9
from ..middleware.request import get_request
10

    
11
class ActValidationState(models.Model):
12

    
13
    class Meta:
14
        ordering = ('-created',)
15

    
16
    act = models.ForeignKey('actes.Act',
17
        verbose_name=u'Acte', editable=False)
18
    state_name = models.CharField(max_length=150)
19
    created = models.DateTimeField(u'Création', auto_now_add=True)
20
    author = \
21
        models.ForeignKey(User,
22
        verbose_name=u'Auteur', editable=False, blank=True, null=True,
23
        on_delete=models.SET_NULL)
24
    previous_state = models.ForeignKey('ActValidationState',
25
        verbose_name=u'Etat précédent',
26
        editable=False, blank=True, null=True)
27
    # To record if the validation has be done by the automated validation
28
    auto = models.BooleanField(default=False,
29
            verbose_name=u'Validaté automatiquement')
30

    
31
    def __repr__(self):
32
        return self.state_name + ' ' + str(self.created)
33

    
34
    def __unicode__(self):
35
        return VALIDATION_STATES[self.state_name]
36

    
37

    
38
class ActManager(models.Manager):
39
    def for_today(self, today=None):
40
        today = today or date.today()
41
        return self.filter(date=today)
42

    
43
    def create_act(self, author=None, **kwargs):
44
        act = Act(**kwargs)
45
        last_validation_state = ActValidationState.objects.create(
46
                act=act,state_name='NON_VALIDE',
47
                author=author, previous_state=None)
48
        act.last_validation_state = last_validation_state
49
        act.save()
50
        return act
51

    
52
    def next_acts(self, patient_record, today=None):
53
        today = today or date.today()
54
        return self.filter(date__gte=today) \
55
                .filter(patient=patient_record) \
56
                .order_by('date')
57

    
58
    def last_acts(self, patient_record, today=None):
59
        today = today or date.today()
60
        return self.filter(date__lte=today) \
61
                .filter(patient=patient_record) \
62
                .order_by('-date')
63

    
64

    
65
class Act(models.Model):
66
    objects = ActManager()
67

    
68
    patient = models.ForeignKey('dossiers.PatientRecord')
69
    date = models.DateField(u'Date', db_index=True)
70
    time = models.TimeField(u'Heure', blank=True, null=True, default=time(), db_index=True)
71
    _duration = models.IntegerField(u'Durée en minutes', blank=True, null=True, default=0)
72
    act_type = models.ForeignKey('ressources.ActType',
73
            verbose_name=u'Type d\'acte')
74
    validation_locked = models.BooleanField(default=False,
75
            verbose_name=u'Vérouillage', db_index=True)
76
    is_billed = models.BooleanField(default=False,
77
            verbose_name=u'Facturé', db_index=True)
78
    already_billed = models.BooleanField(default=False,
79
            verbose_name=u'A déjà été facturé', db_index=True)
80
    is_lost = models.BooleanField(default=False,
81
            verbose_name=u'Acte perdu', db_index=True)
82
    last_validation_state = models.ForeignKey(ActValidationState,
83
            related_name='+',
84
            null=True, default=None,
85
            on_delete=models.SET_NULL)
86
    valide = models.BooleanField(default=False,
87
            verbose_name=u'Validé', db_index=True)
88
    switch_billable = models.BooleanField(default=False,
89
            verbose_name=u'Inverser type facturable')
90
    healthcare = models.ForeignKey('dossiers.HealthCare',
91
            blank=True,
92
            null=True,
93
            on_delete=models.SET_NULL,
94
            verbose_name=u'Prise en charge utilisée pour facturer (CMPP)')
95
    transport_company = models.ForeignKey('ressources.TransportCompany',
96
            blank=True,
97
            null=True,
98
            on_delete=models.SET_NULL,
99
            verbose_name=u'Compagnie de transport')
100
    transport_type = models.ForeignKey('ressources.TransportType',
101
            blank=True,
102
            null=True,
103
            on_delete=models.SET_NULL,
104
            verbose_name=u'Type de transport')
105
    doctors = models.ManyToManyField('personnes.Worker',
106
            limit_choices_to={'type__intervene': True},
107
            verbose_name=u'Intervenants')
108
    pause = models.BooleanField(default=False,
109
            verbose_name=u'Pause facturation', db_index=True)
110
    parent_event = models.ForeignKey('agenda.Event',
111
            verbose_name=u'Rendez-vous lié',
112
            blank=True, null=True,
113
            on_delete=models.SET_NULL)
114
    VALIDATION_CODE_CHOICES = (
115
            ('absent', u'Absent'),
116
            ('present', u'Présent'),
117
            )
118
    attendance = models.CharField(max_length=16,
119
            choices=VALIDATION_CODE_CHOICES,
120
            default='absent',
121
            verbose_name=u'Présence')
122
    old_id = models.CharField(max_length=256,
123
            verbose_name=u'Ancien ID', blank=True, null=True)
124

    
125
    @property
126
    def event(self):
127
        if self.parent_event:
128
            return self.parent_event.today_occurrence(self.date)
129
        return None
130

    
131
    @property
132
    def start_datetime(self):
133
        event = self.event
134
        if event:
135
            return event.start_datetime
136
        return self.date
137

    
138
    @property
139
    def comment(self):
140
        if self.parent_event:
141
            return self.parent_event.description
142
        return None
143

    
144
    @property
145
    def rebilling(self):
146
        if self.already_billed and not self.is_billed and not self.is_lost:
147
            return True
148
        return False
149

    
150
    def get_hc_tag(self):
151
        if self.healthcare:
152
            beg = None
153
            try:
154
                self.healthcare.cmpphealthcaretreatment
155
                beg = 'T'
156
            except:
157
                pass
158
            try:
159
                self.healthcare.cmpphealthcarediagnostic
160
                beg = 'D'
161
            except:
162
                pass
163
            if beg:
164
                acts = list(self.healthcare.act_set.order_by('date').values_list('id', flat=True))
165
                beg += str(acts.index(self.id) + 1)
166
                return beg
167
        return None
168

    
169
    def is_new(self):
170
        return (not self.get_state() or self.is_state('NON_VALIDE'))
171

    
172
    def is_absent(self):
173
        state = self.get_state()
174
        if state and state.state_name in ('ABS_NON_EXC', 'ABS_EXC', 'ABS_INTER', 'ANNUL_NOUS',
175
                'ANNUL_FAMILLE', 'REPORTE', 'ABS_ESS_PPS', 'ENF_HOSP'):
176
            return True
177
        return False
178

    
179
    def is_present(self):
180
        return (not self.is_new() and not self.is_absent())
181

    
182
    def get_state(self):
183
        try:
184
            return self.last_validation_state
185
        except:
186
            return None
187

    
188
    def is_state(self, state_name):
189
        state = self.get_state()
190
        if state and state.state_name == state_name:
191
            return True
192
        return False
193

    
194
    def set_state(self, state_name, author, auto=False,
195
            change_state_check=True):
196
        if not self.id:
197
            self.save()
198
        if self.is_billed:
199
            raise Exception('Billed act state can not be modified')
200
        if not author:
201
            raise Exception('Missing author to set state')
202
        if not state_name in VALIDATION_STATES.keys():
203
            raise Exception("Etat d'acte non existant %s")
204
        current_state = self.get_state()
205
        last_validation_state = ActValidationState.objects.create(
206
                act=self,
207
                state_name=state_name,
208
                author=author,
209
                previous_state=current_state)
210
        self.last_validation_state = last_validation_state
211
        if state_name == 'VALIDE':
212
            self.valide = True
213
        else:
214
            self.valide = False
215
        try:
216
            get_request().record('act-update', '{obj_id} state changed to {state} by {user} from {ip}',
217
                              obj_id=self.id, state=last_validation_state, user=author)
218
        except:
219
            pass
220
        self.save()
221

    
222
    def is_billable(self):
223
        billable = self.act_type.billable
224
        if (billable and not self.switch_billable) or \
225
                (not billable and self.switch_billable):
226
            return True
227
        return False
228

    
229
    # START Specific to sessad healthcare
230
    def was_covered_by_notification(self):
231
        from calebasse.dossiers.models import SessadHealthCareNotification
232
        notifications = \
233
            SessadHealthCareNotification.objects.filter(patient=self.patient,
234
            start_date__lte=self.date, end_date__gte=self.date)
235
        if notifications:
236
            return True
237
    # END Specific to sessad healthcare
238

    
239
    def save(self, *args, **kwargs):
240
        if self.is_billed:
241
            self.already_billed = True
242
        super(Act, self).save(*args, **kwargs)
243

    
244
    def duration(self):
245
        '''Return a displayable duration for this field.'''
246
        hours, remainder = divmod(self._duration, 60)
247
        return '%02d:%02d' % (hours, remainder)
248

    
249
    def __unicode__(self):
250
        if self.time:
251
            return u'{0} le {1} à {2} pour {3} avec {4}'.format(
252
                    self.act_type, self.date.strftime('%d/%m/%Y'),
253
                    self.time.strftime('%H:%M'), self.patient,
254
                    ', '.join(map(unicode, self.doctors.all())))
255
        return u'{0} le {1} pour {2} avec {3}'.format(
256
                self.act_type, self.date.strftime('%d/%m/%Y'), self.patient,
257
                ', '.join(map(unicode, self.doctors.all())))
258

    
259
    def __repr__(self):
260
        return '<%s %r %r>' % (self.__class__.__name__, unicode(self), self.id)
261

    
262
    def cancel(self):
263
        '''Parent event is canceled completely, or partially, act upon it.
264
        '''
265
        new_act = self.actvalidationstate_set.count() > 1
266
        if self.date <= date.today():
267
            if new_act:
268
                self.set_state('ANNUL_NOUS', get_request().user)
269
            self.parent_event = None
270
            self.save()
271
        else:
272
            self.delete()
273

    
274
    def get_invoice_number(self):
275
        try:
276
            if self.is_billed:
277
                return self.invoice_set.order_by('-number')[0].number
278
        except:
279
            pass
280
        return None
281

    
282
    def delete(self):
283
        obj_id = self.id
284
        super(Act, self).delete()
285

    
286
    class Meta:
287
        verbose_name = u"Acte"
288
        verbose_name_plural = u"Actes"
289
        ordering = ['-date', 'patient']
290

    
291

    
292
class ValidationMessage(ServiceLinkedAbstractModel):
293
    validation_date = models.DateTimeField()
294
    what = models.CharField(max_length=256)
295
    who = models.ForeignKey(User)
296
    when = models.DateTimeField(auto_now_add=True)
(4-4/9)