Répository 24.00
Le modèle Repository dans webforJ fournit une manière standardisée de gérer et de requêter des collections d'entités. Il agit comme une couche d'abstraction entre vos composants UI et les données, facilitant le travail avec différentes sources de données tout en maintenant un comportement cohérent.
Pourquoi utiliser un repository
Repository élimine les mises à jour manuelles tout en gardant vos données originales intactes :
// Sans Repository - mises à jour manuelles
List<Customer> customers = loadCustomers();
Table<Customer> table = new Table<>();
table.setItems(customers);
// Ajouter nécessite un rechargement complet
customers.add(newCustomer);
table.setItems(customers); // Doit tout recharger
// Avec Repository - synchronisation automatique
List<Customer> customers = loadCustomers();
CollectionRepository<Customer> repository = new CollectionRepository<>(customers);
Table<Customer> table = new Table<>();
table.setRepository(repository);
// Ajouter se synchronise automatiquement
customers.add(newCustomer);
repository.commit(newCustomer); // Met à jour seulement ce qui a changé
Collection repository
Le CollectionRepository est l'implémentation la plus courante, et enveloppe toute Collection Java :
// Depuis ArrayList
List<Customer> customers = new ArrayList<>();
CollectionRepository<Customer> customerRepo = new CollectionRepository<>(customers);
// Depuis HashSet
Set<String> tags = new HashSet<>();
CollectionRepository<String> tagRepo = new CollectionRepository<>(tags);
// Depuis n'importe quelle Collection
Collection<Employee> employees = getEmployeesFromHR();
CollectionRepository<Employee> employeeRepo = new CollectionRepository<>(employees);
Synchronisation des données
Le Repository agit comme un pont entre vos données et les composants UI. Lorsque les données changent, vous notifiez le repository via la méthode commit() :
List<Product> products = new ArrayList<>();
CollectionRepository<Product> repository = new CollectionRepository<>(products);
// Ajouter un nouveau produit
Product newProduct = new Product("P4", "Gizmo", 79.99, 15);
products.add(newProduct);
repository.commit(); // Tous les composants connectés se mettent à jour
// Mettre à jour un produit existant
products.get(0).setPrice(89.99);
repository.commit(products.get(0)); // Met à jour seulement cette ligne spécifique
// Supprimer un produit
products.remove(2);
repository.commit(); // Rafraîchit la vue
La méthode commit a deux signatures :
commit()- Indique au repository de rafraîchir tout. Cela déclenche unRepositoryCommitEventavec toutes les données actuellescommit(entity)- Cible une entité spécifique. Le repository trouve cette entité par sa clé et met à jour seulement les éléments UI affectés
Cette distinction est importante pour la performance. Lorsque vous mettez à jour un champ dans un tableau de 1000 lignes, commit(entity) met à jour seulement cette cellule alors que commit() rafraîchirait toutes les lignes.
Filtrage des données
Le filtre du repository contrôle quelles données sont transmises aux composants connectés. Votre collection sous-jacente reste inchangée car le filtre agit comme une lentille :
// Filtrer par disponibilité des stocks
repository.setBaseFilter(product -> product.getStock() > 0);
// Filtrer par catégorie
repository.setBaseFilter(product -> "Électronique".equals(product.getCategory()));
// Combiner plusieurs conditions
repository.setBaseFilter(product ->
product.getCategory().equals("Électronique") &&
product.getStock() > 0 &&
product.getPrice() < 100.0
);
// Effacer le filtre
repository.setBaseFilter(null);
Lorsque vous définissez un filtre, le Repository :
- Applique le prédicat à chaque élément de votre collection
- Crée un flux filtré d'éléments correspondants
- Notifie les composants connectés pour mettre à jour leur affichage
Le filtre persiste jusqu'à ce que vous le changiez. Les nouveaux éléments ajoutés à la collection sont automatiquement testés par rapport au filtre actuel.
Travailler avec les clés d'entité
Le repository a besoin d'identifier les entités de manière unique pour prendre en charge des opérations telles que find() et commit(entity). Il y a deux manières de définir comment les entités sont identifiées :
Utilisation de l'interface HasEntityKey
Implémentez HasEntityKey dans votre classe d'entité :
public class Customer implements HasEntityKey {
private String customerId;
private String name;
private String email;
@Override
public Object getEntityKey() {
return customerId;
}
// Constructeur et getters/setters...
}
// Trouver par clé
Optional<Customer> customer = repository.find("C001");
// Mettre à jour un client spécifique
customer.ifPresent(c -> {
c.setEmail("newemail@example.com");
repository.commit(c); // Seule la ligne de ce client se met à jour
});
Utilisation d'un fournisseur de clé personnalisé 25.10
Pour les entités où vous ne pouvez pas ou ne voulez pas implémenter HasEntityKey (comme les entités JPA), utilisez setKeyProvider() :
@Entity
public class Product {
@Id
private Long id;
private String name;
private double price;
// Entité gérée par JPA
}
// Configurez le repository pour utiliser la méthode getId()
CollectionRepository<Product> repository = new CollectionRepository<>(products);
repository.setKeyProvider(Product::getId);
// Maintenant, la recherche fonctionne avec l'ID
Optional<Product> product = repository.find(123L);
Choisir une approche
Les deux approches fonctionnent, mais setKeyProvider() est préférée lorsque :
- Vous travaillez avec des entités JPA qui ont des champs
@Id - Vous ne pouvez pas modifier la classe d'entité
- Vous avez besoin de différentes stratégies de clé pour différents repositories
Utilisez HasEntityKey lorsque :
- Vous contrôlez la classe d'entité
- La logique d'extraction de clé est complexe
- Vous voulez que l'entité définisse sa propre identité
Intégration UI
Repository s'intègre avec des composants conscients des données :
// Créer repository et table
List<Customer> customers = new ArrayList<>();
CollectionRepository<Customer> repository = new CollectionRepository<>(customers);
Table<Customer> table = new Table<>();
table.setRepository(repository);
table.addColumn("ID", Customer::getId);
table.addColumn("Nom", Customer::getName);
table.addColumn("Email", Customer::getEmail);
// Ajouter des données - la table se met à jour automatiquement
customers.add(new Customer("C001", "Alice Johnson", "alice@example.com"));
repository.commit();