summaryrefslogtreecommitdiff
path: root/ishtar_common/jinja_filters.py
blob: cf19ef6c7b3d8589a1fec8799107f4d6705c722c (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
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright (C) 2025 Étienne Loks  <etienne.loks at iggdrasil dot net>
# 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 datetime import datetime
from jinja2.environment import Environment
from jinja2.filters import environmentfilter
import locale
from num2words import num2words
import re

from django.conf import settings


def set_locale():
    language_code = settings.LANGUAGE_CODE.split("-")
    language_code = language_code[0] + "_" + language_code[1].upper()
    for language_suffix in (".utf8", ""):
        try:
            locale.setlocale(locale.LC_TIME, language_code + language_suffix)
            break
        except locale.Error:
            pass


@environmentfilter
def float_format(*args):
    """
    10350.5 -> 10 350,5
    5 -> 5
    5.449999 -> 5,45
    """
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    if value is None or value == "":
        return ""
    try:
        value = float(value)
    except ValueError:
        return ""
    locale.setlocale(locale.LC_ALL, "fr_FR.UTF-8")
    if int(value) != value:
        value = float(f"{value:.2f}")
    return f"{value:n}"


@environmentfilter
def euro_format(*args):
    """
    15000 -> 15 000,00 €
    5 -> 5,00 €
    """
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    value = float_format(value)
    if not value:
        return ""
    parts = value.split(",")
    if len(parts) < 2:
        return value + ",00 €"
    elif len(parts[1]) == 1:
        return value + "0 €"
    return value + " €"


@environmentfilter
def number_to_words(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    if value is None or value == "":
        return ""
    try:
        value = float(value)
    except ValueError:
        return ""
    return num2words(value, lang=settings.LANGUAGE_CODE.split("-")[0])


@environmentfilter
def replace_line_breaks(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    return (value or "").replace("\r\n", "\n")


@environmentfilter
def capfirst_filter(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    return value[0].upper() + value[1:] if value else value


@environmentfilter
def lowerfirst_filter(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    return value[0].lower() + value[1:] if value else value


RE_CAP = re.compile(r"[^-' ]+")
SEP = ("un", "une", "le", "la", "les", "lez", "d", "l", "de", "des", "du", "sur",
       "sous", "en")


@environmentfilter
def capitalize_filter(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    if not value:
        return ""
    value = value.lower()
    res = ""
    for m in RE_CAP.finditer(value):
        start = m.start()
        if start:
            res += value[start - 1]
        v = m.group()
        if v not in SEP:
            v = v[0].upper() + v[1:]
        res += v
    return res


@environmentfilter
def human_date_filter(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    try:
        value = datetime.strptime(value, "%Y-%m-%d")
    except (ValueError, TypeError) as __:
        return ""
    set_locale()
    return value.strftime(settings.DATE_FORMAT)


@environmentfilter
def short_date_filter(*args):
    value = args[0] if len(args) == 1 else args[1]  # jinja simple filter
    try:
        value = datetime.strptime(value, "%Y-%m-%d")
    except (ValueError, TypeError) as __:
        return ""
    set_locale()
    return value.strftime(settings.SHORT_DATE_FORMAT)


@environmentfilter
def splitpart(*args):
    """
    Split string.
    First arg is index of first item to get
    Second arg is last index of items to get not included (not mandatory - default is one item) - 0 to get just one
    Third arg is character for split - default is ","
    Last arg is character for join - default is ","

    {{"9,2,10"|splitpart(1)}} -> 2
    {{"chaise;bureau;papier;paragraphe"|splitpart(0,0,";")}} -> chaise
    {{"182025_C001"|splitpart(1,0,"_")}} -> C001
    {{"chaise;bureau;papier;paragraphe"|splitpart(1,3,";")}} -> bureau;papier
    {{"chaise;bureau;papier;paragraphe"|splitpart(1,4,";","|")}} -> bureau|papier|paragraphe
    """
    # manage delta for environment filter
    delta = 0
    if isinstance(args[0], Environment):
        delta = 1
    value = args[0 + delta]
    index = args[1 + delta]
    index_end = args[2 + delta] if len(args) > (2 + delta) else None
    char = args[3 + delta] if len(args) > (3 + delta) else ","
    merge_character = args[4 + delta] if len(args) > (4 + delta) else None
    if index_end:
        try:
            index_end = int(index_end)
            if not merge_character:  # merge is assumed
                merge_character = char
        except ValueError:
            # old filter use - manage compatibility
            merge_character = char
            char = index_end
            index_end = None
    if not value or (not index and index != 0):
        return ""
    if merge_character is True:  # old filter use
        merge_character = char
    splited = value.split(char)
    if len(splited) <= index:
        return ""
    if not merge_character:
        return splited[index]
    if index_end:
        splited = splited[index:index_end]
    else:
        splited = splited[index:]
    return merge_character.join(splited)