feat: add assessment_* pages

This commit is contained in:
2025-08-04 20:39:58 +02:00
parent 21aa7015f9
commit 3bd9b56a83
4 changed files with 337 additions and 5 deletions

111
models.py
View File

@@ -72,12 +72,11 @@ class Assessment(db.Model):
for grading_element in exercise.grading_elements:
total_elements += total_students
# Compter les notes saisies (valeur non nulle et non vide)
# Compter les notes saisies (valeur non nulle et non vide, y compris '.')
completed_for_element = db.session.query(Grade).filter(
Grade.grading_element_id == grading_element.id,
Grade.value.isnot(None),
Grade.value != '',
Grade.value != '.'
Grade.value != ''
).count()
completed_elements += completed_for_element
@@ -107,6 +106,112 @@ class Assessment(db.Model):
'status': status,
'students_count': total_students
}
def calculate_student_scores(self):
"""Calcule les scores de tous les élèves pour cette évaluation.
Retourne un dictionnaire avec les scores par élève et par exercice."""
from collections import defaultdict
import statistics
students_scores = {}
exercise_scores = defaultdict(lambda: defaultdict(float))
for student in self.class_group.students:
total_score = 0
total_max_points = 0
student_exercises = {}
for exercise in self.exercises:
exercise_score = 0
exercise_max_points = 0
for element in exercise.grading_elements:
grade = Grade.query.filter_by(
student_id=student.id,
grading_element_id=element.id
).first()
# Si une note a été saisie pour cet élément (y compris '.')
if grade and grade.value and grade.value != '':
if element.grading_type == 'points':
if grade.value == '.':
# '.' signifie non répondu = 0 point mais on compte le max
exercise_score += 0
exercise_max_points += element.max_points
else:
try:
exercise_score += float(grade.value)
exercise_max_points += element.max_points
except ValueError:
pass
else: # compétences
if grade.value == '.':
# '.' signifie non évalué = 0 point mais on compte le max
exercise_score += 0
exercise_max_points += (1/3) * 3 * element.max_points # Score max de 3
else:
try:
score_value = float(grade.value)
exercise_score += (1/3) * score_value * element.max_points
exercise_max_points += (1/3) * 3 * element.max_points # Score max de 3
except ValueError:
pass
student_exercises[exercise.id] = {
'score': exercise_score,
'max_points': exercise_max_points,
'title': exercise.title
}
total_score += exercise_score
total_max_points += exercise_max_points
exercise_scores[exercise.id][student.id] = exercise_score
students_scores[student.id] = {
'student': student,
'total_score': round(total_score, 2),
'total_max_points': total_max_points,
'exercises': student_exercises
}
return students_scores, dict(exercise_scores)
def get_assessment_statistics(self):
"""Calcule les statistiques descriptives pour cette évaluation."""
students_scores, _ = self.calculate_student_scores()
scores = [data['total_score'] for data in students_scores.values()]
if not scores:
return {
'count': 0,
'mean': 0,
'median': 0,
'min': 0,
'max': 0,
'std_dev': 0
}
import statistics
import math
return {
'count': len(scores),
'mean': round(statistics.mean(scores), 2),
'median': round(statistics.median(scores), 2),
'min': min(scores),
'max': max(scores),
'std_dev': round(statistics.stdev(scores) if len(scores) > 1 else 0, 2)
}
def get_total_max_points(self):
"""Calcule le total des points maximum pour cette évaluation."""
total = 0
for exercise in self.exercises:
for element in exercise.grading_elements:
if element.grading_type == 'points':
total += element.max_points
else: # compétences
total += (1/3) * 3 * element.max_points # Score max de 3
return total
class Exercise(db.Model):
id = db.Column(db.Integer, primary_key=True)

View File

@@ -178,6 +178,38 @@ def new():
return render_template('assessment_form_unified.html', form=form, title='Nouvelle évaluation complète')
@bp.route('/<int:id>/results')
@handle_db_errors
def results(id):
from sqlalchemy.orm import joinedload
from models import Exercise, GradingElement
assessment = Assessment.query.options(
joinedload(Assessment.class_group),
joinedload(Assessment.exercises).joinedload(Exercise.grading_elements)
).get_or_404(id)
# Calculer les scores des élèves
students_scores, exercise_scores = assessment.calculate_student_scores()
# Trier les élèves par ordre alphabétique
sorted_students = sorted(students_scores.values(),
key=lambda x: (x['student'].last_name.lower(), x['student'].first_name.lower()))
# Calculer les statistiques
statistics = assessment.get_assessment_statistics()
total_max_points = assessment.get_total_max_points()
# Préparer les données pour l'histogramme
scores = [data['total_score'] for data in students_scores.values()]
return render_template('assessment_results.html',
assessment=assessment,
students_scores=sorted_students,
statistics=statistics,
total_max_points=total_max_points,
scores_json=scores)
@bp.route('/<int:id>/delete', methods=['POST'])
@handle_db_errors
def delete(id):

View File

@@ -180,12 +180,12 @@
</svg>
Saisir les notes
</a>
<button class="flex items-center justify-center px-4 py-3 border border-gray-300 rounded-lg text-sm font-medium text-gray-700 hover:bg-gray-50 transition-colors">
<a href="{{ url_for('assessments.results', id=assessment.id) }}" class="flex items-center justify-center px-4 py-3 border border-gray-300 rounded-lg text-sm font-medium text-gray-700 hover:bg-gray-50 transition-colors">
<svg class="w-5 h-5 mr-2 text-purple-600" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M3 17a1 1 0 011-1h12a1 1 0 011 1v1a1 1 0 01-1 1H4a1 1 0 01-1-1v-1zM3 4a1 1 0 011-1h12a1 1 0 011 1v1a1 1 0 01-1 1H4a1 1 0 01-1-1V4z" clip-rule="evenodd"/>
</svg>
Voir les résultats
</button>
</a>
<button class="flex items-center justify-center px-4 py-3 border border-gray-300 rounded-lg text-sm font-medium text-gray-700 hover:bg-gray-50 transition-colors">
<svg class="w-5 h-5 mr-2 text-blue-600" fill="currentColor" viewBox="0 0 20 20">
<path fill-rule="evenodd" d="M3 17a1 1 0 011-1h12a1 1 0 011 1v1a1 1 0 01-1 1H4a1 1 0 01-1-1v-1zM3 4a1 1 0 011-1h12a1 1 0 011 1v1a1 1 0 01-1 1H4a1 1 0 01-1-1V4z" clip-rule="evenodd"/>

View File

@@ -0,0 +1,195 @@
{% extends "base.html" %}
{% block title %}Résultats - {{ assessment.title }} - Gestion Scolaire{% endblock %}
{% block content %}
<div class="space-y-6">
<div class="flex justify-between items-center">
<div>
<a href="{{ url_for('assessments.detail', id=assessment.id) }}" class="text-blue-600 hover:text-blue-800 text-sm font-medium mb-2 inline-block">
← Retour à l'évaluation
</a>
<h1 class="text-2xl font-bold text-gray-900">Résultats - {{ assessment.title }}</h1>
<p class="text-gray-600">{{ assessment.class_group.name }} - {{ assessment.date.strftime('%d/%m/%Y') }} - Trimestre {{ assessment.trimester }}</p>
</div>
</div>
<!-- Statistiques générales -->
<div class="bg-white shadow rounded-lg">
<div class="px-6 py-4 border-b border-gray-200">
<h2 class="text-lg font-medium text-gray-900">Statistiques de l'évaluation</h2>
</div>
<div class="px-6 py-4">
<div class="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-6 gap-4">
<div class="text-center">
<div class="text-2xl font-bold text-blue-600">{{ statistics.count }}</div>
<div class="text-sm text-gray-500">Élèves</div>
</div>
<div class="text-center">
<div class="text-2xl font-bold text-green-600">{{ statistics.mean }}</div>
<div class="text-sm text-gray-500">Moyenne</div>
</div>
<div class="text-center">
<div class="text-2xl font-bold text-purple-600">{{ statistics.median }}</div>
<div class="text-sm text-gray-500">Médiane</div>
</div>
<div class="text-center">
<div class="text-2xl font-bold text-red-600">{{ statistics.min }}</div>
<div class="text-sm text-gray-500">Minimum</div>
</div>
<div class="text-center">
<div class="text-2xl font-bold text-green-600">{{ statistics.max }}</div>
<div class="text-sm text-gray-500">Maximum</div>
</div>
<div class="text-center">
<div class="text-2xl font-bold text-orange-600">{{ statistics.std_dev }}</div>
<div class="text-sm text-gray-500">Écart-type</div>
</div>
</div>
<div class="mt-4 text-sm text-gray-600 text-center">
Total des points de l'évaluation : {{ "%.1f"|format(total_max_points) }}
</div>
</div>
</div>
<!-- Histogramme des notes -->
<div class="bg-white shadow rounded-lg">
<div class="px-6 py-4 border-b border-gray-200">
<h2 class="text-lg font-medium text-gray-900">Distribution des notes</h2>
</div>
<div class="px-6 py-4">
<canvas id="scoresChart" width="400" height="200"></canvas>
</div>
</div>
<!-- Tableau des résultats -->
<div class="bg-white shadow rounded-lg">
<div class="px-6 py-4 border-b border-gray-200">
<h2 class="text-lg font-medium text-gray-900">Résultats par élève</h2>
</div>
<div class="overflow-x-auto">
<table class="min-w-full divide-y divide-gray-200">
<thead class="bg-gray-50">
<tr>
<th class="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider">
Élève
</th>
<th class="px-6 py-3 text-center text-xs font-medium text-gray-500 uppercase tracking-wider">
Note totale
</th>
{% for exercise in assessment.exercises|sort(attribute='order') %}
<th class="px-6 py-3 text-center text-xs font-medium text-gray-500 uppercase tracking-wider">
{{ exercise.title }}
</th>
{% endfor %}
</tr>
</thead>
<tbody class="bg-white divide-y divide-gray-200">
{% for student_data in students_scores %}
<tr class="hover:bg-gray-50">
<td class="px-6 py-4 whitespace-nowrap">
<div class="text-sm font-medium text-gray-900">
{{ student_data.student.last_name }} {{ student_data.student.first_name }}
</div>
</td>
<td class="px-6 py-4 whitespace-nowrap text-center">
<div class="text-sm font-bold text-gray-900">
{{ "%.1f"|format(student_data.total_score) }} / {{ "%.1f"|format(student_data.total_max_points) }}
</div>
</td>
{% for exercise in assessment.exercises|sort(attribute='order') %}
<td class="px-6 py-4 whitespace-nowrap text-center">
<div class="text-sm text-gray-900">
{% if exercise.id in student_data.exercises %}
{{ "%.1f"|format(student_data.exercises[exercise.id].score) }} / {{ "%.1f"|format(student_data.exercises[exercise.id].max_points) }}
{% else %}
-
{% endif %}
</div>
</td>
{% endfor %}
</tr>
{% endfor %}
</tbody>
</table>
</div>
</div>
</div>
<!-- Chart.js -->
<script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
<script>
document.addEventListener('DOMContentLoaded', function() {
const ctx = document.getElementById('scoresChart').getContext('2d');
const scores = {{ scores_json | tojson }};
// Créer les intervalles pour l'histogramme
const totalMaxPoints = {{ total_max_points }};
const minScore = 0;
const maxScore = totalMaxPoints;
const binWidth = 1; // Groupes de 1 point
const numBins = Math.ceil(maxScore - minScore); // Nombre de bins basé sur le total des points
const bins = [];
const labels = [];
for (let i = 0; i < numBins; i++) {
const binStart = minScore + i * binWidth;
const binEnd = minScore + (i + 1) * binWidth;
bins.push(0);
labels.push(`${binStart.toFixed(0)} - ${binEnd.toFixed(0)}`);
}
// Compter les scores dans chaque intervalle
scores.forEach(score => {
let binIndex = Math.floor((score - minScore) / binWidth);
if (binIndex >= numBins) binIndex = numBins - 1; // Pour le score maximum
if (binIndex >= 0) bins[binIndex]++;
});
new Chart(ctx, {
type: 'bar',
data: {
labels: labels,
datasets: [{
label: 'Nombre d\'élèves',
data: bins,
backgroundColor: 'rgba(59, 130, 246, 0.5)',
borderColor: 'rgba(59, 130, 246, 1)',
borderWidth: 1
}]
},
options: {
responsive: true,
plugins: {
title: {
display: true,
text: 'Distribution des notes'
},
legend: {
display: false
}
},
scales: {
y: {
beginAtZero: true,
ticks: {
stepSize: 1
},
title: {
display: true,
text: 'Nombre d\'élèves'
}
},
x: {
title: {
display: true,
text: 'Intervalles de notes'
}
}
}
}
});
});
</script>
{% endblock %}