mirror of
				https://gitlab.com/Chill-Projet/chill-bundles.git
				synced 2025-10-25 06:32:50 +00:00 
			
		
		
		
	Compare commits
	
		
			232 Commits
		
	
	
		
			2.9.0
			...
			244-record
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f5f6eb78a2 | |||
| 4bbad4fc61 | |||
| 86613a9be9 | |||
| 21bd6478ad | |||
| 5849d8d670 | |||
| 568ee079b5 | |||
| bf97b2a50c | |||
| 01785ed494 | |||
| 97d401b7f6 | |||
| 44ccfe92b6 | |||
| b6ea857389 | |||
| f8840d89bf | |||
| 0c9010f065 | |||
| 3871299346 | |||
| e2e0b08210 | |||
| 4df0542932 | |||
| 13854e59de | |||
| 574ad42a76 | |||
| 4736fca679 | |||
| 32ae2f8f0d | |||
| d58c0a867d | |||
| 15f8432ce0 | |||
| ae7637acc6 | |||
| ce391a6de8 | |||
| 950835c10b | |||
| 9ba557a5bf | |||
| 439fecd69f | |||
| f02168950f | |||
| 58c2235b88 | |||
| 42c5577027 | |||
| 036fe8d6f8 | |||
| 51ebc253aa | |||
| 4fdc7fd210 | |||
| 0bf6c07e8d | |||
| 7a12602699 | |||
| 15a927a9f8 | |||
| 0a2805f23f | |||
| 27ce322690 | |||
| fdfc3fb7ec | |||
| aee245cd04 | |||
| adcf183469 | |||
| 86bf5eba4f | |||
| db130ef9df | |||
| 2b903c4d6e | |||
| 2c6b5dfee1 | |||
| 0b50cbfe4c | |||
| 3a6a125bb1 | |||
| e67c5d98ef | |||
| 3c8e59e088 | |||
| 60ede58af0 | |||
| afb79141f0 | |||
| 4d72933edb | |||
| 469e379166 | |||
| f103b228e4 | |||
| d2a31de1be | |||
| 138a537d2b | |||
| c06c861e17 | |||
| 34cbd2605c | |||
| 044bab45ad | |||
| b9890d1302 | |||
| 5b2a2a1bc5 | |||
| 0da12cdc2e | |||
| d1ce99e6d3 | |||
|  | 0ad27328c9 | ||
| 91e6b035bd | |||
| 2adb6105eb | |||
| 33d187f329 | |||
| f02873c6e0 | |||
| 8cc93a8b07 | |||
| f7184ca7bb | |||
| dab80a84d8 | |||
| 68e00dc42f | |||
| 5fae49821f | |||
| 0e599a99a7 | |||
| f0605c6b08 | |||
| 54606403b4 | |||
| 47d829d72d | |||
| fa3b305ab9 | |||
| b26b7a2706 | |||
| fe54e51362 | |||
| 179e3e92ed | |||
| c00c6066a9 | |||
| f424c5464f | |||
| b17b2a8cfb | |||
| 00de657cae | |||
| 673518e0eb | |||
| 4700a0fef7 | |||
| f4d258882f | |||
| fffc4a9c33 | |||
| a344f36592 | |||
| eef93b8a0d | |||
| 75b78d3c99 | |||
| 3d4c439be4 | |||
| 4727a57825 | |||
| 5f441eb5ac | |||
| 2e4e5ee79a | |||
| 1467c708f2 | |||
| 170bb9586d | |||
| c704ffa379 | |||
| 947b7b90e2 | |||
| 992f7761bb | |||
| 35170e1f7c | |||
| 7132dfa3f6 | |||
| 7e09e0ea54 | |||
| ccf8cc4d6e | |||
| 63124f8f92 | |||
| 75d80ebd98 | |||
| 0ea6f36297 | |||
| 975ea417b7 | |||
| af8e02f76b | |||
| cbaeb3d7e8 | |||
| f609ddb315 | |||
| d0bceb59dc | |||
| 2883e085ed | |||
| b05ed86d1e | |||
| c855d0badc | |||
| be57c96a2f | |||
| eb01c7c203 | |||
| 53b4747697 | |||
| 89e19502d3 | |||
| ff344dbb0c | |||
| 8719b4dedd | |||
| d8fa743bc9 | |||
| eaa40d6725 | |||
| 1f47f157ea | |||
| 5ab0d3f8da | |||
| 98fd5cae78 | |||
| 2bb29242e4 | |||
| e6cab938c8 | |||
| f5f4d8fcdd | |||
| e9df26c2f7 | |||
| af6bee2497 | |||
| af585bada3 | |||
| f9763b866d | |||
| 7f18a2fb7d | |||
| 3cdad6caff | |||
| 200ab836c7 | |||
| 3e39c0ced7 | |||
| 83c3621c26 | |||
| b790e2fcf1 | |||
| fde6000d0b | |||
| 3892d1e877 | |||
| 6944773868 | |||
| d5bc9d10d5 | |||
| 3743b336a6 | |||
| 3332413fe0 | |||
| 86e659edd4 | |||
| 2e60ba3137 | |||
| 1663c6f7c7 | |||
| d01b6db5dc | |||
| 847c8238e8 | |||
| 2cc5fa06ae | |||
| 0d741ab886 | |||
| 9cb794fef2 | |||
| fd03855ee6 | |||
| ac9897d9e7 | |||
| 88469dbe8e | |||
| cbc83c0e63 | |||
| 27012d842d | |||
| 4112e59af4 | |||
| 1b0a30baf8 | |||
| 9673b369ef | |||
| 43966c4d5a | |||
| c9f438319c | |||
| 58b45c6f1b | |||
| f756c0d71e | |||
| bb1602934c | |||
| 83fe3ec3fc | |||
| abebb79e8b | |||
| 3a269ba953 | |||
| a4a4aa119e | |||
| 139f5a1ff9 | |||
| 1e43772b17 | |||
| c29b356c02 | |||
| 050a2955ca | |||
| a1132cf82f | |||
| db70ce4a61 | |||
| 4d20a46717 | |||
| dd75cb695c | |||
| 8cb9142800 | |||
| 4847724f6f | |||
| 6280453523 | |||
| 68f56671a4 | |||
| dc860d0c46 | |||
| 6fd80f9f2e | |||
| 3ce78d0c16 | |||
| f43bd2bc58 | |||
| ec85f05e0b | |||
| bfe0d9a137 | |||
| cac092c45a | |||
| 39bb06b991 | |||
| b2aa465b03 | |||
| 0fa8944c29 | |||
| 7494dbfc7c | |||
| 642958bde2 | |||
| 57bd6d1be4 | |||
| bbcbbf078f | |||
| ba778b1cf8 | |||
| 728fd642fa | |||
| 4150eb56b4 | |||
| 2927561c02 | |||
| 5188891108 | |||
| c6deb21606 | |||
| cdfb084fe4 | |||
| 5ce21aadce | |||
| b9000a38d3 | |||
| 89a185a34f | |||
| bc6c3a1089 | |||
| 2c50f484f0 | |||
| 4bc1de01d6 | |||
| e21fe70b75 | |||
| 461995b56b | |||
| e635b73256 | |||
| 54ae17a8d2 | |||
| 617d09ab8a | |||
| f8ee2903b2 | |||
| 36b0844e79 | |||
| 7bff5ce39e | |||
| 2a151d13ed | |||
| 4375ecf49a | |||
| a6e930958b | |||
| 5f805626f7 | |||
| 4c9ea740c8 | |||
| 981dc6a959 | |||
| 11fb9bcd0b | |||
| a4edb34668 | |||
| f799fe0649 | |||
| c7bd60a106 | |||
| 9ec5a633ad | |||
| d54d34be7c | |||
| 01292ba9ae | |||
| 794daa5c3e | 
							
								
								
									
										6
									
								
								.changes/unreleased/Feature-20231218-153151.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.changes/unreleased/Feature-20231218-153151.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | kind: Feature | ||||||
|  | body: Create new filter for persons having a participation in an accompanying period | ||||||
|  |   during a certain time span | ||||||
|  | time: 2023-12-18T15:31:51.489901829+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "231" | ||||||
							
								
								
									
										6
									
								
								.changes/unreleased/Feature-20240122-124849.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.changes/unreleased/Feature-20240122-124849.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | kind: Feature | ||||||
|  | body: '[Export][List of accompanyign period] Add two columns: the list of persons | ||||||
|  |   participating to the period, and their ids' | ||||||
|  | time: 2024-01-22T12:48:49.824833412+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "241" | ||||||
							
								
								
									
										5
									
								
								.changes/unreleased/Feature-20240129-133319.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changes/unreleased/Feature-20240129-133319.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | kind: Feature | ||||||
|  | body: 'Add capability to generate export about change of steps of accompanying period, and generate exports for this' | ||||||
|  | time: 2024-01-29T13:33:19.190365565+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "244" | ||||||
							
								
								
									
										5
									
								
								.changes/unreleased/Feature-20240207-103951.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changes/unreleased/Feature-20240207-103951.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | kind: Feature | ||||||
|  | body: 'Export: group accompanying period by person participating' | ||||||
|  | time: 2024-02-07T10:39:51.97331052+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "253" | ||||||
							
								
								
									
										5
									
								
								.changes/unreleased/Feature-20240207-114629.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changes/unreleased/Feature-20240207-114629.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | kind: Feature | ||||||
|  | body: 'Export: add filter for courses not linked to a reference address' | ||||||
|  | time: 2024-02-07T11:46:29.491027007+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "243" | ||||||
							
								
								
									
										6
									
								
								.changes/unreleased/Fixed-20231129-113138.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.changes/unreleased/Fixed-20231129-113138.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | kind: Fixed | ||||||
|  | body: Fix error in logs about wrong typing of eventArgs in onEditNotificationComment | ||||||
|  |   method | ||||||
|  | time: 2023-11-29T11:31:38.933538592+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "220" | ||||||
							
								
								
									
										6
									
								
								.changes/unreleased/Fixed-20240130-140301.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.changes/unreleased/Fixed-20240130-140301.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | kind: Fixed | ||||||
|  | body: Fix the conditions upon which social actions should be optional or required | ||||||
|  |   in relation to social issues within the activity creation form | ||||||
|  | time: 2024-01-30T14:03:01.942955636+01:00 | ||||||
|  | custom: | ||||||
|  |   Issue: "256" | ||||||
							
								
								
									
										20
									
								
								.changes/v2.10.0.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.changes/v2.10.0.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | |||||||
|  | ## v2.10.0 - 2023-10-24 | ||||||
|  | ### Feature | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] Add a filter "grouping accompanying period by opening date" and "grouping accompanying period by closing date" | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] add a filter and aggregator on accompanying period work: group/filter by handling third party | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] add a filter and aggregator on activites: group/filter activities by people participating to the activities | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] add a grouping on accompanying period export: group by activity type associated to at least one activity within the accompanying period | ||||||
|  | * [export] sort filters and aggregators by title | ||||||
|  | * ([#179](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/179)) [export] create a parameter that will force to skip the filtering by center (ACL) when generating an export | ||||||
|  | ### Fixed | ||||||
|  | * ([#177](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/177)) [export] fix date range selection on filter and grouping "by status of the course at date", on accompanying periods | ||||||
|  |  | ||||||
|  | ### Résumé francophone des changements | ||||||
|  |  | ||||||
|  | - Ajout d'un regroupement sur les parcours: par date de cloture et d'ouverture; | ||||||
|  | - Ajouter d'un filtre et regroupement par tiers traitant sur les actions d'accompagnement; | ||||||
|  | - ajout d'un filtre et regroupement par usager participant sur les échanges | ||||||
|  | - ajout d'un regroupement: par type d'activité associé au parcours; | ||||||
|  | - trie les filtre et regroupements par ordre alphabétique dans els exports | ||||||
|  | - ajout d'un paramètre qui permet de désactiver le filtre par centre dans les exports | ||||||
|  | - correction de l'interface de date dans les filtres et regroupements "par statut du parcours à la date" | ||||||
							
								
								
									
										3
									
								
								.changes/v2.10.1.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.10.1.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.10.1 - 2023-10-24 | ||||||
|  | ### Fixed | ||||||
|  | * Fix export controller when generating an export without any data in session  | ||||||
							
								
								
									
										3
									
								
								.changes/v2.10.2.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.10.2.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.10.2 - 2023-10-26 | ||||||
|  | ### Fixed | ||||||
|  | * ([#175](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/175)) Use injection of translator instead of ->get().  | ||||||
							
								
								
									
										3
									
								
								.changes/v2.10.3.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.10.3.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.10.3 - 2023-10-26 | ||||||
|  | ### Fixed | ||||||
|  | * ([#175](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/175)) Replace old method of getting translator with injection of translatorInterface  | ||||||
							
								
								
									
										3
									
								
								.changes/v2.10.4.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.10.4.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.10.4 - 2023-10-26 | ||||||
|  | ### Fixed | ||||||
|  | * Fix null value constraint errors when merging relationships in doubles  | ||||||
							
								
								
									
										4
									
								
								.changes/v2.10.5.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.changes/v2.10.5.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | ## v2.10.5 - 2023-11-05 | ||||||
|  | ### Fixed | ||||||
|  | * ([#183](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/183)) Fix "problem during download" on some filters, which used a wrong data type  | ||||||
|  | * ([#184](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/184)) Fix filter "activity by date"  | ||||||
							
								
								
									
										4
									
								
								.changes/v2.10.6.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.changes/v2.10.6.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | ## v2.10.6 - 2023-11-07 | ||||||
|  | ### Fixed | ||||||
|  | * ([#182](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/182)) Fix merging of double person files. Adjustement relationship sql statement  | ||||||
|  | * ([#185](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/185)) Export: fix aggregator by geographical unit on person: avoid inconsistencies  | ||||||
							
								
								
									
										6
									
								
								.changes/v2.11.0.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								.changes/v2.11.0.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | ## v2.11.0 - 2023-11-07 | ||||||
|  | ### Feature | ||||||
|  | * ([#194](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/194)) Export: add a filter "filter activity by creator job"  | ||||||
|  | ### Fixed | ||||||
|  | * ([#185](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/185)) Export: fix "group accompanying period by geographical unit": take into account the accompanying periods when the period is not located within an unit  | ||||||
|  | * Fix "group activity by creator job" aggregator  | ||||||
							
								
								
									
										26
									
								
								.changes/v2.12.0.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.changes/v2.12.0.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | ## v2.12.0 - 2023-11-15 | ||||||
|  | ### Feature | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add an aggregator "group activities by presence" | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add a filter "filter activity by activity presence" | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add an aggregator "group activities by person" (only for the activities saved in a person context) | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add a new aggregator "group peoples by postal code" | ||||||
|  | * ([#200](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/200)) Export: split export about person on accompanying period work: one with the people associated with the work, another one with the people associated with the accompanying period | ||||||
|  | * ([#204](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/204)) Add 3 new filters and 3 new aggregators for work action creator (with jobs and scopes) | ||||||
|  |  | ||||||
|  | * ([#202](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/202)) Create export for the average duration of social work actions | ||||||
|  | * ([#206](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/206)) Export: add a export which count persons on accompanying period work | ||||||
|  | * ([#206](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/206)) Export: add an export which count persons on activity | ||||||
|  | * ([#203](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/203)) Export: add clauses on the social work start date and end date within the filter "Filter accompanying period by accompanying period work" | ||||||
|  | ### Fixed | ||||||
|  | * Export: fix typo in filter "filter accompanying period work on end date" | ||||||
|  | * ([#189](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/189)) Export: Fix failure in export linked to household | ||||||
|  | * ([#205](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/205)) Fix loading of accompanying period work referrers | ||||||
|  | ### Traduction francophone des principaux changements | ||||||
|  | * export: ajout d'un regroupement "grouper les échanges par présence de l'usager"; | ||||||
|  | * export: ajout d'un filtre "filtre les échanges par présence de l'usager"; | ||||||
|  | * export: ajout d'un regroupement "regrouper les échanges par personne" (seulement pour les échanges enregistrés dans le contexte de l'usager); | ||||||
|  | * export: ajout d'un regroupement "grouper les usagers par codes postaux" | ||||||
|  | * export: séparation des exports sur les actions: dans l'un, les filtres des usagers portent sur les usagers concernés par l'action, dans l'autre, les filtres portent sur les usagers concernés par le parcours de l'action; | ||||||
|  | * export: ajout de 3 nouveaux filtres et regroupements sur le créateur de l'action, son métier et son service; | ||||||
|  | * export: correction de l'export sur les ménages liés aux parcours; | ||||||
|  | * correction du chargement des actions d'accompagnement | ||||||
							
								
								
									
										3
									
								
								.changes/v2.12.1.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.12.1.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.12.1 - 2023-11-16 | ||||||
|  | ### Fixed | ||||||
|  | * ([#208](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/208)) Export: fix loading of form for "filter action by type, goal and result"  | ||||||
							
								
								
									
										9
									
								
								.changes/v2.13.0.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								.changes/v2.13.0.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | |||||||
|  | ## v2.13.0 - 2023-11-21 | ||||||
|  | ### Feature | ||||||
|  | * ([#173](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/173)) Allow user to add a phonenumber to their profile which will be included in automatically generated documents | ||||||
|  | ### Fixed | ||||||
|  | * ([#211](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/211)) Export: fix loading of "Group activity by type" | ||||||
|  | * ([#190](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/190)) Export: fix loading of "group activity by reasons" | ||||||
|  | * ([#213](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/213)) Export: fix usage of some Collection returned instead of array in export filters | ||||||
|  | * ([#215](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/215)) Use only the string 'both' for gender (with a database migration) | ||||||
|  | * ([#212](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/212)) Clean the database to make working the "Group people by gender" aggregator | ||||||
							
								
								
									
										8
									
								
								.changes/v2.14.0.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								.changes/v2.14.0.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | |||||||
|  | ## v2.14.0 - 2023-11-24 | ||||||
|  | ### Feature | ||||||
|  | * ([#161](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/161)) Export: in filter "Filter accompanying period work (social action) by type, goal and result", order the items alphabetically or with the defined order  | ||||||
|  | ### Fixed | ||||||
|  | * ([#141](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/141)) Export: on filter "action by type goals, and results", restore the fields when editing a saved export  | ||||||
|  | * ([#219](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/219)) Export: fix the list of accompanying period work, when the "calc date" is null  | ||||||
|  | * ([#222](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/222)) Fix rendering of custom fields  | ||||||
|  | * Fix various errors in custom fields administration  | ||||||
							
								
								
									
										5
									
								
								.changes/v2.14.1.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changes/v2.14.1.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | ## v2.14.1 - 2023-11-29 | ||||||
|  | ### Fixed | ||||||
|  | * Export: fix list person with custom fields  | ||||||
|  | * ([#100](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/100)) Add a paginator to budget elements (resource and charge types) in the admin  | ||||||
|  | * Fix error in ListEvaluation when "handling agents" are alone  | ||||||
							
								
								
									
										11
									
								
								.changes/v2.15.0.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.changes/v2.15.0.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | |||||||
|  | ## v2.15.0 - 2023-12-11 | ||||||
|  | ### Feature | ||||||
|  | * ([#191](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/191)) Add export "number of household associate with an exchange"  | ||||||
|  | * ([#235](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/235)) Export: add dates on the filter "filter course by activity type"  | ||||||
|  | ### Fixed | ||||||
|  | * ([#214](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/214)) Fix error when posting an empty comment on an accompanying period.  | ||||||
|  | * ([#233](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/233)) Fix "filter evaluation by evaluation type" (and add select2 to the list of evaluation types to pick)  | ||||||
|  | * ([#234](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/234)) Fix "filter aside activity by date" | ||||||
|  |   | ||||||
|  | * ([#228](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/228)) Fix export of activity for people created before the introduction of the createdAt column on person (during v1)  | ||||||
|  | * ([#246](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/246)) Do not show activities, evaluations and social work when associated to a confidential accompanying period, except for the users which are allowed to see them  | ||||||
							
								
								
									
										5
									
								
								.changes/v2.15.1.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changes/v2.15.1.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | ## v2.15.1 - 2023-12-20 | ||||||
|  | ### Fixed | ||||||
|  | * Fix the household export query to exclude accompanying periods that are in draft state.  | ||||||
|  | ### DX | ||||||
|  | * ([#167](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/167)) Fixed readthedocs compilation by updating readthedocs config file and requirements for Sphinx  | ||||||
							
								
								
									
										5
									
								
								.changes/v2.15.2.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								.changes/v2.15.2.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | ## v2.15.2 - 2024-01-11 | ||||||
|  | ### Fixed | ||||||
|  | * Fix the id_seq used when creating a new accompanying period participation during fusion of two person files  | ||||||
|  | ### DX | ||||||
|  | * Set placeholder to False for expanded EntityType form fields where required is set to False.  | ||||||
| @@ -3,6 +3,7 @@ | |||||||
| * ([#147](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/147)) Add history to scopes and to jobs in administrator section. When user job or main scope of user is changed, automaticaly add a new row in history. | * ([#147](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/147)) Add history to scopes and to jobs in administrator section. When user job or main scope of user is changed, automaticaly add a new row in history. | ||||||
| * ([#146](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/146)) Allow closing motives to be identified as 'canceling the accompanying period' + don't take canceled accompanying periods into account | * ([#146](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/146)) Allow closing motives to be identified as 'canceling the accompanying period' + don't take canceled accompanying periods into account | ||||||
| * [export] add an aggregator for activities: group by job scope's creator aggregator | * [export] add an aggregator for activities: group by job scope's creator aggregator | ||||||
|  | * DX: prepare the code for the upgrade to symfony 5.4 | ||||||
|  |  | ||||||
| ### Traductions francophones des principaux changements | ### Traductions francophones des principaux changements | ||||||
|  |  | ||||||
| @@ -11,3 +12,12 @@ | |||||||
| - modification des motifs de cloture des parcours: ajout d'un chanmp "annule le parcours", qui permet d'indiquer que le motif "annule" le parcours. Les parcours annulés n'apparaissent | - modification des motifs de cloture des parcours: ajout d'un chanmp "annule le parcours", qui permet d'indiquer que le motif "annule" le parcours. Les parcours annulés n'apparaissent | ||||||
|   pas dans les statistiques |   pas dans les statistiques | ||||||
| - ajouter d'un regroupement pour les échanges: grouper par métier et service du créateur de l'échange | - ajouter d'un regroupement pour les échanges: grouper par métier et service du créateur de l'échange | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### Possible BC break in configuration | ||||||
|  |  | ||||||
|  | This release remove the use of deprecated package [symfony/templating](https://symfony.com/components/Templating). | ||||||
|  |  | ||||||
|  | If you use this package in your own bundle (usually `src/` directory, or other dependencies), you should add this dependencies in your local composer.json (`composer require symfony/templating`). | ||||||
|  |  | ||||||
|  | But if you do not need this any more, you must ensure that the configuration key `framework.templating` is removed. This is usually located into `config/packages/framework.yaml`. [See here an example](https://gitea.champs-libres.be/Chill-project/chill-skeleton-basic/commit/cc716beaecc239e6a189f3db62ea95f169a37505#diff-df607fe73ff82c569824a7392edf5e760e998efe) | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.changes/v2.9.1.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.9.1.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.9.1 - 2023-10-17 | ||||||
|  | ### Fixed | ||||||
|  | * Fix the handling of activity form when editing or creating an activity in an accompanying period with multiple centers  | ||||||
							
								
								
									
										3
									
								
								.changes/v2.9.2.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.changes/v2.9.2.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | ## v2.9.2 - 2023-10-17 | ||||||
|  | ### Fixed | ||||||
|  | * Fix possible null values in string's entities  | ||||||
| @@ -11,6 +11,10 @@ cache: | |||||||
| services: | services: | ||||||
|     - name: postgis/postgis:14-3.3-alpine |     - name: postgis/postgis:14-3.3-alpine | ||||||
|       alias: db |       alias: db | ||||||
|  |       command: | ||||||
|  |           - postgres | ||||||
|  |           - "-c" | ||||||
|  |           - max_connections=1000 | ||||||
|     - name: redis |     - name: redis | ||||||
|       alias: redis |       alias: redis | ||||||
|  |  | ||||||
| @@ -28,6 +32,10 @@ variables: | |||||||
|     REDIS_PORT: 6379 |     REDIS_PORT: 6379 | ||||||
|     REDIS_URL: redis://redis:6379 |     REDIS_URL: redis://redis:6379 | ||||||
|     DEFAULT_CARRIER_CODE: BE |     DEFAULT_CARRIER_CODE: BE | ||||||
|  |     # force a timezone | ||||||
|  |     TZ: Europe/Brussels | ||||||
|  |     # avoid direct deprecations (using symfony phpunit bridge: https://symfony.com/doc/4.x/components/phpunit_bridge.html#internal-deprecations | ||||||
|  |     SYMFONY_DEPRECATIONS_HELPER: max[total]=99999999&max[self]=0&max[direct]=0&verbose=0 | ||||||
|  |  | ||||||
| stages: | stages: | ||||||
|     - Composer install |     - Composer install | ||||||
| @@ -45,7 +53,7 @@ build: | |||||||
|         paths: |         paths: | ||||||
|             - .cache/ |             - .cache/ | ||||||
|     artifacts: |     artifacts: | ||||||
|         expire_in: 30 min |         expire_in: 1 day | ||||||
|         paths: |         paths: | ||||||
|             - bin |             - bin | ||||||
|             - vendor/ |             - vendor/ | ||||||
| @@ -59,7 +67,7 @@ code_style: | |||||||
|         paths: |         paths: | ||||||
|             - .cache/ |             - .cache/ | ||||||
|     artifacts: |     artifacts: | ||||||
|         expire_in: 30 min |         expire_in: 1 day | ||||||
|         paths: |         paths: | ||||||
|             - bin |             - bin | ||||||
|             - vendor/ |             - vendor/ | ||||||
| @@ -73,7 +81,7 @@ phpstan_tests: | |||||||
|         paths: |         paths: | ||||||
|             - .cache/ |             - .cache/ | ||||||
|     artifacts: |     artifacts: | ||||||
|         expire_in: 30 min |         expire_in: 1 day | ||||||
|         paths: |         paths: | ||||||
|             - bin |             - bin | ||||||
|             - vendor/ |             - vendor/ | ||||||
| @@ -88,7 +96,7 @@ rector_tests: | |||||||
|         paths: |         paths: | ||||||
|             - .cache/ |             - .cache/ | ||||||
|     artifacts: |     artifacts: | ||||||
|         expire_in: 30 min |         expire_in: 1 day | ||||||
|         paths: |         paths: | ||||||
|             - bin |             - bin | ||||||
|             - vendor/ |             - vendor/ | ||||||
| @@ -115,7 +123,7 @@ unit_tests: | |||||||
|         - php -d memory_limit=3G tests/console doctrine:fixtures:load -n |         - php -d memory_limit=3G tests/console doctrine:fixtures:load -n | ||||||
|         - php -d memory_limit=4G bin/phpunit --colors=never --exclude-group dbIntensive |         - php -d memory_limit=4G bin/phpunit --colors=never --exclude-group dbIntensive | ||||||
|     artifacts: |     artifacts: | ||||||
|         expire_in: 30 min |         expire_in: 1 day | ||||||
|         paths: |         paths: | ||||||
|             - bin |             - bin | ||||||
|             - vendor/ |             - vendor/ | ||||||
|   | |||||||
| @@ -1,10 +1,14 @@ | |||||||
| --- | --- | ||||||
| version: 2 | version: 2 | ||||||
|  |  | ||||||
|  | build: | ||||||
|  |   os: ubuntu-22.04 | ||||||
|  |   tools: | ||||||
|  |     python: "3.7" | ||||||
|  |  | ||||||
| sphinx: | sphinx: | ||||||
|   configuration: docs/source/conf.py |   configuration: docs/source/conf.py | ||||||
|  |  | ||||||
| python: | python: | ||||||
|   version: 3.7 |  | ||||||
|   install: |   install: | ||||||
|     - requirements: docs/requirements.txt |     - requirements: docs/requirements.txt | ||||||
							
								
								
									
										152
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										152
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @@ -6,11 +6,154 @@ adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html), | |||||||
| and is generated by [Changie](https://github.com/miniscruff/changie). | and is generated by [Changie](https://github.com/miniscruff/changie). | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## v2.15.2 - 2024-01-11 | ||||||
|  | ### Fixed | ||||||
|  | * Fix the id_seq used when creating a new accompanying period participation during fusion of two person files  | ||||||
|  | ### DX | ||||||
|  | * Set placeholder to False for expanded EntityType form fields where required is set to False.  | ||||||
|  |  | ||||||
|  | ## v2.15.1 - 2023-12-20 | ||||||
|  | ### Fixed | ||||||
|  | * Fix the household export query to exclude accompanying periods that are in draft state.  | ||||||
|  | ### DX | ||||||
|  | * ([#167](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/167)) Fixed readthedocs compilation by updating readthedocs config file and requirements for Sphinx  | ||||||
|  |  | ||||||
|  | ## v2.15.0 - 2023-12-11 | ||||||
|  | ### Feature | ||||||
|  | * ([#191](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/191)) Add export "number of household associate with an exchange"  | ||||||
|  | * ([#235](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/235)) Export: add dates on the filter "filter course by activity type"  | ||||||
|  | ### Fixed | ||||||
|  | * ([#214](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/214)) Fix error when posting an empty comment on an accompanying period.  | ||||||
|  | * ([#233](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/233)) Fix "filter evaluation by evaluation type" (and add select2 to the list of evaluation types to pick)  | ||||||
|  | * ([#234](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/234)) Fix "filter aside activity by date" | ||||||
|  |   | ||||||
|  | * ([#228](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/228)) Fix export of activity for people created before the introduction of the createdAt column on person (during v1)  | ||||||
|  | * ([#246](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/246)) Do not show activities, evaluations and social work when associated to a confidential accompanying period, except for the users which are allowed to see them  | ||||||
|  |  | ||||||
|  | ## v2.14.1 - 2023-11-29 | ||||||
|  | ### Fixed | ||||||
|  | * Export: fix list person with custom fields  | ||||||
|  | * ([#100](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/100)) Add a paginator to budget elements (resource and charge types) in the admin  | ||||||
|  | * Fix error in ListEvaluation when "handling agents" are alone  | ||||||
|  |  | ||||||
|  | ## v2.14.0 - 2023-11-24 | ||||||
|  | ### Feature | ||||||
|  | * ([#161](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/161)) Export: in filter "Filter accompanying period work (social action) by type, goal and result", order the items alphabetically or with the defined order  | ||||||
|  | ### Fixed | ||||||
|  | * ([#141](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/141)) Export: on filter "action by type goals, and results", restore the fields when editing a saved export  | ||||||
|  | * ([#219](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/219)) Export: fix the list of accompanying period work, when the "calc date" is null  | ||||||
|  | * ([#222](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/222)) Fix rendering of custom fields  | ||||||
|  | * Fix various errors in custom fields administration  | ||||||
|  |  | ||||||
|  | ## v2.13.0 - 2023-11-21 | ||||||
|  | ### Feature | ||||||
|  | * ([#173](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/173)) Allow user to add a phonenumber to their profile which will be included in automatically generated documents | ||||||
|  | ### Fixed | ||||||
|  | * ([#211](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/211)) Export: fix loading of "Group activity by type" | ||||||
|  | * ([#190](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/190)) Export: fix loading of "group activity by reasons" | ||||||
|  | * ([#213](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/213)) Export: fix usage of some Collection returned instead of array in export filters | ||||||
|  | * ([#215](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/215)) Use only the string 'both' for gender (with a database migration) | ||||||
|  | * ([#212](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/212)) Clean the database to make working the "Group people by gender" aggregator | ||||||
|  |  | ||||||
|  | ## v2.12.1 - 2023-11-16 | ||||||
|  | ### Fixed | ||||||
|  | * ([#208](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/208)) Export: fix loading of form for "filter action by type, goal and result"  | ||||||
|  |  | ||||||
|  | ## v2.12.0 - 2023-11-15 | ||||||
|  | ### Feature | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add an aggregator "group activities by presence" | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add a filter "filter activity by activity presence" | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add an aggregator "group activities by person" (only for the activities saved in a person context) | ||||||
|  | * ([#199](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/199)) Export: add a new aggregator "group peoples by postal code" | ||||||
|  | * ([#200](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/200)) Export: split export about person on accompanying period work: one with the people associated with the work, another one with the people associated with the accompanying period | ||||||
|  | * ([#204](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/204)) Add 3 new filters and 3 new aggregators for work action creator (with jobs and scopes) | ||||||
|  |  | ||||||
|  | * ([#202](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/202)) Create export for the average duration of social work actions | ||||||
|  | * ([#206](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/206)) Export: add a export which count persons on accompanying period work | ||||||
|  | * ([#206](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/206)) Export: add an export which count persons on activity | ||||||
|  | * ([#203](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/203)) Export: add clauses on the social work start date and end date within the filter "Filter accompanying period by accompanying period work" | ||||||
|  | ### Fixed | ||||||
|  | * Export: fix typo in filter "filter accompanying period work on end date" | ||||||
|  | * ([#189](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/189)) Export: Fix failure in export linked to household | ||||||
|  | * ([#205](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/205)) Fix loading of accompanying period work referrers | ||||||
|  | ### Traduction francophone des principaux changements | ||||||
|  | * export: ajout d'un regroupement "grouper les échanges par présence de l'usager"; | ||||||
|  | * export: ajout d'un filtre "filtre les échanges par présence de l'usager"; | ||||||
|  | * export: ajout d'un regroupement "regrouper les échanges par personne" (seulement pour les échanges enregistrés dans le contexte de l'usager); | ||||||
|  | * export: ajout d'un regroupement "grouper les usagers par codes postaux" | ||||||
|  | * export: séparation des exports sur les actions: dans l'un, les filtres des usagers portent sur les usagers concernés par l'action, dans l'autre, les filtres portent sur les usagers concernés par le parcours de l'action; | ||||||
|  | * export: ajout de 3 nouveaux filtres et regroupements sur le créateur de l'action, son métier et son service; | ||||||
|  | * export: correction de l'export sur les ménages liés aux parcours; | ||||||
|  | * correction du chargement des actions d'accompagnement | ||||||
|  |  | ||||||
|  | ## v2.11.0 - 2023-11-07 | ||||||
|  | ### Feature | ||||||
|  | * ([#194](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/194)) Export: add a filter "filter activity by creator job"  | ||||||
|  | ### Fixed | ||||||
|  | * ([#185](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/185)) Export: fix "group accompanying period by geographical unit": take into account the accompanying periods when the period is not located within an unit  | ||||||
|  | * Fix "group activity by creator job" aggregator  | ||||||
|  |  | ||||||
|  | ## v2.10.6 - 2023-11-07 | ||||||
|  | ### Fixed | ||||||
|  | * ([#182](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/182)) Fix merging of double person files. Adjustement relationship sql statement  | ||||||
|  | * ([#185](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/185)) Export: fix aggregator by geographical unit on person: avoid inconsistencies  | ||||||
|  |  | ||||||
|  | ## v2.10.5 - 2023-11-05 | ||||||
|  | ### Fixed | ||||||
|  | * ([#183](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/183)) Fix "problem during download" on some filters, which used a wrong data type  | ||||||
|  | * ([#184](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/184)) Fix filter "activity by date"  | ||||||
|  |  | ||||||
|  | ## v2.10.4 - 2023-10-26 | ||||||
|  | ### Fixed | ||||||
|  | * Fix null value constraint errors when merging relationships in doubles  | ||||||
|  |  | ||||||
|  | ## v2.10.3 - 2023-10-26 | ||||||
|  | ### Fixed | ||||||
|  | * ([#175](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/175)) Replace old method of getting translator with injection of translatorInterface  | ||||||
|  |  | ||||||
|  | ## v2.10.2 - 2023-10-26 | ||||||
|  | ### Fixed | ||||||
|  | * ([#175](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/175)) Use injection of translator instead of ->get().  | ||||||
|  |  | ||||||
|  | ## v2.10.1 - 2023-10-24 | ||||||
|  | ### Fixed | ||||||
|  | * Fix export controller when generating an export without any data in session  | ||||||
|  |  | ||||||
|  | ## v2.10.0 - 2023-10-24 | ||||||
|  | ### Feature | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] Add a filter "grouping accompanying period by opening date" and "grouping accompanying period by closing date" | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] add a filter and aggregator on accompanying period work: group/filter by handling third party | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] add a filter and aggregator on activites: group/filter activities by people participating to the activities | ||||||
|  | * ([#172](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/172)) [export] add a grouping on accompanying period export: group by activity type associated to at least one activity within the accompanying period | ||||||
|  | * [export] sort filters and aggregators by title | ||||||
|  | * ([#179](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/179)) [export] create a parameter that will force to skip the filtering by center (ACL) when generating an export | ||||||
|  | ### Fixed | ||||||
|  | * ([#177](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/177)) [export] fix date range selection on filter and grouping "by status of the course at date", on accompanying periods | ||||||
|  |  | ||||||
|  | ### Résumé francophone des changements | ||||||
|  |  | ||||||
|  | - Ajout d'un regroupement sur les parcours: par date de cloture et d'ouverture; | ||||||
|  | - Ajouter d'un filtre et regroupement par tiers traitant sur les actions d'accompagnement; | ||||||
|  | - ajout d'un filtre et regroupement par usager participant sur les échanges | ||||||
|  | - ajout d'un regroupement: par type d'activité associé au parcours; | ||||||
|  | - trie les filtre et regroupements par ordre alphabétique dans els exports | ||||||
|  | - ajout d'un paramètre qui permet de désactiver le filtre par centre dans les exports | ||||||
|  | - correction de l'interface de date dans les filtres et regroupements "par statut du parcours à la date" | ||||||
|  |  | ||||||
|  | ## v2.9.2 - 2023-10-17 | ||||||
|  | ### Fixed | ||||||
|  | * Fix possible null values in string's entities  | ||||||
|  |  | ||||||
|  | ## v2.9.1 - 2023-10-17 | ||||||
|  | ### Fixed | ||||||
|  | * Fix the handling of activity form when editing or creating an activity in an accompanying period with multiple centers  | ||||||
|  |  | ||||||
| ## v2.9.0 - 2023-10-17 | ## v2.9.0 - 2023-10-17 | ||||||
| ### Feature | ### Feature | ||||||
| * ([#147](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/147)) Add history to scopes and to jobs in administrator section. When user job or main scope of user is changed, automaticaly add a new row in history. | * ([#147](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/147)) Add history to scopes and to jobs in administrator section. When user job or main scope of user is changed, automaticaly add a new row in history. | ||||||
| * ([#146](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/146)) Allow closing motives to be identified as 'canceling the accompanying period' + don't take canceled accompanying periods into account | * ([#146](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/146)) Allow closing motives to be identified as 'canceling the accompanying period' + don't take canceled accompanying periods into account | ||||||
| * [export] add an aggregator for activities: group by job scope's creator aggregator | * [export] add an aggregator for activities: group by job scope's creator aggregator | ||||||
|  | * DX: prepare the code for the upgrade to symfony 5.4 | ||||||
|  |  | ||||||
| ### Traductions francophones des principaux changements | ### Traductions francophones des principaux changements | ||||||
|  |  | ||||||
| @@ -20,6 +163,15 @@ and is generated by [Changie](https://github.com/miniscruff/changie). | |||||||
|   pas dans les statistiques |   pas dans les statistiques | ||||||
| - ajouter d'un regroupement pour les échanges: grouper par métier et service du créateur de l'échange | - ajouter d'un regroupement pour les échanges: grouper par métier et service du créateur de l'échange | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ### Possible BC break in configuration | ||||||
|  |  | ||||||
|  | This release remove the use of deprecated package [symfony/templating](https://symfony.com/components/Templating). | ||||||
|  |  | ||||||
|  | If you use this package in your own bundle (usually `src/` directory, or other dependencies), you should add this dependencies in your local composer.json (`composer require symfony/templating`). | ||||||
|  |  | ||||||
|  | But if you do not need this any more, you must ensure that the configuration key `framework.templating` is removed. This is usually located into `config/packages/framework.yaml`. [See here an example](https://gitea.champs-libres.be/Chill-project/chill-skeleton-basic/commit/cc716beaecc239e6a189f3db62ea95f169a37505#diff-df607fe73ff82c569824a7392edf5e760e998efe) | ||||||
|  |  | ||||||
| ## v2.8.0 - 2023-10-05 | ## v2.8.0 - 2023-10-05 | ||||||
|  |  | ||||||
| ### Feature | ### Feature | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ To compile this documentation : | |||||||
| Contribute | Contribute | ||||||
| =========== | =========== | ||||||
|  |  | ||||||
| Issue tracker : https://git.framasoft.org/groups/Chill-project/issues | Issue tracker : https://gitlab.com/Chill-Projet/chill-bundles/-/issues | ||||||
|  |  | ||||||
| Licence | Licence | ||||||
| ======= | ======= | ||||||
|   | |||||||
| @@ -1,6 +1,7 @@ | |||||||
| docutils==0.13.1 | docutils==0.13.1 | ||||||
| Pygments==2.2.0 | Pygments==2.2.0 | ||||||
| sphinx==1.8.5 | sphinx==1.8.5 | ||||||
|  | Jinja2<3.1 | ||||||
| git+https://github.com/fabpot/sphinx-php.git@v2.0.2#egg_name=sphinx-php | git+https://github.com/fabpot/sphinx-php.git@v2.0.2#egg_name=sphinx-php | ||||||
| jsx-lexer===0.0.8 | jsx-lexer===0.0.8 | ||||||
| sphinx_rtd_theme==0.5.0 | sphinx_rtd_theme==0.5.0 | ||||||
|   | |||||||
| @@ -48,8 +48,8 @@ Clone or download the chill-skeleton project and `cd` into the main directory. | |||||||
|  |  | ||||||
| .. code-block:: bash | .. code-block:: bash | ||||||
|  |  | ||||||
|    git clone https://gitlab.com/Chill-Projet/chill-skeleton-basic.git |    git clone https://gitea.champs-libres.be/Chill-project/chill-skeleton-basic.git | ||||||
|    cd chill-app |    cd chill-skeleton-basic | ||||||
|  |  | ||||||
|  |  | ||||||
| As a developer, the code will stay on your computer and will be executed in docker container. To avoid permission problem, the code should be run with the same uid/gid from your current user. This is why we get your current user id with the command ``id -u`` in each following scripts. | As a developer, the code will stay on your computer and will be executed in docker container. To avoid permission problem, the code should be run with the same uid/gid from your current user. This is why we get your current user id with the command ``id -u`` in each following scripts. | ||||||
|   | |||||||
| @@ -1,13 +1,15 @@ | |||||||
|  |  | ||||||
| Configure Chill for calendar sync and SSO with Microsoft Graph (Outlook) | Configure Chill for calendar and absence synchronisation and SSO with Microsoft Graph (Outlook) | ||||||
| ======================================================================== | =============================================================================================== | ||||||
|  |  | ||||||
| Chill offers the possibility to: | Chill offers the possibility to: | ||||||
|  |  | ||||||
| * authenticate users using Microsoft Graph, with relatively small adaptations; | * authenticate users using Microsoft Graph, with relatively small adaptations; | ||||||
| * synchronize calendar in both ways (`see the user manual for a large description of the feature <https://gitea.champs-libres.be/Chill-project/manuals>`_). | * synchronize calendar in both ways (`see the user manual for a large description of the feature <https://gitea.champs-libres.be/Chill-project/manuals>`_). | ||||||
|  |  | ||||||
| Both can be configured separately (synchronising calendars without SSO, or SSO without calendar). When calendar sync is configured without SSL, the user's email address is the key to associate Chill's users with Microsoft's ones. | Both can be configured separately (synchronising calendars without SSO, or SSO without calendar). | ||||||
|  |  | ||||||
|  | Please note that the user's email address is the key to associate Chill's users with Microsoft's ones. | ||||||
|  |  | ||||||
| Configure SSO | Configure SSO | ||||||
| ------------- | ------------- | ||||||
| @@ -186,12 +188,19 @@ Configure chill app | |||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| Configure sync | Configure sync and calendar access | ||||||
| -------------- | ---------------------------------- | ||||||
|  |  | ||||||
| The sync processe might be configured in the same app, or into a different app. | The purpose of this configuration is the following: | ||||||
|  |  | ||||||
| The synchronization processes use Oauth2.0 for authentication and authorization. | - let user read their calendar and shared calendar within Chill (with the same permissions as the one configured in Outlook / Azure); | ||||||
|  | - allow chill instance to write appointment ("Rendez-vous") into their calendar, and invite other users to their appointment; | ||||||
|  | - allow chill instance to be notified if an appoint is added or removed by the user within another interface than Chill: if the appointment match another one created in the Chill interface, the date and time are updated in Chill; | ||||||
|  | - allow chill instance to read the absence of the user and, if set, mark the user as absent in Chill; | ||||||
|  |  | ||||||
|  | The sync processe might be configured in the same app, or into a different app on the Azure side. | ||||||
|  |  | ||||||
|  | The synchronization processes use Oauth 2.0 / OpenID Connect for authentication and authorization. | ||||||
|  |  | ||||||
| .. note:: | .. note:: | ||||||
|  |  | ||||||
| @@ -229,8 +238,9 @@ Some explanation: | |||||||
| The sync daemon must have write access: | The sync daemon must have write access: | ||||||
|  |  | ||||||
| * the daemon must be allowed to read all users and their profile, to establish a link between them and the Chill's users: (:code:`Users.Read.All`); | * the daemon must be allowed to read all users and their profile, to establish a link between them and the Chill's users: (:code:`Users.Read.All`); | ||||||
| * it must also be allowed to read and write into the calendars (:code:`Calendars.ReadWrite.All`) | * it must also be allowed to read and write into the calendars (:code:`Calendars.ReadWrite.All`); | ||||||
| * for sending invitation to other users, the permission (:code:`Mail.Send`) must be granted. | * for sending invitation to other users, the permission (:code:`Mail.Send`) must be granted; | ||||||
|  | * and, for reading the absence status of the user and sync it with chill, it must be able to read the mailboxSettings (:code:`MailboxSettings.Read`). | ||||||
|  |  | ||||||
| At this step, you might choose to accept those permissions for all users, or let them do it by yourself. | At this step, you might choose to accept those permissions for all users, or let them do it by yourself. | ||||||
|  |  | ||||||
|   | |||||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 149 KiB After Width: | Height: | Size: 166 KiB | 
| @@ -6,7 +6,9 @@ Add condition with distinct alias on each export join clauses (Indicators + Filt | |||||||
| These are alias conventions : | These are alias conventions : | ||||||
|  |  | ||||||
| | Entity                                  | Join                                    | Attribute                                  | Alias                                      | | | Entity                                  | Join                                    | Attribute                                  | Alias                                      | | ||||||
| |:----------------------------------------|:----------------------------------------|:-------------------------------------------|:---------------------------------------| | |:----------------------------------------|:----------------------------------------|:-------------------------------------------|:-------------------------------------------| | ||||||
|  | | AccompanyingPeriodStepHistory::class    |                                         |                                            | acpstephistory (contexte ACP_STEP_HISTORY) | | ||||||
|  | |                                         | AccompanyingPeriod::class               | acpstephistory.period                      | acp                                        | | ||||||
| | AccompanyingPeriod::class               |                                         |                                            | acp                                        | | | AccompanyingPeriod::class               |                                         |                                            | acp                                        | | ||||||
| |                                         | AccompanyingPeriodWork::class           | acp.works                                  | acpw                                       | | |                                         | AccompanyingPeriodWork::class           | acp.works                                  | acpw                                       | | ||||||
| |                                         | AccompanyingPeriodParticipation::class  | acp.participations                         | acppart                                    | | |                                         | AccompanyingPeriodParticipation::class  | acp.participations                         | acppart                                    | | ||||||
|   | |||||||
| @@ -28,3 +28,4 @@ Version 1.5.5 | |||||||
|  |  | ||||||
| - [activity] replace dropdown for selecting reasons and use chillEntity for reason rendering | - [activity] replace dropdown for selecting reasons and use chillEntity for reason rendering | ||||||
| - fix bug: error when trying to edit activity of which the type has been deactivated | - fix bug: error when trying to edit activity of which the type has been deactivated | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,4 +13,6 @@ namespace Chill\ActivityBundle; | |||||||
|  |  | ||||||
| use Symfony\Component\HttpKernel\Bundle\Bundle; | use Symfony\Component\HttpKernel\Bundle\Bundle; | ||||||
|  |  | ||||||
| class ChillActivityBundle extends Bundle {} | class ChillActivityBundle extends Bundle | ||||||
|  | { | ||||||
|  | } | ||||||
|   | |||||||
| @@ -67,7 +67,8 @@ final class ActivityController extends AbstractController | |||||||
|         private readonly FilterOrderHelperFactoryInterface $filterOrderHelperFactory, |         private readonly FilterOrderHelperFactoryInterface $filterOrderHelperFactory, | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper, |         private readonly TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|         private readonly PaginatorFactory $paginatorFactory, |         private readonly PaginatorFactory $paginatorFactory, | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Deletes a Activity entity. |      * Deletes a Activity entity. | ||||||
|   | |||||||
| @@ -56,7 +56,7 @@ class ActivityReasonCategoryController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entity = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReasonCategory::class)->find($id); |         $entity = $em->getRepository(ActivityReasonCategory::class)->find($id); | ||||||
|  |  | ||||||
|         if (!$entity) { |         if (!$entity) { | ||||||
|             throw $this->createNotFoundException('Unable to find ActivityReasonCategory entity.'); |             throw $this->createNotFoundException('Unable to find ActivityReasonCategory entity.'); | ||||||
| @@ -79,7 +79,7 @@ class ActivityReasonCategoryController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entities = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReasonCategory::class)->findAll(); |         $entities = $em->getRepository(ActivityReasonCategory::class)->findAll(); | ||||||
|  |  | ||||||
|         return $this->render('@ChillActivity/ActivityReasonCategory/index.html.twig', [ |         return $this->render('@ChillActivity/ActivityReasonCategory/index.html.twig', [ | ||||||
|             'entities' => $entities, |             'entities' => $entities, | ||||||
| @@ -111,7 +111,7 @@ class ActivityReasonCategoryController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entity = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReasonCategory::class)->find($id); |         $entity = $em->getRepository(ActivityReasonCategory::class)->find($id); | ||||||
|  |  | ||||||
|         if (!$entity) { |         if (!$entity) { | ||||||
|             throw $this->createNotFoundException('Unable to find ActivityReasonCategory entity.'); |             throw $this->createNotFoundException('Unable to find ActivityReasonCategory entity.'); | ||||||
| @@ -131,7 +131,7 @@ class ActivityReasonCategoryController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entity = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReasonCategory::class)->find($id); |         $entity = $em->getRepository(ActivityReasonCategory::class)->find($id); | ||||||
|  |  | ||||||
|         if (!$entity) { |         if (!$entity) { | ||||||
|             throw $this->createNotFoundException('Unable to find ActivityReasonCategory entity.'); |             throw $this->createNotFoundException('Unable to find ActivityReasonCategory entity.'); | ||||||
|   | |||||||
| @@ -24,7 +24,9 @@ use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; | |||||||
|  */ |  */ | ||||||
| class ActivityReasonController extends AbstractController | class ActivityReasonController extends AbstractController | ||||||
| { | { | ||||||
|     public function __construct(private readonly ActivityReasonRepository $activityReasonRepository) {} |     public function __construct(private readonly ActivityReasonRepository $activityReasonRepository) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Creates a new ActivityReason entity. |      * Creates a new ActivityReason entity. | ||||||
| @@ -60,7 +62,7 @@ class ActivityReasonController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entity = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReason::class)->find($id); |         $entity = $em->getRepository(ActivityReason::class)->find($id); | ||||||
|  |  | ||||||
|         if (null === $entity) { |         if (null === $entity) { | ||||||
|             throw new NotFoundHttpException('Unable to find ActivityReason entity.'); |             throw new NotFoundHttpException('Unable to find ActivityReason entity.'); | ||||||
| @@ -115,7 +117,7 @@ class ActivityReasonController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entity = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReason::class)->find($id); |         $entity = $em->getRepository(ActivityReason::class)->find($id); | ||||||
|  |  | ||||||
|         if (!$entity) { |         if (!$entity) { | ||||||
|             throw $this->createNotFoundException('Unable to find ActivityReason entity.'); |             throw $this->createNotFoundException('Unable to find ActivityReason entity.'); | ||||||
| @@ -135,7 +137,7 @@ class ActivityReasonController extends AbstractController | |||||||
|     { |     { | ||||||
|         $em = $this->getDoctrine()->getManager(); |         $em = $this->getDoctrine()->getManager(); | ||||||
|  |  | ||||||
|         $entity = $em->getRepository(\Chill\ActivityBundle\Entity\ActivityReason::class)->find($id); |         $entity = $em->getRepository(ActivityReason::class)->find($id); | ||||||
|  |  | ||||||
|         if (!$entity) { |         if (!$entity) { | ||||||
|             throw $this->createNotFoundException('Unable to find ActivityReason entity.'); |             throw $this->createNotFoundException('Unable to find ActivityReason entity.'); | ||||||
|   | |||||||
| @@ -291,7 +291,11 @@ class ActivityType | |||||||
|     public function checkSocialActionsVisibility(ExecutionContextInterface $context, mixed $payload) |     public function checkSocialActionsVisibility(ExecutionContextInterface $context, mixed $payload) | ||||||
|     { |     { | ||||||
|         if ($this->socialIssuesVisible !== $this->socialActionsVisible) { |         if ($this->socialIssuesVisible !== $this->socialActionsVisible) { | ||||||
|             if (!(2 === $this->socialIssuesVisible && 1 === $this->socialActionsVisible)) { |             // if social issues are invisible then social actions cannot be optional or required + if social issues are optional then social actions shouldn't be required | ||||||
|  |             if ( | ||||||
|  |                 (0 === $this->socialIssuesVisible && (1 === $this->socialActionsVisible || 2 === $this->socialActionsVisible)) | ||||||
|  |                 || (1 === $this->socialIssuesVisible && 2 === $this->socialActionsVisible) | ||||||
|  |             ) { | ||||||
|                 $context |                 $context | ||||||
|                     ->buildViolation('The socialActionsVisible value is not compatible with the socialIssuesVisible value') |                     ->buildViolation('The socialActionsVisible value is not compatible with the socialIssuesVisible value') | ||||||
|                     ->atPath('socialActionsVisible') |                     ->atPath('socialActionsVisible') | ||||||
|   | |||||||
| @@ -19,7 +19,9 @@ use Doctrine\ORM\EntityManagerInterface; | |||||||
|  |  | ||||||
| class ActivityEntityListener | class ActivityEntityListener | ||||||
| { | { | ||||||
|     public function __construct(private readonly EntityManagerInterface $em, private readonly AccompanyingPeriodWorkRepository $workRepository) {} |     public function __construct(private readonly EntityManagerInterface $em, private readonly AccompanyingPeriodWorkRepository $workRepository) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function persistActionToCourse(Activity $activity) |     public function persistActionToCourse(Activity $activity) | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -0,0 +1,123 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Aggregator\ACPAggregators; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Entity\Activity; | ||||||
|  | use Chill\ActivityBundle\Entity\ActivityType; | ||||||
|  | use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||||
|  | use Chill\MainBundle\Export\AggregatorInterface; | ||||||
|  | use Chill\MainBundle\Form\Type\PickRollingDateType; | ||||||
|  | use Chill\MainBundle\Service\RollingDate\RollingDateConverterInterface; | ||||||
|  | use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||||
|  | use Chill\PersonBundle\Export\Declarations; | ||||||
|  | use Doctrine\ORM\Query\Expr\Join; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | final readonly class ByActivityTypeAggregator implements AggregatorInterface | ||||||
|  | { | ||||||
|  |     private const PREFIX = 'acp_by_activity_type_agg'; | ||||||
|  |  | ||||||
|  |     public function __construct( | ||||||
|  |         private RollingDateConverterInterface $rollingDateConverter, | ||||||
|  |         private ActivityTypeRepositoryInterface $activityTypeRepository, | ||||||
|  |         private TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|  |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |         $builder | ||||||
|  |             ->add('after_date', PickRollingDateType::class, [ | ||||||
|  |                 'required' => false, | ||||||
|  |                 'label' => 'export.aggregator.acp.by_activity_type.after_date', | ||||||
|  |             ]) | ||||||
|  |             ->add('before_date', PickRollingDateType::class, [ | ||||||
|  |                 'required' => false, | ||||||
|  |                 'label' => 'export.aggregator.acp.by_activity_type.before_date', | ||||||
|  |             ]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             'before_date' => null, | ||||||
|  |             'after_date' => null, | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, mixed $data) | ||||||
|  |     { | ||||||
|  |         return function (int|string|null $value): string { | ||||||
|  |             if ('_header' === $value) { | ||||||
|  |                 return 'export.aggregator.acp.by_activity_type.activity_type'; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if ('' === $value || null === $value || null === $activityType = $this->activityTypeRepository->find($value)) { | ||||||
|  |                 return ''; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return $this->translatableStringHelper->localize($activityType->getName()); | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data) | ||||||
|  |     { | ||||||
|  |         return [self::PREFIX.'_actype_id']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle() | ||||||
|  |     { | ||||||
|  |         return 'export.aggregator.acp.by_activity_type.title'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|  |     { | ||||||
|  |         $p = self::PREFIX; | ||||||
|  |  | ||||||
|  |         // we make a left join, with acp having at least one activity of the given type | ||||||
|  |         $exists = 'EXISTS (SELECT 1 FROM '.Activity::class." {$p}_activity WHERE {$p}_activity.accompanyingPeriod = acp AND {$p}_activity.activityType = {$p}_activity_type"; | ||||||
|  |  | ||||||
|  |         if (null !== $data['after_date']) { | ||||||
|  |             $exists .= " AND {$p}_activity.date > :{$p}_after_date"; | ||||||
|  |             $qb->setParameter("{$p}_after_date", $this->rollingDateConverter->convert($data['after_date'])); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (null !== $data['before_date']) { | ||||||
|  |             $exists .= " AND {$p}_activity.date < :{$p}_before_date"; | ||||||
|  |             $qb->setParameter("{$p}_before_date", $this->rollingDateConverter->convert($data['before_date'])); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $exists .= ')'; | ||||||
|  |  | ||||||
|  |         $qb->leftJoin( | ||||||
|  |             ActivityType::class, | ||||||
|  |             "{$p}_activity_type", | ||||||
|  |             Join::WITH, | ||||||
|  |             $exists | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         $qb | ||||||
|  |             ->addSelect("{$p}_activity_type.id AS {$p}_actype_id") | ||||||
|  |             ->addGroupBy("{$p}_actype_id"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn() | ||||||
|  |     { | ||||||
|  |         return Declarations::ACP_TYPE; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class BySocialActionAggregator implements AggregatorInterface | class BySocialActionAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly SocialActionRender $actionRender, private readonly SocialActionRepository $actionRepository) {} |     public function __construct(private readonly SocialActionRender $actionRender, private readonly SocialActionRepository $actionRepository) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class BySocialIssueAggregator implements AggregatorInterface | class BySocialIssueAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly SocialIssueRepository $issueRepository, private readonly SocialIssueRender $issueRender) {} |     public function __construct(private readonly SocialIssueRepository $issueRepository, private readonly SocialIssueRender $issueRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -0,0 +1,76 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Aggregator; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Repository\ActivityPresenceRepositoryInterface; | ||||||
|  | use Chill\MainBundle\Export\AggregatorInterface; | ||||||
|  | use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | final readonly class ActivityPresenceAggregator implements AggregatorInterface | ||||||
|  | { | ||||||
|  |     public function __construct(private ActivityPresenceRepositoryInterface $activityPresenceRepository, private TranslatableStringHelperInterface $translatableStringHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, mixed $data) | ||||||
|  |     { | ||||||
|  |         return function (int|string|null $value): string { | ||||||
|  |             if ('_header' === $value) { | ||||||
|  |                 return 'export.aggregator.activity.by_activity_presence.header'; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (null === $value || '' === $value || null === $presence = $this->activityPresenceRepository->find($value)) { | ||||||
|  |                 return ''; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return $this->translatableStringHelper->localize($presence->getName()); | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data) | ||||||
|  |     { | ||||||
|  |         return ['activity_presence_aggregator_attendee']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle(): string | ||||||
|  |     { | ||||||
|  |         return 'export.aggregator.activity.by_activity_presence.Group activity by presence'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data): void | ||||||
|  |     { | ||||||
|  |         $qb->addSelect('IDENTITY(activity.attendee) AS activity_presence_aggregator_attendee'); | ||||||
|  |         $qb->addGroupBy('activity_presence_aggregator_attendee'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn() | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -22,7 +22,9 @@ class ActivityTypeAggregator implements AggregatorInterface | |||||||
| { | { | ||||||
|     final public const KEY = 'activity_type_aggregator'; |     final public const KEY = 'activity_type_aggregator'; | ||||||
|  |  | ||||||
|     public function __construct(protected ActivityTypeRepositoryInterface $activityTypeRepository, protected TranslatableStringHelperInterface $translatableStringHelper) {} |     public function __construct(protected ActivityTypeRepositoryInterface $activityTypeRepository, protected TranslatableStringHelperInterface $translatableStringHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -56,20 +58,15 @@ class ActivityTypeAggregator implements AggregatorInterface | |||||||
|  |  | ||||||
|     public function getLabels($key, array $values, $data): \Closure |     public function getLabels($key, array $values, $data): \Closure | ||||||
|     { |     { | ||||||
|         // for performance reason, we load data from db only once |         return function (int|string|null $value): string { | ||||||
|         $this->activityTypeRepository->findBy(['id' => $values]); |  | ||||||
|  |  | ||||||
|         return function ($value): string { |  | ||||||
|             if ('_header' === $value) { |             if ('_header' === $value) { | ||||||
|                 return 'Activity type'; |                 return 'Activity type'; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             if (null === $value || '' === $value) { |             if (null === $value || '' === $value || null === $t = $this->activityTypeRepository->find($value)) { | ||||||
|                 return ''; |                 return ''; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             $t = $this->activityTypeRepository->find($value); |  | ||||||
|  |  | ||||||
|             return $this->translatableStringHelper->localize($t->getName()); |             return $this->translatableStringHelper->localize($t->getName()); | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -22,7 +22,9 @@ class ActivityUserAggregator implements AggregatorInterface | |||||||
| { | { | ||||||
|     final public const KEY = 'activity_user_id'; |     final public const KEY = 'activity_user_id'; | ||||||
|  |  | ||||||
|     public function __construct(private readonly UserRepository $userRepository, private readonly UserRender $userRender) {} |     public function __construct(private readonly UserRepository $userRepository, private readonly UserRender $userRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class ActivityUsersAggregator implements AggregatorInterface | class ActivityUsersAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly UserRepositoryInterface $userRepository, private readonly UserRender $userRender) {} |     public function __construct(private readonly UserRepositoryInterface $userRepository, private readonly UserRender $userRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -27,7 +27,8 @@ class ActivityUsersJobAggregator implements AggregatorInterface | |||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly UserJobRepositoryInterface $userJobRepository, |         private readonly UserJobRepositoryInterface $userJobRepository, | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper |         private readonly TranslatableStringHelperInterface $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -65,7 +66,9 @@ class ActivityUsersJobAggregator implements AggregatorInterface | |||||||
|         return Declarations::ACTIVITY; |         return Declarations::ACTIVITY; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -27,7 +27,8 @@ class ActivityUsersScopeAggregator implements AggregatorInterface | |||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly ScopeRepositoryInterface $scopeRepository, |         private readonly ScopeRepositoryInterface $scopeRepository, | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper |         private readonly TranslatableStringHelperInterface $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -65,7 +66,9 @@ class ActivityUsersScopeAggregator implements AggregatorInterface | |||||||
|         return Declarations::ACTIVITY; |         return Declarations::ACTIVITY; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class ByCreatorAggregator implements AggregatorInterface | class ByCreatorAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly UserRepositoryInterface $userRepository, private readonly UserRender $userRender) {} |     public function __construct(private readonly UserRepositoryInterface $userRepository, private readonly UserRender $userRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class ByThirdpartyAggregator implements AggregatorInterface | class ByThirdpartyAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly ThirdPartyRepository $thirdPartyRepository, private readonly ThirdPartyRender $thirdPartyRender) {} |     public function __construct(private readonly ThirdPartyRepository $thirdPartyRepository, private readonly ThirdPartyRender $thirdPartyRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -14,20 +14,21 @@ namespace Chill\ActivityBundle\Export\Aggregator; | |||||||
| use Chill\ActivityBundle\Export\Declarations; | use Chill\ActivityBundle\Export\Declarations; | ||||||
| use Chill\MainBundle\Entity\User\UserJobHistory; | use Chill\MainBundle\Entity\User\UserJobHistory; | ||||||
| use Chill\MainBundle\Export\AggregatorInterface; | use Chill\MainBundle\Export\AggregatorInterface; | ||||||
| use Chill\MainBundle\Repository\ScopeRepository; | use Chill\MainBundle\Repository\UserJobRepositoryInterface; | ||||||
| use Chill\MainBundle\Templating\TranslatableStringHelper; | use Chill\MainBundle\Templating\TranslatableStringHelper; | ||||||
| use Doctrine\ORM\Query\Expr\Join; | use Doctrine\ORM\Query\Expr\Join; | ||||||
| use Doctrine\ORM\QueryBuilder; | use Doctrine\ORM\QueryBuilder; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
| 
 | 
 | ||||||
| class JobScopeAggregator implements AggregatorInterface | class CreatorJobAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     private const PREFIX = 'acp_agg_creator_job'; |     private const PREFIX = 'acp_agg_creator_job'; | ||||||
| 
 | 
 | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly ScopeRepository $scopeRepository, |         private readonly UserJobRepositoryInterface $userJobRepository, | ||||||
|         private readonly TranslatableStringHelper $translatableStringHelper |         private readonly TranslatableStringHelper $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -65,7 +66,9 @@ class JobScopeAggregator implements AggregatorInterface | |||||||
|         return Declarations::ACTIVITY; |         return Declarations::ACTIVITY; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
| @@ -76,17 +79,15 @@ class JobScopeAggregator implements AggregatorInterface | |||||||
|     { |     { | ||||||
|         return function ($value): string { |         return function ($value): string { | ||||||
|             if ('_header' === $value) { |             if ('_header' === $value) { | ||||||
|                 return 'Scope'; |                 return 'Job'; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             if (null === $value || '' === $value) { |             if (null === $value || '' === $value || null === $s = $this->userJobRepository->find($value)) { | ||||||
|                 return ''; |                 return ''; | ||||||
|             } |             } | ||||||
| 
 | 
 | ||||||
|             $s = $this->scopeRepository->find($value); |  | ||||||
| 
 |  | ||||||
|             return $this->translatableStringHelper->localize( |             return $this->translatableStringHelper->localize( | ||||||
|                 $s->getName() |                 $s->getLabel() | ||||||
|             ); |             ); | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
| @@ -27,7 +27,8 @@ class CreatorScopeAggregator implements AggregatorInterface | |||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly ScopeRepository $scopeRepository, |         private readonly ScopeRepository $scopeRepository, | ||||||
|         private readonly TranslatableStringHelper $translatableStringHelper |         private readonly TranslatableStringHelper $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -65,7 +66,9 @@ class CreatorScopeAggregator implements AggregatorInterface | |||||||
|         return Declarations::ACTIVITY; |         return Declarations::ACTIVITY; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -73,7 +73,6 @@ class DateAggregator implements AggregatorInterface | |||||||
|             'choices' => self::CHOICES, |             'choices' => self::CHOICES, | ||||||
|             'multiple' => false, |             'multiple' => false, | ||||||
|             'expanded' => true, |             'expanded' => true, | ||||||
|             'empty_data' => self::DEFAULT_CHOICE, |  | ||||||
|         ]); |         ]); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class LocationTypeAggregator implements AggregatorInterface | class LocationTypeAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly LocationTypeRepository $locationTypeRepository, private readonly TranslatableStringHelper $translatableStringHelper) {} |     public function __construct(private readonly LocationTypeRepository $locationTypeRepository, private readonly TranslatableStringHelper $translatableStringHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -25,7 +25,9 @@ use Symfony\Component\Validator\Context\ExecutionContextInterface; | |||||||
|  |  | ||||||
| class ActivityReasonAggregator implements AggregatorInterface, ExportElementValidatedInterface | class ActivityReasonAggregator implements AggregatorInterface, ExportElementValidatedInterface | ||||||
| { | { | ||||||
|     public function __construct(protected ActivityReasonCategoryRepository $activityReasonCategoryRepository, protected ActivityReasonRepository $activityReasonRepository, protected TranslatableStringHelper $translatableStringHelper) {} |     public function __construct(protected ActivityReasonCategoryRepository $activityReasonCategoryRepository, protected ActivityReasonRepository $activityReasonRepository, protected TranslatableStringHelper $translatableStringHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -99,12 +101,6 @@ class ActivityReasonAggregator implements AggregatorInterface, ExportElementVali | |||||||
|  |  | ||||||
|     public function getLabels($key, array $values, $data) |     public function getLabels($key, array $values, $data) | ||||||
|     { |     { | ||||||
|         match ($data['level']) { |  | ||||||
|             'reasons' => $this->activityReasonRepository->findBy(['id' => $values]), |  | ||||||
|             'categories' => $this->activityReasonCategoryRepository->findBy(['id' => $values]), |  | ||||||
|             default => throw new \RuntimeException(sprintf("The level data '%s' is invalid.", $data['level'])), |  | ||||||
|         }; |  | ||||||
|  |  | ||||||
|         return function ($value) use ($data) { |         return function ($value) use ($data) { | ||||||
|             if ('_header' === $value) { |             if ('_header' === $value) { | ||||||
|                 return 'reasons' === $data['level'] ? 'Group by reasons' : 'Group by categories of reason'; |                 return 'reasons' === $data['level'] ? 'Group by reasons' : 'Group by categories of reason'; | ||||||
|   | |||||||
| @@ -0,0 +1,67 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Aggregator\PersonAggregators; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\MainBundle\Export\AggregatorInterface; | ||||||
|  | use Chill\PersonBundle\Export\Helper\LabelPersonHelper; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | final readonly class PersonAggregator implements AggregatorInterface | ||||||
|  | { | ||||||
|  |     public function __construct(private LabelPersonHelper $labelPersonHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |         // nothing to add here | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, mixed $data) | ||||||
|  |     { | ||||||
|  |         return $this->labelPersonHelper->getLabel($key, $values, 'export.aggregator.person.by_person.person'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data) | ||||||
|  |     { | ||||||
|  |         return ['activity_by_person_agg']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle() | ||||||
|  |     { | ||||||
|  |         return 'export.aggregator.person.by_person.title'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|  |     { | ||||||
|  |         $qb | ||||||
|  |             ->addSelect('IDENTITY(activity.person) AS activity_by_person_agg') | ||||||
|  |             ->addGroupBy('activity_by_person_agg'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn() | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY_PERSON; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,80 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Aggregator; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Tests\Export\Aggregator\PersonsAggregatorTest; | ||||||
|  | use Chill\MainBundle\Export\AggregatorInterface; | ||||||
|  | use Chill\PersonBundle\Export\Helper\LabelPersonHelper; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @see PersonsAggregatorTest | ||||||
|  |  */ | ||||||
|  | final readonly class PersonsAggregator implements AggregatorInterface | ||||||
|  | { | ||||||
|  |     private const PREFIX = 'act_persons_agg'; | ||||||
|  |  | ||||||
|  |     public function __construct(private LabelPersonHelper $labelPersonHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |         // nothing to add here | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, mixed $data) | ||||||
|  |     { | ||||||
|  |         if ($key !== self::PREFIX.'_pid') { | ||||||
|  |             throw new \UnexpectedValueException('this key should not be handled: '.$key); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return $this->labelPersonHelper->getLabel($key, $values, 'export.aggregator.activity.by_persons.Persons'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data) | ||||||
|  |     { | ||||||
|  |         return [self::PREFIX.'_pid']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle() | ||||||
|  |     { | ||||||
|  |         return 'export.aggregator.activity.by_persons.Group activity by persons'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|  |     { | ||||||
|  |         $p = self::PREFIX; | ||||||
|  |  | ||||||
|  |         $qb | ||||||
|  |             ->leftJoin('activity.persons', "{$p}_p") | ||||||
|  |             ->addSelect("{$p}_p.id AS {$p}_pid") | ||||||
|  |             ->addGroupBy("{$p}_pid"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn() | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -19,7 +19,9 @@ use Symfony\Contracts\Translation\TranslatorInterface; | |||||||
|  |  | ||||||
| class SentReceivedAggregator implements AggregatorInterface | class SentReceivedAggregator implements AggregatorInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly TranslatorInterface $translator) {} |     public function __construct(private readonly TranslatorInterface $translator) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ declare(strict_types=1); | |||||||
|  |  | ||||||
| namespace Chill\ActivityBundle\Export\Export\LinkedToACP; | namespace Chill\ActivityBundle\Export\Export\LinkedToACP; | ||||||
|  |  | ||||||
| use Chill\ActivityBundle\Entity\Activity; |  | ||||||
| use Chill\ActivityBundle\Export\Declarations; | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Repository\ActivityRepository; | ||||||
| use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||||
| use Chill\MainBundle\Export\AccompanyingCourseExportHelper; | use Chill\MainBundle\Export\AccompanyingCourseExportHelper; | ||||||
| use Chill\MainBundle\Export\ExportInterface; | use Chill\MainBundle\Export\ExportInterface; | ||||||
| @@ -21,22 +21,24 @@ use Chill\MainBundle\Export\GroupedExportInterface; | |||||||
| use Chill\PersonBundle\Entity\AccompanyingPeriodParticipation; | use Chill\PersonBundle\Entity\AccompanyingPeriodParticipation; | ||||||
| use Chill\PersonBundle\Entity\Person\PersonCenterHistory; | use Chill\PersonBundle\Entity\Person\PersonCenterHistory; | ||||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||||
| use Doctrine\ORM\EntityManagerInterface; |  | ||||||
| use Doctrine\ORM\EntityRepository; |  | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class AvgActivityDuration implements ExportInterface, GroupedExportInterface | class AvgActivityDuration implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     protected EntityRepository $repository; |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         EntityManagerInterface $em, |         private readonly ActivityRepository $activityRepository, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|     ) { |     ) { | ||||||
|         $this->repository = $em->getRepository(Activity::class); |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
| @@ -91,13 +93,14 @@ class AvgActivityDuration implements ExportInterface, GroupedExportInterface | |||||||
|     { |     { | ||||||
|         $centers = array_map(static fn ($el) => $el['center'], $acl); |         $centers = array_map(static fn ($el) => $el['center'], $acl); | ||||||
|  |  | ||||||
|         $qb = $this->repository->createQueryBuilder('activity'); |         $qb = $this->activityRepository->createQueryBuilder('activity'); | ||||||
|  |  | ||||||
|         $qb |         $qb | ||||||
|             ->join('activity.accompanyingPeriod', 'acp') |             ->join('activity.accompanyingPeriod', 'acp') | ||||||
|             ->select('AVG(activity.durationTime) as export_avg_activity_duration') |             ->select('AVG(activity.durationTime) as export_avg_activity_duration') | ||||||
|             ->andWhere($qb->expr()->isNotNull('activity.durationTime')); |             ->andWhere($qb->expr()->isNotNull('activity.durationTime')); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|                 ->andWhere( |                 ->andWhere( | ||||||
|                     $qb->expr()->exists( |                     $qb->expr()->exists( | ||||||
| @@ -108,6 +111,7 @@ class AvgActivityDuration implements ExportInterface, GroupedExportInterface | |||||||
|                     ) |                     ) | ||||||
|                 ) |                 ) | ||||||
|                 ->setParameter('authorized_centers', $centers); |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,16 +24,21 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | |||||||
| use Doctrine\ORM\EntityManagerInterface; | use Doctrine\ORM\EntityManagerInterface; | ||||||
| use Doctrine\ORM\EntityRepository; | use Doctrine\ORM\EntityRepository; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterface | class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     protected EntityRepository $repository; |     protected EntityRepository $repository; | ||||||
|  |  | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         EntityManagerInterface $em, |         EntityManagerInterface $em, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|     ) { |     ) { | ||||||
|         $this->repository = $em->getRepository(Activity::class); |         $this->repository = $em->getRepository(Activity::class); | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) |     public function buildForm(FormBuilderInterface $builder) | ||||||
| @@ -101,6 +106,7 @@ class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterfac | |||||||
|             ->select('AVG(activity.travelTime) as export_avg_activity_visit_duration') |             ->select('AVG(activity.travelTime) as export_avg_activity_visit_duration') | ||||||
|             ->andWhere($qb->expr()->isNotNull('activity.travelTime')); |             ->andWhere($qb->expr()->isNotNull('activity.travelTime')); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|                 ->andWhere( |                 ->andWhere( | ||||||
|                     $qb->expr()->exists( |                     $qb->expr()->exists( | ||||||
| @@ -111,6 +117,7 @@ class AvgActivityVisitDuration implements ExportInterface, GroupedExportInterfac | |||||||
|                     ) |                     ) | ||||||
|                 ) |                 ) | ||||||
|                 ->setParameter('authorized_centers', $centers); |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,19 +24,26 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | |||||||
| use Doctrine\ORM\EntityManagerInterface; | use Doctrine\ORM\EntityManagerInterface; | ||||||
| use Doctrine\ORM\EntityRepository; | use Doctrine\ORM\EntityRepository; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class CountActivity implements ExportInterface, GroupedExportInterface | class CountActivity implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     protected EntityRepository $repository; |     protected EntityRepository $repository; | ||||||
|  |  | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         EntityManagerInterface $em, |         EntityManagerInterface $em, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|     ) { |     ) { | ||||||
|         $this->repository = $em->getRepository(Activity::class); |         $this->repository = $em->getRepository(Activity::class); | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
| @@ -95,6 +102,7 @@ class CountActivity implements ExportInterface, GroupedExportInterface | |||||||
|             ->createQueryBuilder('activity') |             ->createQueryBuilder('activity') | ||||||
|             ->join('activity.accompanyingPeriod', 'acp'); |             ->join('activity.accompanyingPeriod', 'acp'); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|                 ->andWhere( |                 ->andWhere( | ||||||
|                     $qb->expr()->exists( |                     $qb->expr()->exists( | ||||||
| @@ -105,6 +113,7 @@ class CountActivity implements ExportInterface, GroupedExportInterface | |||||||
|                     ) |                     ) | ||||||
|                 ) |                 ) | ||||||
|                 ->setParameter('authorized_centers', $centers); |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -0,0 +1,149 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Export\LinkedToACP; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Entity\Activity; | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||||
|  | use Chill\MainBundle\Export\AccompanyingCourseExportHelper; | ||||||
|  | use Chill\MainBundle\Export\ExportInterface; | ||||||
|  | use Chill\MainBundle\Export\FormatterInterface; | ||||||
|  | use Chill\MainBundle\Export\GroupedExportInterface; | ||||||
|  | use Chill\PersonBundle\Entity\AccompanyingPeriodParticipation; | ||||||
|  | use Chill\PersonBundle\Entity\Household\HouseholdMember; | ||||||
|  | use Chill\PersonBundle\Entity\Person\PersonCenterHistory; | ||||||
|  | use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||||
|  | use Doctrine\ORM\EntityManagerInterface; | ||||||
|  | use Doctrine\ORM\EntityRepository; | ||||||
|  | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | final readonly class CountHouseholdOnActivity implements ExportInterface, GroupedExportInterface | ||||||
|  | { | ||||||
|  |     private EntityRepository $repository; | ||||||
|  |  | ||||||
|  |     private bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|  |     public function __construct( | ||||||
|  |         EntityManagerInterface $em, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|  |     ) { | ||||||
|  |         $this->repository = $em->getRepository(Activity::class); | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getAllowedFormattersTypes(): array | ||||||
|  |     { | ||||||
|  |         return [FormatterInterface::TYPE_TABULAR]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getDescription(): string | ||||||
|  |     { | ||||||
|  |         return 'export.export.count_household_on_activity.description'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getGroup(): string | ||||||
|  |     { | ||||||
|  |         return 'Exports of activities linked to an accompanying period'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, $data) | ||||||
|  |     { | ||||||
|  |         if ('export_count_activity' !== $key) { | ||||||
|  |             throw new \LogicException("the key {$key} is not used by this export"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return static fn ($value) => '_header' === $value ? 'export.export.count_household_on_activity.header' : $value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data): array | ||||||
|  |     { | ||||||
|  |         return ['export_count_activity']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getResult($query, $data) | ||||||
|  |     { | ||||||
|  |         return $query->getQuery()->getResult(Query::HYDRATE_SCALAR); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle(): string | ||||||
|  |     { | ||||||
|  |         return 'export.export.count_household_on_activity.title'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getType(): string | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||||
|  |     { | ||||||
|  |         $centers = array_map(static fn ($el) => $el['center'], $acl); | ||||||
|  |  | ||||||
|  |         $qb = $this->repository | ||||||
|  |             ->createQueryBuilder('activity') | ||||||
|  |             ->join('activity.persons', 'person') | ||||||
|  |             ->join('activity.accompanyingPeriod', 'acp') | ||||||
|  |             ->join( | ||||||
|  |                 HouseholdMember::class, | ||||||
|  |                 'householdmember', | ||||||
|  |                 Query\Expr\Join::WITH, | ||||||
|  |                 'person.id = IDENTITY(householdmember.person) AND householdmember.startDate <= activity.date AND (householdmember.endDate IS NULL OR householdmember.endDate > activity.date)' | ||||||
|  |             ) | ||||||
|  |             ->join('householdmember.household', 'household'); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|  |             $qb | ||||||
|  |                 ->andWhere( | ||||||
|  |                     $qb->expr()->exists( | ||||||
|  |                         'SELECT 1 FROM '.AccompanyingPeriodParticipation::class.' acl_count_part | ||||||
|  |                     JOIN '.PersonCenterHistory::class.' acl_count_person_history WITH IDENTITY(acl_count_person_history.person) = IDENTITY(acl_count_part.person) | ||||||
|  |                     WHERE acl_count_part.accompanyingPeriod = acp.id AND acl_count_person_history.center IN (:authorized_centers) | ||||||
|  |                     ' | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|  |         $qb->select('COUNT(DISTINCT household.id) as export_count_activity'); | ||||||
|  |  | ||||||
|  |         return $qb; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function requiredRole(): string | ||||||
|  |     { | ||||||
|  |         return ActivityStatsVoter::STATS; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function supportsModifiers(): array | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             Declarations::ACTIVITY, | ||||||
|  |             Declarations::ACTIVITY_ACP, | ||||||
|  |             PersonDeclarations::ACP_TYPE, | ||||||
|  |             PersonDeclarations::PERSON_TYPE, | ||||||
|  |             PersonDeclarations::HOUSEHOLD_TYPE, | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,140 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Export\LinkedToACP; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Entity\Activity; | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||||
|  | use Chill\MainBundle\Export\AccompanyingCourseExportHelper; | ||||||
|  | use Chill\MainBundle\Export\ExportInterface; | ||||||
|  | use Chill\MainBundle\Export\FormatterInterface; | ||||||
|  | use Chill\MainBundle\Export\GroupedExportInterface; | ||||||
|  | use Chill\PersonBundle\Entity\AccompanyingPeriodParticipation; | ||||||
|  | use Chill\PersonBundle\Entity\Person\PersonCenterHistory; | ||||||
|  | use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||||
|  | use Doctrine\ORM\EntityManagerInterface; | ||||||
|  | use Doctrine\ORM\EntityRepository; | ||||||
|  | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | class CountPersonsOnActivity implements ExportInterface, GroupedExportInterface | ||||||
|  | { | ||||||
|  |     protected EntityRepository $repository; | ||||||
|  |  | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|  |     public function __construct( | ||||||
|  |         EntityManagerInterface $em, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|  |     ) { | ||||||
|  |         $this->repository = $em->getRepository(Activity::class); | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getAllowedFormattersTypes(): array | ||||||
|  |     { | ||||||
|  |         return [FormatterInterface::TYPE_TABULAR]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getDescription(): string | ||||||
|  |     { | ||||||
|  |         return 'export.export.count_person_on_activity.description'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getGroup(): string | ||||||
|  |     { | ||||||
|  |         return 'Exports of activities linked to an accompanying period'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, $data) | ||||||
|  |     { | ||||||
|  |         if ('export_count_activity' !== $key) { | ||||||
|  |             throw new \LogicException("the key {$key} is not used by this export"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return static fn ($value) => '_header' === $value ? 'export.export.count_person_on_activity.header' : $value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data): array | ||||||
|  |     { | ||||||
|  |         return ['export_count_activity']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getResult($query, $data) | ||||||
|  |     { | ||||||
|  |         return $query->getQuery()->getResult(Query::HYDRATE_SCALAR); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle(): string | ||||||
|  |     { | ||||||
|  |         return 'export.export.count_person_on_activity.title'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getType(): string | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||||
|  |     { | ||||||
|  |         $centers = array_map(static fn ($el) => $el['center'], $acl); | ||||||
|  |  | ||||||
|  |         $qb = $this->repository | ||||||
|  |             ->createQueryBuilder('activity') | ||||||
|  |             ->join('activity.persons', 'person') | ||||||
|  |             ->join('activity.accompanyingPeriod', 'acp'); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|  |             $qb | ||||||
|  |                 ->andWhere( | ||||||
|  |                     $qb->expr()->exists( | ||||||
|  |                         'SELECT 1 FROM '.AccompanyingPeriodParticipation::class.' acl_count_part | ||||||
|  |                     JOIN '.PersonCenterHistory::class.' acl_count_person_history WITH IDENTITY(acl_count_person_history.person) = IDENTITY(acl_count_part.person) | ||||||
|  |                     WHERE acl_count_part.accompanyingPeriod = acp.id AND acl_count_person_history.center IN (:authorized_centers) | ||||||
|  |                     ' | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|  |         $qb->select('COUNT(DISTINCT person.id) as export_count_activity'); | ||||||
|  |  | ||||||
|  |         return $qb; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function requiredRole(): string | ||||||
|  |     { | ||||||
|  |         return ActivityStatsVoter::STATS; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function supportsModifiers(): array | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             Declarations::ACTIVITY, | ||||||
|  |             Declarations::ACTIVITY_ACP, | ||||||
|  |             PersonDeclarations::ACP_TYPE, | ||||||
|  |             PersonDeclarations::PERSON_TYPE, | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -20,13 +20,19 @@ use Chill\MainBundle\Export\AccompanyingCourseExportHelper; | |||||||
| use Chill\MainBundle\Export\GroupedExportInterface; | use Chill\MainBundle\Export\GroupedExportInterface; | ||||||
| use Chill\MainBundle\Export\Helper\TranslatableStringExportLabelHelper; | use Chill\MainBundle\Export\Helper\TranslatableStringExportLabelHelper; | ||||||
| use Chill\MainBundle\Export\ListInterface; | use Chill\MainBundle\Export\ListInterface; | ||||||
| use Chill\PersonBundle\Entity\Person\PersonCenterHistory; | use Chill\PersonBundle\Export\Helper\FilterListAccompanyingPeriodHelperInterface; | ||||||
| use Doctrine\ORM\EntityManagerInterface; | use Doctrine\ORM\EntityManagerInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class ListActivity implements ListInterface, GroupedExportInterface | final readonly class ListActivity implements ListInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly ListActivityHelper $helper, private readonly EntityManagerInterface $entityManager, private readonly TranslatableStringExportLabelHelper $translatableStringExportLabelHelper) {} |     public function __construct( | ||||||
|  |         private ListActivityHelper $helper, | ||||||
|  |         private EntityManagerInterface $entityManager, | ||||||
|  |         private TranslatableStringExportLabelHelper $translatableStringExportLabelHelper, | ||||||
|  |         private FilterListAccompanyingPeriodHelperInterface $filterListAccompanyingPeriodHelper, | ||||||
|  |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) |     public function buildForm(FormBuilderInterface $builder) | ||||||
|     { |     { | ||||||
| @@ -107,21 +113,15 @@ class ListActivity implements ListInterface, GroupedExportInterface | |||||||
|             ->join('activity.accompanyingPeriod', 'acp') |             ->join('activity.accompanyingPeriod', 'acp') | ||||||
|             ->leftJoin('acp.participations', 'acppart') |             ->leftJoin('acp.participations', 'acppart') | ||||||
|             ->leftJoin('acppart.person', 'person') |             ->leftJoin('acppart.person', 'person') | ||||||
|             ->andWhere('acppart.startDate != acppart.endDate OR acppart.endDate IS NULL') |             ->andWhere('acppart.startDate != acppart.endDate OR acppart.endDate IS NULL'); | ||||||
|             ->andWhere( |  | ||||||
|                 $qb->expr()->exists( |         $this->filterListAccompanyingPeriodHelper->addFilterAccompanyingPeriods($qb, $requiredModifiers, $acl, $data); | ||||||
|                     'SELECT 1 |  | ||||||
|                     FROM '.PersonCenterHistory::class.' acl_count_person_history |         $qb | ||||||
|                     WHERE acl_count_person_history.person = person |  | ||||||
|                     AND acl_count_person_history.center IN (:authorized_centers) |  | ||||||
|                     ' |  | ||||||
|                 ) |  | ||||||
|             ) |  | ||||||
|             // some grouping are necessary |             // some grouping are necessary | ||||||
|             ->addGroupBy('acp.id') |             ->addGroupBy('acp.id') | ||||||
|             ->addOrderBy('activity.date') |             ->addOrderBy('activity.date') | ||||||
|             ->addOrderBy('activity.id') |             ->addOrderBy('activity.id'); | ||||||
|             ->setParameter('authorized_centers', $centers); |  | ||||||
|  |  | ||||||
|         $this->helper->addSelect($qb); |         $this->helper->addSelect($qb); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,16 +24,20 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | |||||||
| use Doctrine\ORM\EntityManagerInterface; | use Doctrine\ORM\EntityManagerInterface; | ||||||
| use Doctrine\ORM\EntityRepository; | use Doctrine\ORM\EntityRepository; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class SumActivityDuration implements ExportInterface, GroupedExportInterface | class SumActivityDuration implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     protected EntityRepository $repository; |     protected EntityRepository $repository; | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         EntityManagerInterface $em, |         EntityManagerInterface $em, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|     ) { |     ) { | ||||||
|         $this->repository = $em->getRepository(Activity::class); |         $this->repository = $em->getRepository(Activity::class); | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) |     public function buildForm(FormBuilderInterface $builder) | ||||||
| @@ -101,6 +105,7 @@ class SumActivityDuration implements ExportInterface, GroupedExportInterface | |||||||
|         $qb->select('SUM(activity.durationTime) as export_sum_activity_duration') |         $qb->select('SUM(activity.durationTime) as export_sum_activity_duration') | ||||||
|             ->andWhere($qb->expr()->isNotNull('activity.durationTime')); |             ->andWhere($qb->expr()->isNotNull('activity.durationTime')); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|                 ->andWhere( |                 ->andWhere( | ||||||
|                     $qb->expr()->exists( |                     $qb->expr()->exists( | ||||||
| @@ -111,6 +116,7 @@ class SumActivityDuration implements ExportInterface, GroupedExportInterface | |||||||
|                     ) |                     ) | ||||||
|                 ) |                 ) | ||||||
|                 ->setParameter('authorized_centers', $centers); |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,16 +24,20 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | |||||||
| use Doctrine\ORM\EntityManagerInterface; | use Doctrine\ORM\EntityManagerInterface; | ||||||
| use Doctrine\ORM\EntityRepository; | use Doctrine\ORM\EntityRepository; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class SumActivityVisitDuration implements ExportInterface, GroupedExportInterface | class SumActivityVisitDuration implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     protected EntityRepository $repository; |     protected EntityRepository $repository; | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         EntityManagerInterface $em, |         EntityManagerInterface $em, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|     ) { |     ) { | ||||||
|         $this->repository = $em->getRepository(Activity::class); |         $this->repository = $em->getRepository(Activity::class); | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) |     public function buildForm(FormBuilderInterface $builder) | ||||||
| @@ -101,6 +105,7 @@ class SumActivityVisitDuration implements ExportInterface, GroupedExportInterfac | |||||||
|         $qb->select('SUM(activity.travelTime) as export_sum_activity_visit_duration') |         $qb->select('SUM(activity.travelTime) as export_sum_activity_visit_duration') | ||||||
|             ->andWhere($qb->expr()->isNotNull('activity.travelTime')); |             ->andWhere($qb->expr()->isNotNull('activity.travelTime')); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|                 ->andWhere( |                 ->andWhere( | ||||||
|                     $qb->expr()->exists( |                     $qb->expr()->exists( | ||||||
| @@ -111,6 +116,7 @@ class SumActivityVisitDuration implements ExportInterface, GroupedExportInterfac | |||||||
|                     ) |                     ) | ||||||
|                 ) |                 ) | ||||||
|                 ->setParameter('authorized_centers', $centers); |                 ->setParameter('authorized_centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); |         AccompanyingCourseExportHelper::addClosingMotiveExclusionClause($qb); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -19,13 +19,23 @@ use Chill\MainBundle\Export\FormatterInterface; | |||||||
| use Chill\MainBundle\Export\GroupedExportInterface; | use Chill\MainBundle\Export\GroupedExportInterface; | ||||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class CountActivity implements ExportInterface, GroupedExportInterface | class CountActivity implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     public function __construct(protected ActivityRepository $activityRepository) {} |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function __construct( | ||||||
|  |         private readonly ActivityRepository $activityRepository, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|  |     ) { | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
| @@ -82,12 +92,13 @@ class CountActivity implements ExportInterface, GroupedExportInterface | |||||||
|  |  | ||||||
|         $qb = $this->activityRepository |         $qb = $this->activityRepository | ||||||
|             ->createQueryBuilder('activity') |             ->createQueryBuilder('activity') | ||||||
|             ->join('activity.person', 'person') |             ->join('activity.person', 'person'); | ||||||
|             ->join('person.centerHistory', 'centerHistory'); |  | ||||||
|  |  | ||||||
|         $qb->select('COUNT(activity.id) as export_count_activity'); |         $qb->select('COUNT(activity.id) as export_count_activity'); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|  |                 ->join('person.centerHistory', 'centerHistory') | ||||||
|                 ->where( |                 ->where( | ||||||
|                     $qb->expr()->andX( |                     $qb->expr()->andX( | ||||||
|                         $qb->expr()->lte('centerHistory.startDate', 'activity.date'), |                         $qb->expr()->lte('centerHistory.startDate', 'activity.date'), | ||||||
| @@ -99,6 +110,7 @@ class CountActivity implements ExportInterface, GroupedExportInterface | |||||||
|                 ) |                 ) | ||||||
|                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) |                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) | ||||||
|                 ->setParameter('centers', $centers); |                 ->setParameter('centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return $qb; |         return $qb; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -0,0 +1,140 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Export\LinkedToPerson; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Repository\ActivityRepository; | ||||||
|  | use Chill\ActivityBundle\Security\Authorization\ActivityStatsVoter; | ||||||
|  | use Chill\MainBundle\Export\ExportInterface; | ||||||
|  | use Chill\MainBundle\Export\FormatterInterface; | ||||||
|  | use Chill\MainBundle\Export\GroupedExportInterface; | ||||||
|  | use Chill\PersonBundle\Entity\Household\HouseholdMember; | ||||||
|  | use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||||
|  | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | final readonly class CountHouseholdOnActivity implements ExportInterface, GroupedExportInterface | ||||||
|  | { | ||||||
|  |     private bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|  |     public function __construct( | ||||||
|  |         private ActivityRepository $activityRepository, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|  |     ) { | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getAllowedFormattersTypes() | ||||||
|  |     { | ||||||
|  |         return [FormatterInterface::TYPE_TABULAR]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getDescription() | ||||||
|  |     { | ||||||
|  |         return 'export.export.count_household_on_activity_person.description'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getGroup(): string | ||||||
|  |     { | ||||||
|  |         return 'Exports of activities linked to a person'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getLabels($key, array $values, $data) | ||||||
|  |     { | ||||||
|  |         if ('export_count_activity' !== $key) { | ||||||
|  |             throw new \LogicException("the key {$key} is not used by this export"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return static fn ($value) => '_header' === $value ? 'export.export.count_household_on_activity_person.header' : $value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getQueryKeys($data) | ||||||
|  |     { | ||||||
|  |         return ['export_count_activity']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getResult($query, $data) | ||||||
|  |     { | ||||||
|  |         return $query->getQuery()->getResult(Query::HYDRATE_SCALAR); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle() | ||||||
|  |     { | ||||||
|  |         return 'export.export.count_household_on_activity_person.title'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getType(): string | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function initiateQuery(array $requiredModifiers, array $acl, array $data = []) | ||||||
|  |     { | ||||||
|  |         $centers = array_map(static fn ($el) => $el['center'], $acl); | ||||||
|  |  | ||||||
|  |         $qb = $this->activityRepository | ||||||
|  |             ->createQueryBuilder('activity') | ||||||
|  |             ->join('activity.person', 'person') | ||||||
|  |             ->join( | ||||||
|  |                 HouseholdMember::class, | ||||||
|  |                 'householdmember', | ||||||
|  |                 Query\Expr\Join::WITH, | ||||||
|  |                 'person = householdmember.person AND householdmember.startDate <= activity.date AND (householdmember.endDate IS NULL OR householdmember.endDate > activity.date)' | ||||||
|  |             ) | ||||||
|  |             ->join('householdmember.household', 'household'); | ||||||
|  |  | ||||||
|  |         $qb->select('COUNT(DISTINCT household.id) as export_count_activity'); | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|  |             $qb | ||||||
|  |                 ->join('person.centerHistory', 'centerHistory') | ||||||
|  |                 ->where( | ||||||
|  |                     $qb->expr()->andX( | ||||||
|  |                         $qb->expr()->lte('centerHistory.startDate', 'activity.date'), | ||||||
|  |                         $qb->expr()->orX( | ||||||
|  |                             $qb->expr()->isNull('centerHistory.endDate'), | ||||||
|  |                             $qb->expr()->gt('centerHistory.endDate', 'activity.date') | ||||||
|  |                         ) | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) | ||||||
|  |                 ->setParameter('centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return $qb; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function requiredRole(): string | ||||||
|  |     { | ||||||
|  |         return ActivityStatsVoter::STATS; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function supportsModifiers() | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             Declarations::ACTIVITY, | ||||||
|  |             Declarations::ACTIVITY_PERSON, | ||||||
|  |             PersonDeclarations::PERSON_TYPE, | ||||||
|  |             PersonDeclarations::HOUSEHOLD_TYPE, | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -23,6 +23,7 @@ use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | |||||||
| use Doctrine\DBAL\Exception\InvalidArgumentException; | use Doctrine\DBAL\Exception\InvalidArgumentException; | ||||||
| use Doctrine\ORM\EntityManagerInterface; | use Doctrine\ORM\EntityManagerInterface; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\Extension\Core\Type\ChoiceType; | use Symfony\Component\Form\Extension\Core\Type\ChoiceType; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
| use Symfony\Component\Validator\Constraints\Callback; | use Symfony\Component\Validator\Constraints\Callback; | ||||||
| @@ -44,8 +45,17 @@ class ListActivity implements ListInterface, GroupedExportInterface | |||||||
|         'person_lastname', |         'person_lastname', | ||||||
|         'person_id', |         'person_id', | ||||||
|     ]; |     ]; | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     public function __construct(protected EntityManagerInterface $entityManager, protected TranslatorInterface $translator, protected TranslatableStringHelperInterface $translatableStringHelper, private readonly ActivityRepository $activityRepository) {} |     public function __construct( | ||||||
|  |         protected EntityManagerInterface $entityManager, | ||||||
|  |         protected TranslatorInterface $translator, | ||||||
|  |         protected TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|  |         private readonly ActivityRepository $activityRepository, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|  |     ) { | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) |     public function buildForm(FormBuilderInterface $builder) | ||||||
|     { |     { | ||||||
| @@ -188,9 +198,10 @@ class ListActivity implements ListInterface, GroupedExportInterface | |||||||
|  |  | ||||||
|         $qb |         $qb | ||||||
|             ->from('ChillActivityBundle:Activity', 'activity') |             ->from('ChillActivityBundle:Activity', 'activity') | ||||||
|             ->join('activity.person', 'actperson') |             ->join('activity.person', 'actperson'); | ||||||
|             ->join('actperson.centerHistory', 'centerHistory'); |  | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|  |             $qb->join('actperson.centerHistory', 'centerHistory'); | ||||||
|             $qb->where( |             $qb->where( | ||||||
|                 $qb->expr()->andX( |                 $qb->expr()->andX( | ||||||
|                     $qb->expr()->lte('centerHistory.startDate', 'activity.date'), |                     $qb->expr()->lte('centerHistory.startDate', 'activity.date'), | ||||||
| @@ -202,6 +213,7 @@ class ListActivity implements ListInterface, GroupedExportInterface | |||||||
|             ) |             ) | ||||||
|                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) |                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) | ||||||
|                 ->setParameter('centers', $centers); |                 ->setParameter('centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         foreach ($this->fields as $f) { |         foreach ($this->fields as $f) { | ||||||
|             if (\in_array($f, $data['fields'], true)) { |             if (\in_array($f, $data['fields'], true)) { | ||||||
|   | |||||||
| @@ -20,6 +20,7 @@ use Chill\MainBundle\Export\FormatterInterface; | |||||||
| use Chill\MainBundle\Export\GroupedExportInterface; | use Chill\MainBundle\Export\GroupedExportInterface; | ||||||
| use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | use Chill\PersonBundle\Export\Declarations as PersonDeclarations; | ||||||
| use Doctrine\ORM\Query; | use Doctrine\ORM\Query; | ||||||
|  | use Symfony\Component\DependencyInjection\ParameterBag\ParameterBagInterface; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -30,19 +31,25 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
| class StatActivityDuration implements ExportInterface, GroupedExportInterface | class StatActivityDuration implements ExportInterface, GroupedExportInterface | ||||||
| { | { | ||||||
|     final public const SUM = 'sum'; |     final public const SUM = 'sum'; | ||||||
|  |     private readonly bool $filterStatsByCenters; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @param string $action the stat to perform |      * @param string $action the stat to perform | ||||||
|      */ |      */ | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly ActivityRepository $activityRepository, |         private readonly ActivityRepository $activityRepository, | ||||||
|  |         ParameterBagInterface $parameterBag, | ||||||
|         /** |         /** | ||||||
|          * The action for this report. |          * The action for this report. | ||||||
|          */ |          */ | ||||||
|         protected string $action = 'sum' |         protected string $action = 'sum' | ||||||
|     ) {} |     ) { | ||||||
|  |         $this->filterStatsByCenters = $parameterBag->get('chill_main')['acl']['filter_stats_by_center']; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
| @@ -119,10 +126,11 @@ class StatActivityDuration implements ExportInterface, GroupedExportInterface | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         $qb->select($select) |         $qb->select($select) | ||||||
|             ->join('activity.person', 'person') |             ->join('activity.person', 'person'); | ||||||
|             ->join('person.centerHistory', 'centerHistory'); |  | ||||||
|  |  | ||||||
|  |         if ($this->filterStatsByCenters) { | ||||||
|             $qb |             $qb | ||||||
|  |                 ->join('person.centerHistory', 'centerHistory') | ||||||
|                 ->where( |                 ->where( | ||||||
|                     $qb->expr()->andX( |                     $qb->expr()->andX( | ||||||
|                         $qb->expr()->lte('centerHistory.startDate', 'activity.date'), |                         $qb->expr()->lte('centerHistory.startDate', 'activity.date'), | ||||||
| @@ -134,6 +142,7 @@ class StatActivityDuration implements ExportInterface, GroupedExportInterface | |||||||
|                 ) |                 ) | ||||||
|                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) |                 ->andWhere($qb->expr()->in('centerHistory.center', ':centers')) | ||||||
|                 ->setParameter('centers', $centers); |                 ->setParameter('centers', $centers); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         return $qb; |         return $qb; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -40,7 +40,8 @@ class ListActivityHelper | |||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper, |         private readonly TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|         private readonly TranslatableStringExportLabelHelper $translatableStringLabelHelper, |         private readonly TranslatableStringExportLabelHelper $translatableStringLabelHelper, | ||||||
|         private readonly UserHelper $userHelper |         private readonly UserHelper $userHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addSelect(QueryBuilder $qb): void |     public function addSelect(QueryBuilder $qb): void | ||||||
|     { |     { | ||||||
| @@ -74,7 +75,9 @@ class ListActivityHelper | |||||||
|             ->addGroupBy('location.id'); |             ->addGroupBy('location.id'); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder) {} |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getAllowedFormattersTypes() |     public function getAllowedFormattersTypes() | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -15,18 +15,24 @@ use Chill\ActivityBundle\Entity\Activity; | |||||||
| use Chill\ActivityBundle\Entity\ActivityType; | use Chill\ActivityBundle\Entity\ActivityType; | ||||||
| use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | use Chill\ActivityBundle\Repository\ActivityTypeRepositoryInterface; | ||||||
| use Chill\MainBundle\Export\FilterInterface; | use Chill\MainBundle\Export\FilterInterface; | ||||||
|  | use Chill\MainBundle\Form\Type\PickRollingDateType; | ||||||
|  | use Chill\MainBundle\Service\RollingDate\RollingDateConverterInterface; | ||||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||||
| use Chill\PersonBundle\Export\Declarations; | use Chill\PersonBundle\Export\Declarations; | ||||||
| use Doctrine\ORM\QueryBuilder; | use Doctrine\ORM\QueryBuilder; | ||||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
| class ActivityTypeFilter implements FilterInterface | final readonly class ActivityTypeFilter implements FilterInterface | ||||||
| { | { | ||||||
|  |     private const BASE_EXISTS = 'SELECT 1 FROM '.Activity::class.' act_type_filter_activity WHERE act_type_filter_activity.accompanyingPeriod = acp'; | ||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly ActivityTypeRepositoryInterface $activityTypeRepository, |         private ActivityTypeRepositoryInterface $activityTypeRepository, | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper |         private TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|     ) {} |         private RollingDateConverterInterface $rollingDateConverter, | ||||||
|  |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -35,15 +41,28 @@ class ActivityTypeFilter implements FilterInterface | |||||||
|  |  | ||||||
|     public function alterQuery(QueryBuilder $qb, $data) |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|     { |     { | ||||||
|         $qb->andWhere( |         $exists = self::BASE_EXISTS; | ||||||
|             $qb->expr()->exists( |  | ||||||
|                 'SELECT 1 FROM '.Activity::class.' act_type_filter_activity |         if (count($data['accepted_activitytypes']) > 0) { | ||||||
|                 WHERE act_type_filter_activity.activityType IN (:act_type_filter_activity_types) AND act_type_filter_activity.accompanyingPeriod = acp' |             $exists .= ' AND act_type_filter_activity.activityType IN (:act_type_filter_activity_types)'; | ||||||
|             ) |  | ||||||
|         ); |  | ||||||
|             $qb->setParameter('act_type_filter_activity_types', $data['accepted_activitytypes']); |             $qb->setParameter('act_type_filter_activity_types', $data['accepted_activitytypes']); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (null !== $data['date_after']) { | ||||||
|  |             $exists .= ' AND act_type_filter_activity.date >= :act_type_filter_activity_date_after'; | ||||||
|  |             $qb->setParameter('act_type_filter_activity_date_after', $this->rollingDateConverter->convert($data['date_after'])); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (null !== $data['date_before']) { | ||||||
|  |             $exists .= ' AND act_type_filter_activity.date >= :act_type_filter_activity_date_before'; | ||||||
|  |             $qb->setParameter('act_type_filter_activity_date_before', $this->rollingDateConverter->convert($data['date_before'])); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (self::BASE_EXISTS !== $exists) { | ||||||
|  |             $qb->andWhere($qb->expr()->exists($exists)); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function applyOn() |     public function applyOn() | ||||||
|     { |     { | ||||||
|         return Declarations::ACP_TYPE; |         return Declarations::ACP_TYPE; | ||||||
| @@ -60,11 +79,27 @@ class ActivityTypeFilter implements FilterInterface | |||||||
|             'multiple' => true, |             'multiple' => true, | ||||||
|             'expanded' => true, |             'expanded' => true, | ||||||
|         ]); |         ]); | ||||||
|  |  | ||||||
|  |         $builder->add('date_after', PickRollingDateType::class, [ | ||||||
|  |             'label' => 'export.filter.activity.acp_by_activity_type.activity after', | ||||||
|  |             'help' => 'export.filter.activity.acp_by_activity_type.activity after help', | ||||||
|  |             'required' => false, | ||||||
|  |         ]); | ||||||
|  |  | ||||||
|  |         $builder->add('date_before', PickRollingDateType::class, [ | ||||||
|  |             'label' => 'export.filter.activity.acp_by_activity_type.activity before', | ||||||
|  |             'help' => 'export.filter.activity.acp_by_activity_type.activity before help', | ||||||
|  |             'required' => false, | ||||||
|  |         ]); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|     { |     { | ||||||
|         return []; |         return [ | ||||||
|  |             'accepted_activitytypes' => [], | ||||||
|  |             'date_after' => null, | ||||||
|  |             'date_before' => null, | ||||||
|  |         ]; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function describeAction($data, $format = 'string'): array |     public function describeAction($data, $format = 'string'): array | ||||||
| @@ -75,8 +110,12 @@ class ActivityTypeFilter implements FilterInterface | |||||||
|             $types[] = $this->translatableStringHelper->localize($aty->getName()); |             $types[] = $this->translatableStringHelper->localize($aty->getName()); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return ['export.filter.activity.acp_by_activity_type.acp_containing_at_least_one_%activitytypes%', [ |         return ['export.filter.activity.acp_by_activity_type.acp_containing_at_least_one_activitytypes', [ | ||||||
|             '%activitytypes%' => implode(', ', $types), |             'activitytypes' => implode(', ', $types), | ||||||
|  |             'has_date_after' => null !== $data['date_after'] ? 1 : 0, | ||||||
|  |             'date_after' => $this->rollingDateConverter->convert($data['date_after']), | ||||||
|  |             'has_date_before' => null !== $data['date_before'] ? 1 : 0, | ||||||
|  |             'date_before' => $this->rollingDateConverter->convert($data['date_before']), | ||||||
|         ]]; |         ]]; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -21,7 +21,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class BySocialActionFilter implements FilterInterface | class BySocialActionFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly SocialActionRender $actionRender) {} |     public function __construct(private readonly SocialActionRender $actionRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -21,7 +21,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class BySocialIssueFilter implements FilterInterface | class BySocialIssueFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly SocialIssueRender $issueRender) {} |     public function __construct(private readonly SocialIssueRender $issueRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -23,7 +23,8 @@ final readonly class PeriodHavingActivityBetweenDatesFilter implements FilterInt | |||||||
| { | { | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         private RollingDateConverterInterface $rollingDateConverter, |         private RollingDateConverterInterface $rollingDateConverter, | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function getTitle() |     public function getTitle() | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -13,21 +13,19 @@ namespace Chill\ActivityBundle\Export\Filter; | |||||||
|  |  | ||||||
| use Chill\ActivityBundle\Export\Declarations; | use Chill\ActivityBundle\Export\Declarations; | ||||||
| use Chill\MainBundle\Export\FilterInterface; | use Chill\MainBundle\Export\FilterInterface; | ||||||
| use Chill\MainBundle\Form\Type\Export\FilterType; |  | ||||||
| use Chill\MainBundle\Form\Type\PickRollingDateType; | use Chill\MainBundle\Form\Type\PickRollingDateType; | ||||||
| use Chill\MainBundle\Service\RollingDate\RollingDate; | use Chill\MainBundle\Service\RollingDate\RollingDate; | ||||||
| use Chill\MainBundle\Service\RollingDate\RollingDateConverterInterface; | use Chill\MainBundle\Service\RollingDate\RollingDateConverterInterface; | ||||||
| use Doctrine\ORM\Query\Expr; | use Doctrine\ORM\Query\Expr; | ||||||
| use Doctrine\ORM\QueryBuilder; | use Doctrine\ORM\QueryBuilder; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
| use Symfony\Component\Form\FormError; |  | ||||||
| use Symfony\Component\Form\FormEvent; |  | ||||||
| use Symfony\Component\Form\FormEvents; |  | ||||||
| use Symfony\Contracts\Translation\TranslatorInterface; | use Symfony\Contracts\Translation\TranslatorInterface; | ||||||
|  |  | ||||||
| class ActivityDateFilter implements FilterInterface | class ActivityDateFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(protected TranslatorInterface $translator, private readonly RollingDateConverterInterface $rollingDateConverter) {} |     public function __construct(protected TranslatorInterface $translator, private readonly RollingDateConverterInterface $rollingDateConverter) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -74,46 +72,6 @@ class ActivityDateFilter implements FilterInterface | |||||||
|             ->add('date_to', PickRollingDateType::class, [ |             ->add('date_to', PickRollingDateType::class, [ | ||||||
|                 'label' => 'Activities before this date', |                 'label' => 'Activities before this date', | ||||||
|             ]); |             ]); | ||||||
|  |  | ||||||
|         $builder->addEventListener(FormEvents::POST_SUBMIT, function (FormEvent $event) { |  | ||||||
|             /** @var \Symfony\Component\Form\FormInterface $filterForm */ |  | ||||||
|             $filterForm = $event->getForm()->getParent(); |  | ||||||
|             $enabled = $filterForm->get(FilterType::ENABLED_FIELD)->getData(); |  | ||||||
|  |  | ||||||
|             if (true === $enabled) { |  | ||||||
|                 // if the filter is enabled, add some validation |  | ||||||
|                 $form = $event->getForm(); |  | ||||||
|                 $date_from = $form->get('date_from')->getData(); |  | ||||||
|                 $date_to = $form->get('date_to')->getData(); |  | ||||||
|  |  | ||||||
|                 // check that fields are not empty |  | ||||||
|                 if (null === $date_from) { |  | ||||||
|                     $form->get('date_from')->addError(new FormError( |  | ||||||
|                         $this->translator->trans('This field ' |  | ||||||
|                         .'should not be empty') |  | ||||||
|                     )); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if (null === $date_to) { |  | ||||||
|                     $form->get('date_to')->addError(new FormError( |  | ||||||
|                         $this->translator->trans('This field ' |  | ||||||
|                         .'should not be empty') |  | ||||||
|                     )); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 // check that date_from is before date_to |  | ||||||
|                 if ( |  | ||||||
|                     (null !== $date_from && null !== $date_to) |  | ||||||
|                     && $date_from >= $date_to |  | ||||||
|                 ) { |  | ||||||
|                     $form->get('date_to')->addError(new FormError( |  | ||||||
|                         $this->translator->trans('This date should be after ' |  | ||||||
|                             .'the date given in "Implied in an activity after ' |  | ||||||
|                             .'this date" field') |  | ||||||
|                     )); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         }); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function getFormDefaultData(): array |     public function getFormDefaultData(): array | ||||||
|   | |||||||
| @@ -0,0 +1,83 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Filter; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Entity\ActivityPresence; | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\MainBundle\Export\FilterInterface; | ||||||
|  | use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||||
|  | use Doctrine\Common\Collections\Collection; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  | use Symfony\Contracts\Translation\TranslatorInterface; | ||||||
|  |  | ||||||
|  | final readonly class ActivityPresenceFilter implements FilterInterface | ||||||
|  | { | ||||||
|  |     public function __construct( | ||||||
|  |         private TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|  |         private TranslatorInterface $translator | ||||||
|  |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle() | ||||||
|  |     { | ||||||
|  |         return 'export.filter.activity.by_presence.Filter activity by activity presence'; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |         $builder->add('presences', EntityType::class, [ | ||||||
|  |             'class' => ActivityPresence::class, | ||||||
|  |             'choice_label' => fn (ActivityPresence $presence) => $this->translatableStringHelper->localize($presence->getName()) | ||||||
|  |                 .($presence->isActive() ? '' : ' ('.$this->translator->trans('inactive').')'), | ||||||
|  |             'multiple' => true, | ||||||
|  |             'expanded' => true, | ||||||
|  |             'label' => 'export.filter.activity.by_presence.presences', | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return []; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function describeAction($data, $format = 'string') | ||||||
|  |     { | ||||||
|  |         $presences = array_map( | ||||||
|  |             fn (ActivityPresence $presence) => $this->translatableStringHelper->localize($presence->getName()), | ||||||
|  |             $data['presences'] instanceof Collection ? $data['presences']->toArray() : $data['presences'] | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         return [ | ||||||
|  |             'export.filter.activity.by_presence.Filtered by activity presence: only %presences%', | ||||||
|  |             ['%presences%' => implode(', ', $presences)], | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|  |     { | ||||||
|  |         $qb | ||||||
|  |             ->andWhere('activity.attendee IN (:activity_presence_filter_presences)') | ||||||
|  |             ->setParameter('activity_presence_filter_presences', $data['presences']); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn() | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -27,7 +27,8 @@ class ActivityTypeFilter implements ExportElementValidatedInterface, FilterInter | |||||||
|     public function __construct( |     public function __construct( | ||||||
|         protected TranslatableStringHelperInterface $translatableStringHelper, |         protected TranslatableStringHelperInterface $translatableStringHelper, | ||||||
|         protected ActivityTypeRepositoryInterface $activityTypeRepository |         protected ActivityTypeRepositoryInterface $activityTypeRepository | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -80,7 +81,7 @@ class ActivityTypeFilter implements ExportElementValidatedInterface, FilterInter | |||||||
|         // collect all the reasons'name used in this filter in one array |         // collect all the reasons'name used in this filter in one array | ||||||
|         $reasonsNames = array_map( |         $reasonsNames = array_map( | ||||||
|             fn (ActivityType $t): string => $this->translatableStringHelper->localize($t->getName()), |             fn (ActivityType $t): string => $this->translatableStringHelper->localize($t->getName()), | ||||||
|             $this->activityTypeRepository->findBy(['id' => $data['types']->toArray()]) |             $this->activityTypeRepository->findBy(['id' => $data['types'] instanceof \Doctrine\Common\Collections\Collection ? $data['types']->toArray() : $data['types']]) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         return ['Filtered by activity type: only %list%', [ |         return ['Filtered by activity type: only %list%', [ | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class ActivityUsersFilter implements FilterInterface | class ActivityUsersFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly UserRender $userRender) {} |     public function __construct(private readonly UserRender $userRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class ByCreatorFilter implements FilterInterface | class ByCreatorFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly UserRender $userRender) {} |     public function __construct(private readonly UserRender $userRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -0,0 +1,117 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Filter; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\MainBundle\Entity\User\UserJobHistory; | ||||||
|  | use Chill\MainBundle\Entity\UserJob; | ||||||
|  | use Chill\MainBundle\Export\FilterInterface; | ||||||
|  | use Chill\MainBundle\Repository\UserJobRepositoryInterface; | ||||||
|  | use Chill\MainBundle\Templating\TranslatableStringHelper; | ||||||
|  | use Doctrine\Common\Collections\Collection; | ||||||
|  | use Doctrine\ORM\Query\Expr\Join; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  | use Symfony\Contracts\Translation\TranslatorInterface; | ||||||
|  |  | ||||||
|  | final readonly class CreatorJobFilter implements FilterInterface | ||||||
|  | { | ||||||
|  |     private const PREFIX = 'acp_act_filter_creator_job'; | ||||||
|  |  | ||||||
|  |     public function __construct( | ||||||
|  |         private TranslatableStringHelper $translatableStringHelper, | ||||||
|  |         private TranslatorInterface $translator, | ||||||
|  |         private UserJobRepositoryInterface $userJobRepository, | ||||||
|  |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|  |     { | ||||||
|  |         $p = self::PREFIX; | ||||||
|  |  | ||||||
|  |         $qb | ||||||
|  |             ->leftJoin('activity.createdBy', "{$p}_user") | ||||||
|  |             ->leftJoin( | ||||||
|  |                 UserJobHistory::class, | ||||||
|  |                 "{$p}_history", | ||||||
|  |                 Join::WITH, | ||||||
|  |                 $qb->expr()->eq("{$p}_history.user", "{$p}_user") | ||||||
|  |             ) | ||||||
|  |             // job_at based on activity.date | ||||||
|  |             ->andWhere( | ||||||
|  |                 $qb->expr()->andX( | ||||||
|  |                     $qb->expr()->lte("{$p}_history.startDate", 'activity.date'), | ||||||
|  |                     $qb->expr()->orX( | ||||||
|  |                         $qb->expr()->isNull("{$p}_history.endDate"), | ||||||
|  |                         $qb->expr()->gt("{$p}_history.endDate", 'activity.date') | ||||||
|  |                     ) | ||||||
|  |                 ) | ||||||
|  |             ) | ||||||
|  |             ->andWhere( | ||||||
|  |                 $qb->expr()->in("{$p}_history.job", ":{$p}_jobs") | ||||||
|  |             ) | ||||||
|  |             ->setParameter( | ||||||
|  |                 "{$p}_jobs", | ||||||
|  |                 $data['jobs'], | ||||||
|  |             ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn(): string | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |         $builder | ||||||
|  |             ->add('jobs', EntityType::class, [ | ||||||
|  |                 'choices' => $this->userJobRepository->findAllOrderedByName(), | ||||||
|  |                 'class' => UserJob::class, | ||||||
|  |                 'choice_label' => fn (UserJob $s) => $this->translatableStringHelper->localize( | ||||||
|  |                     $s->getLabel() | ||||||
|  |                 ).($s->isActive() ? '' : '('.$this->translator->trans('inactive').')'), | ||||||
|  |                 'label' => 'export.filter.activity.by_creator_job.job_form_label', | ||||||
|  |                 'multiple' => true, | ||||||
|  |                 'expanded' => true, | ||||||
|  |             ]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function describeAction($data, $format = 'string'): array | ||||||
|  |     { | ||||||
|  |         $jobs = array_map( | ||||||
|  |             fn (UserJob $job) => $this->translatableStringHelper->localize($job->getLabel()), | ||||||
|  |             $data['jobs'] instanceof Collection ? $data['jobs']->toArray() : $data['jobs'] | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         return ['export.filter.activity.by_creator_job.Filtered activity by user job: only %jobs%', [ | ||||||
|  |             '%jobs%' => implode(', ', $jobs), | ||||||
|  |         ]]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             'jobs' => [], | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle(): string | ||||||
|  |     { | ||||||
|  |         return 'export.filter.activity.by_creator_job.Filter activity by user job'; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -21,13 +21,14 @@ use Doctrine\ORM\QueryBuilder; | |||||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
| 
 | 
 | ||||||
| class UserScopeFilter implements FilterInterface | class CreatorScopeFilter implements FilterInterface | ||||||
| { | { | ||||||
|     private const PREFIX = 'acp_act_filter_user_scope';  // creator ? cfr translation
 |     private const PREFIX = 'acp_act_filter_creator_scope'; | ||||||
| 
 | 
 | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly TranslatableStringHelper $translatableStringHelper |         private readonly TranslatableStringHelper $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -39,7 +40,7 @@ class UserScopeFilter implements FilterInterface | |||||||
|         $p = self::PREFIX; |         $p = self::PREFIX; | ||||||
| 
 | 
 | ||||||
|         $qb |         $qb | ||||||
|             ->leftJoin('activity.user', "{$p}_user")  // createdBy ? cfr translation
 |             ->leftJoin('activity.createdBy', "{$p}_user") | ||||||
|             ->leftJoin( |             ->leftJoin( | ||||||
|                 UserScopeHistory::class, |                 UserScopeHistory::class, | ||||||
|                 "{$p}_history", |                 "{$p}_history", | ||||||
| @@ -28,7 +28,9 @@ class EmergencyFilter implements FilterInterface | |||||||
|  |  | ||||||
|     private const DEFAULT_CHOICE = 'false'; |     private const DEFAULT_CHOICE = 'false'; | ||||||
|  |  | ||||||
|     public function __construct(private readonly TranslatorInterface $translator) {} |     public function __construct(private readonly TranslatorInterface $translator) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -21,7 +21,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class LocationTypeFilter implements FilterInterface | class LocationTypeFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly TranslatableStringHelper $translatableStringHelper) {} |     public function __construct(private readonly TranslatableStringHelper $translatableStringHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ use Chill\ActivityBundle\Repository\ActivityReasonRepository; | |||||||
| use Chill\MainBundle\Export\ExportElementValidatedInterface; | use Chill\MainBundle\Export\ExportElementValidatedInterface; | ||||||
| use Chill\MainBundle\Export\FilterInterface; | use Chill\MainBundle\Export\FilterInterface; | ||||||
| use Chill\MainBundle\Templating\TranslatableStringHelper; | use Chill\MainBundle\Templating\TranslatableStringHelper; | ||||||
|  | use Doctrine\Common\Collections\Collection; | ||||||
| use Doctrine\ORM\Query\Expr; | use Doctrine\ORM\Query\Expr; | ||||||
| use Doctrine\ORM\QueryBuilder; | use Doctrine\ORM\QueryBuilder; | ||||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
| @@ -25,7 +26,9 @@ use Symfony\Component\Validator\Context\ExecutionContextInterface; | |||||||
|  |  | ||||||
| class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInterface | class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(protected TranslatableStringHelper $translatableStringHelper, protected ActivityReasonRepository $activityReasonRepository) {} |     public function __construct(protected TranslatableStringHelper $translatableStringHelper, protected ActivityReasonRepository $activityReasonRepository) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -79,7 +82,7 @@ class ActivityReasonFilter implements ExportElementValidatedInterface, FilterInt | |||||||
|         // collect all the reasons'name used in this filter in one array |         // collect all the reasons'name used in this filter in one array | ||||||
|         $reasonsNames = array_map( |         $reasonsNames = array_map( | ||||||
|             fn (ActivityReason $r): string => '"'.$this->translatableStringHelper->localize($r->getName()).'"', |             fn (ActivityReason $r): string => '"'.$this->translatableStringHelper->localize($r->getName()).'"', | ||||||
|             $this->activityReasonRepository->findBy(['id' => $data['reasons']->toArray()]) |             $this->activityReasonRepository->findBy(['id' => $data['reasons'] instanceof Collection ? $data['reasons']->toArray() : $data['reasons']]) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         return [ |         return [ | ||||||
|   | |||||||
| @@ -32,7 +32,8 @@ final readonly class PersonHavingActivityBetweenDateFilter implements ExportElem | |||||||
|         private TranslatableStringHelper $translatableStringHelper, |         private TranslatableStringHelper $translatableStringHelper, | ||||||
|         private ActivityReasonRepository $activityReasonRepository, |         private ActivityReasonRepository $activityReasonRepository, | ||||||
|         private RollingDateConverterInterface $rollingDateConverter, |         private RollingDateConverterInterface $rollingDateConverter, | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -0,0 +1,89 @@ | |||||||
|  | <?php | ||||||
|  |  | ||||||
|  | declare(strict_types=1); | ||||||
|  |  | ||||||
|  | /* | ||||||
|  |  * Chill is a software for social workers | ||||||
|  |  * | ||||||
|  |  * For the full copyright and license information, please view | ||||||
|  |  * the LICENSE file that was distributed with this source code. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | namespace Chill\ActivityBundle\Export\Filter; | ||||||
|  |  | ||||||
|  | use Chill\ActivityBundle\Export\Declarations; | ||||||
|  | use Chill\ActivityBundle\Tests\Export\Filter\PersonsFilterTest; | ||||||
|  | use Chill\MainBundle\Export\FilterInterface; | ||||||
|  | use Chill\PersonBundle\Form\Type\PickPersonDynamicType; | ||||||
|  | use Chill\PersonBundle\Templating\Entity\PersonRenderInterface; | ||||||
|  | use Doctrine\ORM\QueryBuilder; | ||||||
|  | use Symfony\Component\Form\FormBuilderInterface; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @see PersonsFilterTest | ||||||
|  |  */ | ||||||
|  | final readonly class PersonsFilter implements FilterInterface | ||||||
|  | { | ||||||
|  |     private const PREFIX = 'act_persons_filter'; | ||||||
|  |  | ||||||
|  |     public function __construct(private PersonRenderInterface $personRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function addRole(): ?string | ||||||
|  |     { | ||||||
|  |         return null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function alterQuery(QueryBuilder $qb, $data) | ||||||
|  |     { | ||||||
|  |         $p = self::PREFIX; | ||||||
|  |  | ||||||
|  |         $orX = $qb->expr()->orX(); | ||||||
|  |  | ||||||
|  |         foreach (array_values($data['accepted_persons']) as $key => $person) { | ||||||
|  |             $orX->add($qb->expr()->isMemberOf(":{$p}_p_{$key}", 'activity.persons')); | ||||||
|  |             $qb->setParameter(":{$p}_p_{$key}", $person); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         $qb->andWhere($orX); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function applyOn() | ||||||
|  |     { | ||||||
|  |         return Declarations::ACTIVITY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function buildForm(FormBuilderInterface $builder) | ||||||
|  |     { | ||||||
|  |         $builder->add('accepted_persons', PickPersonDynamicType::class, [ | ||||||
|  |             'multiple' => true, | ||||||
|  |             'label' => 'export.filter.activity.by_persons.persons taking part on the activity', | ||||||
|  |         ]); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getFormDefaultData(): array | ||||||
|  |     { | ||||||
|  |         return [ | ||||||
|  |             'accepted_persons' => [], | ||||||
|  |         ]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function describeAction($data, $format = 'string') | ||||||
|  |     { | ||||||
|  |         $users = []; | ||||||
|  |  | ||||||
|  |         foreach ($data['accepted_persons'] as $u) { | ||||||
|  |             $users[] = $this->personRender->renderString($u, []); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return ['export.filter.activity.by_persons.Filtered activity by persons: only %persons%', [ | ||||||
|  |             '%persons%' => implode(', ', $users), | ||||||
|  |         ]]; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public function getTitle(): string | ||||||
|  |     { | ||||||
|  |         return 'export.filter.activity.by_persons.Filter activity by persons'; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -23,13 +23,15 @@ use Symfony\Contracts\Translation\TranslatorInterface; | |||||||
| class SentReceivedFilter implements FilterInterface | class SentReceivedFilter implements FilterInterface | ||||||
| { | { | ||||||
|     private const CHOICES = [ |     private const CHOICES = [ | ||||||
|         'is sent' => Activity::SENTRECEIVED_SENT, |         'export.filter.activity.by_sent_received.is sent' => Activity::SENTRECEIVED_SENT, | ||||||
|         'is received' => Activity::SENTRECEIVED_RECEIVED, |         'export.filter.activity.by_sent_received.is received' => Activity::SENTRECEIVED_RECEIVED, | ||||||
|     ]; |     ]; | ||||||
|  |  | ||||||
|     private const DEFAULT_CHOICE = Activity::SENTRECEIVED_SENT; |     private const DEFAULT_CHOICE = Activity::SENTRECEIVED_SENT; | ||||||
|  |  | ||||||
|     public function __construct(private readonly TranslatorInterface $translator) {} |     public function __construct(private readonly TranslatorInterface $translator) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -64,6 +66,7 @@ class SentReceivedFilter implements FilterInterface | |||||||
|             'multiple' => false, |             'multiple' => false, | ||||||
|             'expanded' => true, |             'expanded' => true, | ||||||
|             'empty_data' => self::DEFAULT_CHOICE, |             'empty_data' => self::DEFAULT_CHOICE, | ||||||
|  |             'label' => 'export.filter.activity.by_sent_received.Sent or received', | ||||||
|         ]); |         ]); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -21,7 +21,9 @@ use Symfony\Component\Form\FormBuilderInterface; | |||||||
|  |  | ||||||
| class UserFilter implements FilterInterface | class UserFilter implements FilterInterface | ||||||
| { | { | ||||||
|     public function __construct(private readonly UserRender $userRender) {} |     public function __construct(private readonly UserRender $userRender) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -17,6 +17,7 @@ use Chill\MainBundle\Entity\User\UserJobHistory; | |||||||
| use Chill\MainBundle\Entity\UserJob; | use Chill\MainBundle\Entity\UserJob; | ||||||
| use Chill\MainBundle\Export\FilterInterface; | use Chill\MainBundle\Export\FilterInterface; | ||||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||||
|  | use Doctrine\Common\Collections\Collection; | ||||||
| use Doctrine\ORM\QueryBuilder; | use Doctrine\ORM\QueryBuilder; | ||||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
| @@ -27,7 +28,8 @@ class UsersJobFilter implements FilterInterface | |||||||
|  |  | ||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper |         private readonly TranslatableStringHelperInterface $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -80,7 +82,7 @@ class UsersJobFilter implements FilterInterface | |||||||
|                 ', ', |                 ', ', | ||||||
|                 array_map( |                 array_map( | ||||||
|                     fn (UserJob $job) => $this->translatableStringHelper->localize($job->getLabel()), |                     fn (UserJob $job) => $this->translatableStringHelper->localize($job->getLabel()), | ||||||
|                     $data['jobs']->toArray() |                     $data['jobs'] instanceof Collection ? $data['jobs']->toArray() : $data['jobs'] | ||||||
|                 ) |                 ) | ||||||
|             ), |             ), | ||||||
|         ]]; |         ]]; | ||||||
|   | |||||||
| @@ -18,6 +18,7 @@ use Chill\MainBundle\Entity\User\UserScopeHistory; | |||||||
| use Chill\MainBundle\Export\FilterInterface; | use Chill\MainBundle\Export\FilterInterface; | ||||||
| use Chill\MainBundle\Repository\ScopeRepositoryInterface; | use Chill\MainBundle\Repository\ScopeRepositoryInterface; | ||||||
| use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | use Chill\MainBundle\Templating\TranslatableStringHelperInterface; | ||||||
|  | use Doctrine\Common\Collections\Collection; | ||||||
| use Doctrine\ORM\QueryBuilder; | use Doctrine\ORM\QueryBuilder; | ||||||
| use Symfony\Bridge\Doctrine\Form\Type\EntityType; | use Symfony\Bridge\Doctrine\Form\Type\EntityType; | ||||||
| use Symfony\Component\Form\FormBuilderInterface; | use Symfony\Component\Form\FormBuilderInterface; | ||||||
| @@ -29,7 +30,8 @@ class UsersScopeFilter implements FilterInterface | |||||||
|     public function __construct( |     public function __construct( | ||||||
|         private readonly ScopeRepositoryInterface $scopeRepository, |         private readonly ScopeRepositoryInterface $scopeRepository, | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper |         private readonly TranslatableStringHelperInterface $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function addRole(): ?string |     public function addRole(): ?string | ||||||
|     { |     { | ||||||
| @@ -83,7 +85,7 @@ class UsersScopeFilter implements FilterInterface | |||||||
|                 ', ', |                 ', ', | ||||||
|                 array_map( |                 array_map( | ||||||
|                     fn (Scope $s) => $this->translatableStringHelper->localize($s->getName()), |                     fn (Scope $s) => $this->translatableStringHelper->localize($s->getName()), | ||||||
|                     $data['scopes']->toArray() |                     $data['scopes'] instanceof Collection ? $data['scopes']->toArray() : $data['scopes'] | ||||||
|                 ) |                 ) | ||||||
|             ), |             ), | ||||||
|         ]]; |         ]]; | ||||||
|   | |||||||
| @@ -87,8 +87,7 @@ class ActivityType extends AbstractType | |||||||
|         /** @var \Chill\ActivityBundle\Entity\ActivityType $activityType */ |         /** @var \Chill\ActivityBundle\Entity\ActivityType $activityType */ | ||||||
|         $activityType = $options['activityType']; |         $activityType = $options['activityType']; | ||||||
|  |  | ||||||
|         // TODO revoir la gestion des center au niveau du form des activité. |         if (null !== $options['data']->getPerson()) { | ||||||
|         if ($options['center'] instanceof Center && null !== $options['data']->getPerson()) { |  | ||||||
|             $builder->add('scope', ScopePickerType::class, [ |             $builder->add('scope', ScopePickerType::class, [ | ||||||
|                 'center' => $options['center'], |                 'center' => $options['center'], | ||||||
|                 'role' => ActivityVoter::CREATE === (string) $options['role'] ? ActivityVoter::CREATE_PERSON : (string) $options['role'], |                 'role' => ActivityVoter::CREATE === (string) $options['role'] ? ActivityVoter::CREATE_PERSON : (string) $options['role'], | ||||||
| @@ -184,6 +183,7 @@ class ActivityType extends AbstractType | |||||||
|             $builder->add('attendee', EntityType::class, [ |             $builder->add('attendee', EntityType::class, [ | ||||||
|                 'label' => $activityType->getLabel('attendee'), |                 'label' => $activityType->getLabel('attendee'), | ||||||
|                 'required' => $activityType->isRequired('attendee'), |                 'required' => $activityType->isRequired('attendee'), | ||||||
|  |                 'placeholder' => false, | ||||||
|                 'expanded' => true, |                 'expanded' => true, | ||||||
|                 'class' => ActivityPresence::class, |                 'class' => ActivityPresence::class, | ||||||
|                 'choice_label' => fn (ActivityPresence $activityPresence) => $this->translatableStringHelper->localize($activityPresence->getName()), |                 'choice_label' => fn (ActivityPresence $activityPresence) => $this->translatableStringHelper->localize($activityPresence->getName()), | ||||||
| @@ -192,7 +192,7 @@ class ActivityType extends AbstractType | |||||||
|             ]); |             ]); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if ($activityType->isVisible('user') && $options['center'] instanceof Center) { |         if ($activityType->isVisible('user')) { | ||||||
|             $builder->add('user', PickUserDynamicType::class, [ |             $builder->add('user', PickUserDynamicType::class, [ | ||||||
|                 'label' => $activityType->getLabel('user'), |                 'label' => $activityType->getLabel('user'), | ||||||
|                 'required' => $activityType->isRequired('user'), |                 'required' => $activityType->isRequired('user'), | ||||||
| @@ -402,10 +402,10 @@ class ActivityType extends AbstractType | |||||||
|  |  | ||||||
|         $resolver |         $resolver | ||||||
|             ->setRequired(['center', 'role', 'activityType', 'accompanyingPeriod']) |             ->setRequired(['center', 'role', 'activityType', 'accompanyingPeriod']) | ||||||
|             ->setAllowedTypes('center', ['null', Center::class]) |             ->setAllowedTypes('center', ['null', Center::class, 'array']) | ||||||
|             ->setAllowedTypes('role', ['string']) |             ->setAllowedTypes('role', ['string']) | ||||||
|             ->setAllowedTypes('activityType', \Chill\ActivityBundle\Entity\ActivityType::class) |             ->setAllowedTypes('activityType', \Chill\ActivityBundle\Entity\ActivityType::class) | ||||||
|             ->setAllowedTypes('accompanyingPeriod', [\Chill\PersonBundle\Entity\AccompanyingPeriod::class, 'null']); |             ->setAllowedTypes('accompanyingPeriod', [AccompanyingPeriod::class, 'null']); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public function getBlockPrefix(): string |     public function getBlockPrefix(): string | ||||||
|   | |||||||
| @@ -25,7 +25,9 @@ use Symfony\Component\OptionsResolver\OptionsResolver; | |||||||
|  |  | ||||||
| class ActivityTypeType extends AbstractType | class ActivityTypeType extends AbstractType | ||||||
| { | { | ||||||
|     public function __construct(private readonly TranslatableStringHelper $translatableStringHelper) {} |     public function __construct(private readonly TranslatableStringHelper $translatableStringHelper) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function buildForm(FormBuilderInterface $builder, array $options) |     public function buildForm(FormBuilderInterface $builder, array $options) | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -28,7 +28,8 @@ class PickActivityReasonType extends AbstractType | |||||||
|         private readonly ActivityReasonRepository $activityReasonRepository, |         private readonly ActivityReasonRepository $activityReasonRepository, | ||||||
|         private readonly ActivityReasonRender $reasonRender, |         private readonly ActivityReasonRender $reasonRender, | ||||||
|         private readonly TranslatableStringHelperInterface $translatableStringHelper |         private readonly TranslatableStringHelperInterface $translatableStringHelper | ||||||
|     ) {} |     ) { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function configureOptions(OptionsResolver $resolver) |     public function configureOptions(OptionsResolver $resolver) | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -23,7 +23,9 @@ use Symfony\Contracts\Translation\TranslatorInterface; | |||||||
|  */ |  */ | ||||||
| class TranslatableActivityReasonCategoryType extends AbstractType | class TranslatableActivityReasonCategoryType extends AbstractType | ||||||
| { | { | ||||||
|     public function __construct(private readonly TranslatableStringHelperInterface $translatableStringHelper, private readonly TranslatorInterface $translator) {} |     public function __construct(private readonly TranslatableStringHelperInterface $translatableStringHelper, private readonly TranslatorInterface $translator) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function configureOptions(OptionsResolver $resolver) |     public function configureOptions(OptionsResolver $resolver) | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -20,7 +20,9 @@ use Symfony\Component\OptionsResolver\OptionsResolver; | |||||||
|  |  | ||||||
| class TranslatableActivityType extends AbstractType | class TranslatableActivityType extends AbstractType | ||||||
| { | { | ||||||
|     public function __construct(protected TranslatableStringHelperInterface $translatableStringHelper, protected ActivityTypeRepositoryInterface $activityTypeRepository) {} |     public function __construct(protected TranslatableStringHelperInterface $translatableStringHelper, protected ActivityTypeRepositoryInterface $activityTypeRepository) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function configureOptions(OptionsResolver $resolver) |     public function configureOptions(OptionsResolver $resolver) | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -23,7 +23,9 @@ use Symfony\Contracts\Translation\TranslatorInterface; | |||||||
|  */ |  */ | ||||||
| class AccompanyingCourseMenuBuilder implements LocalMenuBuilderInterface | class AccompanyingCourseMenuBuilder implements LocalMenuBuilderInterface | ||||||
| { | { | ||||||
|     public function __construct(protected Security $security, protected TranslatorInterface $translator) {} |     public function __construct(protected Security $security, protected TranslatorInterface $translator) | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|     public function buildMenu($menuId, MenuItem $menu, array $parameters) |     public function buildMenu($menuId, MenuItem $menu, array $parameters) | ||||||
|     { |     { | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user