summaryrefslogtreecommitdiff
path: root/archaeological_finds/wizards.py
blob: 0465792b7db324ca10dad790759452a9182ea4b0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright (C) 2012-2016  Étienne Loks  <etienne.loks_AT_peacefrogsDOTnet>

# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 of the
# License, or (at your option) any later version.

# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU Affero General Public License for more details.

# You should have received a copy of the GNU Affero General Public License
# along with this program.  If not, see <http://www.gnu.org/licenses/>.

# See the file COPYING for details.

from django.contrib import messages
from django.core.exceptions import ObjectDoesNotExist, PermissionDenied
from ishtar_common.utils import ugettext_lazy as _, pgettext

from ishtar_common.forms import reverse_lazy
from ishtar_common.wizards import (
    Wizard,
    DeletionWizard,
    SearchWizard,
    MultipleDeletionWizard,
)
from archaeological_operations.wizards import OperationAdministrativeActWizard

from archaeological_operations.models import AdministrativeAct
from archaeological_context_records.models import ContextRecord
from . import models


class FindSearch(SearchWizard):
    model = models.Find


class FindWizard(Wizard):
    model = models.Find
    wizard_done_window = reverse_lazy("show-find")
    redirect_url = "find_modification"

    def get_current_contextrecord(self):
        step = self.steps.current
        if not step:
            return
        if step.endswith("_creation"):  # a context record has been selected
            main_form_key = "selecrecord-" + self.url_name
            try:
                idx = int(self.session_get_value(main_form_key, "pk"))
                return ContextRecord.objects.get(pk=idx)
            except (TypeError, ValueError, ObjectDoesNotExist):
                pass
        current_item = self.get_current_object()
        if current_item:
            base_finds = current_item.base_finds.all()
            if base_finds:
                return base_finds[0].context_record

    def get_current_basefinds(self):
        find = self.get_current_object()
        if not find:
            return []
        return find.base_finds.all()

    def get_form_kwargs(self, step=None):
        kwargs = super(FindWizard, self).get_form_kwargs(step)
        if step not in (
            "find-find_creation",
            "find-find_modification",
            "simplefind-find_modification",
        ):
            return kwargs
        kwargs["context_record"] = self.get_current_contextrecord()
        if step == "simplefind-find_modification":
            kwargs["base_finds"] = self.get_current_basefinds()
        return kwargs

    def get_context_data(self, form, **kwargs):
        """
        Get the operation and context record "reminder" on top of wizard forms
        """
        context = super(FindWizard, self).get_context_data(form, **kwargs)
        current_cr = self.get_current_contextrecord()
        if not current_cr or self.steps.current.startswith("select-"):
            return context
        context["reminders"] = (
            (_("Operation"), str(current_cr.operation)),
            (_("Context record"), str(current_cr)),
        )
        return context

    def get_extra_model(self, dct, m2m, form_list):
        dct = super(FindWizard, self).get_extra_model(dct, m2m, form_list)
        dct["order"] = 1
        if "pk" in dct and type(dct["pk"]) == ContextRecord:
            dct["base_finds__context_record"] = dct.pop("pk")
        return dct


class FindModificationWizard(FindWizard):
    modification = True
    main_item_select_keys = ("selec-", "selecw-")
    filter_owns = {
        "selec-find_modification": ["pk"],
        "selecw-find_modification": ["pk"],
    }
    wizard_templates = {
        "simplefind-find_modification": "ishtar/wizard/wizard_simplefind.html",
    }


class FindDeletionWizard(MultipleDeletionWizard):
    model = models.Find
    main_item_select_keys = ("selec-", "selecw-")
    fields = [
        "label",
        "material_types",
        "datings",
        "find_number",
        "object_types",
        "description",
        "conservatory_state",
        "mark",
        "preservation_to_considers",
        "integrities",
        "remarkabilities",
        "volume",
        "weight",
        "length",
        "width",
        "height",
        "diameter",
        "comment",
    ]
    redirect_url = "find_deletion"


class TreatmentSearch(SearchWizard):
    model = models.Treatment


class TreatmentBase(Wizard):
    model = models.Treatment
    wizard_done_window = reverse_lazy("show-treatment")
    base_url = ""
    saved_args = {"treatment_type_list": []}
    redirect_url = "treatment_search"

    def get_current_finds(self):
        step = self.steps.current
        if not step:
            return
        find_form_key = "selecfind-" + self.base_url
        find_ids = self.session_get_value(find_form_key, "resulting_pk")
        try:
            return [
                models.Find.objects.get(pk=int(find_id.strip()))
                for find_id in find_ids.split(",")
            ]
        except (TypeError, ValueError, AttributeError, ObjectDoesNotExist):
            pass

    def get_form_initial(self, step, data=None):
        initial = super(TreatmentBase, self).get_form_initial(step)
        base_step = "basetreatment-" + self.base_url
        if step != base_step:
            return initial
        finds = self.get_current_finds()
        if not finds:
            return initial
        locations = [find.container.location.pk for find in finds if find.container]
        # no location or multiple locations
        if not locations or len(set(locations)) != 1:
            return initial
        if not initial:
            initial = {}
        initial["location"] = locations[0]
        return initial

    def get_extra_model(self, dct, m2m, form_list):
        dct = super(TreatmentBase, self).get_extra_model(dct, m2m, form_list)
        dct["treatment_type_list"] = []
        for k, v in m2m:
            if k == "treatment_type":
                if type(v) not in (list, tuple):
                    v = [v]
                dct["treatment_type_list"] += v
        return dct


class TreatmentWizard(TreatmentBase):
    basket_step = "basetreatment-treatment_creation"
    saved_args = {"items": [], "treatment_type_list": []}
    base_url = "treatment_creation"

    def get_form_kwargs(self, step, **kwargs):
        kwargs = super(TreatmentWizard, self).get_form_kwargs(step, **kwargs)
        if self.basket_step not in step:
            return kwargs
        kwargs["user"] = self.request.user
        return kwargs

    def get_extra_model(self, dct, m2m, form_list):
        """
        Get items concerned by the treatment
        """
        dct = super(TreatmentWizard, self).get_extra_model(dct, m2m, form_list)
        if "resulting_pk" in dct:
            dct["items"] = []
            pks = dct.pop("resulting_pk")
            if isinstance(pks, models.Find):
                pks = [pks]
            if not isinstance(pks, (list, tuple)):
                pks = str(pks).split(",")

            for pk in pks:
                if isinstance(pk, models.Find):
                    find = pk
                else:
                    try:
                        find = models.Find.objects.get(pk=pk)
                    except models.Find.DoesNotExist:
                        raise PermissionDenied
                dct["items"].append(find)
        if "basket" in dct:
            basket = dct.pop("basket")
            if basket.user.pk != dct["history_modifier"].pk:
                raise PermissionDenied
            dct["items"] = list(basket.items.all())

        if "items" in dct:
            for find in dct["items"]:
                if "own" in self.current_right and not find.is_own(
                    dct["history_modifier"]
                ):
                    raise PermissionDenied
        return dct


class TreatmentModificationWizard(TreatmentWizard):
    modification = True


class TreatmentN1Wizard(TreatmentBase):
    saved_args = {
        "upstream_items": [],
        "resulting_find": None,
        "treatment_type_list": [],
    }
    base_url = "treatment_creation_n1"

    def _update_simple_initial_from_finds(self, initial, find, k):
        r_k = "resulting_" + k
        if getattr(find, k) and r_k in initial:
            # pop k when value is inconsistent between finds
            if initial[r_k] and initial[r_k] != getattr(find, k).pk:
                initial.pop(r_k)
            else:
                initial[r_k] = getattr(find, k).pk
        return initial

    def _update_multi_initial_from_finds(self, initial, find, k):
        r_k = "resulting_" + k
        for value in getattr(find, k + "s").all():
            if value.pk not in initial[r_k]:
                initial[r_k].append(value.pk)
        return initial

    def _update_num_initial_from_finds(self, initial, find, k):
        r_k = "resulting_" + k
        if not getattr(find, k):
            return initial
        if initial[r_k] is None:
            initial[r_k] = 0
        initial[r_k] += getattr(find, k)
        return initial

    def _update_char_initial_from_finds(self, initial, find, k, sep=" ; "):
        r_k = "resulting_" + k
        value = getattr(find, k)
        if not value:
            return initial
        value = value.strip()
        if not initial[r_k]:
            initial[r_k] = value
        else:
            # new value is entirely inside the current value
            if (
                value == initial[r_k]
                or (value + sep) in initial[r_k]
                or (sep + value) in initial[r_k]
            ):
                return initial
            initial[r_k] += sep + value
        return initial

    def get_form_initial(self, step, data=None):
        initial = super(TreatmentN1Wizard, self).get_form_initial(step)
        if step != "resultingfind-treatment_creation_n1":
            return initial
        finds = self.get_current_finds()
        if not finds:
            return initial
        simple_key = ["material_type_quality"]
        multi_key = ["material_type", "object_type", "communicabilitie"]
        numeric_key = [
            "find_number",
            "min_number_of_individuals",
            "length",
            "width",
            "height",
            "diameter",
            "circumference",
            "thickness",
            "volume",
            "weight",
            "clutter_long_side",
            "clutter_short_side",
            "clutter_height"
        ]
        desc_key = ["decoration", "inscription", "comment", "dating_comment",
                    "description", "dimensions_comment"]
        char_key = ["manufacturing_place"]

        for k in simple_key + numeric_key + desc_key + char_key:
            initial["resulting_" + k] = None
        for k in multi_key:
            initial["resulting_" + k] = []

        for find in finds:
            for k in simple_key:
                initial = self._update_simple_initial_from_finds(initial, find, k)
            for k in multi_key:
                initial = self._update_multi_initial_from_finds(initial, find, k)
            for k in numeric_key:
                initial = self._update_num_initial_from_finds(initial, find, k)
            for k in char_key:
                initial = self._update_char_initial_from_finds(
                    initial, find, k, sep=" ; "
                )
            for k in desc_key:
                initial = self._update_char_initial_from_finds(
                    initial, find, k, sep="\n"
                )

        for k in list(initial.keys()):
            if initial[k] is None:
                initial.pop(k)
        return initial

    def get_extra_model(self, dct, m2m, form_list):
        """
        Get items concerned by the treatment
        """
        dct = super(TreatmentN1Wizard, self).get_extra_model(dct, m2m, form_list)
        if "resulting_pk" not in dct:
            return dct

        dct["upstream_items"] = []
        # manage upstream items
        pks = dct.pop("resulting_pk")
        if hasattr(pks, "split"):
            pks = pks.split(",")  # string
        for pk in pks:
            if isinstance(pk, models.Find):
                find = pk
            else:
                try:
                    find = models.Find.objects.get(pk=pk)
                except models.Find.DoesNotExist:
                    raise PermissionDenied
            dct["upstream_items"].append(find)

        for find in dct["upstream_items"]:
            if "own" in self.current_right and not find.is_own(dct["history_modifier"]):
                raise PermissionDenied

        # extract data of the new find
        dct["resulting_find"] = {}
        for k in list(dct.keys()):
            if k.startswith("resulting_") and k != "resulting_find":
                dct["resulting_find"][k[len("resulting_") :]] = dct.pop(k)
        return dct


class Treatment1NWizard(TreatmentBase):
    saved_args = {
        "upstream_item": None,
        "resulting_finds": None,
        "treatment_type_list": [],
    }
    base_url = "treatment_creation_1n"
    redirect_url = "find_modification"
    open_created_in_redirect = False

    def get_form_kwargs(self, step, **kwargs):
        kwargs = super(Treatment1NWizard, self).get_form_kwargs(step, **kwargs)
        if step != "resultingfind-treatment_creation_1n":
            return kwargs
        kwargs["user"] = self.request.user
        return kwargs

    def get_form_initial(self, step, data=None):
        initial = super(Treatment1NWizard, self).get_form_initial(step)
        if step != "resultingfinds-treatment_creation_1n":
            return initial
        finds = self.get_current_finds()
        if not finds:
            return initial
        lbl = finds[0].label
        initial["resultings_basket_name"] = str(_("Basket")) + " - " + lbl
        initial["resultings_label"] = lbl + "-"
        return initial

    def get_extra_model(self, dct, m2m, form_list):
        """
        Get items concerned by the treatment
        """
        dct = super(Treatment1NWizard, self).get_extra_model(dct, m2m, form_list)
        if "resulting_pk" not in dct:
            return dct

        # manage upstream item
        pk = dct.pop("resulting_pk")
        if isinstance(pk, models.Find):
            find = pk
        else:
            try:
                find = models.Find.objects.get(pk=pk)
            except models.Find.DoesNotExist:
                raise PermissionDenied
        dct["upstream_item"] = find

        if "own" in self.current_right and not find.is_own(dct["history_modifier"]):
            raise PermissionDenied

        # extract attributes to generate the new find
        dct["resulting_finds"] = {}
        for k in list(dct.keys()):
            if k.startswith("resultings_"):
                dct["resulting_finds"][k[len("resultings_") :]] = dct.pop(k)
        messages.add_message(
            self.request,
            messages.INFO,
            str(
                _(
                    'The new basket: "{}" have been created with the '
                    "resulting items. This search have been pinned."
                )
            ).format(dct["resulting_finds"]["basket_name"]),
        )
        self.request.session["pin-search-find"] = '{}="{}"'.format(
            str(pgettext("key for text search", "basket")),
            dct["resulting_finds"]["basket_name"],
        )
        self.request.session["find"] = ""
        return dct


class TreatmentDeletionWizard(DeletionWizard):
    model = models.Treatment
    wizard_confirm = "ishtar/wizard/wizard_treatement_deletion.html"
    fields = [
        "label",
        "other_reference",
        "year",
        "index",
        "treatment_types",
        "location",
        "person",
        "organization",
        "external_id",
        "comment",
        "description",
        "goal",
        "start_date",
        "end_date",
        "container",
    ]
    redirect_url = "treatment_deletion"


class TreatmentAdministrativeActWizard(OperationAdministrativeActWizard):
    model = models.Treatment
    current_obj_slug = "administrativeacttreatment"
    ref_object_key = "treatment"
    redirect_url = "treatment_admacttreatment_modification"

    def get_reminder(self):
        return


class TreatmentEditAdministrativeActWizard(TreatmentAdministrativeActWizard):
    model = AdministrativeAct
    edit = True

    def get_associated_item(self, dct):
        return self.get_current_object().treatment


class TreatmentFileSearch(SearchWizard):
    model = models.TreatmentFile


class TreatmentFileWizard(Wizard):
    model = models.TreatmentFile
    wizard_done_window = reverse_lazy("show-treatmentfile")
    redirect_url = "treatmentfile_modification"


class TreatmentFileModificationWizard(TreatmentFileWizard):
    modification = True


class TreatmentFileDeletionWizard(MultipleDeletionWizard):
    model = models.TreatmentFile
    fields = [
        "name",
        "internal_reference",
        "external_id",
        "year",
        "index",
        "type",
        "in_charge",
        "reception_date",
        "creation_date",
        "end_date",
        "comment",
    ]
    redirect_url = "treatmentfile_deletion"


class TreatmentFileAdministrativeActWizard(OperationAdministrativeActWizard):
    model = models.TreatmentFile
    current_obj_slug = "administrativeacttreatmentfile"
    ref_object_key = "treatment_file"
    redirect_url = "treatmentfle_admacttreatmentfle_modification"

    def get_reminder(self):
        form_key = "selec-" + self.url_name
        if self.url_name.endswith("_administrativeactop"):
            # modification and deletion are suffixed with '_modification'
            # and '_deletion' so it is creation
            pk = self.session_get_value(form_key, "pk")
            try:
                return (
                    (
                        _("Treatment request"),
                        str(models.TreatmentFile.objects.get(pk=pk)),
                    ),
                )
            except models.TreatmentFile.DoesNotExist:
                return
        else:
            admin_id = self.session_get_value(form_key, "pk")
            try:
                admin = AdministrativeAct.objects.get(pk=admin_id)
                if not admin.operation:
                    return
                return ((_("Operation"), str(admin.operation)),)
            except AdministrativeAct.DoesNotExist:
                return


class TreatmentFileEditAdministrativeActWizard(TreatmentFileAdministrativeActWizard):
    model = AdministrativeAct
    edit = True

    def get_associated_item(self, dct):
        return self.get_current_object().treatment_file


class FindBasketSearch(SearchWizard):
    model = models.FindBasket


class FindBasketWizard(Wizard):
    model = models.FindBasket
    wizard_done_window = reverse_lazy("show-findbasket")
    redirect_url = "find_basket_modification"


class FindBasketEditWizard(FindBasketWizard):
    edit = True
    alt_is_own_method = "get_write_query_owns"

    def get_form_kwargs(self, step, **kwargs):
        kwargs = super(FindBasketEditWizard, self).get_form_kwargs(step, **kwargs)
        if step != "basket-find_basket_modification":
            return kwargs
        kwargs["basket_pk"] = self.get_current_object().pk
        kwargs["user"] = self.request.user
        return kwargs


class FindBasketDeletionWizard(DeletionWizard):
    model = models.FindBasket
    redirect_url = "find_basket_deletion"
    wizard_confirm = "ishtar/wizard/wizard_findbasket_deletion.html"