summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--app.py26
-rw-r--r--configuration.py8
-rw-r--r--data/changelog_data.txt8
-rw-r--r--gestion_donnees.py286
-rw-r--r--gestion_erreurs.py4
-rw-r--r--static/outilspage.js19
-rw-r--r--static/requetes.js4
-rw-r--r--static/style.css6
-rw-r--r--templates/index.html45
-rw-r--r--trace_courbe.py47
10 files changed, 298 insertions, 155 deletions
diff --git a/app.py b/app.py
index c4dfc93..a803d90 100644
--- a/app.py
+++ b/app.py
@@ -8,7 +8,7 @@ import base64
8from matplotlib.backends.backend_agg import FigureCanvasAgg as FigureCanvas 8from matplotlib.backends.backend_agg import FigureCanvasAgg as FigureCanvas
9 9
10from configuration import config_init,CONFIG 10from configuration import config_init,CONFIG
11from gestion_erreurs import initialise_erreurs, debug, niveau_debug 11from gestion_erreurs import initialise_erreurs, debug, niveau_debug, warning
12import gestion_donnees as donnees 12import gestion_donnees as donnees
13from trace_courbe import cree_figure 13from trace_courbe import cree_figure
14import faq as f 14import faq as f
@@ -55,7 +55,27 @@ def apropos():
55def courbe_image(ext): 55def 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
15CONFIG["extradata"] = ["calculextradata", "calculextratemps"] 15CONFIG["extradata"] = ["calculextradata", "calculextratemps"]
16 16
17# Liste des symboles autorisés pour les enfants (matplotlib).
18# Sous forme "item: description"
19CONFIG["liste_symboles"] = {"o": "&#x25CF;", "^": "&#x25B2;", "v": "&#x25BC;", "<": "&#x25C0;", ">": "&#x25B6;", "s": "&#x25A0;", "p": "&#11039;", "P": "&#x2795;", "x": "&times", "D": "&#x25C6;" }
20
21
17# fichiers 22# fichiers
18 23
19CONFIG["chemin_oms"] = "data_OMS/" 24CONFIG["chemin_oms"] = "data_OMS/"
@@ -109,6 +114,9 @@ DEFAUT["couleurs"]["fond"]= "#FFFFFF" # blanc
109DEFAUT["couleurs"]["cadretxt"] = "#000000" # noir 114DEFAUT["couleurs"]["cadretxt"] = "#000000" # noir
110DEFAUT["couleurs"]["grille"] = "#7f7f7f" # gris 115DEFAUT["couleurs"]["grille"] = "#7f7f7f" # gris
111 116
117DEFAUT["symbole"] = "o" # Symbole par défaut
118
119
112# Remplissage du formulaire, autres 120# Remplissage du formulaire, autres
113DEFAUT["age_0"]= "0j" 121DEFAUT["age_0"]= "0j"
114DEFAUT["legende"] = "oui" 122DEFAUT["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):
101def convertit_donnee_vers_python(chaine,typedonnee,liste_err): 101def 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
165def web_vers_python(data,liste_err): 169def 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
202def gere_configuration(data,liste_err): 208def 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
216def 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
517def 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):
21def warning(message,listeerreurs): 21def 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
26def debug(message,listeerreurs): 28def 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
145function 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
157function 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
147function upload_file(data) { 147function 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&nbsp;: 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&nbsp;: 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 «&nbsp;principal&nbsp;».</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&nbsp;:</strong> la courbe a eu quelques soucis à se générer. Voici la liste des erreurs.</p> 185 <p><strong>Alerte&nbsp;:</strong> il y a eu quelques soucis lors de la génération de la (des) courbes. En voici la liste&nbsp;:</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
19def cree_figure(conf,l_jours,l_poids,typedonnee,liste_extracalculs, liste_err): 19def 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 ?