74f0f981d8
Le contact principal (firstName, lastName, phonePrimary, phoneSecondary, email) n'est plus porte par l'entite Client : les contacts vivent uniquement dans ClientContact (onglet Contact). RG-1.01 et RG-1.02 supprimees du Client (equivalent RG-1.05 / RG-1.14 sur ClientContact). - Migration (namespace racine DoctrineMigrations, ordre par timestamp) : backfill des clients sans contact vers client_contact (position 0) puis DROP des 5 colonnes inline. down() best-effort documente. - Entite Client : retrait des 5 props + getters/setters + groupes. - ClientProcessor : MAIN_FIELDS / changedBusinessFields / normalize alleges, validateMainContact (RG-1.01) supprimee. - Recherche repertoire : companyName seul (D1). - Export XLSX : colonnes de contact retirees (D2). - Fixtures + catalogue de commentaires SQL alignes. - Tests fonctionnels et unitaires mis a jour.
227 lines
7.5 KiB
PHP
227 lines
7.5 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace App\Module\Commercial\Infrastructure\Doctrine;
|
|
|
|
use App\Module\Commercial\Domain\Entity\Client;
|
|
use App\Module\Commercial\Domain\Repository\ClientRepositoryInterface;
|
|
use Doctrine\Bundle\DoctrineBundle\Repository\ServiceEntityRepository;
|
|
use Doctrine\ORM\QueryBuilder;
|
|
use Doctrine\Persistence\ManagerRegistry;
|
|
|
|
/**
|
|
* @extends ServiceEntityRepository<Client>
|
|
*/
|
|
class DoctrineClientRepository extends ServiceEntityRepository implements ClientRepositoryInterface
|
|
{
|
|
public function __construct(ManagerRegistry $registry)
|
|
{
|
|
parent::__construct($registry, Client::class);
|
|
}
|
|
|
|
public function findById(int $id): ?Client
|
|
{
|
|
return $this->find($id);
|
|
}
|
|
|
|
public function save(Client $client): void
|
|
{
|
|
$this->getEntityManager()->persist($client);
|
|
$this->getEntityManager()->flush();
|
|
}
|
|
|
|
public function createListQueryBuilder(
|
|
bool $includeArchived = false,
|
|
?string $search = null,
|
|
array $categoryCodes = [],
|
|
array $siteIds = [],
|
|
bool $archivedOnly = false,
|
|
): QueryBuilder {
|
|
// SELECTION uniquement (filtres + tri) : pas de fetch-join to-many ici.
|
|
// L'hydratation des collections affichees (Catégories / Site(s)) est
|
|
// deleguee a hydrateListCollections() une fois le jeu borne, pour ne pas
|
|
// imposer un produit cartesien aux chemins non pagines (export,
|
|
// ?pagination=false) — ERP-100.
|
|
$qb = $this->createQueryBuilder('c')
|
|
->andWhere('c.deletedAt IS NULL')
|
|
->orderBy('c.companyName', 'ASC')
|
|
;
|
|
|
|
// Perimetre d'archivage : archivedOnly prioritaire sur includeArchived.
|
|
if ($archivedOnly) {
|
|
$qb->andWhere('c.isArchived = true');
|
|
} elseif (!$includeArchived) {
|
|
$qb->andWhere('c.isArchived = false');
|
|
}
|
|
|
|
$this->applySearch($qb, $search);
|
|
$this->applyCategoryCodes($qb, $categoryCodes);
|
|
$this->applySiteIds($qb, $siteIds);
|
|
|
|
return $qb;
|
|
}
|
|
|
|
public function hydrateListCollections(array $clients): void
|
|
{
|
|
if ([] === $clients) {
|
|
return;
|
|
}
|
|
|
|
// Ids des clients deja charges (entites managees). On rehydrate leurs
|
|
// collections via l'identity map : les requetes ci-dessous renvoient les
|
|
// MEMES instances Client, dont les collections sont alors remplies.
|
|
$ids = [];
|
|
foreach ($clients as $client) {
|
|
$id = $client->getId();
|
|
if (null !== $id) {
|
|
$ids[] = $id;
|
|
}
|
|
}
|
|
if ([] === $ids) {
|
|
return;
|
|
}
|
|
|
|
// 1re passe : categories (colonne « Catégories »). Produit c x cat seul.
|
|
$this->createQueryBuilder('c')
|
|
->leftJoin('c.categories', 'cat')->addSelect('cat')
|
|
->where('c.id IN (:ids)')->setParameter('ids', $ids)
|
|
->getQuery()
|
|
->getResult()
|
|
;
|
|
|
|
// 2e passe : adresses + sites (colonne « Site(s) », sites portes par les
|
|
// adresses — RG-1.10). Le join addr -> site reste imbrique mais n'est
|
|
// plus multiplie par les categories : le cartesien global est casse.
|
|
$this->createQueryBuilder('c')
|
|
->leftJoin('c.addresses', 'addr')->addSelect('addr')
|
|
->leftJoin('addr.sites', 'site')->addSelect('site')
|
|
->where('c.id IN (:ids)')->setParameter('ids', $ids)
|
|
->getQuery()
|
|
->getResult()
|
|
;
|
|
}
|
|
|
|
/**
|
|
* Recherche fuzzy insensible a la casse sur companyName (D1, refonte contact).
|
|
* Depuis la suppression du contact inline du Client, la recherche ne porte
|
|
* plus que sur le nom d'entreprise (les anciens criteres lastName / email
|
|
* vivaient sur les colonnes inline disparues). Les metacaracteres LIKE
|
|
* (%, _, \) saisis sont echappes pour rester litteraux.
|
|
*/
|
|
private function applySearch(QueryBuilder $qb, ?string $search): void
|
|
{
|
|
if (null === $search || '' === trim($search)) {
|
|
return;
|
|
}
|
|
|
|
$escaped = str_replace(['\\', '%', '_'], ['\\\\', '\%', '\_'], trim($search));
|
|
$pattern = '%'.mb_strtolower($escaped, 'UTF-8').'%';
|
|
|
|
$qb->andWhere('LOWER(c.companyName) LIKE :search')
|
|
->setParameter('search', $pattern)
|
|
;
|
|
}
|
|
|
|
/**
|
|
* Restreint aux clients possedant au moins une categorie dont le code figure
|
|
* dans la liste (OR — ERP-78). Alimente le filtre « Catégories » du drawer
|
|
* (multi) ainsi que les selects « distributeur »/« courtier » (un seul code,
|
|
* RG-1.03). Sous-requete IN (plutot qu'un JOIN sur la collection M2M) pour ne
|
|
* pas perturber le DISTINCT / ORDER BY principal.
|
|
*
|
|
* @param list<string> $categoryCodes
|
|
*/
|
|
private function applyCategoryCodes(QueryBuilder $qb, array $categoryCodes): void
|
|
{
|
|
$codes = $this->normalizeStringList($categoryCodes);
|
|
if ([] === $codes) {
|
|
return;
|
|
}
|
|
|
|
$sub = $this->getEntityManager()->createQueryBuilder()
|
|
->select('c2.id')
|
|
->from(Client::class, 'c2')
|
|
->join('c2.categories', 'cat2')
|
|
->where('cat2.code IN (:categoryCodes)')
|
|
;
|
|
|
|
$qb->andWhere($qb->expr()->in('c.id', $sub->getDQL()))
|
|
->setParameter('categoryCodes', $codes)
|
|
;
|
|
}
|
|
|
|
/**
|
|
* Restreint aux clients ayant au moins une adresse rattachee a l'un des
|
|
* sites donnes (OR — RG-1.10 : les sites vivent sur les adresses, pas sur le
|
|
* client). Sous-requete IN pour ne pas perturber le tri/pagination principal.
|
|
*
|
|
* @param list<int> $siteIds
|
|
*/
|
|
private function applySiteIds(QueryBuilder $qb, array $siteIds): void
|
|
{
|
|
$ids = $this->normalizeIntList($siteIds);
|
|
if ([] === $ids) {
|
|
return;
|
|
}
|
|
|
|
$sub = $this->getEntityManager()->createQueryBuilder()
|
|
->select('c3.id')
|
|
->from(Client::class, 'c3')
|
|
->join('c3.addresses', 'addr3')
|
|
->join('addr3.sites', 'site3')
|
|
->where('site3.id IN (:siteIds)')
|
|
;
|
|
|
|
$qb->andWhere($qb->expr()->in('c.id', $sub->getDQL()))
|
|
->setParameter('siteIds', $ids)
|
|
;
|
|
}
|
|
|
|
/**
|
|
* Nettoie une liste de chaines : trim, retrait des vides, reindexation.
|
|
* Defensive : tolere des elements scalaires non-string (cast) et ignore le
|
|
* reste sans lever de TypeError, le contrat etant justement de normaliser une
|
|
* entree potentiellement brute (query params).
|
|
*
|
|
* @param array<mixed> $values
|
|
*
|
|
* @return list<string>
|
|
*/
|
|
private function normalizeStringList(array $values): array
|
|
{
|
|
$out = [];
|
|
foreach ($values as $value) {
|
|
if (is_string($value) || is_int($value) || is_float($value)) {
|
|
$trimmed = trim((string) $value);
|
|
if ('' !== $trimmed) {
|
|
$out[] = $trimmed;
|
|
}
|
|
}
|
|
}
|
|
|
|
return $out;
|
|
}
|
|
|
|
/**
|
|
* Nettoie une liste d'identifiants : cast int, retrait des <= 0, reindexation.
|
|
* Defensive (cf. normalizeStringList) : accepte des entiers ou des chaines
|
|
* numeriques ('1', '2') sans TypeError, ignore le reste.
|
|
*
|
|
* @param array<mixed> $values
|
|
*
|
|
* @return list<int>
|
|
*/
|
|
private function normalizeIntList(array $values): array
|
|
{
|
|
$out = [];
|
|
foreach ($values as $value) {
|
|
if (is_numeric($value) && (int) $value > 0) {
|
|
$out[] = (int) $value;
|
|
}
|
|
}
|
|
|
|
return $out;
|
|
}
|
|
}
|