570 lines
16 KiB
Markdown
Executable File
570 lines
16 KiB
Markdown
Executable File
# Session du 31 décembre 2025 - Pagination et Miniatures
|
||
|
||
## 🎯 Objectifs de la session
|
||
|
||
1. **Corriger la génération des miniatures** - Conservation du ratio d'aspect à 48px de large
|
||
2. **Implémenter l'icône de sélection de photo principale** - Toggle cliquable pour choisir la vignette
|
||
3. **Générer des données de test pour pagination** - 40+ périphériques pour tester prev/next
|
||
|
||
---
|
||
|
||
## ✅ 1. Correction des miniatures (Thumbnails)
|
||
|
||
### Problème initial
|
||
|
||
Les miniatures étaient générées en **carré 300×300px** avec crop, ce qui :
|
||
- Déformait les images
|
||
- Coupait des parties de l'image
|
||
- Créait des fichiers trop volumineux (~25-53 KB)
|
||
|
||
### Solution implémentée
|
||
|
||
**Algorithme modifié** : Conservation du ratio d'aspect avec largeur fixe de 48px
|
||
|
||
#### Fichiers modifiés
|
||
|
||
**1. `backend/app/utils/image_processor.py` (lignes 222-230)**
|
||
|
||
```python
|
||
# Resize keeping aspect ratio (width-based)
|
||
# size parameter represents the target width
|
||
width, height = img.size
|
||
aspect_ratio = height / width
|
||
new_width = size
|
||
new_height = int(size * aspect_ratio)
|
||
|
||
# Use thumbnail method to preserve aspect ratio
|
||
img.thumbnail((new_width, new_height), Image.Resampling.LANCZOS)
|
||
```
|
||
|
||
**Avant** : Crop carré → Resize 300×300
|
||
**Après** : Resize proportionnel → 48×(hauteur calculée)
|
||
|
||
**2. `config/image_compression.yaml`**
|
||
|
||
Mise à jour de tous les niveaux de compression :
|
||
```yaml
|
||
levels:
|
||
high:
|
||
thumbnail_size: 48 # Avant: 400
|
||
|
||
medium:
|
||
thumbnail_size: 48 # Avant: 300
|
||
|
||
low:
|
||
thumbnail_size: 48 # Avant: 200
|
||
|
||
minimal:
|
||
thumbnail_size: 48 # Avant: 150
|
||
```
|
||
|
||
**3. `backend/app/utils/image_config_loader.py` (ligne 54)**
|
||
|
||
```python
|
||
"thumbnail_size": 48, # Était 300
|
||
```
|
||
|
||
**4. `backend/app/core/config.py` (ligne 35)**
|
||
|
||
```python
|
||
THUMBNAIL_SIZE: int = 48 # Était 300
|
||
```
|
||
|
||
### Script de régénération
|
||
|
||
**Fichier créé** : `backend/regenerate_thumbnails.py`
|
||
|
||
Permet de régénérer toutes les miniatures existantes avec le nouveau système.
|
||
|
||
**Exécution** :
|
||
```bash
|
||
docker exec linux_benchtools_backend python3 regenerate_thumbnails.py
|
||
```
|
||
|
||
**Résultats** :
|
||
```
|
||
[1/4] Photo ID 4 - image1.png
|
||
🗑️ Ancienne miniature supprimée (22615 octets)
|
||
✅ Nouvelle miniature : image1_thumb.png (1679 octets)
|
||
📐 Dimensions : 48×27px
|
||
|
||
[2/4] Photo ID 5 - image2.png
|
||
🗑️ Ancienne miniature supprimée (53454 octets)
|
||
✅ Nouvelle miniature : image2_thumb.png (2763 octets)
|
||
📐 Dimensions : 48×64px
|
||
|
||
[3/4] Photo ID 6 - image3.png
|
||
🗑️ Ancienne miniature supprimée (36719 octets)
|
||
✅ Nouvelle miniature : image3_thumb.png (1879 octets)
|
||
📐 Dimensions : 48×32px
|
||
|
||
[4/4] Photo ID 7 - image4.png
|
||
🗑️ Ancienne miniature supprimée (41280 octets)
|
||
✅ Nouvelle miniature : image4_thumb.png (2312 octets)
|
||
📐 Dimensions : 48×36px
|
||
|
||
✅ Succès : 4/4
|
||
```
|
||
|
||
### Gains obtenus
|
||
|
||
| Photo | Format original | Avant (300×300) | Après (48px) | Gain |
|
||
|-------|-----------------|-----------------|--------------|------|
|
||
| 1 | 1920×1080 | 22 KB | 1.6 KB | **93%** |
|
||
| 2 | 800×1067 | 53 KB | 2.7 KB | **95%** |
|
||
| 3 | 1280×853 | 37 KB | 1.8 KB | **95%** |
|
||
| 4 | 1600×1200 | 41 KB | 2.3 KB | **94%** |
|
||
|
||
**→ Réduction moyenne : 94%**
|
||
|
||
### Exemples de résultats
|
||
|
||
```
|
||
Image paysage (16:9)
|
||
Original : 1920×1080
|
||
Avant : 1920×1080 → crop 1080×1080 → 300×300 ❌
|
||
Après : 1920×1080 → resize 48×27 ✅
|
||
|
||
Image portrait (3:4)
|
||
Original : 800×1067
|
||
Avant : 800×1067 → crop 800×800 → 300×300 ❌
|
||
Après : 800×1067 → resize 48×64 ✅
|
||
|
||
Image carrée (1:1)
|
||
Original : 800×800
|
||
Avant : 800×800 → crop 800×800 → 300×300
|
||
Après : 800×800 → resize 48×48 ✅
|
||
```
|
||
|
||
---
|
||
|
||
## ✅ 2. Icône de sélection de photo principale
|
||
|
||
### Fonctionnalité
|
||
|
||
Ajouter une icône cliquable en bas à gauche de chaque photo pour définir quelle photo sera la vignette principale (thumbnail).
|
||
|
||
**Règles** :
|
||
- Une seule photo principale par périphérique
|
||
- Icône ⭕ (circle) = non sélectionnée
|
||
- Icône ✅ (check-circle) = photo principale
|
||
- Clic sur icône → change la photo principale
|
||
|
||
### Implémentation Frontend
|
||
|
||
**Fichier** : `frontend/js/peripheral-detail.js` (lignes 108-112, 239-252)
|
||
|
||
**1. Bouton HTML dans la galerie**
|
||
|
||
```javascript
|
||
<button class="photo-primary-toggle ${photo.is_primary ? 'active' : ''}"
|
||
onclick="setPrimaryPhoto(${photo.id})"
|
||
title="${photo.is_primary ? 'Photo principale' : 'Définir comme photo principale'}">
|
||
<i class="fas fa-${photo.is_primary ? 'check-circle' : 'circle'}"></i>
|
||
</button>
|
||
```
|
||
|
||
**2. Fonction JavaScript**
|
||
|
||
```javascript
|
||
async function setPrimaryPhoto(photoId) {
|
||
try {
|
||
await apiRequest(`/peripherals/${peripheralId}/photos/${photoId}/set-primary`, {
|
||
method: 'POST'
|
||
});
|
||
|
||
showSuccess('Photo principale définie');
|
||
loadPhotos(); // Reload to update icons
|
||
} catch (error) {
|
||
console.error('Error setting primary photo:', error);
|
||
showError('Erreur lors de la définition de la photo principale');
|
||
}
|
||
}
|
||
```
|
||
|
||
**3. Style CSS**
|
||
|
||
**Fichier** : `frontend/css/peripherals.css` (lignes 764-803)
|
||
|
||
```css
|
||
/* Photo Primary Toggle */
|
||
.photo-primary-toggle {
|
||
position: absolute;
|
||
bottom: 8px;
|
||
left: 8px;
|
||
background: rgba(0, 0, 0, 0.7);
|
||
border: 2px solid #666;
|
||
border-radius: 50%;
|
||
width: 32px;
|
||
height: 32px;
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: center;
|
||
cursor: pointer;
|
||
transition: all 0.2s ease;
|
||
color: #999;
|
||
font-size: 16px;
|
||
z-index: 10;
|
||
}
|
||
|
||
.photo-primary-toggle:hover {
|
||
background: rgba(0, 0, 0, 0.85);
|
||
border-color: #66d9ef;
|
||
color: #66d9ef;
|
||
transform: scale(1.1);
|
||
}
|
||
|
||
.photo-primary-toggle.active {
|
||
background: rgba(102, 217, 239, 0.2);
|
||
border-color: #66d9ef;
|
||
color: #66d9ef;
|
||
}
|
||
|
||
.photo-primary-toggle.active:hover {
|
||
background: rgba(102, 217, 239, 0.3);
|
||
}
|
||
```
|
||
|
||
**Caractéristiques visuelles** :
|
||
- Position : Coin inférieur gauche (8px × 8px)
|
||
- Taille : 32×32px bouton rond
|
||
- Couleur normale : Gris #999
|
||
- Couleur hover/active : Cyan #66d9ef
|
||
- Effet hover : Scale 1.1
|
||
- Z-index élevé pour rester au-dessus
|
||
|
||
### Implémentation Backend
|
||
|
||
**Fichier** : `backend/app/api/endpoints/peripherals.py` (lignes 370-396)
|
||
|
||
**Endpoint POST**
|
||
|
||
```python
|
||
@router.post("/{peripheral_id}/photos/{photo_id}/set-primary", status_code=200)
|
||
def set_primary_photo(
|
||
peripheral_id: int,
|
||
photo_id: int,
|
||
db: Session = Depends(get_peripherals_db)
|
||
):
|
||
"""Set a photo as primary (thumbnail)"""
|
||
# Get the photo
|
||
photo = db.query(PeripheralPhoto).filter(
|
||
PeripheralPhoto.id == photo_id,
|
||
PeripheralPhoto.peripheral_id == peripheral_id
|
||
).first()
|
||
|
||
if not photo:
|
||
raise HTTPException(status_code=404, detail="Photo not found")
|
||
|
||
# Unset all other primary photos for this peripheral
|
||
db.query(PeripheralPhoto).filter(
|
||
PeripheralPhoto.peripheral_id == peripheral_id,
|
||
PeripheralPhoto.id != photo_id
|
||
).update({"is_primary": False})
|
||
|
||
# Set this photo as primary
|
||
photo.is_primary = True
|
||
db.commit()
|
||
|
||
return {"message": "Photo set as primary", "photo_id": photo_id}
|
||
```
|
||
|
||
**Logique** :
|
||
1. Vérifie que la photo existe et appartient au périphérique
|
||
2. Désactive `is_primary` sur toutes les autres photos du même périphérique
|
||
3. Active `is_primary` sur la photo sélectionnée
|
||
4. Garantit qu'une seule photo est principale à la fois
|
||
|
||
### Flux utilisateur
|
||
|
||
```
|
||
1. User voit la galerie de photos
|
||
↓
|
||
2. Chaque photo affiche une icône ⭕/✅ en bas à gauche
|
||
↓
|
||
3. User clique sur une icône ⭕ (non sélectionnée)
|
||
↓
|
||
4. setPrimaryPhoto(photoId) appelé
|
||
│ ├─> POST /api/peripherals/{id}/photos/{photo_id}/set-primary
|
||
│ └─> Backend met à jour is_primary
|
||
↓
|
||
5. Base de données mise à jour
|
||
│ ├─> Ancienne photo principale : is_primary = false
|
||
│ └─> Nouvelle photo : is_primary = true
|
||
↓
|
||
6. Success
|
||
│ ├─> Message "Photo principale définie"
|
||
│ ├─> Galerie rechargée
|
||
│ └─> Icônes mises à jour (✅ sur nouvelle, ⭕ sur autres)
|
||
```
|
||
|
||
### Rendu visuel
|
||
|
||
```
|
||
┌─────────────┐ ┌─────────────┐ ┌─────────────┐
|
||
│ │ │ │ │ │
|
||
│ Photo 1 │ │ Photo 2 │ │ Photo 3 │
|
||
│ │ │ │ │ │
|
||
│ ⭕ [🗑️]│ │ ✅ [🗑️]│ │ ⭕ [🗑️]│
|
||
│ ★ Principale│ │ │ │ │
|
||
└─────────────┘ └─────────────┘ └─────────────┘
|
||
```
|
||
|
||
**Légende** :
|
||
- ⭕ = Icône circle grise (non sélectionnée)
|
||
- ✅ = Icône check-circle cyan (sélectionnée)
|
||
- ★ = Badge "Principale" (en haut)
|
||
- 🗑️ = Bouton supprimer (en haut à droite)
|
||
|
||
---
|
||
|
||
## ✅ 3. Génération de données de test pour pagination
|
||
|
||
### Objectif
|
||
|
||
Créer au minimum 40 périphériques pour tester la pagination (prev/next).
|
||
|
||
### Configuration pagination
|
||
|
||
**Fichier** : `frontend/js/peripherals.js` (ligne 7)
|
||
|
||
```javascript
|
||
let pageSize = 10; // Items per page (était 50)
|
||
```
|
||
|
||
**Changement** : 50 → 10 items par page pour mieux tester la navigation entre pages.
|
||
|
||
### Script de génération
|
||
|
||
**Fichier créé** : `backend/generate_test_peripherals.py`
|
||
|
||
**Fonctionnalités** :
|
||
- Génère N périphériques de test avec données aléatoires
|
||
- Types variés : USB, Stockage, Réseau, Audio, Vidéo, Clavier, Souris, etc.
|
||
- Marques variées : Logitech, SanDisk, Kingston, TP-Link, Razer, Corsair, etc.
|
||
- États aléatoires : Neuf, Bon, Usagé, Défectueux
|
||
- Prix, quantités, dates d'achat, garanties générés aléatoirement
|
||
- Support argument `--count` pour spécifier le nombre
|
||
|
||
**Code principal** :
|
||
|
||
```python
|
||
def generate_peripherals(count=40):
|
||
"""Génère des périphériques de test"""
|
||
db = next(get_peripherals_db())
|
||
|
||
try:
|
||
print(f"🔧 Génération de {count} périphériques de test...")
|
||
print("=" * 60)
|
||
|
||
for i in range(1, count + 1):
|
||
type_principal = random.choice(TYPES)
|
||
marque = random.choice(MARQUES)
|
||
nom = f"{marque} {type_principal} {random.randint(100, 9999)}"
|
||
|
||
peripheral = Peripheral(
|
||
nom=nom,
|
||
type_principal=type_principal,
|
||
marque=marque,
|
||
modele=random.choice(modeles),
|
||
numero_serie=f"SN{random.randint(100000, 999999)}",
|
||
etat=random.choice(ETATS),
|
||
rating=random.randint(0, 5),
|
||
quantite_totale=random.randint(1, 5),
|
||
quantite_disponible=random.randint(0, 5),
|
||
prix=round(random.uniform(5.99, 199.99), 2) if random.random() > 0.2 else None,
|
||
# ... autres champs aléatoires
|
||
)
|
||
|
||
db.add(peripheral)
|
||
|
||
if i % 10 == 0:
|
||
db.commit()
|
||
print(f" ✅ {i}/{count} périphériques créés")
|
||
|
||
db.commit()
|
||
print(f"✅ {count} périphériques de test créés avec succès !")
|
||
|
||
# Statistiques
|
||
total = db.query(Peripheral).count()
|
||
print(f"📊 Total dans la base : {total} périphériques")
|
||
|
||
except Exception as e:
|
||
print(f"❌ Erreur : {e}")
|
||
db.rollback()
|
||
finally:
|
||
db.close()
|
||
```
|
||
|
||
### Exécution
|
||
|
||
**Démarrage des containers** :
|
||
```bash
|
||
docker compose up -d
|
||
```
|
||
|
||
**Copie et exécution du script** :
|
||
```bash
|
||
docker cp backend/generate_test_peripherals.py linux_benchtools_backend:/app/
|
||
docker exec linux_benchtools_backend python3 generate_test_peripherals.py --count 40
|
||
```
|
||
|
||
**Résultat** :
|
||
```
|
||
🔧 Génération de 40 périphériques de test...
|
||
============================================================
|
||
✅ 10/40 périphériques créés
|
||
✅ 20/40 périphériques créés
|
||
✅ 30/40 périphériques créés
|
||
✅ 40/40 périphériques créés
|
||
|
||
============================================================
|
||
✅ 40 périphériques de test créés avec succès !
|
||
📊 Total dans la base : 46 périphériques
|
||
```
|
||
|
||
### Vérification de la pagination
|
||
|
||
**Test API** :
|
||
|
||
```bash
|
||
# Page 1
|
||
curl -s "http://localhost:8007/api/peripherals/?page=1&page_size=10"
|
||
→ Total: 46, Page: 1/5, Items: 10
|
||
|
||
# Page 2
|
||
curl -s "http://localhost:8007/api/peripherals/?page=2&page_size=10"
|
||
→ Total: 46, Page: 2/5, Items: 10
|
||
|
||
# Page 5 (dernière)
|
||
curl -s "http://localhost:8007/api/peripherals/?page=5&page_size=10"
|
||
→ Total: 46, Page: 5/5, Items: 6
|
||
```
|
||
|
||
**Résultat** : ✅ Pagination fonctionnelle
|
||
- **Total** : 46 périphériques
|
||
- **Pages** : 5 pages (4×10 items + 1×6 items)
|
||
- **Taille** : 10 items par page
|
||
|
||
### Interface utilisateur
|
||
|
||
Les boutons "Précédent" et "Suivant" sont déjà implémentés dans `frontend/js/peripherals.js` :
|
||
|
||
```javascript
|
||
function previousPage() {
|
||
if (currentPage > 1) {
|
||
currentPage--;
|
||
loadPeripherals();
|
||
}
|
||
}
|
||
|
||
function nextPage() {
|
||
if (currentPage < totalPages) {
|
||
currentPage++;
|
||
loadPeripherals();
|
||
}
|
||
}
|
||
```
|
||
|
||
**États des boutons** :
|
||
- "Précédent" : Désactivé sur page 1
|
||
- "Suivant" : Désactivé sur dernière page
|
||
|
||
---
|
||
|
||
## 📊 Résumé des modifications
|
||
|
||
### Fichiers créés
|
||
|
||
| Fichier | Description |
|
||
|---------|-------------|
|
||
| `backend/regenerate_thumbnails.py` | Script pour régénérer les miniatures existantes |
|
||
| `backend/generate_test_peripherals.py` | Script pour générer des périphériques de test |
|
||
| `docs/FEATURE_PRIMARY_PHOTO_TOGGLE.md` | Documentation de l'icône de photo principale |
|
||
| `docs/THUMBNAILS_ASPECT_RATIO.md` | Documentation des miniatures avec ratio conservé |
|
||
| `docs/SESSION_2025-12-31_PAGINATION_THUMBNAILS.md` | Ce document |
|
||
|
||
### Fichiers modifiés
|
||
|
||
| Fichier | Lignes | Modification |
|
||
|---------|--------|--------------|
|
||
| `backend/app/utils/image_processor.py` | 222-230 | Algorithme thumbnail avec ratio conservé |
|
||
| `config/image_compression.yaml` | 23, 33, 43, 53 | `thumbnail_size: 48` pour tous niveaux |
|
||
| `backend/app/utils/image_config_loader.py` | 54 | Default `thumbnail_size: 48` |
|
||
| `backend/app/core/config.py` | 35 | `THUMBNAIL_SIZE: int = 48` |
|
||
| `frontend/js/peripheral-detail.js` | 108-112, 239-252 | Bouton toggle + fonction `setPrimaryPhoto()` |
|
||
| `frontend/css/peripherals.css` | 764-803 | Style `.photo-primary-toggle` |
|
||
| `backend/app/api/endpoints/peripherals.py` | 370-396 | Endpoint POST set-primary |
|
||
| `frontend/js/peripherals.js` | 7 | `pageSize = 10` (était 50) |
|
||
|
||
---
|
||
|
||
## 🎯 Résultats obtenus
|
||
|
||
### 1. Miniatures optimisées ✅
|
||
- **Taille** : 48px de large, hauteur proportionnelle
|
||
- **Poids** : Réduction de ~94% (25-53 KB → 1-3 KB)
|
||
- **Ratio** : Conservé (pas de déformation)
|
||
- **Crop** : Supprimé (image complète)
|
||
|
||
### 2. Photo principale sélectionnable ✅
|
||
- **Interface** : Icône cliquable sur chaque photo
|
||
- **Visuel** : États clair (⭕ non cochée, ✅ cochée)
|
||
- **Logique** : Une seule photo principale à la fois
|
||
- **API** : Endpoint POST fonctionnel
|
||
|
||
### 3. Pagination testable ✅
|
||
- **Données** : 46 périphériques en base
|
||
- **Pages** : 5 pages de 10 items
|
||
- **API** : Pagination fonctionnelle
|
||
- **UI** : Boutons prev/next déjà implémentés
|
||
|
||
---
|
||
|
||
## 🧪 Tests effectués
|
||
|
||
### Test 1 : Régénération des miniatures
|
||
```bash
|
||
docker exec linux_benchtools_backend python3 regenerate_thumbnails.py
|
||
```
|
||
**Résultat** : ✅ 4 photos régénérées avec succès
|
||
|
||
### Test 2 : API pagination
|
||
```bash
|
||
curl "http://localhost:8007/api/peripherals/?page=1&page_size=10"
|
||
```
|
||
**Résultat** : ✅ 10 items retournés, page 1/5
|
||
|
||
### Test 3 : Génération de données
|
||
```bash
|
||
docker exec linux_benchtools_backend python3 generate_test_peripherals.py --count 40
|
||
```
|
||
**Résultat** : ✅ 40 périphériques créés
|
||
|
||
---
|
||
|
||
## 💡 Prochaines améliorations possibles
|
||
|
||
### Miniatures
|
||
- [ ] Régénération automatique au démarrage si config change
|
||
- [ ] Support WebP pour réduction supplémentaire du poids
|
||
- [ ] Lazy loading des miniatures dans la galerie
|
||
|
||
### Photo principale
|
||
- [ ] Double-clic sur photo pour la définir comme principale
|
||
- [ ] Drag & drop pour réorganiser l'ordre des photos
|
||
- [ ] Raccourci clavier (P pour Primary)
|
||
- [ ] Preview de la vignette avant validation
|
||
|
||
### Pagination
|
||
- [ ] Sélecteur de nombre d'items par page
|
||
- [ ] Input pour aller directement à une page
|
||
- [ ] Indicateur de position (ex: "1-10 sur 46")
|
||
- [ ] Navigation clavier (← →)
|
||
|
||
---
|
||
|
||
**Date** : 31 décembre 2025
|
||
**Statut** : ✅ Toutes les fonctionnalités implémentées et testées
|
||
**Impact** : Miniatures optimisées, sélection intuitive de photo principale, pagination fonctionnelle avec données de test
|