Delete app.py
Browse files
app.py
DELETED
@@ -1,409 +0,0 @@
|
|
1 |
-
from flask import Flask, render_template, request, send_file
|
2 |
-
import os
|
3 |
-
import convertapi
|
4 |
-
from docx import Document
|
5 |
-
from docx.shared import Pt, Cm, RGBColor
|
6 |
-
from docx.enum.text import WD_ALIGN_PARAGRAPH
|
7 |
-
from docx.enum.table import WD_TABLE_ALIGNMENT
|
8 |
-
from docx.oxml.ns import nsdecls
|
9 |
-
from docx.oxml import parse_xml
|
10 |
-
|
11 |
-
# Configuration de ConvertAPI
|
12 |
-
convertapi.api_secret = 'secret_8wCI6pgOP9AxLVJG'
|
13 |
-
|
14 |
-
# --- Classe de génération de document ---
|
15 |
-
class EvaluationGymnique:
|
16 |
-
# Le reste de la classe reste inchangé...
|
17 |
-
def __init__(self):
|
18 |
-
self.document = Document()
|
19 |
-
self.document.sections[0].page_height = Cm(29.7)
|
20 |
-
self.document.sections[0].page_width = Cm(21)
|
21 |
-
self.document.sections[0].left_margin = Cm(1.5)
|
22 |
-
self.document.sections[0].right_margin = Cm(1.5)
|
23 |
-
self.document.sections[0].top_margin = Cm(1)
|
24 |
-
self.document.sections[0].bottom_margin = Cm(1)
|
25 |
-
|
26 |
-
# Informations d'en-tête par défaut
|
27 |
-
self.centre_examen = "Centre d'examen"
|
28 |
-
self.type_examen = "Bac Général"
|
29 |
-
self.serie = "Série"
|
30 |
-
self.etablissement = "Établissement"
|
31 |
-
self.session = "2025"
|
32 |
-
self.nom_candidat = "Candidat"
|
33 |
-
|
34 |
-
# Liste vide pour les éléments techniques
|
35 |
-
self.elements_techniques = []
|
36 |
-
self.appreciations = ["M", "PM", "NM", "NR"]
|
37 |
-
|
38 |
-
def ajouter_entete_colore(self):
|
39 |
-
header_paragraph = self.document.add_paragraph()
|
40 |
-
header_paragraph.alignment = WD_ALIGN_PARAGRAPH.CENTER
|
41 |
-
header_paragraph.space_after = Pt(6)
|
42 |
-
header_run = header_paragraph.add_run("ÉVALUATION GYMNASTIQUE")
|
43 |
-
header_run.bold = True
|
44 |
-
header_run.font.size = Pt(14)
|
45 |
-
header_run.font.color.rgb = RGBColor(0, 32, 96)
|
46 |
-
|
47 |
-
header_table = self.document.add_table(rows=3, cols=2)
|
48 |
-
header_table.style = 'Table Grid'
|
49 |
-
header_table.autofit = False
|
50 |
-
|
51 |
-
for row in header_table.rows:
|
52 |
-
row.height = Cm(0.8)
|
53 |
-
for row in header_table.rows:
|
54 |
-
for cell in row.cells:
|
55 |
-
shading_elm = parse_xml(f'<w:shd {nsdecls("w")} w:fill="D9E2F3"/>')
|
56 |
-
cell._tc.get_or_add_tcPr().append(shading_elm)
|
57 |
-
|
58 |
-
# Première ligne
|
59 |
-
cell = header_table.cell(0, 0)
|
60 |
-
paragraph = cell.paragraphs[0]
|
61 |
-
run = paragraph.add_run("Centre d'examen: ")
|
62 |
-
run.bold = True
|
63 |
-
run.font.size = Pt(10)
|
64 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
65 |
-
paragraph.add_run(self.centre_examen).font.size = Pt(10)
|
66 |
-
|
67 |
-
cell = header_table.cell(0, 1)
|
68 |
-
paragraph = cell.paragraphs[0]
|
69 |
-
run = paragraph.add_run("Examen: ")
|
70 |
-
run.bold = True
|
71 |
-
run.font.size = Pt(10)
|
72 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
73 |
-
paragraph.add_run(self.type_examen).font.size = Pt(10)
|
74 |
-
|
75 |
-
# Deuxième ligne
|
76 |
-
cell = header_table.cell(1, 0)
|
77 |
-
paragraph = cell.paragraphs[0]
|
78 |
-
run = paragraph.add_run("Série: ")
|
79 |
-
run.bold = True
|
80 |
-
run.font.size = Pt(10)
|
81 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
82 |
-
paragraph.add_run(self.serie).font.size = Pt(10)
|
83 |
-
|
84 |
-
cell = header_table.cell(1, 1)
|
85 |
-
paragraph = cell.paragraphs[0]
|
86 |
-
run = paragraph.add_run("Établissement: ")
|
87 |
-
run.bold = True
|
88 |
-
run.font.size = Pt(10)
|
89 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
90 |
-
paragraph.add_run(self.etablissement).font.size = Pt(10)
|
91 |
-
|
92 |
-
# Troisième ligne
|
93 |
-
cell = header_table.cell(2, 0)
|
94 |
-
paragraph = cell.paragraphs[0]
|
95 |
-
run = paragraph.add_run("Session: ")
|
96 |
-
run.bold = True
|
97 |
-
run.font.size = Pt(10)
|
98 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
99 |
-
paragraph.add_run(self.session).font.size = Pt(10)
|
100 |
-
|
101 |
-
cell = header_table.cell(2, 1)
|
102 |
-
paragraph = cell.paragraphs[0]
|
103 |
-
run = paragraph.add_run("Candidat: ")
|
104 |
-
run.bold = True
|
105 |
-
run.font.size = Pt(10)
|
106 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
107 |
-
paragraph.add_run(self.nom_candidat).font.size = Pt(10)
|
108 |
-
|
109 |
-
self.document.add_paragraph().space_after = Pt(4)
|
110 |
-
|
111 |
-
def creer_tableau_elements(self):
|
112 |
-
table = self.document.add_table(rows=len(self.elements_techniques) + 1, cols=5)
|
113 |
-
table.style = 'Table Grid'
|
114 |
-
table.alignment = WD_TABLE_ALIGNMENT.CENTER
|
115 |
-
|
116 |
-
# Configuration des largeurs de colonnes
|
117 |
-
for cell in table.columns[0].cells:
|
118 |
-
cell.width = Cm(8)
|
119 |
-
for cell in table.columns[1].cells:
|
120 |
-
cell.width = Cm(3)
|
121 |
-
for cell in table.columns[2].cells:
|
122 |
-
cell.width = Cm(2)
|
123 |
-
for cell in table.columns[3].cells:
|
124 |
-
cell.width = Cm(2.5)
|
125 |
-
for cell in table.columns[4].cells:
|
126 |
-
cell.width = Cm(2.5)
|
127 |
-
|
128 |
-
for row in table.rows:
|
129 |
-
row.height = Cm(1)
|
130 |
-
|
131 |
-
# En-tête du tableau
|
132 |
-
header_row = table.rows[0]
|
133 |
-
for cell in header_row.cells:
|
134 |
-
shading_elm = parse_xml(f'<w:shd {nsdecls("w")} w:fill="BDD7EE"/>')
|
135 |
-
cell._tc.get_or_add_tcPr().append(shading_elm)
|
136 |
-
|
137 |
-
headers = ["ELEMENTS TECHNIQUES", "CATEGORIES D'ELEMENTS TECHNIQUES ET PONDERATION", "", "APPRECIATIONS", "POINTS Accordés"]
|
138 |
-
for i, header in enumerate(headers):
|
139 |
-
cell = table.cell(0, i)
|
140 |
-
cell.text = header
|
141 |
-
cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
142 |
-
run = cell.paragraphs[0].runs[0]
|
143 |
-
run.bold = True
|
144 |
-
run.font.size = Pt(9)
|
145 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
146 |
-
table.cell(0, 1).merge(table.cell(0, 2))
|
147 |
-
|
148 |
-
# Ajout des éléments techniques
|
149 |
-
for i, element in enumerate(self.elements_techniques, 1):
|
150 |
-
element_cell = table.cell(i, 0)
|
151 |
-
element_cell.text = element["nom"]
|
152 |
-
element_cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.LEFT
|
153 |
-
run = element_cell.paragraphs[0].runs[0]
|
154 |
-
run.bold = True
|
155 |
-
run.font.size = Pt(9)
|
156 |
-
|
157 |
-
categorie_cell = table.cell(i, 1)
|
158 |
-
categorie_cell.text = element["categorie"]
|
159 |
-
categorie_cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
160 |
-
run = categorie_cell.paragraphs[0].runs[0]
|
161 |
-
run.bold = True
|
162 |
-
run.font.size = Pt(9)
|
163 |
-
run.italic = True
|
164 |
-
|
165 |
-
points_cell = table.cell(i, 2)
|
166 |
-
points_cell.text = str(element["points"])
|
167 |
-
points_cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
168 |
-
run = points_cell.paragraphs[0].runs[0]
|
169 |
-
run.bold = True
|
170 |
-
run.font.size = Pt(9)
|
171 |
-
run.italic = True
|
172 |
-
|
173 |
-
def ajouter_note_jury(self):
|
174 |
-
para = self.document.add_paragraph()
|
175 |
-
para.space_before = Pt(4)
|
176 |
-
para.space_after = Pt(4)
|
177 |
-
run = para.add_run("NB1 : Zone réservée aux membres du jury ! Le jury cochera le point correspondant au niveau de réalisation de l'élément gymnique par le candidat.")
|
178 |
-
run.bold = True
|
179 |
-
run.font.color.rgb = RGBColor(255, 0, 0)
|
180 |
-
run.font.size = Pt(9)
|
181 |
-
|
182 |
-
def creer_tableau_recapitulatif(self):
|
183 |
-
note_table = self.document.add_table(rows=3, cols=13)
|
184 |
-
note_table.style = 'Table Grid'
|
185 |
-
note_table.alignment = WD_TABLE_ALIGNMENT.CENTER
|
186 |
-
|
187 |
-
for row in note_table.rows:
|
188 |
-
row.height = Cm(0.6)
|
189 |
-
for cell in note_table.rows[0].cells:
|
190 |
-
shading_elm = parse_xml(f'<w:shd {nsdecls("w")} w:fill="BDD7EE"/>')
|
191 |
-
cell._tc.get_or_add_tcPr().append(shading_elm)
|
192 |
-
|
193 |
-
for col, (type_lettre, points) in enumerate([("A", "1pt"), ("B", "1,5pt"), ("C", "2pts"), ("D", "2,5pts"), ("E", "3pts")]):
|
194 |
-
idx = col * 2
|
195 |
-
if idx + 1 < len(note_table.columns):
|
196 |
-
cell = note_table.cell(0, idx)
|
197 |
-
cell.merge(note_table.cell(0, idx + 1))
|
198 |
-
cell.text = f"Type {type_lettre}\n{points}"
|
199 |
-
cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
200 |
-
for run in cell.paragraphs[0].runs:
|
201 |
-
run.bold = True
|
202 |
-
run.font.size = Pt(9)
|
203 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
204 |
-
|
205 |
-
for col, (titre, points) in enumerate([("ROV", "2pts"), ("Projet", "2pts"), ("Réalisation", "16pts")], 10):
|
206 |
-
if col < len(note_table.columns):
|
207 |
-
cell = note_table.cell(0, col)
|
208 |
-
cell.text = f"{titre}\n{points}"
|
209 |
-
cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
210 |
-
for run in cell.paragraphs[0].runs:
|
211 |
-
run.bold = True
|
212 |
-
run.font.size = Pt(9)
|
213 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
214 |
-
|
215 |
-
for col in range(5):
|
216 |
-
idx = col * 2
|
217 |
-
if idx + 1 < len(note_table.columns):
|
218 |
-
neg_cell = note_table.cell(1, idx)
|
219 |
-
neg_cell.text = "NEG"
|
220 |
-
neg_cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
221 |
-
for run in neg_cell.paragraphs[0].runs:
|
222 |
-
run.italic = True
|
223 |
-
run.font.size = Pt(9)
|
224 |
-
|
225 |
-
note_cell = note_table.cell(1, idx + 1)
|
226 |
-
note_cell.text = "Note"
|
227 |
-
note_cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
228 |
-
for run in note_cell.paragraphs[0].runs:
|
229 |
-
run.italic = True
|
230 |
-
run.font.size = Pt(9)
|
231 |
-
|
232 |
-
for col in range(10, 13):
|
233 |
-
if col < len(note_table.columns):
|
234 |
-
cell = note_table.cell(1, col)
|
235 |
-
cell.text = "Note"
|
236 |
-
cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
237 |
-
for run in cell.paragraphs[0].runs:
|
238 |
-
run.italic = True
|
239 |
-
run.font.size = Pt(9)
|
240 |
-
|
241 |
-
for col, type_lettre in enumerate(["A", "B", "C", "D", "E"]):
|
242 |
-
idx = col * 2
|
243 |
-
if idx < len(note_table.columns):
|
244 |
-
neg_cell = note_table.cell(2, idx)
|
245 |
-
neg_cell.text = ""
|
246 |
-
neg_cell.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
247 |
-
|
248 |
-
def ajouter_note_candidat_avec_cadre(self):
|
249 |
-
note_table = self.document.add_table(rows=1, cols=1)
|
250 |
-
note_table.style = 'Table Grid'
|
251 |
-
note_table.alignment = WD_TABLE_ALIGNMENT.CENTER
|
252 |
-
|
253 |
-
cell = note_table.cell(0, 0)
|
254 |
-
shading_elm = parse_xml(f'<w:shd {nsdecls("w")} w:fill="E2EFDA"/>')
|
255 |
-
cell._tc.get_or_add_tcPr().append(shading_elm)
|
256 |
-
|
257 |
-
p = cell.paragraphs[0]
|
258 |
-
run = p.add_run("NB2: Après le choix des catégories d'éléments gymniques par le candidat, ce dernier remplira la colonne de pointage selon l'orientation suivante: A (0.25; 0.5; 0.75; 1) B (0.25; 0.5; 0.75; 1; 1.25; 1.5) C (0.5; 0.75; 1; 1.25; 1.5; 2) D (0.75; 1; 1.25; 1.5; 2; 2.5) et E (0.75; 1; 1.5; 2; 2.5; 3) également, le candidat devra fournir 2 copies de son projet sur une page! (appréciations: NR, NM, PM, M).")
|
259 |
-
run.italic = True
|
260 |
-
run.font.size = Pt(8)
|
261 |
-
|
262 |
-
def ajouter_zone_note(self):
|
263 |
-
zone_note = self.document.add_table(rows=1, cols=2)
|
264 |
-
zone_note.style = 'Table Grid'
|
265 |
-
zone_note.alignment = WD_TABLE_ALIGNMENT.RIGHT
|
266 |
-
|
267 |
-
# Définir une largeur fixe pour les deux cellules
|
268 |
-
cell_width = Cm(2)
|
269 |
-
|
270 |
-
cell_libelle = zone_note.cell(0, 0)
|
271 |
-
cell_libelle.width = cell_width # Même largeur que la cellule de note
|
272 |
-
cell_libelle.text = "Note finale"
|
273 |
-
cell_libelle.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.RIGHT
|
274 |
-
run = cell_libelle.paragraphs[0].runs[0]
|
275 |
-
run.bold = True
|
276 |
-
run.font.size = Pt(12)
|
277 |
-
run.font.color.rgb = RGBColor(0, 32, 96)
|
278 |
-
|
279 |
-
cell_saisie = zone_note.cell(0, 1)
|
280 |
-
cell_saisie.width = cell_width # Largeur fixe
|
281 |
-
zone_note.rows[0].height = cell_width # Hauteur identique à la largeur pour être carré
|
282 |
-
shading_elm = parse_xml(f'<w:shd {nsdecls("w")} w:fill="F2F2F2"/>')
|
283 |
-
cell_saisie._tc.get_or_add_tcPr().append(shading_elm)
|
284 |
-
cell_saisie.text = "/20"
|
285 |
-
cell_saisie.paragraphs[0].alignment = WD_ALIGN_PARAGRAPH.CENTER
|
286 |
-
run = cell_saisie.paragraphs[0].runs[0]
|
287 |
-
run.bold = True
|
288 |
-
run.font.size = Pt(12)
|
289 |
-
|
290 |
-
# Ajouter plus d'espace avant la liste des correcteurs
|
291 |
-
self.document.add_paragraph().space_after = Pt(12)
|
292 |
-
self.document.add_paragraph().space_after = Pt(12) # Un deuxième paragraphe pour plus d'espace
|
293 |
-
|
294 |
-
def ajouter_lignes_correcteurs(self):
|
295 |
-
# Commencer avec un peu d'espace
|
296 |
-
para_intro = self.document.add_paragraph()
|
297 |
-
para_intro.space_before = Pt(6)
|
298 |
-
|
299 |
-
# Ajouter les lignes des correcteurs
|
300 |
-
for role in ["Projet", "principal", "ROV"]:
|
301 |
-
para = self.document.add_paragraph()
|
302 |
-
para.space_before = Pt(4)
|
303 |
-
para.space_after = Pt(4)
|
304 |
-
run = para.add_run(f"Correcteur {role} : ")
|
305 |
-
run.bold = True
|
306 |
-
para.add_run(".................................................................")
|
307 |
-
|
308 |
-
# Méthodes de modification des données
|
309 |
-
def modifier_centre_examen(self, nom):
|
310 |
-
self.centre_examen = nom
|
311 |
-
|
312 |
-
def modifier_type_examen(self, type_examen):
|
313 |
-
self.type_examen = type_examen
|
314 |
-
|
315 |
-
def modifier_serie(self, serie):
|
316 |
-
self.serie = serie
|
317 |
-
|
318 |
-
def modifier_etablissement(self, nom):
|
319 |
-
self.etablissement = nom
|
320 |
-
|
321 |
-
def modifier_session(self, annee):
|
322 |
-
self.session = annee
|
323 |
-
|
324 |
-
def modifier_candidat(self, nom):
|
325 |
-
self.nom_candidat = nom
|
326 |
-
|
327 |
-
def ajouter_element(self, nom, categorie, points):
|
328 |
-
self.elements_techniques.append({
|
329 |
-
"nom": nom,
|
330 |
-
"categorie": categorie,
|
331 |
-
"points": float(points)
|
332 |
-
})
|
333 |
-
|
334 |
-
def generer_document(self, nom_fichier="evaluation_gymnastique.docx"):
|
335 |
-
self.ajouter_entete_colore()
|
336 |
-
self.creer_tableau_elements()
|
337 |
-
self.ajouter_note_jury()
|
338 |
-
self.creer_tableau_recapitulatif()
|
339 |
-
self.ajouter_lignes_correcteurs()
|
340 |
-
self.ajouter_zone_note()
|
341 |
-
self.ajouter_note_candidat_avec_cadre()
|
342 |
-
self.document.save(nom_fichier)
|
343 |
-
return nom_fichier
|
344 |
-
|
345 |
-
# --- Application Flask ---
|
346 |
-
app = Flask(__name__)
|
347 |
-
|
348 |
-
@app.route("/", methods=["GET", "POST"])
|
349 |
-
def index():
|
350 |
-
if request.method == "POST":
|
351 |
-
# Récupération des informations depuis le formulaire
|
352 |
-
centre_examen = request.form.get("centre_examen", "Centre d'examen")
|
353 |
-
type_examen = request.form.get("type_examen", "Bac Général")
|
354 |
-
serie = request.form.get("serie", "Série")
|
355 |
-
etablissement = request.form.get("etablissement", "Établissement")
|
356 |
-
session_value = request.form.get("session", "2025")
|
357 |
-
nom_candidat = request.form.get("nom_candidat", "Candidat")
|
358 |
-
|
359 |
-
# Création et configuration du document
|
360 |
-
evaluation = EvaluationGymnique()
|
361 |
-
evaluation.modifier_centre_examen(centre_examen)
|
362 |
-
evaluation.modifier_type_examen(type_examen)
|
363 |
-
evaluation.modifier_serie(serie)
|
364 |
-
evaluation.modifier_etablissement(etablissement)
|
365 |
-
evaluation.modifier_session(session_value)
|
366 |
-
evaluation.modifier_candidat(nom_candidat)
|
367 |
-
|
368 |
-
# Récupération des éléments techniques ajoutés dynamiquement
|
369 |
-
element_names = request.form.getlist("new_element_name")
|
370 |
-
element_categories = request.form.getlist("new_element_categorie")
|
371 |
-
element_points = request.form.getlist("new_element_points")
|
372 |
-
for name, cat, pts in zip(element_names, element_categories, element_points):
|
373 |
-
if name and cat and pts:
|
374 |
-
evaluation.ajouter_element(name, cat, pts)
|
375 |
-
|
376 |
-
# Génération du document DOCX
|
377 |
-
docx_filename = "evaluation_personnalisee.docx"
|
378 |
-
evaluation.generer_document(docx_filename)
|
379 |
-
|
380 |
-
# Vérification du format de sortie demandé
|
381 |
-
output_format = request.form.get("format", "docx")
|
382 |
-
if output_format == "pdf":
|
383 |
-
try:
|
384 |
-
# Utilisation de ConvertAPI pour convertir en PDF
|
385 |
-
pdf_result = convertapi.convert('pdf', {
|
386 |
-
'File': docx_filename,
|
387 |
-
'FileName': 'evaluation_personnalisee',
|
388 |
-
'ConvertMetadata': 'false',
|
389 |
-
'ConvertHeadings': 'false'
|
390 |
-
}, from_format='doc')
|
391 |
-
|
392 |
-
# Télécharger et sauvegarder le PDF localement
|
393 |
-
pdf_filename = "evaluation_personnalisee.pdf"
|
394 |
-
pdf_result.save_files('.')
|
395 |
-
|
396 |
-
# Renvoyer le fichier PDF généré
|
397 |
-
return send_file(pdf_filename, as_attachment=True)
|
398 |
-
except Exception as e:
|
399 |
-
# En cas d'erreur avec l'API, afficher l'erreur et proposer le DOCX
|
400 |
-
print(f"Erreur lors de la conversion en PDF: {e}")
|
401 |
-
return send_file(docx_filename, as_attachment=True)
|
402 |
-
else:
|
403 |
-
# Renvoyer le fichier DOCX si demandé
|
404 |
-
return send_file(docx_filename, as_attachment=True)
|
405 |
-
|
406 |
-
return render_template("index.html")
|
407 |
-
|
408 |
-
if __name__ == "__main__":
|
409 |
-
app.run(debug=True)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|