diff options
author | Denise sur Lya <sekhmet@lya> | 2021-06-29 23:04:02 +0200 |
---|---|---|
committer | Denise sur Lya <sekhmet@lya> | 2021-06-29 23:04:02 +0200 |
commit | ca61f310d80e05ea43efc148787f214a567743f0 (patch) | |
tree | 0f8bda76932d6ed30bf274cd441fb070a684e868 | |
parent | 639d9cecc866004800974bf4d6ffc1c1a3a59fc1 (diff) | |
download | oms-ca61f310d80e05ea43efc148787f214a567743f0.tar.gz oms-ca61f310d80e05ea43efc148787f214a567743f0.tar.zst oms-ca61f310d80e05ea43efc148787f214a567743f0.zip |
nouveauté : courbes multiples
-rw-r--r-- | app.py | 26 | ||||
-rw-r--r-- | configuration.py | 8 | ||||
-rw-r--r-- | data/changelog_data.txt | 8 | ||||
-rw-r--r-- | gestion_donnees.py | 286 | ||||
-rw-r--r-- | gestion_erreurs.py | 4 | ||||
-rw-r--r-- | static/outilspage.js | 19 | ||||
-rw-r--r-- | static/requetes.js | 4 | ||||
-rw-r--r-- | static/style.css | 6 | ||||
-rw-r--r-- | templates/index.html | 45 | ||||
-rw-r--r-- | trace_courbe.py | 47 |
10 files changed, 298 insertions, 155 deletions
@@ -8,7 +8,7 @@ import base64 | |||
8 | from matplotlib.backends.backend_agg import FigureCanvasAgg as FigureCanvas | 8 | from matplotlib.backends.backend_agg import FigureCanvasAgg as FigureCanvas |
9 | 9 | ||
10 | from configuration import config_init,CONFIG | 10 | from configuration import config_init,CONFIG |
11 | from gestion_erreurs import initialise_erreurs, debug, niveau_debug | 11 | from gestion_erreurs import initialise_erreurs, debug, niveau_debug, warning |
12 | import gestion_donnees as donnees | 12 | import gestion_donnees as donnees |
13 | from trace_courbe import cree_figure | 13 | from trace_courbe import cree_figure |
14 | import faq as f | 14 | import faq as f |
@@ -55,7 +55,27 @@ def apropos(): | |||
55 | def courbe_image(ext): | 55 | def courbe_image(ext): |
56 | liste_err = initialise_erreurs() | 56 | liste_err = initialise_erreurs() |
57 | data = flask.request.form | 57 | data = flask.request.form |
58 | 58 | enfants_add = [] # Enfants additionnels | |
59 | for i in range(2, 11): | ||
60 | if 'fichier_donnees_'+str(i) in flask.request.files: # UUn enfant à ajouter | ||
61 | fichier = flask.request.files["fichier_donnees_"+str(i)] | ||
62 | chaine = fichier.read() # On récupère la chaîne | ||
63 | if len(chaine)>=5: # Si elle a une longueur à peu près raisonnable | ||
64 | debug("Un fichier de données additionnel trouvé", liste_err) | ||
65 | formulaire_2 = donnees.fichier_json_vers_configdonnees(chaine, liste_err) | ||
66 | if formulaire_2 =={}: | ||
67 | warning("Le fichier de données additionnel est vide ou mal formaté", liste_err) | ||
68 | else: | ||
69 | #debug("Form 2 : "+str(formulaire_2), liste_err) | ||
70 | # Récupérer sous forme python | ||
71 | conf2, ljours2, listes_donnees2 = donnees.web_vers_python(formulaire_2,liste_err, court=True) | ||
72 | # Et le symbole | ||
73 | symb2 = donnees.gere_symbole(data.get("symbole_donnees_"+str(i))) | ||
74 | |||
75 | enfant2 = donnees.eclate_donnees_additionnelles(conf2, ljours2, listes_donnees2, symb2) | ||
76 | enfants_add.append(enfant2) | ||
77 | |||
78 | |||
59 | # récupérer les données du formulaire proprement | 79 | # récupérer les données du formulaire proprement |
60 | config,listes_jours,listes_donnees = donnees.web_vers_python(data,liste_err) | 80 | config,listes_jours,listes_donnees = donnees.web_vers_python(data,liste_err) |
61 | debug(" * On a récupéré et traité les données du formulaire web",liste_err) | 81 | debug(" * On a récupéré et traité les données du formulaire web",liste_err) |
@@ -75,7 +95,7 @@ def courbe_image(ext): | |||
75 | for typed in CONFIG["liste_typedonnees"]: | 95 | for typed in CONFIG["liste_typedonnees"]: |
76 | if config["tracevide"] == "oui" or listes_jours[typed] != []: | 96 | if config["tracevide"] == "oui" or listes_jours[typed] != []: |
77 | debug("On trace la courbe de "+typed,liste_err) | 97 | debug("On trace la courbe de "+typed,liste_err) |
78 | fig = cree_figure(config,listes_jours[typed],listes_donnees[typed],typed,liste_extracalculs, liste_err) | 98 | fig = cree_figure(config,listes_jours[typed],listes_donnees[typed],typed,liste_extracalculs, liste_err, enfants_add) |
79 | output = io.BytesIO() | 99 | output = io.BytesIO() |
80 | FigureCanvas(fig).print_png(output) | 100 | FigureCanvas(fig).print_png(output) |
81 | plt.close(fig) | 101 | plt.close(fig) |
diff --git a/configuration.py b/configuration.py index 2a0cb28..75088fe 100644 --- a/configuration.py +++ b/configuration.py | |||
@@ -14,6 +14,11 @@ CONFIG["arrondis_typedonnees"] = {"poids": 0.01, "taille": 1} | |||
14 | # Liste des calculs additionnels | 14 | # Liste des calculs additionnels |
15 | CONFIG["extradata"] = ["calculextradata", "calculextratemps"] | 15 | CONFIG["extradata"] = ["calculextradata", "calculextratemps"] |
16 | 16 | ||
17 | # Liste des symboles autorisés pour les enfants (matplotlib). | ||
18 | # Sous forme "item: description" | ||
19 | CONFIG["liste_symboles"] = {"o": "●", "^": "▲", "v": "▼", "<": "◀", ">": "▶", "s": "■", "p": "⬟", "P": "➕", "x": "×", "D": "◆" } | ||
20 | |||
21 | |||
17 | # fichiers | 22 | # fichiers |
18 | 23 | ||
19 | CONFIG["chemin_oms"] = "data_OMS/" | 24 | CONFIG["chemin_oms"] = "data_OMS/" |
@@ -109,6 +114,9 @@ DEFAUT["couleurs"]["fond"]= "#FFFFFF" # blanc | |||
109 | DEFAUT["couleurs"]["cadretxt"] = "#000000" # noir | 114 | DEFAUT["couleurs"]["cadretxt"] = "#000000" # noir |
110 | DEFAUT["couleurs"]["grille"] = "#7f7f7f" # gris | 115 | DEFAUT["couleurs"]["grille"] = "#7f7f7f" # gris |
111 | 116 | ||
117 | DEFAUT["symbole"] = "o" # Symbole par défaut | ||
118 | |||
119 | |||
112 | # Remplissage du formulaire, autres | 120 | # Remplissage du formulaire, autres |
113 | DEFAUT["age_0"]= "0j" | 121 | DEFAUT["age_0"]= "0j" |
114 | DEFAUT["legende"] = "oui" | 122 | DEFAUT["legende"] = "oui" |
diff --git a/data/changelog_data.txt b/data/changelog_data.txt index 25f7791..5726bce 100644 --- a/data/changelog_data.txt +++ b/data/changelog_data.txt | |||
@@ -1,3 +1,11 @@ | |||
1 | "Version 2.4","29/06/2021","<p>Plus de courbes ! | ||
2 | <ul> | ||
3 | <li>Les données et la configuration sont celles de l'enfant qu'on a saisi en ""principal""</li> | ||
4 | <li>Pour les enfants en plus, il faut déjà avoir le fichier de données déjà généré.</li> | ||
5 | <li>On peut choisir ses symboles pour la courbe ""principale"" ainsi que les autres.</li> | ||
6 | </ul> | ||
7 | </p>" | ||
8 | |||
1 | "Version 2.31","28/04/2021","<p>Les calculs de données sont améliorés : | 9 | "Version 2.31","28/04/2021","<p>Les calculs de données sont améliorés : |
2 | <ul> | 10 | <ul> |
3 | <li>Si l'âge demandé (ou la donnée) est entre les données saisies, c'est une simple interpolation linéaire entre les données existantes,</li> | 11 | <li>Si l'âge demandé (ou la donnée) est entre les données saisies, c'est une simple interpolation linéaire entre les données existantes,</li> |
diff --git a/gestion_donnees.py b/gestion_donnees.py index 7c22997..8fee99c 100644 --- a/gestion_donnees.py +++ b/gestion_donnees.py | |||
@@ -101,9 +101,13 @@ def simplifie_nom(chaine): | |||
101 | def convertit_donnee_vers_python(chaine,typedonnee,liste_err): | 101 | def convertit_donnee_vers_python(chaine,typedonnee,liste_err): |
102 | """ convertit une chaine vers un float qui est le type donnee voulu. | 102 | """ convertit une chaine vers un float qui est le type donnee voulu. |
103 | La virgule peut être . ou , et on vire d'éventuels espaces. | 103 | La virgule peut être . ou , et on vire d'éventuels espaces. |
104 | Taille invalide : on renvoie 0 avec un warning.""" | 104 | Taille invalide : on renvoie 0 avec un warning. |
105 | Si la chaine est en fait déjà au format float, on laisse tel quel""" | ||
106 | if type(chaine) == float: | ||
107 | return chaine | ||
105 | chaine2 = chaine.replace(",",".") | 108 | chaine2 = chaine.replace(",",".") |
106 | chaine2 = chaine2.replace(" ","") | 109 | chaine2 = chaine2.replace(" ","") |
110 | |||
107 | 111 | ||
108 | try: | 112 | try: |
109 | donnee = float(chaine2) | 113 | donnee = float(chaine2) |
@@ -162,12 +166,14 @@ def delta_date(date1,datenaissance): | |||
162 | 166 | ||
163 | ################### On regroupe tout ce qui gère les données en une fonction | 167 | ################### On regroupe tout ce qui gère les données en une fonction |
164 | 168 | ||
165 | def web_vers_python(data,liste_err): | 169 | def web_vers_python(data,liste_err, court=False): |
166 | """ prend en argument le dictionnaire de requête et renvoie la config, et les | 170 | """ prend en argument le dictionnaire de requête et renvoie la config, et les |
167 | tableaux de donnée""" | 171 | tableaux de donnée |
172 | court : si True est précisé, on ne met que le nom dans la config (enfant | ||
173 | additionnel)""" | ||
168 | 174 | ||
169 | # Régler la configuration | 175 | # Régler la configuration |
170 | config = gere_configuration(data,liste_err) | 176 | config = gere_configuration(data,liste_err, court) |
171 | 177 | ||
172 | # récupérer les données | 178 | # récupérer les données |
173 | listes_jours = {} | 179 | listes_jours = {} |
@@ -178,7 +184,7 @@ def web_vers_python(data,liste_err): | |||
178 | # Si on veut extrapoler au-delà du jour maxi, on adapte | 184 | # Si on veut extrapoler au-delà du jour maxi, on adapte |
179 | 185 | ||
180 | # Si on a choisi la même échelle de données | 186 | # Si on a choisi la même échelle de données |
181 | if config["memechelle"] == "oui": | 187 | if config.get("memechelle") == "oui": |
182 | config["non_sauve"]["maxi"] = calcule_max_graphique([j for lj in listes_jours.values() for j in lj]) | 188 | config["non_sauve"]["maxi"] = calcule_max_graphique([j for lj in listes_jours.values() for j in lj]) |
183 | # En cas d'extrapolation, on prend le maxi | 189 | # En cas d'extrapolation, on prend le maxi |
184 | if config["non_sauve"]["calculextradata_type"] !="" and config["non_sauve"]["calculextradata_age"]>config["non_sauve"]["maxi"]: | 190 | if config["non_sauve"]["calculextradata_type"] !="" and config["non_sauve"]["calculextradata_age"]>config["non_sauve"]["maxi"]: |
@@ -199,10 +205,20 @@ def gere_checkbox(chaine): | |||
199 | else: | 205 | else: |
200 | return "" | 206 | return "" |
201 | 207 | ||
202 | def gere_configuration(data,liste_err): | 208 | def gere_symbole(chaine): |
209 | """ prend en arg une chaîne genre "o", ">" et vérifie si c'est un symbole valide. | ||
210 | Renvoie ce symbole-là ou le défaut""" | ||
211 | if chaine in CONFIG["liste_symboles"]: | ||
212 | return chaine | ||
213 | else: | ||
214 | return DEFAUT["symbole"] | ||
215 | |||
216 | def gere_configuration(data,liste_err, court=False): | ||
203 | """ prend en argument le dictionnaire de requête (configuration imparfaite), et | 217 | """ prend en argument le dictionnaire de requête (configuration imparfaite), et |
204 | construit le dictionnaire de configuration qui va bien. | 218 | construit le dictionnaire de configuration qui va bien. |
205 | Vérifie que chaque entrée est cohérente évidemment.""" | 219 | Vérifie que chaque entrée est cohérente évidemment. |
220 | court : si mis à True, on ne met que le nom dans la configuraion, | ||
221 | ainsi que la date de naissance et le sexe""" | ||
206 | # Initialisation | 222 | # Initialisation |
207 | configuration = {"non_sauve": {}} | 223 | configuration = {"non_sauve": {}} |
208 | 224 | ||
@@ -210,137 +226,141 @@ def gere_configuration(data,liste_err): | |||
210 | nom = data.get("nom","") | 226 | nom = data.get("nom","") |
211 | # Par contre s'il est trop long on le tronque | 227 | # Par contre s'il est trop long on le tronque |
212 | configuration["nom"] = nom[:CONFIG["longueur_max_nom_bebe"]] | 228 | configuration["nom"] = nom[:CONFIG["longueur_max_nom_bebe"]] |
213 | 229 | ||
214 | sexe = data.get("sexe","") | ||
215 | if not (sexe in ["F","M","N"]): | ||
216 | warning("Le sexe de l'enfant est invalide ! "+sexe,liste_err) | ||
217 | sexe = "N" | ||
218 | configuration["sexe"] = sexe | ||
219 | |||
220 | naissance = data.get("naissance","") | 230 | naissance = data.get("naissance","") |
221 | if naissance !="": | 231 | if naissance !="": |
222 | naissance = convertit_date_vers_python(naissance,liste_err) | 232 | naissance = convertit_date_vers_python(naissance,liste_err) |
223 | configuration["naissance"] = naissance | 233 | configuration["naissance"] = naissance |
224 | 234 | ||
225 | prematurite = data.get("prematurite","") | 235 | sexe = data.get("sexe","") |
226 | j = convertit_jours_vers_python(prematurite,liste_err) | 236 | if not (sexe in ["F","M","N"]): |
227 | configuration["prematurite"] = convertit_age_vers_texte(j) | 237 | warning("Le sexe de l'enfant est invalide ! "+sexe,liste_err) |
228 | 238 | sexe = "N" | |
229 | configuration["agecorrige"] = gere_checkbox(data.get("agecorrige","")) | 239 | configuration["sexe"] = sexe |
230 | 240 | ||
231 | # Type de courbe. Au pire on met P | 241 | if not(court): |
232 | tyc = data.get("typecourbe","") | 242 | |
233 | if not (tyc in ["P","Z"]): | 243 | prematurite = data.get("prematurite","") |
234 | tyc = "P" | 244 | j = convertit_jours_vers_python(prematurite,liste_err) |
235 | configuration["typecourbe"] = tyc | 245 | configuration["prematurite"] = convertit_age_vers_texte(j) |
236 | |||
237 | # unité | ||
238 | unite = data.get("unite","") | ||
239 | if not (unite in CONFIG["liste_unites"]): | ||
240 | unite = "" | ||
241 | #warning("L'unité "+unite+" n'est pas reconnue !",liste_err) | ||
242 | configuration["unite"] = unite | ||
243 | |||
244 | # grille | ||
245 | configuration["grille"] = gere_checkbox(data.get("grille","")) | ||
246 | |||
247 | # tracer ou non les courbes vides | ||
248 | configuration["tracevide"] = gere_checkbox(data.get("tracevide","")) | ||
249 | 246 | ||
250 | # Même échelle sur tous les graphiques | 247 | configuration["agecorrige"] = gere_checkbox(data.get("agecorrige","")) |
251 | configuration["memechelle"] = gere_checkbox(data.get("memechelle","")) | 248 | |
252 | 249 | # Type de courbe. Au pire on met P | |
253 | 250 | tyc = data.get("typecourbe","") | |
254 | # maxi. 0 signifie qu'on veut pas de maxi | 251 | if not (tyc in ["P","Z"]): |
255 | maxi = data.get("maxi","") | 252 | tyc = "P" |
256 | if maxi == "": | 253 | configuration["typecourbe"] = tyc |
257 | configuration["maxi"] = 0 | 254 | |
258 | else: | 255 | # unité |
259 | configuration["maxi"] = int(convertit_jours_vers_python(maxi,liste_err)) | 256 | unite = data.get("unite","") |
257 | if not (unite in CONFIG["liste_unites"]): | ||
258 | unite = "" | ||
259 | #warning("L'unité "+unite+" n'est pas reconnue !",liste_err) | ||
260 | configuration["unite"] = unite | ||
261 | |||
262 | # grille | ||
263 | configuration["grille"] = gere_checkbox(data.get("grille","")) | ||
264 | |||
265 | # tracer ou non les courbes vides | ||
266 | configuration["tracevide"] = gere_checkbox(data.get("tracevide","")) | ||
267 | |||
268 | # Même échelle sur tous les graphiques | ||
269 | configuration["memechelle"] = gere_checkbox(data.get("memechelle","")) | ||
260 | 270 | ||
261 | # dimensions du graphique | 271 | |
262 | largeur = data.get("largeur","") | 272 | # maxi. 0 signifie qu'on veut pas de maxi |
263 | if largeur == "": | 273 | maxi = data.get("maxi","") |
264 | largeur = DEFAUT["largeur_graphique"] | 274 | if maxi == "": |
265 | else: | 275 | configuration["maxi"] = 0 |
266 | try: | 276 | else: |
267 | largeur = int(largeur) | 277 | configuration["maxi"] = int(convertit_jours_vers_python(maxi,liste_err)) |
268 | except: | 278 | |
269 | warning("La largeur "+largeur+"est invalide !",liste_err) | 279 | # dimensions du graphique |
280 | largeur = data.get("largeur","") | ||
281 | if largeur == "": | ||
270 | largeur = DEFAUT["largeur_graphique"] | 282 | largeur = DEFAUT["largeur_graphique"] |
271 | if largeur > CONFIG["largeur_graphique_max"]: | 283 | else: |
272 | largeur = CONFIG["largeur_graphique_max"] | 284 | try: |
273 | warning("Largeur du graphique trop grande !",liste_err) | 285 | largeur = int(largeur) |
274 | elif largeur < CONFIG["largeur_graphique_min"]: | 286 | except: |
275 | largeur = CONFIG["largeur_graphique_min"] | 287 | warning("La largeur "+largeur+"est invalide !",liste_err) |
276 | warning("Largeur du graphique trop petite !",liste_err) | 288 | largeur = DEFAUT["largeur_graphique"] |
277 | configuration["largeur"] = largeur | 289 | if largeur > CONFIG["largeur_graphique_max"]: |
278 | 290 | largeur = CONFIG["largeur_graphique_max"] | |
279 | hauteur = data.get("hauteur","") | 291 | warning("Largeur du graphique trop grande !",liste_err) |
280 | if hauteur == "": | 292 | elif largeur < CONFIG["largeur_graphique_min"]: |
281 | hauteur = DEFAUT["hauteur_graphique"] | 293 | largeur = CONFIG["largeur_graphique_min"] |
282 | else: | 294 | warning("Largeur du graphique trop petite !",liste_err) |
283 | try: | 295 | configuration["largeur"] = largeur |
284 | hauteur = int(hauteur) | 296 | |
285 | except: | 297 | hauteur = data.get("hauteur","") |
286 | warning("La hauteur "+hauteur+"est invalide !",liste_err) | 298 | if hauteur == "": |
287 | hauteur = DEFAUT["hauteur_graphique"] | 299 | hauteur = DEFAUT["hauteur_graphique"] |
288 | if hauteur > CONFIG["hauteur_graphique_max"]: | 300 | else: |
289 | hauteur = CONFIG["hauteur_graphique_max"] | 301 | try: |
290 | warning("Hauteur du graphique trop grande !",liste_err) | 302 | hauteur = int(hauteur) |
291 | elif hauteur < CONFIG["hauteur_graphique_min"]: | 303 | except: |
292 | hauteur = CONFIG["hauteur_graphique_min"] | 304 | warning("La hauteur "+hauteur+"est invalide !",liste_err) |
293 | warning("Hauteur du graphique trop petite !",liste_err) | 305 | hauteur = DEFAUT["hauteur_graphique"] |
294 | configuration["hauteur"] = hauteur | 306 | if hauteur > CONFIG["hauteur_graphique_max"]: |
295 | 307 | hauteur = CONFIG["hauteur_graphique_max"] | |
296 | # existence et position de la légende | 308 | warning("Hauteur du graphique trop grande !",liste_err) |
297 | configuration["legende"] = gere_checkbox(data.get("legende","")) | 309 | elif hauteur < CONFIG["hauteur_graphique_min"]: |
298 | 310 | hauteur = CONFIG["hauteur_graphique_min"] | |
299 | positionlegende = data.get("positionlegende","") | 311 | warning("Hauteur du graphique trop petite !",liste_err) |
300 | if not(positionlegende in ['upper left','upper right','lower left','lower right']): | 312 | configuration["hauteur"] = hauteur |
301 | positionlegende = "upper left" | 313 | |
302 | configuration["positionlegende"] = positionlegende | 314 | # existence et position de la légende |
315 | configuration["legende"] = gere_checkbox(data.get("legende","")) | ||
316 | |||
317 | positionlegende = data.get("positionlegende","") | ||
318 | if not(positionlegende in ['upper left','upper right','lower left','lower right']): | ||
319 | positionlegende = "upper left" | ||
320 | configuration["positionlegende"] = positionlegende | ||
321 | |||
322 | configuration["couleurs"] = {} | ||
323 | # gérer les couleurs | ||
324 | for clecouleur in DEFAUT["couleurs"]: | ||
325 | coul = rgb_vers_tuple(data.get("couleur_"+clecouleur,""),CONFIG["couleurs"][clecouleur],liste_err) | ||
326 | configuration["couleurs"][clecouleur] = coul | ||
327 | |||
328 | # symbole | ||
329 | configuration["symbole"] = gere_symbole( data.get("symbole", "")) | ||
303 | 330 | ||
304 | configuration["couleurs"] = {} | 331 | configuration["non_sauve"]["grilleamelio"] = gere_checkbox(data.get("grilleamelio","")) |
305 | # gérer les couleurs | ||
306 | for clecouleur in DEFAUT["couleurs"]: | ||
307 | coul = rgb_vers_tuple(data.get("couleur_"+clecouleur,""),CONFIG["couleurs"][clecouleur],liste_err) | ||
308 | configuration["couleurs"][clecouleur] = coul | ||
309 | |||
310 | |||
311 | configuration["non_sauve"]["grilleamelio"] = gere_checkbox(data.get("grilleamelio","")) | ||
312 | |||
313 | |||
314 | #### La partie extrapolation n'a pas besoin d'être sauvée | ||
315 | configuration["non_sauve"]["prolongercourbes"] = gere_checkbox(data.get("prolongercourbes","")) | ||
316 | |||
317 | # Valeur par défaut : 1 | ||
318 | debug(data.get("nbextradata", "aaargh"), liste_err) | ||
319 | nbextradata = data.get("nbextradata",1) | ||
320 | try: | ||
321 | nbextradata = int(nbextradata) | ||
322 | except: | ||
323 | warning("Le nombre de données sur lequel on extrapole est invalide : "+nbextradata, liste_err) | ||
324 | nbextradata = 1 | ||
325 | configuration["non_sauve"]["nbextradata"] = nbextradata | ||
326 | 332 | ||
327 | if data.get("calculextradata_type","") in CONFIG["liste_typedonnees"]: | ||
328 | configuration["non_sauve"]["calculextradata_type"] = data.get("calculextradata_type","") | ||
329 | configuration["non_sauve"]["calculextradata_age"] = convertit_jours_vers_python(data.get("calculextradata_age","0j"),liste_err) | ||
330 | else: | ||
331 | configuration["non_sauve"]["calculextradata_type"] = "" | ||
332 | # On ne met rien dans l'âge, pas la peine | ||
333 | 333 | ||
334 | ctyped = data.get("calculextratemps_type","") | 334 | #### La partie extrapolation n'a pas besoin d'être sauvée |
335 | if ctyped in CONFIG["liste_typedonnees"]: | 335 | configuration["non_sauve"]["prolongercourbes"] = gere_checkbox(data.get("prolongercourbes","")) |
336 | configuration["non_sauve"]["calculextratemps_type"] = ctyped | ||
337 | configuration["non_sauve"]["calculextratemps_val"] = convertit_donnee_vers_python(data.get("calculextratemps_val",""), ctyped, liste_err) | ||
338 | else: | ||
339 | configuration["non_sauve"]["calculextratemps_type"] = "" | ||
340 | 336 | ||
341 | # Tracer les calculs sur la grille | 337 | # Valeur par défaut : 1 |
342 | configuration["non_sauve"]["calculextradata_trace"] = gere_checkbox(data.get("calculextradata_trace")) | 338 | debug(data.get("nbextradata", "aaargh"), liste_err) |
343 | configuration["non_sauve"]["calculextratemps_trace"] = gere_checkbox(data.get("calculextratemps_trace")) | 339 | nbextradata = data.get("nbextradata",1) |
340 | try: | ||
341 | nbextradata = int(nbextradata) | ||
342 | except: | ||
343 | warning("Le nombre de données sur lequel on extrapole est invalide : "+nbextradata, liste_err) | ||
344 | nbextradata = 1 | ||
345 | configuration["non_sauve"]["nbextradata"] = nbextradata | ||
346 | |||
347 | if data.get("calculextradata_type","") in CONFIG["liste_typedonnees"]: | ||
348 | configuration["non_sauve"]["calculextradata_type"] = data.get("calculextradata_type","") | ||
349 | configuration["non_sauve"]["calculextradata_age"] = convertit_jours_vers_python(data.get("calculextradata_age","0j"),liste_err) | ||
350 | else: | ||
351 | configuration["non_sauve"]["calculextradata_type"] = "" | ||
352 | # On ne met rien dans l'âge, pas la peine | ||
353 | |||
354 | ctyped = data.get("calculextratemps_type","") | ||
355 | if ctyped in CONFIG["liste_typedonnees"]: | ||
356 | configuration["non_sauve"]["calculextratemps_type"] = ctyped | ||
357 | configuration["non_sauve"]["calculextratemps_val"] = convertit_donnee_vers_python(data.get("calculextratemps_val",""), ctyped, liste_err) | ||
358 | else: | ||
359 | configuration["non_sauve"]["calculextratemps_type"] = "" | ||
360 | |||
361 | # Tracer les calculs sur la grille | ||
362 | configuration["non_sauve"]["calculextradata_trace"] = gere_checkbox(data.get("calculextradata_trace")) | ||
363 | configuration["non_sauve"]["calculextratemps_trace"] = gere_checkbox(data.get("calculextratemps_trace")) | ||
344 | 364 | ||
345 | 365 | ||
346 | return configuration | 366 | return configuration |
@@ -492,4 +512,18 @@ def fichier_json_vers_configdonnees(chaine,liste_err): | |||
492 | return valform | 512 | return valform |
493 | 513 | ||
494 | 514 | ||
495 | \ No newline at end of file | 515 | |
516 | #### Pour l'insertion d'une 2e (ou plus) courbe sur le graphique, ue fonction qui sépare tout ça | ||
517 | def eclate_donnees_additionnelles(conf, ljours, ldonnees, symb): | ||
518 | """ conf est la config (on ne garde que le nom) pour un enfant additionnel, | ||
519 | ljours et ldonnees les dictionnaires de listes contenant les données. | ||
520 | symb est le symbole choisi pour cette courbe additionnelle (déjà vérifié) | ||
521 | On fabrique un joli dictionnaire typed -> (conf lj, ldonnee) avec le nom de l'enfant, | ||
522 | et les données pour chaque typed""" | ||
523 | |||
524 | retour = {} | ||
525 | conf["symbole"] = symb # On ajoute le symbole additionnel | ||
526 | for typed in CONFIG["liste_typedonnees"]: | ||
527 | retour[typed] = (conf, ljours[typed], ldonnees[typed]) | ||
528 | |||
529 | return retour \ No newline at end of file | ||
diff --git a/gestion_erreurs.py b/gestion_erreurs.py index 910aabd..0e75dd3 100644 --- a/gestion_erreurs.py +++ b/gestion_erreurs.py | |||
@@ -21,7 +21,9 @@ def erreur(message,listeerreurs): | |||
21 | def warning(message,listeerreurs): | 21 | def warning(message,listeerreurs): |
22 | """ En cas d'avertissement mais on peut quand même continuer """ | 22 | """ En cas d'avertissement mais on peut quand même continuer """ |
23 | print("** Warning : "+message) | 23 | print("** Warning : "+message) |
24 | listeerreurs[1].append("Alerte : "+message) | 24 | message = "Alerte : "+message |
25 | if message not in listeerreurs[1]: | ||
26 | listeerreurs[1].append(message) | ||
25 | 27 | ||
26 | def debug(message,listeerreurs): | 28 | def debug(message,listeerreurs): |
27 | global niveau_debug | 29 | global niveau_debug |
diff --git a/static/outilspage.js b/static/outilspage.js index 9491495..9b3f51d 100644 --- a/static/outilspage.js +++ b/static/outilspage.js | |||
@@ -141,3 +141,22 @@ function change_mode_dates(mode) | |||
141 | elem.innerHTML = "<span onclick=\"change_mode_dates('text')\" class=\"bouton\">Cliquez ici</span> pour saisir les dates comme du texte." | 141 | elem.innerHTML = "<span onclick=\"change_mode_dates('text')\" class=\"bouton\">Cliquez ici</span> pour saisir les dates comme du texte." |
142 | } | 142 | } |
143 | } | 143 | } |
144 | |||
145 | function revele_enfants() | ||
146 | { | ||
147 | // révèle tous les champs pour enfants additionnels | ||
148 | var listelt = document.getElementsByClassName("enfants_cache_def") ; | ||
149 | for(i=0; i<listelt.length; i++) { | ||
150 | listelt[i].style.display = "block" ; | ||
151 | } | ||
152 | // Enlève le bouton pour les afficher | ||
153 | var bouton = document.getElementById("reveleenfants") ; | ||
154 | bouton.style.display = "none" ; | ||
155 | } | ||
156 | |||
157 | function vide_champ(nom) | ||
158 | { | ||
159 | // vide le champ d'id proposé | ||
160 | var elt = document.getElementById(nom) ; | ||
161 | elt.value = "" ; | ||
162 | } | ||
diff --git a/static/requetes.js b/static/requetes.js index 22abc57..5644469 100644 --- a/static/requetes.js +++ b/static/requetes.js | |||
@@ -144,6 +144,6 @@ function download_file(filename,mimetype,data) { | |||
144 | 144 | ||
145 | } | 145 | } |
146 | 146 | ||
147 | function upload_file(data) { | 147 | function upload_file(nomformulaire) { |
148 | document.getElementById('form_import_donnees').submit(); | 148 | document.getElementById(nomformulaire).submit(); |
149 | } | 149 | } |
diff --git a/static/style.css b/static/style.css index 929bfd4..b048f11 100644 --- a/static/style.css +++ b/static/style.css | |||
@@ -52,7 +52,7 @@ img#courbe { | |||
52 | display: none; | 52 | display: none; |
53 | } | 53 | } |
54 | 54 | ||
55 | #pref_graphique, #extra { | 55 | #pref_graphique, #extra, #multi{ |
56 | display: none; | 56 | display: none; |
57 | } | 57 | } |
58 | 58 | ||
@@ -60,6 +60,10 @@ img#courbe { | |||
60 | display: none; | 60 | display: none; |
61 | } | 61 | } |
62 | 62 | ||
63 | .enfants_cache_def{ | ||
64 | display: none; | ||
65 | } | ||
66 | |||
63 | .icon_button { | 67 | .icon_button { |
64 | display: block; | 68 | display: block; |
65 | cursor: pointer; | 69 | cursor: pointer; |
diff --git a/templates/index.html b/templates/index.html index 32c45e1..65754a5 100644 --- a/templates/index.html +++ b/templates/index.html | |||
@@ -15,7 +15,7 @@ | |||
15 | <img src="static/icons/import.png"> | 15 | <img src="static/icons/import.png"> |
16 | <span class="icon_legend">Importer un fichier</span> | 16 | <span class="icon_legend">Importer un fichier</span> |
17 | </label> | 17 | </label> |
18 | <input type="file" name="fichier_donnees" id="fichier_donnees" oninput="upload_file()"> | 18 | <input type="file" name="fichier_donnees" id="fichier_donnees" oninput="upload_file('form_import_donnees')"> |
19 | </form> | 19 | </form> |
20 | </div> | 20 | </div> |
21 | 21 | ||
@@ -51,7 +51,7 @@ | |||
51 | <h3>Saisir les données</h3> | 51 | <h3>Saisir les données</h3> |
52 | <p><span id="bouton_afficher_donneesplus" class="bouton" onclick="affiche_cache_classe('donneesplus')">Afficher</span> la saisie des tailles.</p> | 52 | <p><span id="bouton_afficher_donneesplus" class="bouton" onclick="affiche_cache_classe('donneesplus')">Afficher</span> la saisie des tailles.</p> |
53 | <p>Syntaxe pour l'âge : utiliser j, s, m, a comme des "unités" (jours, semaines, mois, années). Vous pouvez mixer les unités, par exemple "3a2m5j" pour 3 ans, 2 mois et 5 jours. Les espaces sont ignorées, et il faut saisir des nombres entiers.</p> | 53 | <p>Syntaxe pour l'âge : utiliser j, s, m, a comme des "unités" (jours, semaines, mois, années). Vous pouvez mixer les unités, par exemple "3a2m5j" pour 3 ans, 2 mois et 5 jours. Les espaces sont ignorées, et il faut saisir des nombres entiers.</p> |
54 | <p>Saisir le poids en kilogrammes (par exemple "2.62" ou "2,62" pour 2 kilogrammes et 620 grammes), ou en grammes (2620 par exemple).</p> | 54 | <p>Saisir le poids en kilogrammes (par exemple "2.62" ou "2,62" pour 2 kilogrammes et 620 grammes), ou en grammes ("2620" par exemple).</p> |
55 | <p class="donneesplus">Saisir la taille en centimètres. Note : il y a généralement un centimètre de moins quand on mesure l'enfant debout par rapport à allongé. On peut voir sur la courbe de référence une "cassure" à deux ans qui correspond au changement du type de mesure.</p> | 55 | <p class="donneesplus">Saisir la taille en centimètres. Note : il y a généralement un centimètre de moins quand on mesure l'enfant debout par rapport à allongé. On peut voir sur la courbe de référence une "cassure" à deux ans qui correspond au changement du type de mesure.</p> |
56 | <p>Il faut saisir la date ou l'âge. Si les deux sont saisis, seul l'âge comptera.</p> | 56 | <p>Il faut saisir la date ou l'âge. Si les deux sont saisis, seul l'âge comptera.</p> |
57 | 57 | ||
@@ -73,8 +73,7 @@ | |||
73 | </table> | 73 | </table> |
74 | <p class="bouton" onclick="ajoutelignes()">Cliquer ici pour ajouter des lignes</p> | 74 | <p class="bouton" onclick="ajoutelignes()">Cliquer ici pour ajouter des lignes</p> |
75 | 75 | ||
76 | 76 | <!-- Préférences du graphique --> | |
77 | |||
78 | <h3>Préférences du graphique</h3> | 77 | <h3>Préférences du graphique</h3> |
79 | <div class="bouton" onclick="affiche_cache('pref_graphique',this)">Afficher</div> | 78 | <div class="bouton" onclick="affiche_cache('pref_graphique',this)">Afficher</div> |
80 | 79 | ||
@@ -109,16 +108,23 @@ | |||
109 | 108 | ||
110 | <li>Couleur courbe du bas <input type="color" name="couleur_courbe1" value="{{ valform.couleurs.courbe1 }}"> | Couleur courbe du milieu <input type="color" name="couleur_courbe2" value="{{ valform.couleurs.courbe2 }}"> | Couleur courbe du haut <input type="color" name="couleur_courbe3" value="{{ valform.couleurs.courbe3 }}"> </li> | 109 | <li>Couleur courbe du bas <input type="color" name="couleur_courbe1" value="{{ valform.couleurs.courbe1 }}"> | Couleur courbe du milieu <input type="color" name="couleur_courbe2" value="{{ valform.couleurs.courbe2 }}"> | Couleur courbe du haut <input type="color" name="couleur_courbe3" value="{{ valform.couleurs.courbe3 }}"> </li> |
111 | <li>Couleur du fond <input type="color" name="couleur_fond" value="{{ valform.couleurs.fond }}"> | Couleur de la grille <input type="color" name="couleur_grille" value="{{ valform.couleurs.grille }}"> | Couleur des axes, texte et courbe <input type="color" name="couleur_cadretxt" value="{{ valform.couleurs.cadretxt }}"></li> | 110 | <li>Couleur du fond <input type="color" name="couleur_fond" value="{{ valform.couleurs.fond }}"> | Couleur de la grille <input type="color" name="couleur_grille" value="{{ valform.couleurs.grille }}"> | Couleur des axes, texte et courbe <input type="color" name="couleur_cadretxt" value="{{ valform.couleurs.cadretxt }}"></li> |
111 | <li>Symbole pour les points de la courbe : | ||
112 | <select name="symbole"> | ||
113 | {% for symb in CONFIG.liste_symboles %} | ||
114 | <option value="{{ symb }}" {% if valform.symbole==symb %} selected{% endif %}>{{CONFIG.liste_symboles[symb] | safe}}</option> | ||
115 | {% endfor %} | ||
116 | </select> | ||
117 | </li> | ||
112 | </ul> | 118 | </ul> |
113 | 119 | ||
114 | 120 | ||
115 | |||
116 | <!-- Section calculs/Extrapolation !--> | 121 | <!-- Section calculs/Extrapolation !--> |
117 | 122 | ||
118 | <h3>Extrapolation</h3> | 123 | <h3>Extrapolation</h3> |
119 | <div class="bouton" onclick="affiche_cache('extra',this)">Afficher</div> | 124 | <div class="bouton" onclick="affiche_cache('extra',this)">Afficher</div> |
120 | 125 | ||
121 | <ul id="extra"> | 126 | <div id="extra"> |
127 | <ul> | ||
122 | <li>Calculer le prolongement de la courbe à partir des <input type="text" name="nbextradata" size="4" value="1"> dernières données (mettre 0 pour "toutes").</li> | 128 | <li>Calculer le prolongement de la courbe à partir des <input type="text" name="nbextradata" size="4" value="1"> dernières données (mettre 0 pour "toutes").</li> |
123 | <li><input type="checkbox" name="prolongercourbes">Tracer l'extrapolation sur le graphique.</li> | 129 | <li><input type="checkbox" name="prolongercourbes">Tracer l'extrapolation sur le graphique.</li> |
124 | 130 | ||
@@ -140,11 +146,32 @@ | |||
140 | <input type="checkbox" name="calculextratemps_trace"> ... et le voir sur le graphique.</li> | 146 | <input type="checkbox" name="calculextratemps_trace"> ... et le voir sur le graphique.</li> |
141 | 147 | ||
142 | </ul> | 148 | </ul> |
143 | |||
144 | <p>Les résultats de calculs seront affichés en-dessous des courbes.</p> | 149 | <p>Les résultats de calculs seront affichés en-dessous des courbes.</p> |
150 | </div> | ||
151 | |||
152 | <h3>Courbes multiples</h3> | ||
153 | <div class="bouton" onclick="affiche_cache('multi',this)">Afficher</div> | ||
154 | |||
155 | <div id="multi"> | ||
156 | <p>Si vous souhaitez tracer sur le même graphique les courbes de différents enfants, c'est ici. Il faut importer le fichier de données additionnel de l'autre enfant (ou des autres enfants). Seul le nom de l'enfant et ses données seront utilisés, les autres paramètres (courbe de référence suivant le sexe, maximum du graphique, ...) seront ceux de l'enfant « principal ».</p> | ||
157 | <ul> | ||
158 | {% for numero_enfant in (2, 3, 4, 5, 6, 7, 8, 9, 10) %} | ||
159 | <li {% if numero_enfant>4 %} class="enfants_cache_def"{% endif %}><input type="file" name="fichier_donnees_{{ numero_enfant }}" id="fichier_donnees_{{ numero_enfant }}"> | ||
160 | Symbole :<select name="symbole_donnees_{{ numero_enfant }}"> | ||
161 | {% for symb in CONFIG.liste_symboles %} | ||
162 | <option value="{{ symb }}" {% if numero_enfant == loop.index %} selected{% endif %}>{{ CONFIG.liste_symboles[symb] | safe}}</option> | ||
163 | {% endfor %} | ||
164 | </select> | ||
165 | <span class="bouton petit" onclick="vide_champ('fichier_donnees_{{ numero_enfant }}')">Enlever ce fichier</span></li> | ||
166 | {% endfor %} | ||
167 | </ul> | ||
168 | <p class="bouton" onclick="revele_enfants()" id="reveleenfants">Plus d'enfants O_o</p> | ||
169 | </div> | ||
170 | |||
145 | 171 | ||
172 | </form> <!-- Fin du gros méga formulaire --> | ||
146 | 173 | ||
147 | </form> | 174 | <!-- #################################################################################### --> |
148 | <!-- section retour --> | 175 | <!-- section retour --> |
149 | <hr> | 176 | <hr> |
150 | 177 | ||
@@ -155,7 +182,7 @@ | |||
155 | 182 | ||
156 | <hr> | 183 | <hr> |
157 | <div id="courbe_warnings"> | 184 | <div id="courbe_warnings"> |
158 | <p><strong>Alerte :</strong> la courbe a eu quelques soucis à se générer. Voici la liste des erreurs.</p> | 185 | <p><strong>Alerte :</strong> il y a eu quelques soucis lors de la génération de la (des) courbes. En voici la liste :</p> |
159 | <ul></ul> | 186 | <ul></ul> |
160 | </div> | 187 | </div> |
161 | <div id="courbe_erreurs"> | 188 | <div id="courbe_erreurs"> |
diff --git a/trace_courbe.py b/trace_courbe.py index 0c7a1d0..1188fc4 100644 --- a/trace_courbe.py +++ b/trace_courbe.py | |||
@@ -16,13 +16,16 @@ import matplotlib.pyplot as plt | |||
16 | # Ainsi que les calculs additionnels. | 16 | # Ainsi que les calculs additionnels. |
17 | 17 | ||
18 | 18 | ||
19 | def cree_figure(conf,l_jours,l_poids,typedonnee,liste_extracalculs, liste_err): | 19 | def cree_figure(conf,l_jours,l_poids,typedonnee,liste_extracalculs, liste_err, enfants_add = []): |
20 | """ conf est le dictionnaire de config. l_jours et l_poids les listes des temps (en jours) et de données | 20 | """ conf est le dictionnaire de config. l_jours et l_poids les listes des temps (en jours) et de données |
21 | (donc pas forcément du poids) | 21 | (donc pas forcément du poids) |
22 | typedonnee est le type de données (voir CONFIG["liste_typedonnees"] | 22 | typedonnee est le type de données (voir CONFIG["liste_typedonnees"] |
23 | liste_err la liste des erreurs à compléter (voir gestion_erreurs)) | 23 | liste_err la liste des erreurs à compléter (voir gestion_erreurs)) |
24 | Renvoie la figure tracée, et les calculs additionnels sont mis sous forme de chaîne dans la liste | 24 | Renvoie la figure tracée, et les calculs additionnels sont mis sous forme de chaîne dans la liste |
25 | liste_extracalculs""" | 25 | liste_extracalculs |
26 | |||
27 | Les enfants en plus sont dans la liste enfants_add. Pour chaque item de la liste, il faut prendre | ||
28 | item[typed] pour avoir accès au nom, ljours, et ldonnees""" | ||
26 | debug("debut de cree_figure. Config : "+str(conf),liste_err) | 29 | debug("debut de cree_figure. Config : "+str(conf),liste_err) |
27 | try: | 30 | try: |
28 | liste_data_labels_p,liste_data_labels_z = oms.renvoie_liste_labels(conf,CONFIG["liste_data_choisie_p"],CONFIG["liste_data_choisie_z"],liste_err) | 31 | liste_data_labels_p,liste_data_labels_z = oms.renvoie_liste_labels(conf,CONFIG["liste_data_choisie_p"],CONFIG["liste_data_choisie_z"],liste_err) |
@@ -107,16 +110,7 @@ def cree_figure(conf,l_jours,l_poids,typedonnee,liste_extracalculs, liste_err): | |||
107 | erreur("Type de courbe invalide"+conf["typecourbe"],liste_err) | 110 | erreur("Type de courbe invalide"+conf["typecourbe"],liste_err) |
108 | return "" | 111 | return "" |
109 | 112 | ||
110 | # Si y'a un nom on met "courbe de machin" | 113 | ## On finira le titre plus tard quand on aura su qui est concerné |
111 | if conf["nom"] !="": | ||
112 | titre += " de " +conf["nom"] | ||
113 | |||
114 | if prema>0: | ||
115 | titre+= ", préma de "+conf["prematurite"] | ||
116 | if conf["agecorrige"] == "oui": | ||
117 | titre+=" (courbe en âge corrigé)" | ||
118 | else: | ||
119 | titre+=" (courbe en âge réel, données OMS décalées)" | ||
120 | 114 | ||
121 | #debug("cree_figure : géré le type de courbe ok. Liste des data labels : "+str(liste_data_labels),liste_err) | 115 | #debug("cree_figure : géré le type de courbe ok. Liste des data labels : "+str(liste_data_labels),liste_err) |
122 | debug("Fichier d'où extraire les données : "+fichier_oms,liste_err) | 116 | debug("Fichier d'où extraire les données : "+fichier_oms,liste_err) |
@@ -161,8 +155,35 @@ def cree_figure(conf,l_jours,l_poids,typedonnee,liste_extracalculs, liste_err): | |||
161 | 155 | ||
162 | ### Tracé pour de bon | 156 | ### Tracé pour de bon |
163 | if l_jours != []: | 157 | if l_jours != []: |
164 | ax.plot(l_jours_conv,l_poids,label=conf["nom"],color=conf["couleurs"]["cadretxt"],marker='o') | 158 | ax.plot(l_jours_conv,l_poids,label=conf["nom"],color=conf["couleurs"]["cadretxt"],marker=conf["symbole"]) |
165 | debug("Tracé de la courbe enfant, avec les jours "+str(l_jours_conv),liste_err) | 159 | debug("Tracé de la courbe enfant, avec les jours "+str(l_jours_conv),liste_err) |
160 | |||
161 | |||
162 | listenoms = [] | ||
163 | if conf["nom"] != "": # Ajouter le nom de l'enfant | ||
164 | listenoms.append(conf["nom"]) | ||
165 | |||
166 | for enfant in enfants_add: # Enfants additionnels éventuels | ||
167 | conf_add, lj, ld = enfant[typedonnee] # On récuère les données | ||
168 | if lj != []: # pas la peine de tracer si y'a rien à tracer | ||
169 | # Ajouter le nom de cet enfant-là | ||
170 | listenoms.append(conf_add["nom"]) | ||
171 | lj_conv = u.convertit_tableau(lj,unite,liste_err) | ||
172 | ax.plot(lj_conv, ld, label=conf_add["nom"], color=conf["couleurs"]["cadretxt"],marker=conf_add["symbole"]) | ||
173 | if conf["sexe"] != conf_add["sexe"]: | ||
174 | warning("Attention, tous les enfants n'ont pas le même sexe. La courbe de référence est celle de "+conf["nom"]+" et ne sera pas forcément pertinente pour les autres. Vous pouvez éventuellement essayer la courbe neutre. Remarque : cette alerte s'affichera quand même.", liste_err) | ||
175 | |||
176 | # Si y'a un nom on met "courbe de machin" | ||
177 | if listenoms != []: | ||
178 | titre += " de " +", ".join(listenoms) | ||
179 | |||
180 | if prema>0: | ||
181 | titre+= ", préma de "+conf["prematurite"] | ||
182 | if conf["agecorrige"] == "oui": | ||
183 | titre+=" (courbe en âge corrigé)" | ||
184 | else: | ||
185 | titre+=" (courbe en âge réel, données OMS décalées)" | ||
186 | |||
166 | 187 | ||
167 | #### extrapolatios éventuelles | 188 | #### extrapolatios éventuelles |
168 | # a-t-on demndé des calculs ? | 189 | # a-t-on demndé des calculs ? |