mirror of
https://gitlab.com/Chill-Projet/chill-bundles.git
synced 2025-09-11 17:25:02 +00:00
Compare commits
279 Commits
v3.4.2
...
321-text-e
Author | SHA1 | Date | |
---|---|---|---|
b5c9e65986
|
|||
8b2af35e97
|
|||
dc44c46667 | |||
ba571c1a69 | |||
6a364705f2 | |||
b6d454691a | |||
6d7a6932a9 | |||
|
2faf194b15 | ||
f207599d86 | |||
b0959f8cc5 | |||
4c5dee5f0a | |||
f6c98aa0d5 | |||
6d13d184d5 | |||
af36eccfaf | |||
483a20a43f | |||
6d8e2ad825 | |||
86388a63a8 | |||
|
5ea55ebfe5 | ||
f97dc8f931 | |||
|
a9c3aab528 | ||
1181377bd6 | |||
|
2275b7c560 | ||
4a8d298ae5 | |||
3e7f03d331
|
|||
b830952b9e
|
|||
ad17313c61 | |||
620515ad15
|
|||
50c377ee22 | |||
cc7e7a90ee | |||
1d4ef19051
|
|||
8337a724d1
|
|||
8ca377d5d4 | |||
224e0bae43 | |||
3aa4fac80d | |||
|
a7517eb647 | ||
e278e636e0 | |||
|
40e373a9c7 | ||
1c1f418b18 | |||
bf0e14b43a | |||
203a098054
|
|||
d58acff541
|
|||
5858e05a42
|
|||
b9b4fafe14
|
|||
fe6949ea26 | |||
8a444a12f4 | |||
8b7b5ceed7
|
|||
b0e826d05a
|
|||
6fb9c3af3f
|
|||
7f101ba616
|
|||
cea82fac10 | |||
1344fc33e1 | |||
c8e09a28e6 | |||
c52d4b2a0e
|
|||
7f326d5441
|
|||
2840c06476 | |||
7ddf84ea5a | |||
f202625ea8
|
|||
7a9168fcdb
|
|||
40eb71f95a | |||
84b7cc8145 | |||
08af530726 | |||
03b2496817
|
|||
f638ce71fd | |||
b39997f00a | |||
38b21a2159 | |||
17db571244 | |||
8c8c16c1a1 | |||
046d3ec9f1 | |||
596833f1a5 | |||
66e4bab558 | |||
d1c9926bb1 | |||
a71d066765 | |||
217ac7b9e7 | |||
be901822bc | |||
2dcce7b826 | |||
7ed10efcd1 | |||
350661a4fa | |||
08207b656a | |||
8de63de6d6 | |||
51804b10c0 | |||
02f555efae | |||
d2fcb6945b | |||
dcd1777a70 | |||
a6eb28175a | |||
c89e3785ef | |||
9f17ec4841 | |||
b277a7749a | |||
c8b6b6e33a | |||
10eaebf610 | |||
7d78512823 | |||
0a34f9086f
|
|||
739e0b1692 | |||
8db8f5fdf5
|
|||
d0cd4792d6 | |||
6d196ead94 | |||
4047d5fd5b | |||
9aac80d834 | |||
7560dc57c6 | |||
10314845f6 | |||
9b84bc4d69 | |||
a2fcf039be | |||
b4d887a372 | |||
0aaa7122da | |||
1bc7f85874 | |||
1d2fd000aa | |||
fc32f9eca9 | |||
ab35e8c034
|
|||
2aded2974f | |||
f84c1632b2
|
|||
|
03717a1a87 | ||
|
02c524dd79 | ||
506df432b0 | |||
c32c18b0e2 | |||
321d569ee9 | |||
cd40eb3932 | |||
f0f2531fa3 | |||
183a220e7b | |||
9df127a82c | |||
04a1412562 | |||
3aef0a185e | |||
578bce31b9 | |||
99e4824137 | |||
dacaaea235 | |||
096466e79e
|
|||
7285e5c2b0 | |||
37227a3aeb | |||
7569667189
|
|||
b0993f4062 | |||
7c79b65f48 | |||
b8f25bcd45 | |||
f4efb0e975 | |||
c641baec78 | |||
cc150e32f0 | |||
|
bc7f0907ab | ||
26cf6459b4 | |||
d0fa6dd512 | |||
|
fbdc0d32f0 | ||
|
5f31473c90 | ||
|
98cf167040 | ||
|
6c37d798bf | ||
03748a7e84 | |||
9e3431f397 | |||
912861dbff | |||
35f25daf7c | |||
21274155b5 | |||
3f7c136d6b
|
|||
5d9c573853 | |||
9a5fd67842 | |||
2755bc12c4 | |||
9e191f1b5b | |||
ab684a20ad
|
|||
bc92b52498 | |||
be5655e537
|
|||
ceb0bd982e
|
|||
47c0af3623
|
|||
f6f2efee2c
|
|||
59fd9fc63f | |||
ec2c08681e | |||
fedcbb9a70
|
|||
3f1a4fe353 | |||
fc27c73dab | |||
20bfd5b717
|
|||
5e3a1eb2ab
|
|||
b02820407c
|
|||
594ed4a5b4
|
|||
88fbf7bc1c
|
|||
aa26e67f6f
|
|||
21ac3eaab4 | |||
2ff500b00e | |||
19fa308c06 | |||
1b831bc424 | |||
573118e514 | |||
0cabf5654a | |||
cfb547d55f | |||
a915c35026 | |||
018f8aef5c | |||
de6385ba21 | |||
edb51dd3cd | |||
c379bccad4 | |||
bd9ad8a569 | |||
0cdd9184a3 | |||
cb5fd2b69d | |||
feebcf6662
|
|||
2a61197999 | |||
0a53a9a9d1 | |||
eea1e40663 | |||
1b0771eb07 | |||
3a74c48104 | |||
6de4861b98
|
|||
b4a1e824ac
|
|||
d87cf925e2
|
|||
ce3cce7b95 | |||
6c97654e5e
|
|||
0787e61c22
|
|||
73bcfb82b7
|
|||
812e4047d0
|
|||
999ac3af2b
|
|||
0c628c39db
|
|||
c65f1d495d
|
|||
83f7086bb0
|
|||
c1e449f48e
|
|||
1f6de3cb11
|
|||
3a2548ed89
|
|||
d7652658f2
|
|||
67b5bc6dba
|
|||
e25c1e1816
|
|||
282b7f7fbb | |||
ab311eaecb
|
|||
b37d7fb907 | |||
57b8dacba0 | |||
edcc01149b | |||
27b2d77fdb | |||
96bb98f854
|
|||
68ed2db51e
|
|||
184bb095d8 | |||
78c8e94765 | |||
c8d1a91953 | |||
3e8e2b0fa8 | |||
|
2c9c700ca7 | ||
|
110db30748 | ||
|
1f96f76f87 | ||
|
65902ea231 | ||
|
88c0b1570d | ||
4933d2251c
|
|||
c939ff4a4e
|
|||
82fb98348b
|
|||
d56d00421a | |||
de11fa86c6 | |||
4cc254a3e5 | |||
22ce16ef49 | |||
1e52a93cbe | |||
16fe07cce7 | |||
e8962782ed | |||
64a853fb6d | |||
ebfd48e41f | |||
a6aa2a81c2 | |||
98d29c2134 | |||
60386ae9ac | |||
559901e528 | |||
05bc69fd33 | |||
49da62d364 | |||
3af7824d01
|
|||
033053c437
|
|||
633bb00154
|
|||
f5c1b5cf8a | |||
ccd71da4e4
|
|||
1eadb3bbdb
|
|||
|
0bb5a79cae
|
||
|
bd3198e42b
|
||
|
96dfddc55f
|
||
|
da37a3db5f
|
||
|
c2882b1079
|
||
b9e515f4e6 | |||
|
df2ea7e1ba
|
||
|
d59cda9cc4
|
||
54e10cacd3 | |||
3ca126804b | |||
40d733c290 | |||
7a98bb5a06
|
|||
2ce8f540fe
|
|||
6c4d8990cc | |||
351e9c3fcc
|
|||
1b65cac1df
|
|||
aa0785fc71 | |||
610227815a | |||
9d9f062417 | |||
0454e5d758 | |||
2d6d2a1f58 | |||
c971e34675 | |||
be2a119163 | |||
22ecb11227 | |||
dd854ea339 | |||
f05c25853c | |||
90798b12e5 | |||
f91f5ce27e | |||
96f73b419d | |||
273f91fd00 | |||
27ecae4486 | |||
f90f1c7ef8 |
6
.changes/unreleased/DX-20250430-144550.yaml
Normal file
6
.changes/unreleased/DX-20250430-144550.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
kind: DX
|
||||||
|
body: Remove dead code for wopi-link module
|
||||||
|
time: 2025-04-30T14:45:50.406111606+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "352"
|
||||||
|
SchemaChange: No schema change
|
7
.changes/unreleased/Feature-20250424-142211.yaml
Normal file
7
.changes/unreleased/Feature-20250424-142211.yaml
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
kind: Feature
|
||||||
|
body: Add the document file name to the document title when a user upload a document,
|
||||||
|
unless there is already a document title.
|
||||||
|
time: 2025-04-24T14:22:11.800975422+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "377"
|
||||||
|
SchemaChange: No schema change
|
6
.changes/unreleased/Feature-20250520-095628.yaml
Normal file
6
.changes/unreleased/Feature-20250520-095628.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
kind: Feature
|
||||||
|
body: Add desactivation date for social action and issue csv export
|
||||||
|
time: 2025-05-20T09:56:28.108941934+02:00
|
||||||
|
custom:
|
||||||
|
Issue: ""
|
||||||
|
SchemaChange: No schema change
|
7
.changes/unreleased/Fixed-20250424-133943.yaml
Normal file
7
.changes/unreleased/Fixed-20250424-133943.yaml
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
kind: Fixed
|
||||||
|
body: trying to prevent bug of typeerror in doc-history + improved display of document
|
||||||
|
history
|
||||||
|
time: 2025-04-24T13:39:43.878468232+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "376"
|
||||||
|
SchemaChange: No schema change
|
7
.changes/unreleased/Fixed-20250424-163746.yaml
Normal file
7
.changes/unreleased/Fixed-20250424-163746.yaml
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
kind: Fixed
|
||||||
|
body: Display previous participation in acc course work even if the person has left
|
||||||
|
the acc course
|
||||||
|
time: 2025-04-24T16:37:46.970203594+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "381"
|
||||||
|
SchemaChange: No schema change
|
6
.changes/unreleased/Fixed-20250505-102715.yaml
Normal file
6
.changes/unreleased/Fixed-20250505-102715.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
kind: Fixed
|
||||||
|
body: Fix display of text in calendar events
|
||||||
|
time: 2025-05-05T10:27:15.461493066+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "372"
|
||||||
|
SchemaChange: No schema change
|
6
.changes/unreleased/Fixed-20250514-145339.yaml
Normal file
6
.changes/unreleased/Fixed-20250514-145339.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
kind: Fixed
|
||||||
|
body: Add missing translation for user_group.no_user_groups
|
||||||
|
time: 2025-05-14T14:53:39.53927329+02:00
|
||||||
|
custom:
|
||||||
|
Issue: ""
|
||||||
|
SchemaChange: No schema change
|
6
.changes/unreleased/UX-20250423-172624.yaml
Normal file
6
.changes/unreleased/UX-20250423-172624.yaml
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
kind: UX
|
||||||
|
body: Remove default filter in_progress for the page 'my tasks'; Allows for new tasks to be displayed upon opening of the page
|
||||||
|
time: 2025-04-23T17:26:24.45777387+02:00
|
||||||
|
custom:
|
||||||
|
Issue: "374"
|
||||||
|
SchemaChange: No schema change
|
6
.changes/v3.10.0.md
Normal file
6
.changes/v3.10.0.md
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
## v3.10.0 - 2025-03-17
|
||||||
|
### Feature
|
||||||
|
* ([#363](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/363)) Display social actions grouped per social issue within activity form
|
||||||
|
### Fixed
|
||||||
|
* ([#362](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/362)) Fix Dependency Injection, which prevented to save the CalendarRange
|
||||||
|
* ([#368](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/368)) fix search query for user groups
|
3
.changes/v3.10.1.md
Normal file
3
.changes/v3.10.1.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.10.1 - 2025-03-17
|
||||||
|
### DX
|
||||||
|
* Remove yarn dependency to symfony/ux-translator, to ease the build process
|
3
.changes/v3.10.2.md
Normal file
3
.changes/v3.10.2.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.10.2 - 2025-03-17
|
||||||
|
### Fixed
|
||||||
|
* Replace a ts-expect-error with a ts-ignore
|
3
.changes/v3.10.3.md
Normal file
3
.changes/v3.10.3.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.10.3 - 2025-03-18
|
||||||
|
### DX
|
||||||
|
* Eslint fixes
|
19
.changes/v3.11.0.md
Normal file
19
.changes/v3.11.0.md
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
## v3.11.0 - 2025-04-17
|
||||||
|
### Feature
|
||||||
|
* ([#365](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/365)) Add counters of actions and activities, with 2 boxes to (1) show the number of active actions on total actions and (2) show the number of activities in a accompanying period, and pills in menus for showing the number of active actions and the number of activities.
|
||||||
|
* ([#364](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/364)) Added a second phone number "telephone2" to the thirdParty entity. Adapted twig templates and vuejs apps to handle this phone number
|
||||||
|
|
||||||
|
**Schema Change**: Add columns or tables
|
||||||
|
* Signature: add a button to go directly to the signature zone, even if there is only one
|
||||||
|
### Fixed
|
||||||
|
* Fixed wrong translations in the on-the-fly for creation of thirdParty
|
||||||
|
* Fixed update of phone number in on-the-fly edition of thirdParty
|
||||||
|
* Fixed closing of modal when editing thirdParty in accompanying course works
|
||||||
|
* Shorten the delay between two execution of AccompanyingPeriodStepChangeCronjob, to ensure at least one execution in a day
|
||||||
|
* ([#102](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/102)) Fix display of title in document list
|
||||||
|
* When cleaning the old stored object versions, do not throw an error if the stored object is not found on disk
|
||||||
|
* Add consistent log prefix and key to logs when stale workflows are automatically canceled
|
||||||
|
* ([#380](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/380)) Remove the "not null" validation constraint on recently added properties on HouseholdComposition
|
||||||
|
|
||||||
|
### DX
|
||||||
|
* Add new chill-col style for displaying title and aside in a flex table
|
4
.changes/v3.4.3.md
Normal file
4
.changes/v3.4.3.md
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
## v3.4.3 - 2024-12-05
|
||||||
|
### Fixed
|
||||||
|
* Remove the "not null" constraint on person supplementary phones
|
||||||
|
* Remove doctrine annotation that prevent from adding documents to activities
|
6
.changes/v3.5.0.md
Normal file
6
.changes/v3.5.0.md
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
## v3.5.0 - 2024-12-09
|
||||||
|
### Feature
|
||||||
|
* ([#318](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/318)) Show all the pages of the documents in the signature app
|
||||||
|
### Fixed
|
||||||
|
* Wrap the signature's change state into a transaction, to avoid race conditions
|
||||||
|
* Fix display of gender label
|
4
.changes/v3.5.1.md
Normal file
4
.changes/v3.5.1.md
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
## v3.5.1 - 2024-12-16
|
||||||
|
### Fixed
|
||||||
|
* Filiation: fix the display of the gender label in the graph
|
||||||
|
* Wrap handling of PdfSignedMessage into transactions
|
3
.changes/v3.5.2.md
Normal file
3
.changes/v3.5.2.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.5.2 - 2024-12-19
|
||||||
|
### Fixed
|
||||||
|
* ([#345](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/345)) Export: activity filtering of users that were associated to an activity between certain dates. Results contained activities that were not within the specified date range"
|
3
.changes/v3.5.3.md
Normal file
3
.changes/v3.5.3.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.5.3 - 2025-01-07
|
||||||
|
### Fixed
|
||||||
|
* Fix the EntityToJsonTransformer to return an empty array if the value is ""
|
9
.changes/v3.6.0.md
Normal file
9
.changes/v3.6.0.md
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
## v3.6.0 - 2025-01-16
|
||||||
|
### Feature
|
||||||
|
* Importer for addresses does not fails when the postal code is not found with some addresses, and compute a recap list of all addresses that could not be imported. This recap list can be send by email.
|
||||||
|
* ([#346](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/346)) Create a driver for storing documents on disk (instead of openstack object store)
|
||||||
|
|
||||||
|
* Add address importer from french Base d'Adresse Nationale (BAN)
|
||||||
|
* ([#343](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/343)) Add csv export for social issues and social actions
|
||||||
|
### Fixed
|
||||||
|
* Export: fix missing alias in activity between certain dates filter. Condition added for alias.
|
62
.changes/v3.7.0.md
Normal file
62
.changes/v3.7.0.md
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
## v3.7.0 - 2025-01-21
|
||||||
|
### Feature
|
||||||
|
* Use the Notifier component from Symfony to sens short messages (SMS). This allow to use more provider.
|
||||||
|
### Fixed
|
||||||
|
* ([#348](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/348)) [export] Fix aggregation of referrer's scope and job: fix the date range comparison
|
||||||
|
|
||||||
|
### Warning on configuration of Notifier component
|
||||||
|
|
||||||
|
If installed in an symfony app where the recipes are activated, this configuration should be added automatically:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
chatter_transports:
|
||||||
|
texter_transports:
|
||||||
|
ovhcloud: '%env(OVHCLOUD_DSN)%'
|
||||||
|
channel_policy:
|
||||||
|
# use chat/slack, chat/telegram, sms/twilio or sms/nexmo
|
||||||
|
urgent: ['email']
|
||||||
|
high: ['email']
|
||||||
|
medium: ['email']
|
||||||
|
low: ['email']
|
||||||
|
admin_recipients:
|
||||||
|
- { email: admin@example.com }
|
||||||
|
```
|
||||||
|
|
||||||
|
Actually, you should either:
|
||||||
|
|
||||||
|
- remove the configuration of ovhcloud added by the recipe
|
||||||
|
- or remove the previous configuration of chill, to avoid keeping legacy configuration
|
||||||
|
|
||||||
|
#### Remove the added configuration and keep the legacy configuration
|
||||||
|
|
||||||
|
To remove the configuration:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
chatter_transports:
|
||||||
|
texter_transports:
|
||||||
|
- ovhcloud: '%env(OVHCLOUD_DSN)%'
|
||||||
|
```
|
||||||
|
|
||||||
|
In that case, the previous configuration, which was stored under the `chill_main.short_messages.dsn` will be reconfigured into the Notifier component's configuration.
|
||||||
|
|
||||||
|
#### Properly configure SMS
|
||||||
|
|
||||||
|
You can also properly configure it, as [described in the OVH cloud provider repository](https://github.com/symfony/ovh-cloud-notifier/tree/5.4?tab=readme-ov-file#dsn-example) (where the scheme is `ovhcloud`):
|
||||||
|
|
||||||
|
**NOTE**: You have access to all notifier available with the [Notifier component](https://symfony.com/doc/current/notifier.html#notifier-sms-channel). You are not restricted to use OVH as a provider.
|
||||||
|
|
||||||
|
```diff
|
||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
chatter_transports:
|
||||||
|
texter_transports:
|
||||||
|
+ ovhcloud: '%env(OVHCLOUD_DSN)%' # this value should be located in a variable, and have `ovhcloud://` as a scheme
|
||||||
|
|
||||||
|
chill_main:
|
||||||
|
- short_messages:
|
||||||
|
- dsn: '%env(string:SHORT_MESSAGE_DSN)%'
|
||||||
|
```
|
3
.changes/v3.7.1.md
Normal file
3
.changes/v3.7.1.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.7.1 - 2025-01-21
|
||||||
|
### Fixed
|
||||||
|
* Fix legacy configuration processor for notifier component
|
11
.changes/v3.8.0.md
Normal file
11
.changes/v3.8.0.md
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
## v3.8.0 - 2025-02-03
|
||||||
|
### Feature
|
||||||
|
* Improve the UX of the news item admin form to prevent wrong usage
|
||||||
|
* ([#319](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/319)) Notification list: display the concerned person's badges in the list
|
||||||
|
* ([#320](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/320)) Show the first 3 persons directly in the accompanying period's banner
|
||||||
|
* ([#334](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/334)) Suggest current user when creating an activity
|
||||||
|
* ([#331](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/331)) Add attachments to workflows
|
||||||
|
### Fixed
|
||||||
|
* ([#350](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/350)) Add validation error to manual selection of person in PersonDuplicateController
|
||||||
|
* ([#354](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/354)) Fix document category creation
|
||||||
|
* ([#351](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/351)) Add definitive whitespace between span elements in vue PersonText component
|
3
.changes/v3.8.1.md
Normal file
3
.changes/v3.8.1.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.8.1 - 2025-02-05
|
||||||
|
### Fixed
|
||||||
|
* Fix household link in the parcours banner
|
3
.changes/v3.8.2.md
Normal file
3
.changes/v3.8.2.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.8.2 - 2025-02-10
|
||||||
|
### Fixed
|
||||||
|
* ([#358](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/358)) Remove "filter" button on list of documents in the workflow's "add attachement" modal
|
10
.changes/v3.9.0.md
Normal file
10
.changes/v3.9.0.md
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
## v3.9.0 - 2025-02-27
|
||||||
|
### Feature
|
||||||
|
* ([#349](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/349)) Suggest all referrers within actions of the accompanying period when creating an activity
|
||||||
|
* ([#343](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/343)) Add possibility to export a csv with all social issues and social actions
|
||||||
|
* ([#360](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/360)) Restore document to previous kept version when a workflow is canceled
|
||||||
|
* ([#341](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/341)) Add a list of third parties from within the admin (csv download)
|
||||||
|
### Fixed
|
||||||
|
* fix generation of document with accompanying period context, and list of activities and works
|
||||||
|
### DX
|
||||||
|
* ([#333](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/333)) Create an unique source of trust for translations
|
3
.changes/v3.9.1.md
Normal file
3
.changes/v3.9.1.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.9.1 - 2025-02-27
|
||||||
|
### Fixed
|
||||||
|
* Fix post/patch request with missing 'type' property for gender
|
3
.changes/v3.9.2.md
Normal file
3
.changes/v3.9.2.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
## v3.9.2 - 2025-02-27
|
||||||
|
### Fixed
|
||||||
|
* Use fetchResults method to fetch all social issues instead of only the first page
|
@@ -7,15 +7,29 @@ versionFormat: '## {{.Version}} - {{.Time.Format "2006-01-02"}}'
|
|||||||
kindFormat: '### {{.Kind}}'
|
kindFormat: '### {{.Kind}}'
|
||||||
# Note: it is possible to add a `.custom.Long` text manually into the yaml file produced by `changie new`. This will add a long description.
|
# Note: it is possible to add a `.custom.Long` text manually into the yaml file produced by `changie new`. This will add a long description.
|
||||||
changeFormat: >-
|
changeFormat: >-
|
||||||
* {{ if not (eq .Custom.Issue "") }}([#{{ .Custom.Issue }}](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/{{ .Custom.Issue }})) {{ end }}{{.Body}} {{ if and (.Custom.Long) (not (eq .Custom.Long "")) }}
|
* {{ if not (eq .Custom.Issue "") }}([#{{ .Custom.Issue }}](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/{{ .Custom.Issue }})) {{ end }}{{ .Body }} {{ if and .Custom.SchemaChange (ne .Custom.SchemaChange "No schema change") }}
|
||||||
|
|
||||||
|
**Schema Change**: {{ .Custom.SchemaChange }}
|
||||||
|
{{- end -}}
|
||||||
|
|
||||||
|
{{ if and (.Custom.Long) (not (eq .Custom.Long "")) }}{{ .Custom.Long }}{{ end }}
|
||||||
|
|
||||||
{{ .Custom.Long }}{{ end }}
|
|
||||||
custom:
|
custom:
|
||||||
|
- key: SchemaChange
|
||||||
|
label: Is a schema change required?
|
||||||
|
optional: false
|
||||||
|
type: enum
|
||||||
|
enumOptions:
|
||||||
|
- "No schema change"
|
||||||
|
- "Add columns or tables"
|
||||||
|
- "Drop or rename table or columns, or enforce new constraint that must be manually fixed"
|
||||||
|
|
||||||
- key: Issue
|
- key: Issue
|
||||||
label: Issue number (on chill-bundles repository) (optional)
|
label: Issue number (on chill-bundles repository) (optional)
|
||||||
optional: true
|
optional: true
|
||||||
type: int
|
type: int
|
||||||
minInt: 1
|
minInt: 1
|
||||||
|
|
||||||
body:
|
body:
|
||||||
# allow multiline messages
|
# allow multiline messages
|
||||||
block: true
|
block: true
|
||||||
|
4
.env
4
.env
@@ -88,3 +88,7 @@ REDIS_HOST=redis
|
|||||||
REDIS_PORT=6379
|
REDIS_PORT=6379
|
||||||
REDIS_URL=redis://${REDIS_HOST}:${REDIS_PORT}
|
REDIS_URL=redis://${REDIS_HOST}:${REDIS_PORT}
|
||||||
###< chill-project/chill-bundles ###
|
###< chill-project/chill-bundles ###
|
||||||
|
|
||||||
|
###> symfony/ovh-cloud-notifier ###
|
||||||
|
# OVHCLOUD_DSN=ovhcloud://APPLICATION_KEY:APPLICATION_SECRET@default?consumer_key=CONSUMER_KEY&service_name=SERVICE_NAME
|
||||||
|
###< symfony/ovh-cloud-notifier ###
|
||||||
|
2386
.eslint-baseline.json
Normal file
2386
.eslint-baseline.json
Normal file
File diff suppressed because it is too large
Load Diff
8
.gitignore
vendored
8
.gitignore
vendored
@@ -5,12 +5,15 @@ composer.lock
|
|||||||
docs/build/
|
docs/build/
|
||||||
.php_cs.cache
|
.php_cs.cache
|
||||||
.cache/*
|
.cache/*
|
||||||
|
yarn.lock
|
||||||
|
|
||||||
docker/db/data
|
docker/db/data
|
||||||
docker/rabbitmq/data
|
docker/rabbitmq/data
|
||||||
|
|
||||||
# in this development bundle, we want to ignore directories related to a real app
|
# in this development bundle, we want to ignore directories related to a real app
|
||||||
assets/*
|
assets/*
|
||||||
|
!assets/translator.ts
|
||||||
|
!assets/ux-translator
|
||||||
migrations/*
|
migrations/*
|
||||||
templates/*
|
templates/*
|
||||||
translations/*
|
translations/*
|
||||||
@@ -51,3 +54,8 @@ phpstan.neon
|
|||||||
npm-debug.log
|
npm-debug.log
|
||||||
yarn-error.log
|
yarn-error.log
|
||||||
###< symfony/webpack-encore-bundle ###
|
###< symfony/webpack-encore-bundle ###
|
||||||
|
|
||||||
|
###> friendsofphp/php-cs-fixer ###
|
||||||
|
/.php-cs-fixer.php
|
||||||
|
/.php-cs-fixer.cache
|
||||||
|
###< friendsofphp/php-cs-fixer ###
|
||||||
|
@@ -5,6 +5,7 @@ cache:
|
|||||||
paths:
|
paths:
|
||||||
- /vendor/
|
- /vendor/
|
||||||
- .cache
|
- .cache
|
||||||
|
- node_modules/
|
||||||
|
|
||||||
# Bring in any services we need http://docs.gitlab.com/ee/ci/docker/using_docker_images.html#what-is-a-service
|
# Bring in any services we need http://docs.gitlab.com/ee/ci/docker/using_docker_images.html#what-is-a-service
|
||||||
# See http://docs.gitlab.com/ee/ci/services/README.html for examples.
|
# See http://docs.gitlab.com/ee/ci/services/README.html for examples.
|
||||||
@@ -103,6 +104,32 @@ rector_tests:
|
|||||||
paths:
|
paths:
|
||||||
- vendor/
|
- vendor/
|
||||||
|
|
||||||
|
lint:
|
||||||
|
stage: Tests
|
||||||
|
image: node:20-alpine
|
||||||
|
before_script:
|
||||||
|
- apk add --no-cache python3 make g++ py3-setuptools
|
||||||
|
- export PYTHON="$(which python3)"
|
||||||
|
- export PATH="./node_modules/.bin:$PATH"
|
||||||
|
script:
|
||||||
|
- yarn install --ignore-optional
|
||||||
|
- npx eslint-baseline "src/**/*.{js,ts,vue}"
|
||||||
|
cache:
|
||||||
|
paths:
|
||||||
|
- node_modules/
|
||||||
|
|
||||||
|
# psalm_tests:
|
||||||
|
# stage: Tests
|
||||||
|
# image: gitea.champs-libres.be/chill-project/chill-skeleton-basic/base-image:php82
|
||||||
|
# script:
|
||||||
|
# - bin/psalm
|
||||||
|
# allow_failure: true
|
||||||
|
# artifacts:
|
||||||
|
# expire_in: 30 min
|
||||||
|
# paths:
|
||||||
|
# - bin
|
||||||
|
# - tests/app/vendor/
|
||||||
|
|
||||||
unit_tests:
|
unit_tests:
|
||||||
stage: Tests
|
stage: Tests
|
||||||
image: gitea.champs-libres.be/chill-project/chill-skeleton-basic/base-image:php82
|
image: gitea.champs-libres.be/chill-project/chill-skeleton-basic/base-image:php82
|
||||||
|
@@ -25,7 +25,7 @@ $config = new PhpCsFixer\Config();
|
|||||||
$config
|
$config
|
||||||
->setFinder($finder)
|
->setFinder($finder)
|
||||||
->setRiskyAllowed(true)
|
->setRiskyAllowed(true)
|
||||||
->setCacheFile('.cache/php-cs-fixer.cache')
|
->setCacheFile('var/php-cs-fixer.cache')
|
||||||
->setUsingCache(true)
|
->setUsingCache(true)
|
||||||
->setParallelConfig(PhpCsFixer\Runner\Parallel\ParallelConfigFactory::detect())
|
->setParallelConfig(PhpCsFixer\Runner\Parallel\ParallelConfigFactory::detect())
|
||||||
;
|
;
|
||||||
|
160
CHANGELOG.md
160
CHANGELOG.md
@@ -6,6 +6,166 @@ 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).
|
||||||
|
|
||||||
|
|
||||||
|
## v3.10.3 - 2025-03-18
|
||||||
|
### DX
|
||||||
|
* Eslint fixes
|
||||||
|
|
||||||
|
## v3.10.2 - 2025-03-17
|
||||||
|
### Fixed
|
||||||
|
* Replace a ts-expect-error with a ts-ignore
|
||||||
|
|
||||||
|
## v3.10.1 - 2025-03-17
|
||||||
|
### DX
|
||||||
|
* Remove yarn dependency to symfony/ux-translator, to ease the build process
|
||||||
|
|
||||||
|
## v3.10.0 - 2025-03-17
|
||||||
|
### Feature
|
||||||
|
* ([#363](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/363)) Display social actions grouped per social issue within activity form
|
||||||
|
### Fixed
|
||||||
|
* ([#362](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/362)) Fix Dependency Injection, which prevented to save the CalendarRange
|
||||||
|
* ([#368](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/368)) fix search query for user groups
|
||||||
|
|
||||||
|
## v3.9.2 - 2025-02-27
|
||||||
|
### Fixed
|
||||||
|
* Use fetchResults method to fetch all social issues instead of only the first page
|
||||||
|
|
||||||
|
## v3.9.1 - 2025-02-27
|
||||||
|
### Fixed
|
||||||
|
* Fix post/patch request with missing 'type' property for gender
|
||||||
|
|
||||||
|
## v3.9.0 - 2025-02-27
|
||||||
|
### Feature
|
||||||
|
* ([#349](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/349)) Suggest all referrers within actions of the accompanying period when creating an activity
|
||||||
|
* ([#343](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/343)) Add possibility to export a csv with all social issues and social actions
|
||||||
|
* ([#360](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/360)) Restore document to previous kept version when a workflow is canceled
|
||||||
|
* ([#341](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/341)) Add a list of third parties from within the admin (csv download)
|
||||||
|
### Fixed
|
||||||
|
* fix generation of document with accompanying period context, and list of activities and works
|
||||||
|
### DX
|
||||||
|
* ([#333](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/333)) Create an unique source of trust for translations
|
||||||
|
|
||||||
|
## v3.8.2 - 2025-02-10
|
||||||
|
### Fixed
|
||||||
|
* ([#358](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/358)) Remove "filter" button on list of documents in the workflow's "add attachement" modal
|
||||||
|
|
||||||
|
## v3.8.1 - 2025-02-05
|
||||||
|
### Fixed
|
||||||
|
* Fix household link in the parcours banner
|
||||||
|
|
||||||
|
## v3.8.0 - 2025-02-03
|
||||||
|
### Feature
|
||||||
|
* Improve the UX of the news item admin form to prevent wrong usage
|
||||||
|
* ([#319](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/319)) Notification list: display the concerned person's badges in the list
|
||||||
|
* ([#320](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/320)) Show the first 3 persons directly in the accompanying period's banner
|
||||||
|
* ([#334](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/334)) Suggest current user when creating an activity
|
||||||
|
* ([#331](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/331)) Add attachments to workflows
|
||||||
|
### Fixed
|
||||||
|
* ([#350](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/350)) Add validation error to manual selection of person in PersonDuplicateController
|
||||||
|
* ([#354](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/354)) Fix document category creation
|
||||||
|
* ([#351](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/351)) Add definitive whitespace between span elements in vue PersonText component
|
||||||
|
|
||||||
|
## v3.7.1 - 2025-01-21
|
||||||
|
### Fixed
|
||||||
|
* Fix legacy configuration processor for notifier component
|
||||||
|
|
||||||
|
## v3.7.0 - 2025-01-21
|
||||||
|
### Feature
|
||||||
|
* Use the Notifier component from Symfony to sens short messages (SMS). This allow to use more provider.
|
||||||
|
### Fixed
|
||||||
|
* ([#348](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/348)) [export] Fix aggregation of referrer's scope and job: fix the date range comparison
|
||||||
|
|
||||||
|
### Warning on configuration of Notifier component
|
||||||
|
|
||||||
|
If installed in an symfony app where the recipes are activated, this configuration should be added automatically:
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
chatter_transports:
|
||||||
|
texter_transports:
|
||||||
|
ovhcloud: '%env(OVHCLOUD_DSN)%'
|
||||||
|
channel_policy:
|
||||||
|
# use chat/slack, chat/telegram, sms/twilio or sms/nexmo
|
||||||
|
urgent: ['email']
|
||||||
|
high: ['email']
|
||||||
|
medium: ['email']
|
||||||
|
low: ['email']
|
||||||
|
admin_recipients:
|
||||||
|
- { email: admin@example.com }
|
||||||
|
```
|
||||||
|
|
||||||
|
Actually, you should either:
|
||||||
|
|
||||||
|
- remove the configuration of ovhcloud added by the recipe
|
||||||
|
- or remove the previous configuration of chill, to avoid keeping legacy configuration
|
||||||
|
|
||||||
|
#### Remove the added configuration and keep the legacy configuration
|
||||||
|
|
||||||
|
To remove the configuration:
|
||||||
|
|
||||||
|
```diff
|
||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
chatter_transports:
|
||||||
|
texter_transports:
|
||||||
|
- ovhcloud: '%env(OVHCLOUD_DSN)%'
|
||||||
|
```
|
||||||
|
|
||||||
|
In that case, the previous configuration, which was stored under the `chill_main.short_messages.dsn` will be reconfigured into the Notifier component's configuration.
|
||||||
|
|
||||||
|
#### Properly configure SMS
|
||||||
|
|
||||||
|
You can also properly configure it, as [described in the OVH cloud provider repository](https://github.com/symfony/ovh-cloud-notifier/tree/5.4?tab=readme-ov-file#dsn-example) (where the scheme is `ovhcloud`):
|
||||||
|
|
||||||
|
**NOTE**: You have access to all notifier available with the [Notifier component](https://symfony.com/doc/current/notifier.html#notifier-sms-channel). You are not restricted to use OVH as a provider.
|
||||||
|
|
||||||
|
```diff
|
||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
chatter_transports:
|
||||||
|
texter_transports:
|
||||||
|
+ ovhcloud: '%env(OVHCLOUD_DSN)%' # this value should be located in a variable, and have `ovhcloud://` as a scheme
|
||||||
|
|
||||||
|
chill_main:
|
||||||
|
- short_messages:
|
||||||
|
- dsn: '%env(string:SHORT_MESSAGE_DSN)%'
|
||||||
|
```
|
||||||
|
|
||||||
|
## v3.6.0 - 2025-01-16
|
||||||
|
### Feature
|
||||||
|
* Importer for addresses does not fails when the postal code is not found with some addresses, and compute a recap list of all addresses that could not be imported. This recap list can be send by email.
|
||||||
|
* ([#346](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/346)) Create a driver for storing documents on disk (instead of openstack object store)
|
||||||
|
|
||||||
|
* Add address importer from french Base d'Adresse Nationale (BAN)
|
||||||
|
* ([#343](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/343)) Add csv export for social issues and social actions
|
||||||
|
### Fixed
|
||||||
|
* Export: fix missing alias in activity between certain dates filter. Condition added for alias.
|
||||||
|
|
||||||
|
## v3.5.3 - 2025-01-07
|
||||||
|
### Fixed
|
||||||
|
* Fix the EntityToJsonTransformer to return an empty array if the value is ""
|
||||||
|
|
||||||
|
## v3.5.2 - 2024-12-19
|
||||||
|
### Fixed
|
||||||
|
* ([#345](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/345)) Export: activity filtering of users that were associated to an activity between certain dates. Results contained activities that were not within the specified date range"
|
||||||
|
|
||||||
|
## v3.5.1 - 2024-12-16
|
||||||
|
### Fixed
|
||||||
|
* Filiation: fix the display of the gender label in the graph
|
||||||
|
* Wrap handling of PdfSignedMessage into transactions
|
||||||
|
|
||||||
|
## v3.5.0 - 2024-12-09
|
||||||
|
### Feature
|
||||||
|
* ([#318](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/318)) Show all the pages of the documents in the signature app
|
||||||
|
### Fixed
|
||||||
|
* Wrap the signature's change state into a transaction, to avoid race conditions
|
||||||
|
* Fix display of gender label
|
||||||
|
|
||||||
|
## v3.4.3 - 2024-12-05
|
||||||
|
### Fixed
|
||||||
|
* Remove the "not null" constraint on person supplementary phones
|
||||||
|
* Remove doctrine annotation that prevent from adding documents to activities
|
||||||
|
|
||||||
## v3.4.2 - 2024-12-05
|
## v3.4.2 - 2024-12-05
|
||||||
### Fixed
|
### Fixed
|
||||||
* ([#329](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/329)) Fix the serialization of gender for the generation of documents
|
* ([#329](https://gitlab.com/Chill-Projet/chill-bundles/-/issues/329)) Fix the serialization of gender for the generation of documents
|
||||||
|
386
CONVENTIONS-en.md
Normal file
386
CONVENTIONS-en.md
Normal file
@@ -0,0 +1,386 @@
|
|||||||
|
# Chill Conventions
|
||||||
|
|
||||||
|
In Progress
|
||||||
|
|
||||||
|
## Translations
|
||||||
|
|
||||||
|
Per bundle, all translations for Twig pages are located in a single file: translations/messages.fr.yaml.
|
||||||
|
|
||||||
|
## File Locations
|
||||||
|
|
||||||
|
Controllers, form types, and Twig templates are placed in the root folders Controller, Form, and Resources/views, respectively.
|
||||||
|
Admin pages are no longer placed in subfolders under Admin.
|
||||||
|
|
||||||
|
## Assets: Entrypoint Naming
|
||||||
|
|
||||||
|
There are three types of entry points:
|
||||||
|
|
||||||
|
* Vue application-specific (often for a single page): Prefixed with vue_.
|
||||||
|
* Reusable JavaScript/CSS code:
|
||||||
|
Examples include:
|
||||||
|
* ckeditor
|
||||||
|
* async_upload (used for forms)
|
||||||
|
* bootstrap
|
||||||
|
* chill.js
|
||||||
|
* ...
|
||||||
|
=> We prefix with `mod_`
|
||||||
|
|
||||||
|
* Page-specific CSS/JS:
|
||||||
|
* Often reuses functionalities like ShowHide.
|
||||||
|
=> We prefix with `page_`.
|
||||||
|
|
||||||
|
## Folder Structure
|
||||||
|
|
||||||
|
````
|
||||||
|
# Under Resources/public:
|
||||||
|
|
||||||
|
- chill/ => Contains the theme (Chill).
|
||||||
|
- chillmain.scss: Compiled into the chill entrypoint.
|
||||||
|
- lib/ => Libraries never used as entrypoints but reused elsewhere.
|
||||||
|
- Examples: ShowHide, Collection, Select2.
|
||||||
|
- module/: Ends up in reusable entrypoints (mod_).
|
||||||
|
- bootstrap
|
||||||
|
- custom.scss
|
||||||
|
- custom/
|
||||||
|
- variable.scss
|
||||||
|
- ...
|
||||||
|
- AsyncUpload.
|
||||||
|
- vue/ => Vue applications only (vue_).
|
||||||
|
- Examples: _components, app.
|
||||||
|
- page/ => Page-specific assets (page_).
|
||||||
|
- login
|
||||||
|
- person
|
||||||
|
- personvendee
|
||||||
|
- household_edit_metadata
|
||||||
|
- index.js
|
||||||
|
````
|
||||||
|
|
||||||
|
# Stylesheet Organization
|
||||||
|
|
||||||
|
1. The mod_bootstrap (module bootstrap) entry point is the first level. All parts (modules) of bootstrap are included in the bootstrap.js file located in ChillMainBundle/Resources/public/module/bootstrap.
|
||||||
|
|
||||||
|
* At the beginning, this file imports the variables.scss file, which determines most of the bootstrap settings as customized. This file overrides the original, and many variables are adjusted for Chill.
|
||||||
|
* Care must be taken to ensure this file can always be compared to the original bootstrap file. In the event of a bootstrap update, a diff must be generated and this diff applied to the variable file of the new version.
|
||||||
|
* At the end, it imports the custom.scss file, which includes bootstrap adaptations to prepare it for our Chill theme.
|
||||||
|
* This `custom.scss` file can be split into smaller files using `@import 'custom/...'`.
|
||||||
|
* The idea is that this first bootstrap layer sets an important part of the application’s styles, particularly those related to layout positioning, responsive breakpoints, and the default margins and spacing applied to elements being manipulated.
|
||||||
|
|
||||||
|
2. The chill entry point is the second level. It contains the Chill theme, which is recognizable in the application.
|
||||||
|
|
||||||
|
* Each bundle has a `Resources/public/chill` folder containing a main sass file, which can optionally be split using `@imports`. All these files are compiled into a single Chill entry point, which serves as the application’s theme and overrides bootstrap.
|
||||||
|
* The chillmain.scss file should contain the most general style cascades, those applied to many areas of the application.
|
||||||
|
* The chillperson.scss file also includes styles specific to different contexts related to people: person, household, and accompanying course.
|
||||||
|
* Some smaller bundles contain only styles specific to their functionality.
|
||||||
|
|
||||||
|
3. The vue_ entry points are used for Vue components. Vue files can contain an scss style block. These styles are specific to the component and its inheritance, with the scoped tag precisely defining their scope (see the documentation).
|
||||||
|
|
||||||
|
4. The page_ entry points are used to add assets specific to certain pages, most often scripts and styles.
|
||||||
|
|
||||||
|
## HTML Tagging and Style Cascades
|
||||||
|
|
||||||
|
The following example shows how to tag a code element without overdoing it. Note that:
|
||||||
|
|
||||||
|
* It is not necessary to tag all inner classes.
|
||||||
|
* The parent class should not be repeated in all child classes. Sass cascading allows for flexible HTML structuring without overloading the tag hierarchy.
|
||||||
|
* Often, the first class will have variations created with additional classes that start in the same way: bloc-dark simply adds the dark version of bloc. We do not use bloc dark because we don’t want the dark class of bloc to interact with the same dark class of table. As a result, we will have an element bloc bloc-dark and another element table table-dark.
|
||||||
|
|
||||||
|
```html
|
||||||
|
<div class="bloc bloc-dark my-bloc">
|
||||||
|
<h3>My Title</h3>
|
||||||
|
<ul class="record_actions">
|
||||||
|
<li>
|
||||||
|
<a class="btn btn-edit"></a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
Finally, it is important to define what a block, an action zone, and a button are. These three elements exist independently and are the only ones we tag.
|
||||||
|
|
||||||
|
For example, to style the title, we simply specify h3 within the block cascade.
|
||||||
|
|
||||||
|
```scss
|
||||||
|
div.bloc {
|
||||||
|
// un bloc générique, utilisé à plusieurs endroits
|
||||||
|
&.bloc-dark {
|
||||||
|
// la version sombre du bloc
|
||||||
|
}
|
||||||
|
h3 {}
|
||||||
|
ul {
|
||||||
|
// une liste standard dans bloc
|
||||||
|
li {
|
||||||
|
// des items de liste standard dans bloc
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
div.mon-bloc {
|
||||||
|
// des exceptions spécifiques à mon-bloc,
|
||||||
|
// qui sont des adaptations de bloc
|
||||||
|
}
|
||||||
|
|
||||||
|
ul.record_actions {
|
||||||
|
// va uniformiser tous les record_actions de l'application
|
||||||
|
li {
|
||||||
|
//...
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn {
|
||||||
|
// les boutons de bootstrap
|
||||||
|
.btn-edit {
|
||||||
|
// chill étends les boutons bootstrap pour ses propres besoins
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
```
|
||||||
|
## Render box
|
||||||
|
|
||||||
|
## URL
|
||||||
|
|
||||||
|
## Route Naming Conventions
|
||||||
|
|
||||||
|
:::warning
|
||||||
|
These rules have not always been followed in the past. They are desired for the future.
|
||||||
|
:::
|
||||||
|
|
||||||
|
Routes follow this structure:
|
||||||
|
|
||||||
|
`chill_(api|crud)_bundle_(api)_entity_action`
|
||||||
|
|
||||||
|
1. First, chill_ (for all Chill modules).
|
||||||
|
2. Then, crud or api, optional, automatically added if the route is generated by the configuration.
|
||||||
|
3. Then, a string indicating the bundle (`main`, `person`, `activity`, ...).
|
||||||
|
4. Then, api, if the route is an API route.
|
||||||
|
5. Then, a string indicating the entity the route targets, and possibly the sub-entities.
|
||||||
|
6. Then, an action (`list`, `view`, `edit`, `new`, ...).
|
||||||
|
|
||||||
|
Indicating `api` in the fourth position allows distinguishing API routes generated by the configuration (which are all prefixed with `chill_api`) from those generated manually. (For example: `chill_api_household__index` and `chill_person_api_household_members_move`).
|
||||||
|
|
||||||
|
If points 4 and 5 are missing, they are replaced by other elements to ensure the uniqueness of the route and its clear understanding.
|
||||||
|
|
||||||
|
#### HTML pages
|
||||||
|
|
||||||
|
:::warning
|
||||||
|
These rules have not always been followed in the past. They are desired for the future.
|
||||||
|
:::
|
||||||
|
|
||||||
|
Syntaxe:
|
||||||
|
|
||||||
|
```
|
||||||
|
/{_locale}/bundle/entity/{id}/action
|
||||||
|
/{_locale}/bundle/entity/sub-entity/{id}/action
|
||||||
|
```
|
||||||
|
|
||||||
|
The following elements should be included in the list:
|
||||||
|
|
||||||
|
1. The locale;
|
||||||
|
2. An identifier for the bundle;
|
||||||
|
3. The entity it relates to;
|
||||||
|
4. Any sub-entities that the URL refers to;
|
||||||
|
5. The action.
|
||||||
|
|
||||||
|
```
|
||||||
|
# list of activities for a person
|
||||||
|
/fr/activity/person/25/activity/list
|
||||||
|
|
||||||
|
# new activity
|
||||||
|
/fr/activity/activity/new?person_id=25
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
#### Pour les API
|
||||||
|
|
||||||
|
:::info
|
||||||
|
Automatically generated routes are prefixed with `chill_api`
|
||||||
|
:::
|
||||||
|
|
||||||
|
Syntaxe:
|
||||||
|
|
||||||
|
```
|
||||||
|
/api/1.0/bundle/entity/{id}/action
|
||||||
|
/api/1.0/bundle/entity/sub-entity/{id}/action
|
||||||
|
```
|
||||||
|
|
||||||
|
The following elements should be included in the list:
|
||||||
|
|
||||||
|
1. The string `/api/` followed by the version (e.g., 1.0);
|
||||||
|
2. An identifier for the bundle;
|
||||||
|
3. The entity it relates to;
|
||||||
|
4. Any sub-entities that the URL refers to;
|
||||||
|
5. The action.
|
||||||
|
|
||||||
|
These elements may be interspersed with the entity identifier. In this case, the identifier should be placed immediately after the entity it relates to.
|
||||||
|
|
||||||
|
#### URLS for admin pages
|
||||||
|
Same conventions as for other html pages, **but `admin` is added in second position**. Soit:
|
||||||
|
|
||||||
|
`/{_locale}/admin/bundle/entity/{id}/action`
|
||||||
|
|
||||||
|
## Database table naming convention
|
||||||
|
|
||||||
|
When creating a new entity and the corresponding database table, we follow the following naming convention for the database table:
|
||||||
|
|
||||||
|
`chill_{bundle_identifier}_{entity_name}`.
|
||||||
|
Eg. chill_person_spoken_languages
|
||||||
|
|
||||||
|
## UI Rules
|
||||||
|
|
||||||
|
### Page Titles
|
||||||
|
#### Every page must have a title (in the head tag and page header).
|
||||||
|
|
||||||
|
Each page contains a title in the <head> tag. This title is typically the same as the header title on the page.
|
||||||
|
|
||||||
|
Tip: It is possible to use the block function in Twig for this.
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
```htmlmixed=
|
||||||
|
{% block title "Titre de la page" %}
|
||||||
|
|
||||||
|
{% block content %}
|
||||||
|
<h1>
|
||||||
|
{{ block('title')}}
|
||||||
|
</h1>
|
||||||
|
{% endblock %}
|
||||||
|
```
|
||||||
|
|
||||||
|
### entity_render usage
|
||||||
|
|
||||||
|
#### in Twig
|
||||||
|
Always use chill_entity_render_box for rendering entities like:
|
||||||
|
* User
|
||||||
|
* Person
|
||||||
|
* SocialAction
|
||||||
|
* SocialIssue
|
||||||
|
* Address
|
||||||
|
* Thirdparty
|
||||||
|
* ...
|
||||||
|
|
||||||
|
Example:
|
||||||
|
|
||||||
|
```
|
||||||
|
address|chill_entity_render_box
|
||||||
|
```
|
||||||
|
Justification:
|
||||||
|
|
||||||
|
1. Customization by installation: Some elements are sometimes customized during installation (for example, the name of each user will be followed by the name of the service).
|
||||||
|
2. To streamline and make displays consistent: Ensures uniformity in the way information is displayed across different pages or sections.
|
||||||
|
3. To simplify Twig code: By using blocks and centralizing title logic, it reduces repetition and makes the Twig code easier to maintain.
|
||||||
|
|
||||||
|
* Three rendering modes:
|
||||||
|
* inline
|
||||||
|
* block
|
||||||
|
* item.
|
||||||
|
|
||||||
|
#### In Vue
|
||||||
|
|
||||||
|
There is always a renderbox equivalent in vue.
|
||||||
|
|
||||||
|
#### HTML Links to Sections
|
||||||
|
|
||||||
|
Always include links/icons for accessing records, such as person or household details, if the user has access.
|
||||||
|
|
||||||
|
### Form Guidelines
|
||||||
|
|
||||||
|
#### Vocabulary:
|
||||||
|
* `Create` in a `bt bt-create` for links to the form to create an entity (to access the form).
|
||||||
|
* `Save` in a `bt bt-save` for "Save" buttons (in either an edit or create form).
|
||||||
|
* `Save and new`
|
||||||
|
* `Save and view`
|
||||||
|
* `Edit` in a `bt bt-edit` for links to the edit form.
|
||||||
|
* `Duplicate` (specify where it can be seen).
|
||||||
|
* `Cancel` for leaving an edit page with a link to the list, or the `returnPath`.
|
||||||
|
|
||||||
|
#### After Saving:
|
||||||
|
|
||||||
|
Redirect to the returnPath if available; otherwise, redirect to the view page.
|
||||||
|
|
||||||
|
### Sticky Form Buttons:
|
||||||
|
|
||||||
|
Buttons like "Cancel" and "Save" must be within a "sticky-form" bar at the bottom of the form.
|
||||||
|
|
||||||
|
If relevant:
|
||||||
|
|
||||||
|
* The banner contains a "Cancel" button that returns to the previous page. It is mandatory for forms, but optional for lists or "summary" pages.
|
||||||
|
* This "Cancel" button is always on the left.
|
||||||
|
|
||||||
|
```
|
||||||
|
<ul class="record_actions sticky-form-buttons">
|
||||||
|
<li class="cancel">
|
||||||
|
<a href="{{ chill_entity_return_path('route_name' { 'route': 'option' } )}}">{{ return_path_label }}</a>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<!-- action 1 -->
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
```
|
||||||
|
|
||||||
|
### Flash Messages
|
||||||
|
|
||||||
|
#### Display a flash message upon entity creation:
|
||||||
|
|
||||||
|
Everytime an entity has been created
|
||||||
|
|
||||||
|
> "The entity has been created."
|
||||||
|
|
||||||
|
The name of the element can be replaced with something more relevant:
|
||||||
|
|
||||||
|
> * The activity has been created
|
||||||
|
> * The appointment has been created
|
||||||
|
> * ...
|
||||||
|
|
||||||
|
#### On Saving an Entity
|
||||||
|
|
||||||
|
Each time an entity is saved, a flash message should appear:
|
||||||
|
|
||||||
|
> The data has been modified
|
||||||
|
|
||||||
|
#### Form Error (Validation Error)
|
||||||
|
|
||||||
|
At the top of the form, a flash message should indicate that validations have failed:
|
||||||
|
|
||||||
|
> This form contains errors
|
||||||
|
|
||||||
|
Errors should appear attached to the field they concern. However, it is acceptable to display errors at the root of the form if it is technically difficult to attach errors.
|
||||||
|
|
||||||
|
### Return Links
|
||||||
|
|
||||||
|
Each time a link is provided, check whether the function chill_return_path, chill_forward_return_path, or chill_return_path_or should be used.
|
||||||
|
* From the list page, to the opening of an element, or the creation button => use chill_path_add_return_path
|
||||||
|
* In these edit pages:
|
||||||
|
* use chill_return_path_or in the "Cancel" button;
|
||||||
|
* for the "Save and view" and "Save and close" buttons => ?
|
||||||
|
|
||||||
|
### Assets for Suggestion Lists
|
||||||
|
|
||||||
|
Create a list of suggestions to add (the entire item is clickable)
|
||||||
|
|
||||||
|
```html
|
||||||
|
<ul class="list-suggest add-items">
|
||||||
|
<li>
|
||||||
|
<span>item</span>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
```
|
||||||
|
|
||||||
|
Create a list of suggestions to remove (with a clickable red cross, the anchor <a> is empty)
|
||||||
|
|
||||||
|
```html
|
||||||
|
<ul class="list-suggest remove-items">
|
||||||
|
<li>
|
||||||
|
<span>
|
||||||
|
item
|
||||||
|
</span>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
```
|
||||||
|
|
||||||
|
Create a removable title (with a clickable red cross, the anchor <a> is empty)
|
||||||
|
|
||||||
|
```html
|
||||||
|
<div class="item-title">
|
||||||
|
<span>title</span>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
|
The classes `cols` or `inline` can be added alongside `list-suggest` to modify the layout of the list. In the last example, add a `removable` class to the `<span>` if you want to make the item removable.
|
@@ -4,7 +4,6 @@ en cours de rédaction
|
|||||||
|
|
||||||
## Translations
|
## Translations
|
||||||
|
|
||||||
|
|
||||||
Par bundle, toutes les traductions des pages twig se trouvent dans un seul fichier `translations/messages.fr.yaml`.
|
Par bundle, toutes les traductions des pages twig se trouvent dans un seul fichier `translations/messages.fr.yaml`.
|
||||||
|
|
||||||
## Emplacement des fichiers
|
## Emplacement des fichiers
|
||||||
@@ -142,7 +141,6 @@ ul.record_actions {
|
|||||||
|
|
||||||
## Render box
|
## Render box
|
||||||
|
|
||||||
|
|
||||||
## URL
|
## URL
|
||||||
|
|
||||||
### Nommage des routes
|
### Nommage des routes
|
||||||
@@ -234,6 +232,13 @@ Même conventions que dans les autres pages html de l'application, **mais `admin
|
|||||||
|
|
||||||
`/{_locale}/admin/bundle/entity/{id}/action`
|
`/{_locale}/admin/bundle/entity/{id}/action`
|
||||||
|
|
||||||
|
### Nommage des tables de base de donnée
|
||||||
|
|
||||||
|
Lors de la création d'une nouvelle entité et de la table de base de données correspondante, nous suivons la convention d'appellation suivante pour la table de base de données :
|
||||||
|
|
||||||
|
`chill_{bundle_identifier}_{nom_de_l'entité}`.
|
||||||
|
|
||||||
|
Par exemple : chill_person_spoken_languages
|
||||||
|
|
||||||
## Règles UI chill
|
## Règles UI chill
|
||||||
|
|
||||||
@@ -293,8 +298,6 @@ A prevoir:
|
|||||||
|
|
||||||
> quand on passe l’option render: bloc, on peut placer le render_box dans une boucle for plus large qui fonctionne avec la classe flex-table ou la classe flex-bloc, ce qui donnera un affichage en rangée (table) ou en blocs. [name=Mathieu]
|
> quand on passe l’option render: bloc, on peut placer le render_box dans une boucle for plus large qui fonctionne avec la classe flex-table ou la classe flex-bloc, ce qui donnera un affichage en rangée (table) ou en blocs. [name=Mathieu]
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#### En vue
|
#### En vue
|
||||||
|
|
||||||
Il existe systématiquement une "box" équivalente en vue.
|
Il existe systématiquement une "box" équivalente en vue.
|
7
assets/translator.ts
Normal file
7
assets/translator.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { trans, setLocale, setLocaleFallbacks } from "./ux-translator";
|
||||||
|
|
||||||
|
setLocaleFallbacks({"en": "fr", "nl": "fr", "fr": "en"});
|
||||||
|
setLocale('fr');
|
||||||
|
|
||||||
|
export { trans };
|
||||||
|
export * from '../var/translations';
|
3
assets/ux-translator/README.md
Normal file
3
assets/ux-translator/README.md
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
This directory import the symfony ux-translator files directly into chill-bundles.
|
||||||
|
|
||||||
|
This remove the yarn dependencies from the real package, which breaks our installation.
|
1
assets/ux-translator/dist/formatters/formatter.d.ts
vendored
Normal file
1
assets/ux-translator/dist/formatters/formatter.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export declare function format(id: string, parameters: Record<string, string | number>, locale: string): string;
|
1
assets/ux-translator/dist/formatters/intl-formatter.d.ts
vendored
Normal file
1
assets/ux-translator/dist/formatters/intl-formatter.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export declare function formatIntl(id: string, parameters: Record<string, string | number>, locale: string): string;
|
27
assets/ux-translator/dist/translator.d.ts
vendored
Normal file
27
assets/ux-translator/dist/translator.d.ts
vendored
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
export type DomainType = string;
|
||||||
|
export type LocaleType = string;
|
||||||
|
export type TranslationsType = Record<DomainType, {
|
||||||
|
parameters: ParametersType;
|
||||||
|
}>;
|
||||||
|
export type NoParametersType = Record<string, never>;
|
||||||
|
export type ParametersType = Record<string, string | number | Date> | NoParametersType;
|
||||||
|
export type RemoveIntlIcuSuffix<T> = T extends `${infer U}+intl-icu` ? U : T;
|
||||||
|
export type DomainsOf<M> = M extends Message<infer Translations, LocaleType> ? keyof Translations : never;
|
||||||
|
export type LocaleOf<M> = M extends Message<TranslationsType, infer Locale> ? Locale : never;
|
||||||
|
export type ParametersOf<M, D extends DomainType> = M extends Message<infer Translations, LocaleType> ? Translations[D] extends {
|
||||||
|
parameters: infer Parameters;
|
||||||
|
} ? Parameters : never : never;
|
||||||
|
export interface Message<Translations extends TranslationsType, Locale extends LocaleType> {
|
||||||
|
id: string;
|
||||||
|
translations: {
|
||||||
|
[domain in DomainType]: {
|
||||||
|
[locale in Locale]: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
export declare function setLocale(locale: LocaleType | null): void;
|
||||||
|
export declare function getLocale(): LocaleType;
|
||||||
|
export declare function throwWhenNotFound(enabled: boolean): void;
|
||||||
|
export declare function setLocaleFallbacks(localeFallbacks: Record<LocaleType, LocaleType>): void;
|
||||||
|
export declare function getLocaleFallbacks(): Record<LocaleType, LocaleType>;
|
||||||
|
export declare function trans<M extends Message<TranslationsType, LocaleType>, D extends DomainsOf<M>, P extends ParametersOf<M, D>>(...args: P extends NoParametersType ? [message: M, parameters?: P, domain?: RemoveIntlIcuSuffix<D>, locale?: LocaleOf<M>] : [message: M, parameters: P, domain?: RemoveIntlIcuSuffix<D>, locale?: LocaleOf<M>]): string;
|
1
assets/ux-translator/dist/translator_controller.d.ts
vendored
Normal file
1
assets/ux-translator/dist/translator_controller.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export * from './translator';
|
283
assets/ux-translator/dist/translator_controller.js
vendored
Normal file
283
assets/ux-translator/dist/translator_controller.js
vendored
Normal file
@@ -0,0 +1,283 @@
|
|||||||
|
import { IntlMessageFormat } from 'intl-messageformat';
|
||||||
|
|
||||||
|
function strtr(string, replacePairs) {
|
||||||
|
const regex = Object.entries(replacePairs).map(([from]) => {
|
||||||
|
return from.replace(/([-[\]{}()*+?.\\^$|#,])/g, '\\$1');
|
||||||
|
});
|
||||||
|
if (regex.length === 0) {
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
return string.replace(new RegExp(regex.join('|'), 'g'), (matched) => replacePairs[matched].toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
function format(id, parameters, locale) {
|
||||||
|
if (null === id || '' === id) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
if (typeof parameters['%count%'] === 'undefined' || Number.isNaN(parameters['%count%'])) {
|
||||||
|
return strtr(id, parameters);
|
||||||
|
}
|
||||||
|
const number = Number(parameters['%count%']);
|
||||||
|
let parts = [];
|
||||||
|
if (/^\|+$/.test(id)) {
|
||||||
|
parts = id.split('|');
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
parts = id.match(/(?:\|\||[^|])+/g) || [];
|
||||||
|
}
|
||||||
|
const intervalRegex = /^(?<interval>({\s*(-?\d+(\.\d+)?[\s*,\s*\-?\d+(.\d+)?]*)\s*})|(?<left_delimiter>[[\]])\s*(?<left>-Inf|-?\d+(\.\d+)?)\s*,\s*(?<right>\+?Inf|-?\d+(\.\d+)?)\s*(?<right_delimiter>[[\]]))\s*(?<message>.*?)$/s;
|
||||||
|
const standardRules = [];
|
||||||
|
for (let part of parts) {
|
||||||
|
part = part.trim().replace(/\|\|/g, '|');
|
||||||
|
const matches = part.match(intervalRegex);
|
||||||
|
if (matches) {
|
||||||
|
const matchGroups = matches.groups || {};
|
||||||
|
if (matches[2]) {
|
||||||
|
for (const n of matches[3].split(',')) {
|
||||||
|
if (number === Number(n)) {
|
||||||
|
return strtr(matchGroups.message, parameters);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const leftNumber = '-Inf' === matchGroups.left ? Number.NEGATIVE_INFINITY : Number(matchGroups.left);
|
||||||
|
const rightNumber = ['Inf', '+Inf'].includes(matchGroups.right)
|
||||||
|
? Number.POSITIVE_INFINITY
|
||||||
|
: Number(matchGroups.right);
|
||||||
|
if (('[' === matchGroups.left_delimiter ? number >= leftNumber : number > leftNumber) &&
|
||||||
|
(']' === matchGroups.right_delimiter ? number <= rightNumber : number < rightNumber)) {
|
||||||
|
return strtr(matchGroups.message, parameters);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const ruleMatch = part.match(/^\w+:\s*(.*?)$/);
|
||||||
|
standardRules.push(ruleMatch ? ruleMatch[1] : part);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const position = getPluralizationRule(number, locale);
|
||||||
|
if (typeof standardRules[position] === 'undefined') {
|
||||||
|
if (1 === parts.length && typeof standardRules[0] !== 'undefined') {
|
||||||
|
return strtr(standardRules[0], parameters);
|
||||||
|
}
|
||||||
|
throw new Error(`Unable to choose a translation for "${id}" with locale "${locale}" for value "${number}". Double check that this translation has the correct plural options (e.g. "There is one apple|There are %count% apples").`);
|
||||||
|
}
|
||||||
|
return strtr(standardRules[position], parameters);
|
||||||
|
}
|
||||||
|
function getPluralizationRule(number, locale) {
|
||||||
|
number = Math.abs(number);
|
||||||
|
let _locale = locale;
|
||||||
|
if (locale === 'pt_BR' || locale === 'en_US_POSIX') {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
_locale = _locale.length > 3 ? _locale.substring(0, _locale.indexOf('_')) : _locale;
|
||||||
|
switch (_locale) {
|
||||||
|
case 'af':
|
||||||
|
case 'bn':
|
||||||
|
case 'bg':
|
||||||
|
case 'ca':
|
||||||
|
case 'da':
|
||||||
|
case 'de':
|
||||||
|
case 'el':
|
||||||
|
case 'en':
|
||||||
|
case 'en_US_POSIX':
|
||||||
|
case 'eo':
|
||||||
|
case 'es':
|
||||||
|
case 'et':
|
||||||
|
case 'eu':
|
||||||
|
case 'fa':
|
||||||
|
case 'fi':
|
||||||
|
case 'fo':
|
||||||
|
case 'fur':
|
||||||
|
case 'fy':
|
||||||
|
case 'gl':
|
||||||
|
case 'gu':
|
||||||
|
case 'ha':
|
||||||
|
case 'he':
|
||||||
|
case 'hu':
|
||||||
|
case 'is':
|
||||||
|
case 'it':
|
||||||
|
case 'ku':
|
||||||
|
case 'lb':
|
||||||
|
case 'ml':
|
||||||
|
case 'mn':
|
||||||
|
case 'mr':
|
||||||
|
case 'nah':
|
||||||
|
case 'nb':
|
||||||
|
case 'ne':
|
||||||
|
case 'nl':
|
||||||
|
case 'nn':
|
||||||
|
case 'no':
|
||||||
|
case 'oc':
|
||||||
|
case 'om':
|
||||||
|
case 'or':
|
||||||
|
case 'pa':
|
||||||
|
case 'pap':
|
||||||
|
case 'ps':
|
||||||
|
case 'pt':
|
||||||
|
case 'so':
|
||||||
|
case 'sq':
|
||||||
|
case 'sv':
|
||||||
|
case 'sw':
|
||||||
|
case 'ta':
|
||||||
|
case 'te':
|
||||||
|
case 'tk':
|
||||||
|
case 'ur':
|
||||||
|
case 'zu':
|
||||||
|
return 1 === number ? 0 : 1;
|
||||||
|
case 'am':
|
||||||
|
case 'bh':
|
||||||
|
case 'fil':
|
||||||
|
case 'fr':
|
||||||
|
case 'gun':
|
||||||
|
case 'hi':
|
||||||
|
case 'hy':
|
||||||
|
case 'ln':
|
||||||
|
case 'mg':
|
||||||
|
case 'nso':
|
||||||
|
case 'pt_BR':
|
||||||
|
case 'ti':
|
||||||
|
case 'wa':
|
||||||
|
return number < 2 ? 0 : 1;
|
||||||
|
case 'be':
|
||||||
|
case 'bs':
|
||||||
|
case 'hr':
|
||||||
|
case 'ru':
|
||||||
|
case 'sh':
|
||||||
|
case 'sr':
|
||||||
|
case 'uk':
|
||||||
|
return 1 === number % 10 && 11 !== number % 100
|
||||||
|
? 0
|
||||||
|
: number % 10 >= 2 && number % 10 <= 4 && (number % 100 < 10 || number % 100 >= 20)
|
||||||
|
? 1
|
||||||
|
: 2;
|
||||||
|
case 'cs':
|
||||||
|
case 'sk':
|
||||||
|
return 1 === number ? 0 : number >= 2 && number <= 4 ? 1 : 2;
|
||||||
|
case 'ga':
|
||||||
|
return 1 === number ? 0 : 2 === number ? 1 : 2;
|
||||||
|
case 'lt':
|
||||||
|
return 1 === number % 10 && 11 !== number % 100
|
||||||
|
? 0
|
||||||
|
: number % 10 >= 2 && (number % 100 < 10 || number % 100 >= 20)
|
||||||
|
? 1
|
||||||
|
: 2;
|
||||||
|
case 'sl':
|
||||||
|
return 1 === number % 100 ? 0 : 2 === number % 100 ? 1 : 3 === number % 100 || 4 === number % 100 ? 2 : 3;
|
||||||
|
case 'mk':
|
||||||
|
return 1 === number % 10 ? 0 : 1;
|
||||||
|
case 'mt':
|
||||||
|
return 1 === number
|
||||||
|
? 0
|
||||||
|
: 0 === number || (number % 100 > 1 && number % 100 < 11)
|
||||||
|
? 1
|
||||||
|
: number % 100 > 10 && number % 100 < 20
|
||||||
|
? 2
|
||||||
|
: 3;
|
||||||
|
case 'lv':
|
||||||
|
return 0 === number ? 0 : 1 === number % 10 && 11 !== number % 100 ? 1 : 2;
|
||||||
|
case 'pl':
|
||||||
|
return 1 === number
|
||||||
|
? 0
|
||||||
|
: number % 10 >= 2 && number % 10 <= 4 && (number % 100 < 12 || number % 100 > 14)
|
||||||
|
? 1
|
||||||
|
: 2;
|
||||||
|
case 'cy':
|
||||||
|
return 1 === number ? 0 : 2 === number ? 1 : 8 === number || 11 === number ? 2 : 3;
|
||||||
|
case 'ro':
|
||||||
|
return 1 === number ? 0 : 0 === number || (number % 100 > 0 && number % 100 < 20) ? 1 : 2;
|
||||||
|
case 'ar':
|
||||||
|
return 0 === number
|
||||||
|
? 0
|
||||||
|
: 1 === number
|
||||||
|
? 1
|
||||||
|
: 2 === number
|
||||||
|
? 2
|
||||||
|
: number % 100 >= 3 && number % 100 <= 10
|
||||||
|
? 3
|
||||||
|
: number % 100 >= 11 && number % 100 <= 99
|
||||||
|
? 4
|
||||||
|
: 5;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatIntl(id, parameters, locale) {
|
||||||
|
if (id === '') {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
const intlMessage = new IntlMessageFormat(id, [locale.replace('_', '-')], undefined, { ignoreTag: true });
|
||||||
|
parameters = { ...parameters };
|
||||||
|
Object.entries(parameters).forEach(([key, value]) => {
|
||||||
|
if (key.includes('%') || key.includes('{')) {
|
||||||
|
delete parameters[key];
|
||||||
|
parameters[key.replace(/[%{} ]/g, '').trim()] = value;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return intlMessage.format(parameters);
|
||||||
|
}
|
||||||
|
|
||||||
|
let _locale = null;
|
||||||
|
let _localeFallbacks = {};
|
||||||
|
let _throwWhenNotFound = false;
|
||||||
|
function setLocale(locale) {
|
||||||
|
_locale = locale;
|
||||||
|
}
|
||||||
|
function getLocale() {
|
||||||
|
return (_locale ||
|
||||||
|
document.documentElement.getAttribute('data-symfony-ux-translator-locale') ||
|
||||||
|
(document.documentElement.lang ? document.documentElement.lang.replace('-', '_') : null) ||
|
||||||
|
'en');
|
||||||
|
}
|
||||||
|
function throwWhenNotFound(enabled) {
|
||||||
|
_throwWhenNotFound = enabled;
|
||||||
|
}
|
||||||
|
function setLocaleFallbacks(localeFallbacks) {
|
||||||
|
_localeFallbacks = localeFallbacks;
|
||||||
|
}
|
||||||
|
function getLocaleFallbacks() {
|
||||||
|
return _localeFallbacks;
|
||||||
|
}
|
||||||
|
function trans(message, parameters = {}, domain = 'messages', locale = null) {
|
||||||
|
if (typeof domain === 'undefined') {
|
||||||
|
domain = 'messages';
|
||||||
|
}
|
||||||
|
if (typeof locale === 'undefined' || null === locale) {
|
||||||
|
locale = getLocale();
|
||||||
|
}
|
||||||
|
if (typeof message.translations === 'undefined') {
|
||||||
|
return message.id;
|
||||||
|
}
|
||||||
|
const localesFallbacks = getLocaleFallbacks();
|
||||||
|
const translationsIntl = message.translations[`${domain}+intl-icu`];
|
||||||
|
if (typeof translationsIntl !== 'undefined') {
|
||||||
|
while (typeof translationsIntl[locale] === 'undefined') {
|
||||||
|
locale = localesFallbacks[locale];
|
||||||
|
if (!locale) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (locale) {
|
||||||
|
return formatIntl(translationsIntl[locale], parameters, locale);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const translations = message.translations[domain];
|
||||||
|
if (typeof translations !== 'undefined') {
|
||||||
|
while (typeof translations[locale] === 'undefined') {
|
||||||
|
locale = localesFallbacks[locale];
|
||||||
|
if (!locale) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (locale) {
|
||||||
|
return format(translations[locale], parameters, locale);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (_throwWhenNotFound) {
|
||||||
|
throw new Error(`No translation message found with id "${message.id}".`);
|
||||||
|
}
|
||||||
|
return message.id;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { getLocale, getLocaleFallbacks, setLocale, setLocaleFallbacks, throwWhenNotFound, trans };
|
1
assets/ux-translator/dist/utils.d.ts
vendored
Normal file
1
assets/ux-translator/dist/utils.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export declare function strtr(string: string, replacePairs: Record<string, string | number>): string;
|
34
assets/ux-translator/package.json
Normal file
34
assets/ux-translator/package.json
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
{
|
||||||
|
"name": "@symfony/ux-translator",
|
||||||
|
"description": "Symfony Translator for JavaScript",
|
||||||
|
"license": "MIT",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"main": "dist/translator_controller.js",
|
||||||
|
"types": "dist/translator_controller.d.ts",
|
||||||
|
"scripts": {
|
||||||
|
"build": "node ../../../bin/build_package.js .",
|
||||||
|
"watch": "node ../../../bin/build_package.js . --watch",
|
||||||
|
"test": "../../../bin/test_package.sh .",
|
||||||
|
"check": "biome check",
|
||||||
|
"ci": "biome ci"
|
||||||
|
},
|
||||||
|
"symfony": {
|
||||||
|
"importmap": {
|
||||||
|
"intl-messageformat": "^10.5.11",
|
||||||
|
"@symfony/ux-translator": "path:%PACKAGE%/dist/translator_controller.js",
|
||||||
|
"@app/translations": "path:var/translations/index.js",
|
||||||
|
"@app/translations/configuration": "path:var/translations/configuration.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"intl-messageformat": "^10.5.11"
|
||||||
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"intl-messageformat": {
|
||||||
|
"optional": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"intl-messageformat": "^10.5.11"
|
||||||
|
}
|
||||||
|
}
|
@@ -13,8 +13,10 @@
|
|||||||
"ext-json": "*",
|
"ext-json": "*",
|
||||||
"ext-openssl": "*",
|
"ext-openssl": "*",
|
||||||
"ext-redis": "*",
|
"ext-redis": "*",
|
||||||
|
"ext-zlib": "*",
|
||||||
"champs-libres/wopi-bundle": "dev-master@dev",
|
"champs-libres/wopi-bundle": "dev-master@dev",
|
||||||
"champs-libres/wopi-lib": "dev-master@dev",
|
"champs-libres/wopi-lib": "dev-master@dev",
|
||||||
|
"doctrine/data-fixtures": "^1.8",
|
||||||
"doctrine/doctrine-bundle": "^2.1",
|
"doctrine/doctrine-bundle": "^2.1",
|
||||||
"doctrine/doctrine-migrations-bundle": "^3.0",
|
"doctrine/doctrine-migrations-bundle": "^3.0",
|
||||||
"doctrine/orm": "^2.13.0",
|
"doctrine/orm": "^2.13.0",
|
||||||
@@ -56,7 +58,9 @@
|
|||||||
"symfony/messenger": "^5.4",
|
"symfony/messenger": "^5.4",
|
||||||
"symfony/mime": "^5.4",
|
"symfony/mime": "^5.4",
|
||||||
"symfony/monolog-bundle": "^3.5",
|
"symfony/monolog-bundle": "^3.5",
|
||||||
|
"symfony/notifier": "^5.4",
|
||||||
"symfony/options-resolver": "^5.4",
|
"symfony/options-resolver": "^5.4",
|
||||||
|
"symfony/ovh-cloud-notifier": "^5.4",
|
||||||
"symfony/process": "^5.4",
|
"symfony/process": "^5.4",
|
||||||
"symfony/property-access": "^5.4",
|
"symfony/property-access": "^5.4",
|
||||||
"symfony/property-info": "^5.4",
|
"symfony/property-info": "^5.4",
|
||||||
@@ -71,6 +75,7 @@
|
|||||||
"symfony/templating": "^5.4",
|
"symfony/templating": "^5.4",
|
||||||
"symfony/translation": "^5.4",
|
"symfony/translation": "^5.4",
|
||||||
"symfony/twig-bundle": "^5.4",
|
"symfony/twig-bundle": "^5.4",
|
||||||
|
"symfony/ux-translator": "^2.22",
|
||||||
"symfony/validator": "^5.4",
|
"symfony/validator": "^5.4",
|
||||||
"symfony/webpack-encore-bundle": "^1.11",
|
"symfony/webpack-encore-bundle": "^1.11",
|
||||||
"symfony/workflow": "^5.4",
|
"symfony/workflow": "^5.4",
|
||||||
@@ -85,6 +90,7 @@
|
|||||||
"require-dev": {
|
"require-dev": {
|
||||||
"doctrine/doctrine-fixtures-bundle": "^3.3",
|
"doctrine/doctrine-fixtures-bundle": "^3.3",
|
||||||
"fakerphp/faker": "^1.13",
|
"fakerphp/faker": "^1.13",
|
||||||
|
"friendsofphp/php-cs-fixer": "3.65.0",
|
||||||
"jangregor/phpstan-prophecy": "^1.0",
|
"jangregor/phpstan-prophecy": "^1.0",
|
||||||
"nelmio/alice": "^3.8",
|
"nelmio/alice": "^3.8",
|
||||||
"nikic/php-parser": "^4.15",
|
"nikic/php-parser": "^4.15",
|
||||||
@@ -95,6 +101,7 @@
|
|||||||
"phpstan/phpstan-strict-rules": "^1.0",
|
"phpstan/phpstan-strict-rules": "^1.0",
|
||||||
"phpunit/phpunit": "^10.5.24",
|
"phpunit/phpunit": "^10.5.24",
|
||||||
"rector/rector": "^1.1.0",
|
"rector/rector": "^1.1.0",
|
||||||
|
"symfony/amqp-messenger": "^5.4.45",
|
||||||
"symfony/debug-bundle": "^5.4",
|
"symfony/debug-bundle": "^5.4",
|
||||||
"symfony/dotenv": "^5.4",
|
"symfony/dotenv": "^5.4",
|
||||||
"symfony/flex": "^2.4",
|
"symfony/flex": "^2.4",
|
||||||
@@ -156,7 +163,9 @@
|
|||||||
"cache:clear": "symfony-cmd",
|
"cache:clear": "symfony-cmd",
|
||||||
"assets:install %PUBLIC_DIR%": "symfony-cmd"
|
"assets:install %PUBLIC_DIR%": "symfony-cmd"
|
||||||
},
|
},
|
||||||
"php-cs-fixer": "php-cs-fixer fix --config=./.php-cs-fixer.dist.php --show-progress=none"
|
"php-cs-fixer": "php-cs-fixer fix --config=./.php-cs-fixer.dist.php --show-progress=none",
|
||||||
|
"phpstan": "phpstan --no-progress",
|
||||||
|
"rector": "rector --no-progress-bar"
|
||||||
},
|
},
|
||||||
"extra": {
|
"extra": {
|
||||||
"symfony": {
|
"symfony": {
|
||||||
|
@@ -36,4 +36,5 @@ return [
|
|||||||
Chill\BudgetBundle\ChillBudgetBundle::class => ['all' => true],
|
Chill\BudgetBundle\ChillBudgetBundle::class => ['all' => true],
|
||||||
Chill\WopiBundle\ChillWopiBundle::class => ['all' => true],
|
Chill\WopiBundle\ChillWopiBundle::class => ['all' => true],
|
||||||
Symfony\Bundle\WebProfilerBundle\WebProfilerBundle::class => ['dev' => true, 'test' => true],
|
Symfony\Bundle\WebProfilerBundle\WebProfilerBundle::class => ['dev' => true, 'test' => true],
|
||||||
|
Symfony\UX\Translator\UxTranslatorBundle::class => ['all' => true],
|
||||||
];
|
];
|
||||||
|
@@ -1,4 +1,7 @@
|
|||||||
chill_doc_store:
|
chill_doc_store:
|
||||||
|
use_driver: openstack
|
||||||
|
local_storage:
|
||||||
|
storage_path: '%kernel.project_dir%/var/storage'
|
||||||
openstack:
|
openstack:
|
||||||
temp_url:
|
temp_url:
|
||||||
temp_url_key: '%env(resolve:ASYNC_UPLOAD_TEMP_URL_KEY)%' # Required
|
temp_url_key: '%env(resolve:ASYNC_UPLOAD_TEMP_URL_KEY)%' # Required
|
||||||
|
11
config/packages/chill_workflow_signature_documents.yaml
Normal file
11
config/packages/chill_workflow_signature_documents.yaml
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
chill_main:
|
||||||
|
workflow_signature:
|
||||||
|
base_signer:
|
||||||
|
document_kinds:
|
||||||
|
- { key: id_card, labels: [ { lang: fr, label: "Carte d'identité" } ] }
|
||||||
|
- { key: passport, labels: [ { lang: fr, label: "Passeport" } ] }
|
||||||
|
- { key: drivers_license, labels: [ { lang: fr, label: "Permis de conduire" } ] }
|
||||||
|
- { key: visa_long_stay, labels: [ { lang: fr, label: "Visa de long séjour" } ] }
|
||||||
|
- { key: resident_permit, labels: [ { lang: fr, label: "Carte de séjour" } ] }
|
||||||
|
- { key: residency_card, labels: [ { lang: fr, label: "Carte de résident" } ] }
|
||||||
|
- { key: provisionary_residency_permit, labels: [ { lang: fr, label: "Autorisation provisoire de séjour" } ] }
|
13
config/packages/notifier.yaml
Normal file
13
config/packages/notifier.yaml
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
framework:
|
||||||
|
notifier:
|
||||||
|
texter_transports:
|
||||||
|
#ovhcloud: '%env(OVHCLOUD_DSN)%'
|
||||||
|
#ovhcloud: '%env(SHORT_MESSAGE_DSN)%'
|
||||||
|
channel_policy:
|
||||||
|
# use chat/slack, chat/telegram, sms/twilio or sms/nexmo
|
||||||
|
urgent: ['email']
|
||||||
|
high: ['email']
|
||||||
|
medium: ['email']
|
||||||
|
low: ['email']
|
||||||
|
admin_recipients:
|
||||||
|
- { email: admin@example.com }
|
3
config/packages/ux_translator.yaml
Normal file
3
config/packages/ux_translator.yaml
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
ux_translator:
|
||||||
|
# The directory where the JavaScript translations are dumped
|
||||||
|
dump_directory: '%kernel.project_dir%/var/translations'
|
@@ -220,6 +220,7 @@ framework:
|
|||||||
- attenteModification
|
- attenteModification
|
||||||
- attenteMiseEnForme
|
- attenteMiseEnForme
|
||||||
- attenteValidationMiseEnForme
|
- attenteValidationMiseEnForme
|
||||||
|
- attenteSignature
|
||||||
- attenteVisa
|
- attenteVisa
|
||||||
- postSignature
|
- postSignature
|
||||||
- attenteTraitement
|
- attenteTraitement
|
||||||
|
19
config/routes/chill_assets_dev.yaml
Normal file
19
config/routes/chill_assets_dev.yaml
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
when@dev:
|
||||||
|
sass_assets:
|
||||||
|
path: /_dev/assets
|
||||||
|
controller: Symfony\Bundle\FrameworkBundle\Controller\TemplateController
|
||||||
|
defaults:
|
||||||
|
template: '@ChillMain/Dev/dev.assets.html.twig'
|
||||||
|
|
||||||
|
sass_assets_test1:
|
||||||
|
path: /_dev/assets_test1
|
||||||
|
controller: Symfony\Bundle\FrameworkBundle\Controller\TemplateController
|
||||||
|
defaults:
|
||||||
|
template: '@ChillMain/Dev/dev.assets.test1.html.twig'
|
||||||
|
|
||||||
|
sass_assets_test2:
|
||||||
|
path: /_dev/assets_test2
|
||||||
|
controller: Symfony\Bundle\FrameworkBundle\Controller\TemplateController
|
||||||
|
defaults:
|
||||||
|
template: '@ChillMain/Dev/dev.assets.test2.html.twig'
|
||||||
|
|
12
config/routes/chill_swagger.yaml
Normal file
12
config/routes/chill_swagger.yaml
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
when@dev:
|
||||||
|
swagger_ui:
|
||||||
|
path: /_dev/swagger
|
||||||
|
controller: Symfony\Bundle\FrameworkBundle\Controller\TemplateController
|
||||||
|
defaults:
|
||||||
|
template: '@ChillMain/Dev/swagger-ui/index.html.twig'
|
||||||
|
|
||||||
|
swagger_specs:
|
||||||
|
path: /_dev/specs.yaml
|
||||||
|
controller: Symfony\Bundle\FrameworkBundle\Controller\TemplateController
|
||||||
|
defaults:
|
||||||
|
template: api/specs.yaml
|
71
docs/source/development/es-lint.rst
Normal file
71
docs/source/development/es-lint.rst
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
ESLint
|
||||||
|
======
|
||||||
|
|
||||||
|
To improve the quality of our JS and VueJS code, ESLint and eslint-plugin-vue are implemented within the chill-bundles project.
|
||||||
|
|
||||||
|
Commands
|
||||||
|
--------
|
||||||
|
|
||||||
|
To run ESLint, you can simply use the ``eslint`` command within the chill-bundles directory.
|
||||||
|
This runs eslint **not** taking the baseline into account, thus showing all existing errors in the project.
|
||||||
|
|
||||||
|
A script was also added to package.json allowing you to execute ``yarn run eslint``.
|
||||||
|
This will run eslint, but **taking the baseline into account**, thus only alerting to newly created errors.
|
||||||
|
|
||||||
|
The eslint command is configured to also run ``prettier`` which will simply format the code to look more uniform (takes care indentation for example).
|
||||||
|
|
||||||
|
Interesting options that can be used in combination with eslint are:
|
||||||
|
|
||||||
|
- ``--quiet`` to only get errors and silence the warnings
|
||||||
|
- ``--fix`` to have ESLint fix what it can, automatically. This will not fix everything.
|
||||||
|
|
||||||
|
Baseline
|
||||||
|
--------
|
||||||
|
|
||||||
|
To allow us the time to fix linting errors/warnings a baseline has been created using the following command
|
||||||
|
- ``npx eslint-baseline "**/*.{js,vue}"``
|
||||||
|
|
||||||
|
The baseline has been commited and the gitlab CI setup to only fail upon new errors/warnings being created.
|
||||||
|
When fixing errors/warnings manually, please update the baseline.
|
||||||
|
|
||||||
|
1. Delete the current baseline file
|
||||||
|
2. Run the above command locally, this will automatically create a new baseline that should be commited
|
||||||
|
|
||||||
|
Rules
|
||||||
|
-----
|
||||||
|
|
||||||
|
We use Vue 3, so the rules can be configured as follows within the ``eslint.config.mjs`` file:
|
||||||
|
|
||||||
|
- ``.configs["flat/base"]`` ... Settings and rules to enable correct ESLint parsing.
|
||||||
|
|
||||||
|
Configurations for using Vue.js 3.x:
|
||||||
|
|
||||||
|
- ``.configs["flat/essential"]`` : Base rules plus rules to prevent errors or unintended behavior.
|
||||||
|
- ``.configs["flat/strongly-recommended"]`` ... Above, plus rules to considerably improve code readability and/or dev experience.
|
||||||
|
- ``.configs["flat/recommended"]`` ... Above, plus rules to enforce subjective community defaults to ensure consistency.
|
||||||
|
|
||||||
|
Detailed information about which rules each set includes can be found here:
|
||||||
|
`https://eslint.vuejs.org/rules/ <https://eslint.vuejs.org/rules/>`_
|
||||||
|
|
||||||
|
Manual Rule Configuration
|
||||||
|
-------------------------
|
||||||
|
|
||||||
|
We can also manually configure certain rules or override rules that are part of the ruleset specified above.
|
||||||
|
|
||||||
|
For example, if we want to turn off a certain rule, we can do so as follows:
|
||||||
|
|
||||||
|
.. code-block:: javascript
|
||||||
|
|
||||||
|
rules: {
|
||||||
|
'vue/multi-word-component': 'off'
|
||||||
|
}
|
||||||
|
|
||||||
|
We could also change the severity of a certain rule from 'error' to 'warning', for example.
|
||||||
|
|
||||||
|
Within specific ``.js`` or ``.vue`` files, we can also override a certain rule only for that specific file by adding a comment:
|
||||||
|
|
||||||
|
.. code-block:: javascript
|
||||||
|
|
||||||
|
/* eslint multi-word-component: "off", no-child-content: "error"
|
||||||
|
--------
|
||||||
|
Here's a description about why this configuration is necessary. */
|
@@ -31,6 +31,7 @@ As Chill relies on the `symfony <http://symfony.com>`_ framework, reading the fr
|
|||||||
Exports <exports.rst>
|
Exports <exports.rst>
|
||||||
Embeddable comments <embeddable-comments.rst>
|
Embeddable comments <embeddable-comments.rst>
|
||||||
Run tests <run-tests.rst>
|
Run tests <run-tests.rst>
|
||||||
|
ESLint <es-lint.rst>
|
||||||
Useful snippets <useful-snippets.rst>
|
Useful snippets <useful-snippets.rst>
|
||||||
manual/index.rst
|
manual/index.rst
|
||||||
Assets <assets.rst>
|
Assets <assets.rst>
|
||||||
|
@@ -1,39 +1,37 @@
|
|||||||
import { ShowHide } from 'ShowHide/show_hide.js';
|
import { ShowHide } from "ShowHide/show_hide.js";
|
||||||
|
|
||||||
var
|
|
||||||
div_accompagnement = document.getElementById("form_accompagnement"),
|
|
||||||
div_accompagnement_comment = document.getElementById("form_accompagnement_comment"),
|
|
||||||
div_caf_id = document.getElementById("cafId"),
|
|
||||||
div_caf_inscription_date = document.getElementById("cafInscriptionDate"),
|
|
||||||
;
|
|
||||||
|
|
||||||
|
var div_accompagnement = document.getElementById("form_accompagnement"),
|
||||||
|
div_accompagnement_comment = document.getElementById(
|
||||||
|
"form_accompagnement_comment",
|
||||||
|
),
|
||||||
|
div_caf_id = document.getElementById("cafId"),
|
||||||
|
div_caf_inscription_date = document.getElementById("cafInscriptionDate");
|
||||||
// let show/hide the div_accompagnement_comment if the input with value `'autre'` is checked
|
// let show/hide the div_accompagnement_comment if the input with value `'autre'` is checked
|
||||||
new ShowHide({
|
new ShowHide({
|
||||||
"froms": [div_accompagnement],
|
froms: [div_accompagnement],
|
||||||
"test": function(froms, event) {
|
test: function (froms, event) {
|
||||||
for (let el of froms.values()) {
|
for (let el of froms.values()) {
|
||||||
for (let input of el.querySelectorAll('input').values()) {
|
for (let input of el.querySelectorAll("input").values()) {
|
||||||
if (input.value === 'autre') {
|
if (input.value === "autre") {
|
||||||
return input.checked;
|
return input.checked;
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return false;
|
}
|
||||||
},
|
|
||||||
"container": [div_accompagnement_comment]
|
return false;
|
||||||
|
},
|
||||||
|
container: [div_accompagnement_comment],
|
||||||
});
|
});
|
||||||
|
|
||||||
// let show the date input only if the the id is filled
|
// let show the date input only if the the id is filled
|
||||||
new ShowHide({
|
new ShowHide({
|
||||||
froms: [ div_caf_id ],
|
froms: [div_caf_id],
|
||||||
test: function(froms, event) {
|
test: function (froms, event) {
|
||||||
for (let el of froms.values()) {
|
for (let el of froms.values()) {
|
||||||
return el.querySelector("input").value !== "";
|
return el.querySelector("input").value !== "";
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
container: [ div_caf_inscription_date ],
|
container: [div_caf_inscription_date],
|
||||||
// using this option, we use the event `input` instead of `change`
|
// using this option, we use the event `input` instead of `change`
|
||||||
event_name: 'input'
|
event_name: "input",
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -16,7 +16,7 @@ Welcome to Chill documentation!
|
|||||||
|
|
||||||
Chill is a free software for social workers.
|
Chill is a free software for social workers.
|
||||||
|
|
||||||
Chill rely on the php framework `Symfony <http://symfony.com>`_.
|
Chill rely on the php framework `Symfony <http://symfony.com>`_.
|
||||||
|
|
||||||
Contents of this documentation:
|
Contents of this documentation:
|
||||||
|
|
||||||
@@ -42,7 +42,7 @@ Contribute
|
|||||||
User manual
|
User manual
|
||||||
===========
|
===========
|
||||||
|
|
||||||
An user manual exists in French and currently focuses on describing the main concept of the software.
|
An user manual exists in French and currently focuses on describing the main concept of the software.
|
||||||
|
|
||||||
`Read (and contribute) to the manual <https://fr.wikibooks.org/wiki/Chill>`_
|
`Read (and contribute) to the manual <https://fr.wikibooks.org/wiki/Chill>`_
|
||||||
|
|
||||||
@@ -55,12 +55,11 @@ Available bundles
|
|||||||
* Chill Person, to deal with persons,
|
* Chill Person, to deal with persons,
|
||||||
* chill custom fields, to add custom fields to some entities,
|
* chill custom fields, to add custom fields to some entities,
|
||||||
* chill activity: to add activities to people,
|
* chill activity: to add activities to people,
|
||||||
* chill report: to add report to people,
|
* chill report: to add report to people,
|
||||||
* chill event: to gather people into events,
|
* chill event: to gather people into events,
|
||||||
* chill docs store: to store documents to people, but also entities,
|
* chill docs store: to store documents to people, but also entities,
|
||||||
* chill task: to register task with people,
|
* chill task: to register task with people,
|
||||||
* chill third party: to register third parties,
|
* chill third party: to register third parties,
|
||||||
* chill family members: to register family members
|
|
||||||
|
|
||||||
You will also found the following projects :
|
You will also found the following projects :
|
||||||
|
|
||||||
|
84
docs/source/installation/document-storage.rst
Normal file
84
docs/source/installation/document-storage.rst
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
Document storage
|
||||||
|
################
|
||||||
|
|
||||||
|
You can store document on two different ways:
|
||||||
|
|
||||||
|
- on disk
|
||||||
|
- in the cloud, using object storage: currently only `openstack swift <https://docs.openstack.org/api-ref/object-store/index.html>`_ is supported.
|
||||||
|
|
||||||
|
Comparison
|
||||||
|
==========
|
||||||
|
|
||||||
|
Storing documents within the cloud is particularily suitable for "portable" deployments, like in kubernetes, or within container
|
||||||
|
without having to manage volumes to store documents. But you'll have to subscribe on a commercial offer.
|
||||||
|
|
||||||
|
Storing documents on disk is more easy to configure, but more difficult to manage: if you use container, you will have to
|
||||||
|
manager volumes to attach documents on disk. You'll have to do some backup of the directory. If chill is load-balanced (and
|
||||||
|
multiple instances of chill are run), you will have to find a way to share the directories in read-write mode for every instance.
|
||||||
|
|
||||||
|
On Disk
|
||||||
|
=======
|
||||||
|
|
||||||
|
Configure Chill like this:
|
||||||
|
|
||||||
|
.. code-block:: yaml
|
||||||
|
|
||||||
|
# file config/packages/chill_doc_store.yaml
|
||||||
|
chill_doc_store:
|
||||||
|
use_driver: local_storage
|
||||||
|
local_storage:
|
||||||
|
storage_path: '%kernel.project_dir%/var/storage'
|
||||||
|
|
||||||
|
In this configuration, documents will be stored in :code:`var/storage` within your app directory. But this path can be
|
||||||
|
elsewhere on the disk. Be aware that the directory must be writable by the user executing the chill app (php-fpm or www-data).
|
||||||
|
|
||||||
|
Documents will be stored in subpathes within that directory. The files will be encrypted, the key is stored in the database.
|
||||||
|
|
||||||
|
In the cloud, using openstack object store
|
||||||
|
##########################################
|
||||||
|
|
||||||
|
You must subscribe to a commercial offer for object store.
|
||||||
|
|
||||||
|
Chill use some features to allow documents to be stored in the cloud without being uploaded first to the chill server:
|
||||||
|
|
||||||
|
- `Form POST Middelware <https://docs.openstack.org/swift/latest/api/form_post_middleware.html>`_;
|
||||||
|
- `Temporary URL Middelware <https://docs.openstack.org/swift/latest/api/temporary_url_middleware.html>`_.
|
||||||
|
|
||||||
|
A secret key must be generated and configured, and CORS must be configured depending on the domain you will use to serve Chill.
|
||||||
|
|
||||||
|
At first, create a container and get the base path to the container. For instance, on OVH, if you create a container named "mychill",
|
||||||
|
you will be able to retrieve the base path of the container within the OVH interface, like this:
|
||||||
|
|
||||||
|
- base_path: :code:`https://storage.gra.cloud.ovh.net/v1/AUTH_123456789/mychill/` => will be variable :code:`ASYNC_UPLOAD_TEMP_URL_BASE_PATH`
|
||||||
|
- container: :code:`mychill` => will be variable :code:`ASYNC_UPLOAD_TEMP_URL_CONTAINER`
|
||||||
|
|
||||||
|
You can also generate a key, which should have at least 20 characters. This key will go in the variable :code:`ASYNC_UPLOAD_TEMP_URL_KEY`.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
|
||||||
|
See the `documentation of symfony <https://symfony.com/doc/current/configuration.html#config-env-vars>`_ on how to store variables, and how to encrypt them if needed.
|
||||||
|
|
||||||
|
Configure the storage like this:
|
||||||
|
|
||||||
|
.. code-block:: yaml
|
||||||
|
|
||||||
|
# file config/packages/chill_doc_store.yaml
|
||||||
|
chill_doc_store:
|
||||||
|
use_driver: openstack
|
||||||
|
openstack:
|
||||||
|
temp_url:
|
||||||
|
temp_url_key: '%env(resolve:ASYNC_UPLOAD_TEMP_URL_KEY)%' # Required
|
||||||
|
container: '%env(resolve:ASYNC_UPLOAD_TEMP_URL_CONTAINER)%' # Required
|
||||||
|
temp_url_base_path: '%env(resolve:ASYNC_UPLOAD_TEMP_URL_BASE_PATH)%' # Required
|
||||||
|
|
||||||
|
Chill is able to configure the container in order to store document. Grab an Openstack Token (for instance, using :code:`openstack token issue` or
|
||||||
|
the web interface of your openstack provider), and run this command:
|
||||||
|
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
symfony console async-upload:configure --os_token=OPENSTACK_TOKEN -d https://mychill.mydomain.example
|
||||||
|
|
||||||
|
# or, without symfony-cli
|
||||||
|
bin/console async-upload:configure --os_token=OPENSTACK_TOKEN -d https://mychill.mydomain.example
|
||||||
|
|
||||||
|
|
@@ -29,8 +29,7 @@ We strongly encourage you to initialize a git repository at this step, to track
|
|||||||
# add the flex endpoints required for custom recipes
|
# add the flex endpoints required for custom recipes
|
||||||
cat <<< "$(jq '.extra.symfony += {"endpoint": ["flex://defaults", "https://gitlab.com/api/v4/projects/57371968/repository/files/index.json/raw?ref=main"]}' composer.json)" > composer.json
|
cat <<< "$(jq '.extra.symfony += {"endpoint": ["flex://defaults", "https://gitlab.com/api/v4/projects/57371968/repository/files/index.json/raw?ref=main"]}' composer.json)" > composer.json
|
||||||
# install chill and some dependencies
|
# install chill and some dependencies
|
||||||
# TODO fix the suffix "alpha1" and replace by ^3.0.0 when version 3.0.0 will be released
|
symfony composer require chill-project/chill-bundles ^3.7.1 champs-libres/wopi-lib dev-master@dev champs-libres/wopi-bundle dev-master@dev symfony/amqp-messenger
|
||||||
symfony composer require chill-project/chill-bundles v3.0.0-RC3 champs-libres/wopi-lib dev-master@dev champs-libres/wopi-bundle dev-master@dev
|
|
||||||
|
|
||||||
We encourage you to accept the inclusion of the "Docker configuration from recipes": this is the documented way to run the database.
|
We encourage you to accept the inclusion of the "Docker configuration from recipes": this is the documented way to run the database.
|
||||||
You must also accept to configure recipes from the contrib repository, unless you want to configure the bundles manually).
|
You must also accept to configure recipes from the contrib repository, unless you want to configure the bundles manually).
|
||||||
@@ -48,7 +47,7 @@ You must also accept to configure recipes from the contrib repository, unless yo
|
|||||||
|
|
||||||
If you encounter this error during assets compilation (:code:`yarn run encore production`) (repeated multiple times):
|
If you encounter this error during assets compilation (:code:`yarn run encore production`) (repeated multiple times):
|
||||||
|
|
||||||
.. code-block:: txt
|
.. code-block::
|
||||||
|
|
||||||
[tsl] ERROR in /tmp/chill/v1/public/bundles/chillcalendar/types.ts(2,65)
|
[tsl] ERROR in /tmp/chill/v1/public/bundles/chillcalendar/types.ts(2,65)
|
||||||
TS2307: Cannot find module '../../../ChillMainBundle/Resources/public/types' or its corresponding type declarations.
|
TS2307: Cannot find module '../../../ChillMainBundle/Resources/public/types' or its corresponding type declarations.
|
||||||
@@ -74,14 +73,22 @@ or in the :code:`.env.local` file, which should not be committed to the git repo
|
|||||||
You do not need to set variables for the smtp server, redis server and relatorio server, as they are generated automatically
|
You do not need to set variables for the smtp server, redis server and relatorio server, as they are generated automatically
|
||||||
by the symfony server, from the docker compose services.
|
by the symfony server, from the docker compose services.
|
||||||
|
|
||||||
The only required variable is the :code:`ADMIN_PASSWORD`. You can generate a hashed and salted admin password using the command
|
The required variables are:
|
||||||
:code:`symfony console security:hash-password <your password> 'Symfony\Component\Security\Core\User\User'`. Then,
|
|
||||||
|
- the :code:`ADMIN_PASSWORD`;
|
||||||
|
- the :code:`OVHCLOUD_DSN` variable;
|
||||||
|
|
||||||
|
:code:`ADMIN_PASSWORD`
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
|
You can generate a hashed and salted admin password using the command
|
||||||
|
:code:`symfony console security:hash-password <your password> 'Symfony\Component\Security\Core\User\User'`.Then,
|
||||||
you can either:
|
you can either:
|
||||||
|
|
||||||
- add this password to the :code:`.env.local` file, you must escape the character :code:`$`: if the generated password
|
- add this password to the :code:`.env.local` file, you must escape the character :code:`$`: if the generated password
|
||||||
is :code:`$2y$13$iyvJLuT4YEa6iWXyQV4/N.hNHpNG8kXlYDkkt5MkYy4FXcSwYAwmm`, your :code:`.env.local` file will be:
|
is :code:`$2y$13$iyvJLuT4YEa6iWXyQV4/N.hNHpNG8kXlYDkkt5MkYy4FXcSwYAwmm`, your :code:`.env.local` file will be:
|
||||||
|
|
||||||
.. code-block:: env
|
.. code-block:: bash
|
||||||
|
|
||||||
ADMIN_PASSWORD=\$2y\$13\$iyvJLuT4YEa6iWXyQV4/N.hNHpNG8kXlYDkkt5MkYy4FXcSwYAwmm
|
ADMIN_PASSWORD=\$2y\$13\$iyvJLuT4YEa6iWXyQV4/N.hNHpNG8kXlYDkkt5MkYy4FXcSwYAwmm
|
||||||
# note: if you copy-paste the line above, the password will be "admin".
|
# note: if you copy-paste the line above, the password will be "admin".
|
||||||
@@ -89,12 +96,24 @@ you can either:
|
|||||||
- add the generated password to the secrets manager (**note**: you must add the generated hashed password to the secrets env,
|
- add the generated password to the secrets manager (**note**: you must add the generated hashed password to the secrets env,
|
||||||
not the password in clear text).
|
not the password in clear text).
|
||||||
|
|
||||||
- set up the jwt authentication bundle
|
:code:`OVHCLOUD_DSN` and sending SMS messages
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
|
This is a temporary dependency, for ensuring compatibility for previous behaviour.
|
||||||
|
|
||||||
|
You can set it to :code:`null://null` if you do not plan to use sending SMS.
|
||||||
|
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
OVHCLOUD_DSN=null://null
|
||||||
|
|
||||||
|
If you plan to do it, you can configure the notifier component `as described in the symfony documentation <https://symfony.com/doc/current/notifier.html#notifier-sms-channel>`_.
|
||||||
|
|
||||||
|
|
||||||
Some environment variables are available for the JWT authentication bundle in the :code:`.env` file.
|
Some environment variables are available for the JWT authentication bundle in the :code:`.env` file.
|
||||||
|
|
||||||
Prepare migrations and other tools
|
Prepare database, messenger queue, and other configuration
|
||||||
----------------------------------
|
----------------------------------------------------------
|
||||||
|
|
||||||
To continue the installation process, you will have to run migrations:
|
To continue the installation process, you will have to run migrations:
|
||||||
|
|
||||||
@@ -109,17 +128,22 @@ To continue the installation process, you will have to run migrations:
|
|||||||
symfony console messenger:setup-transports
|
symfony console messenger:setup-transports
|
||||||
# prepare some views
|
# prepare some views
|
||||||
symfony console chill:db:sync-views
|
symfony console chill:db:sync-views
|
||||||
|
# load languages data
|
||||||
|
symfony console chill:main:languages:populate
|
||||||
# generate jwt token, required for some api features (webdav access, ...)
|
# generate jwt token, required for some api features (webdav access, ...)
|
||||||
symfony console lexik:jwt:generate-keypair
|
symfony console lexik:jwt:generate-keypair
|
||||||
|
|
||||||
.. warning::
|
.. note::
|
||||||
|
|
||||||
If you encounter an error while running :code:`symfony console messenger:setup-transports`, you can set up the messenger
|
If you encounter this error:
|
||||||
transport to redis, by adding this in the :code:`.env.local` or :code:`.env` file:
|
|
||||||
|
.. code-block::
|
||||||
|
|
||||||
|
No transport supports the given Messenger DSN.
|
||||||
|
|
||||||
|
Please check that you installed the package `symfony/amqp-messenger`.
|
||||||
|
|
||||||
.. code-block:: env
|
|
||||||
|
|
||||||
MESSENGER_TRANSPORT_DSN=redis://${REDIS_HOST}:${REDIS_PORT}/messages
|
|
||||||
|
|
||||||
Start your web server locally
|
Start your web server locally
|
||||||
-----------------------------
|
-----------------------------
|
||||||
@@ -323,6 +347,7 @@ Going further
|
|||||||
:maxdepth: 2
|
:maxdepth: 2
|
||||||
|
|
||||||
prod.rst
|
prod.rst
|
||||||
|
document-storage.rst
|
||||||
load-addresses.rst
|
load-addresses.rst
|
||||||
prod-calendar-sms-sending.rst
|
prod-calendar-sms-sending.rst
|
||||||
msgraph-configure.rst
|
msgraph-configure.rst
|
||||||
|
@@ -41,16 +41,18 @@ Postal code are loaded from this database. There is no need to load postal codes
|
|||||||
The data are prepared for Chill (`See this repository <https://gitea.champs-libres.be/Chill-project/belgian-bestaddresses-transform/releases>`_).
|
The data are prepared for Chill (`See this repository <https://gitea.champs-libres.be/Chill-project/belgian-bestaddresses-transform/releases>`_).
|
||||||
One can select postal code by his first number (:code:`1xxx` for postal codes from 1000 to 1999), or a limited list for development purpose.
|
One can select postal code by his first number (:code:`1xxx` for postal codes from 1000 to 1999), or a limited list for development purpose.
|
||||||
|
|
||||||
|
The command expects a language code as first argument.
|
||||||
|
|
||||||
.. code-block:: bash
|
.. code-block:: bash
|
||||||
|
|
||||||
# load postal code from 1000 to 3999:
|
# load postal code from 1000 to 3999:
|
||||||
bin/console chill:main:address-ref-from-best-addresse 1xxx 2xxx 3xxx
|
bin/console chill:main:address-ref-from-best-addresse fr 1xxx 2xxx 3xxx
|
||||||
|
|
||||||
# load only an extract (for dev purposes)
|
# load only an extract (for dev purposes)
|
||||||
bin/console chill:main:address-ref-from-best-addresse extract
|
bin/console chill:main:address-ref-from-best-addresse fr extract
|
||||||
|
|
||||||
# load full addresses (discouraged)
|
# load full addresses (discouraged)
|
||||||
bin/console chill:main:address-ref-from-best-addresse full
|
bin/console chill:main:address-ref-from-best-addresse fr full
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
|
40
eslint.config.mjs
Normal file
40
eslint.config.mjs
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
import eslintPluginVue from "eslint-plugin-vue";
|
||||||
|
import ts from "typescript-eslint";
|
||||||
|
import eslintPluginPrettier from "eslint-plugin-prettier";
|
||||||
|
|
||||||
|
export default ts.config(
|
||||||
|
...ts.configs.recommended,
|
||||||
|
...ts.configs.stylistic,
|
||||||
|
...eslintPluginVue.configs["flat/essential"],
|
||||||
|
{
|
||||||
|
files: ["**/*.vue"],
|
||||||
|
languageOptions: {
|
||||||
|
parserOptions: {
|
||||||
|
parser: "@typescript-eslint/parser",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
ignores: [
|
||||||
|
"**/vendor/*",
|
||||||
|
"**/import-png.d.ts",
|
||||||
|
"**/chill.webpack.config.js",
|
||||||
|
"**/var/*",
|
||||||
|
"**/docker/*",
|
||||||
|
"**/node_modules/*",
|
||||||
|
"**/public/build/*"
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
plugins: {
|
||||||
|
prettier: eslintPluginPrettier,
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
"prettier/prettier": "error",
|
||||||
|
// override/add rules settings here, such as:
|
||||||
|
"vue/multi-word-component-names": "off",
|
||||||
|
"@typescript-eslint/no-require-imports": "off",
|
||||||
|
"@typescript-eslint/ban-ts-comment": "off"
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
40
package.json
40
package.json
@@ -6,33 +6,41 @@
|
|||||||
"@apidevtools/swagger-cli": "^4.0.4",
|
"@apidevtools/swagger-cli": "^4.0.4",
|
||||||
"@babel/core": "^7.20.5",
|
"@babel/core": "^7.20.5",
|
||||||
"@babel/preset-env": "^7.20.2",
|
"@babel/preset-env": "^7.20.2",
|
||||||
"@ckeditor/ckeditor5-build-classic": "^41.4.2",
|
"@ckeditor/ckeditor5-vue": "^7.3.0",
|
||||||
"@ckeditor/ckeditor5-dev-translations": "^40.2.0",
|
"@eslint/js": "^9.14.0",
|
||||||
"@ckeditor/ckeditor5-dev-utils": "^40.2.0",
|
"@hotwired/stimulus": "^3.0.0",
|
||||||
"@ckeditor/ckeditor5-dev-webpack-plugin": "^31.1.13",
|
"@luminateone/eslint-baseline": "^1.0.9",
|
||||||
"@ckeditor/ckeditor5-markdown-gfm": "^41.4.2",
|
"@symfony/stimulus-bridge": "^3.2.0",
|
||||||
"@ckeditor/ckeditor5-theme-lark": "^41.4.2",
|
"@symfony/ux-translator": "file:vendor/symfony/ux-translator/assets",
|
||||||
"@ckeditor/ckeditor5-vue": "^5.1.0",
|
|
||||||
"@symfony/webpack-encore": "^4.1.0",
|
"@symfony/webpack-encore": "^4.1.0",
|
||||||
"@tsconfig/node14": "^1.0.1",
|
"@tsconfig/node20": "^20.1.4",
|
||||||
"@types/dompurify": "^3.0.5",
|
"@types/dompurify": "^3.0.5",
|
||||||
|
"@types/eslint__js": "^8.42.3",
|
||||||
|
"@typescript-eslint/parser": "^8.12.2",
|
||||||
"bindings": "^1.5.0",
|
"bindings": "^1.5.0",
|
||||||
"bootstrap": "5.2.3",
|
"bootstrap": "5.2.3",
|
||||||
"chokidar": "^3.5.1",
|
"chokidar": "^3.5.1",
|
||||||
|
"ckeditor5": "^44.1.0",
|
||||||
"dompurify": "^3.1.0",
|
"dompurify": "^3.1.0",
|
||||||
|
"eslint": "^9.14.0",
|
||||||
|
"eslint-config-prettier": "^9.1.0",
|
||||||
|
"eslint-plugin-prettier": "^5.2.1",
|
||||||
|
"eslint-plugin-vue": "^9.30.0",
|
||||||
"fork-awesome": "^1.1.7",
|
"fork-awesome": "^1.1.7",
|
||||||
|
"intl-messageformat": "^10.5.11",
|
||||||
"jquery": "^3.6.0",
|
"jquery": "^3.6.0",
|
||||||
"marked": "^12.0.1",
|
|
||||||
"node-sass": "^8.0.0",
|
"node-sass": "^8.0.0",
|
||||||
"popper.js": "^1.16.1",
|
"popper.js": "^1.16.1",
|
||||||
"postcss-loader": "^7.0.2",
|
"postcss-loader": "^7.0.2",
|
||||||
|
"prettier": "^3.3.3",
|
||||||
"raw-loader": "^4.0.2",
|
"raw-loader": "^4.0.2",
|
||||||
"sass-loader": "^14.0.0",
|
"sass-loader": "^14.0.0",
|
||||||
"select2": "^4.0.13",
|
"select2": "^4.0.13",
|
||||||
"select2-bootstrap-theme": "0.1.0-beta.10",
|
"select2-bootstrap-theme": "0.1.0-beta.10",
|
||||||
"style-loader": "^3.3.1",
|
"style-loader": "^3.3.1",
|
||||||
"ts-loader": "^9.3.1",
|
"ts-loader": "^9.3.1",
|
||||||
"typescript": "^5.4.5",
|
"typescript": "^5.6.3",
|
||||||
|
"typescript-eslint": "^8.13.0",
|
||||||
"vue-loader": "^17.0.0",
|
"vue-loader": "^17.0.0",
|
||||||
"webpack": "^5.75.0",
|
"webpack": "^5.75.0",
|
||||||
"webpack-cli": "^5.0.1"
|
"webpack-cli": "^5.0.1"
|
||||||
@@ -45,11 +53,13 @@
|
|||||||
"@fullcalendar/timegrid": "^6.1.4",
|
"@fullcalendar/timegrid": "^6.1.4",
|
||||||
"@fullcalendar/vue3": "^6.1.4",
|
"@fullcalendar/vue3": "^6.1.4",
|
||||||
"@popperjs/core": "^2.9.2",
|
"@popperjs/core": "^2.9.2",
|
||||||
|
"@tsconfig/node20": "^20.1.4",
|
||||||
"@types/dompurify": "^3.0.5",
|
"@types/dompurify": "^3.0.5",
|
||||||
"@types/leaflet": "^1.9.3",
|
"@types/leaflet": "^1.9.3",
|
||||||
"bootstrap-icons": "^1.11.3",
|
"bootstrap-icons": "^1.11.3",
|
||||||
"dropzone": "^5.7.6",
|
"dropzone": "^5.7.6",
|
||||||
"es6-promise": "^4.2.8",
|
"es6-promise": "^4.2.8",
|
||||||
|
"intl-messageformat": "^10.5.11",
|
||||||
"leaflet": "^1.7.1",
|
"leaflet": "^1.7.1",
|
||||||
"marked": "^12.0.2",
|
"marked": "^12.0.2",
|
||||||
"masonry-layout": "^4.2.2",
|
"masonry-layout": "^4.2.2",
|
||||||
@@ -63,13 +73,19 @@
|
|||||||
"vuex": "^4.0.0"
|
"vuex": "^4.0.0"
|
||||||
},
|
},
|
||||||
"browserslist": [
|
"browserslist": [
|
||||||
"Firefox ESR"
|
"defaults and fully supports es6-module and not dead"
|
||||||
],
|
],
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev-server": "encore dev-server",
|
"dev-server": "encore dev-server",
|
||||||
"dev": "encore dev",
|
"dev": "encore dev",
|
||||||
"watch": "encore dev --watch",
|
"watch": "encore dev --watch",
|
||||||
"build": "encore production --progress"
|
"build": "encore production --progress",
|
||||||
|
"specs-build": "yaml-merge src/Bundle/ChillMainBundle/chill.api.specs.yaml src/Bundle/ChillPersonBundle/chill.api.specs.yaml src/Bundle/ChillCalendarBundle/chill.api.specs.yaml src/Bundle/ChillThirdPartyBundle/chill.api.specs.yaml src/Bundle/ChillDocStoreBundle/chill.api.specs.yaml> templates/api/specs.yaml",
|
||||||
|
"specs-validate": "swagger-cli validate templates/api/specs.yaml",
|
||||||
|
"specs-create-dir": "mkdir -p templates/api",
|
||||||
|
"specs": "yarn run specs-create-dir && yarn run specs-build && yarn run specs-validate",
|
||||||
|
"version": "node --version",
|
||||||
|
"eslint": "npx eslint-baseline --fix \"src/**/*.{js,ts,vue}\""
|
||||||
},
|
},
|
||||||
"private": true
|
"private": true
|
||||||
}
|
}
|
||||||
|
@@ -20,6 +20,10 @@ return static function (RectorConfig $rectorConfig): void {
|
|||||||
__DIR__ . '/src',
|
__DIR__ . '/src',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
$rectorConfig->skip([
|
||||||
|
\Rector\Php55\Rector\String_\StringClassNameToClassConstantRector::class => __DIR__ . 'src/Bundle/ChillMainBundle/Service/Notifier/LegacyOvhCloudFactory.php'
|
||||||
|
]);
|
||||||
|
|
||||||
$rectorConfig->symfonyContainerXml(__DIR__ . '/var/cache/dev/test/App_KernelTestDebugContainer.xml ');
|
$rectorConfig->symfonyContainerXml(__DIR__ . '/var/cache/dev/test/App_KernelTestDebugContainer.xml ');
|
||||||
$rectorConfig->symfonyContainerPhp(__DIR__ . '/tests/symfony-container.php');
|
$rectorConfig->symfonyContainerPhp(__DIR__ . '/tests/symfony-container.php');
|
||||||
|
|
||||||
|
@@ -84,11 +84,6 @@ class Activity implements AccompanyingPeriodLinkedWithSocialIssuesEntityInterfac
|
|||||||
*/
|
*/
|
||||||
#[Assert\Valid(traverse: true)]
|
#[Assert\Valid(traverse: true)]
|
||||||
#[ORM\ManyToMany(targetEntity: StoredObject::class, cascade: ['persist'])]
|
#[ORM\ManyToMany(targetEntity: StoredObject::class, cascade: ['persist'])]
|
||||||
#[ORM\JoinTable(
|
|
||||||
name: 'activity_storedobject',
|
|
||||||
joinColumns: new ORM\JoinColumn(name: 'activity_id', referencedColumnName: 'id', nullable: false),
|
|
||||||
inverseJoinColumns: new ORM\InverseJoinColumn(name: 'storedobject_id', referencedColumnName: 'id', unique: true, nullable: false)
|
|
||||||
)]
|
|
||||||
private Collection $documents;
|
private Collection $documents;
|
||||||
|
|
||||||
#[ORM\Column(type: \Doctrine\DBAL\Types\Types::TIME_MUTABLE, nullable: true)]
|
#[ORM\Column(type: \Doctrine\DBAL\Types\Types::TIME_MUTABLE, nullable: true)]
|
||||||
|
@@ -55,6 +55,10 @@ final readonly class PersonHavingActivityBetweenDateFilter implements ExportElem
|
|||||||
.' AND '
|
.' AND '
|
||||||
.'(person_person_having_activity.id = person.id OR person MEMBER OF activity_person_having_activity.persons)');
|
.'(person_person_having_activity.id = person.id OR person MEMBER OF activity_person_having_activity.persons)');
|
||||||
|
|
||||||
|
if (\in_array('activity', $qb->getAllAliases(), true)) {
|
||||||
|
$sqb->andWhere('activity_person_having_activity.id = activity.id');
|
||||||
|
}
|
||||||
|
|
||||||
if (isset($data['reasons']) && [] !== $data['reasons']) {
|
if (isset($data['reasons']) && [] !== $data['reasons']) {
|
||||||
// add clause activity reason
|
// add clause activity reason
|
||||||
$sqb->join('activity_person_having_activity.reasons', 'reasons_person_having_activity');
|
$sqb->join('activity_person_having_activity.reasons', 'reasons_person_having_activity');
|
||||||
|
@@ -11,6 +11,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Chill\ActivityBundle\Menu;
|
namespace Chill\ActivityBundle\Menu;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Entity\Activity;
|
||||||
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
||||||
use Chill\MainBundle\Routing\LocalMenuBuilderInterface;
|
use Chill\MainBundle\Routing\LocalMenuBuilderInterface;
|
||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
@@ -23,22 +24,30 @@ 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,
|
||||||
|
private readonly \Doctrine\Persistence\ManagerRegistry $managerRegistry,
|
||||||
|
) {}
|
||||||
|
|
||||||
public function buildMenu($menuId, MenuItem $menu, array $parameters)
|
public function buildMenu($menuId, MenuItem $menu, array $parameters)
|
||||||
{
|
{
|
||||||
$period = $parameters['accompanyingCourse'];
|
$period = $parameters['accompanyingCourse'];
|
||||||
|
|
||||||
|
$activities = $this->managerRegistry->getManager()->getRepository(Activity::class)->findBy(
|
||||||
|
['accompanyingPeriod' => $period]
|
||||||
|
);
|
||||||
|
|
||||||
if (
|
if (
|
||||||
AccompanyingPeriod::STEP_DRAFT !== $period->getStep()
|
AccompanyingPeriod::STEP_DRAFT !== $period->getStep()
|
||||||
&& $this->security->isGranted(ActivityVoter::SEE, $period)
|
&& $this->security->isGranted(ActivityVoter::SEE, $period)
|
||||||
) {
|
) {
|
||||||
$menu->addChild($this->translator->trans('Activity'), [
|
$menu->addChild($this->translator->trans('Activities'), [
|
||||||
'route' => 'chill_activity_activity_list',
|
'route' => 'chill_activity_activity_list',
|
||||||
'routeParameters' => [
|
'routeParameters' => [
|
||||||
'accompanying_period_id' => $period->getId(),
|
'accompanying_period_id' => $period->getId(),
|
||||||
], ])
|
], ])
|
||||||
->setExtras(['order' => 40]);
|
->setExtras(['order' => 40, 'counter' => count($activities) > 0 ? count($activities) : null]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -11,6 +11,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
namespace Chill\ActivityBundle\Menu;
|
namespace Chill\ActivityBundle\Menu;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityACLAwareRepositoryInterface;
|
||||||
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
||||||
use Chill\MainBundle\Routing\LocalMenuBuilderInterface;
|
use Chill\MainBundle\Routing\LocalMenuBuilderInterface;
|
||||||
use Chill\PersonBundle\Entity\Person;
|
use Chill\PersonBundle\Entity\Person;
|
||||||
@@ -23,13 +24,20 @@ use Symfony\Contracts\Translation\TranslatorInterface;
|
|||||||
*/
|
*/
|
||||||
final readonly class PersonMenuBuilder implements LocalMenuBuilderInterface
|
final readonly class PersonMenuBuilder implements LocalMenuBuilderInterface
|
||||||
{
|
{
|
||||||
public function __construct(private AuthorizationCheckerInterface $authorizationChecker, private TranslatorInterface $translator) {}
|
public function __construct(
|
||||||
|
private readonly ActivityACLAwareRepositoryInterface $activityACLAwareRepository,
|
||||||
|
private AuthorizationCheckerInterface $authorizationChecker,
|
||||||
|
private TranslatorInterface $translator,
|
||||||
|
) {}
|
||||||
|
|
||||||
public function buildMenu($menuId, MenuItem $menu, array $parameters)
|
public function buildMenu($menuId, MenuItem $menu, array $parameters)
|
||||||
{
|
{
|
||||||
/** @var Person $person */
|
/** @var Person $person */
|
||||||
$person = $parameters['person'];
|
$person = $parameters['person'];
|
||||||
|
|
||||||
|
|
||||||
|
$count = $this->activityACLAwareRepository->countByPerson($person, ActivityVoter::SEE);
|
||||||
|
|
||||||
if ($this->authorizationChecker->isGranted(ActivityVoter::SEE, $person)) {
|
if ($this->authorizationChecker->isGranted(ActivityVoter::SEE, $person)) {
|
||||||
$menu->addChild(
|
$menu->addChild(
|
||||||
$this->translator->trans('Activities'),
|
$this->translator->trans('Activities'),
|
||||||
@@ -38,7 +46,7 @@ final readonly class PersonMenuBuilder implements LocalMenuBuilderInterface
|
|||||||
'routeParameters' => ['person_id' => $person->getId()],
|
'routeParameters' => ['person_id' => $person->getId()],
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
->setExtra('order', 201);
|
->setExtras(['order' => 201, 'counter' => $count > 0 ? $count : null]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -15,10 +15,13 @@ use Chill\ActivityBundle\Entity\Activity;
|
|||||||
use Chill\ActivityBundle\Repository\ActivityRepository;
|
use Chill\ActivityBundle\Repository\ActivityRepository;
|
||||||
use Chill\MainBundle\Entity\Notification;
|
use Chill\MainBundle\Entity\Notification;
|
||||||
use Chill\MainBundle\Notification\NotificationHandlerInterface;
|
use Chill\MainBundle\Notification\NotificationHandlerInterface;
|
||||||
|
use Chill\MainBundle\Templating\TranslatableStringHelperInterface;
|
||||||
|
use Symfony\Component\Translation\TranslatableMessage;
|
||||||
|
use Symfony\Contracts\Translation\TranslatableInterface;
|
||||||
|
|
||||||
final readonly class ActivityNotificationHandler implements NotificationHandlerInterface
|
final readonly class ActivityNotificationHandler implements NotificationHandlerInterface
|
||||||
{
|
{
|
||||||
public function __construct(private ActivityRepository $activityRepository) {}
|
public function __construct(private ActivityRepository $activityRepository, private TranslatableStringHelperInterface $translatableStringHelper) {}
|
||||||
|
|
||||||
public function getTemplate(Notification $notification, array $options = []): string
|
public function getTemplate(Notification $notification, array $options = []): string
|
||||||
{
|
{
|
||||||
@@ -37,4 +40,30 @@ final readonly class ActivityNotificationHandler implements NotificationHandlerI
|
|||||||
{
|
{
|
||||||
return Activity::class === $notification->getRelatedEntityClass();
|
return Activity::class === $notification->getRelatedEntityClass();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getTitle(Notification $notification, array $options = []): TranslatableInterface
|
||||||
|
{
|
||||||
|
if (null === $activity = $this->getRelatedEntity($notification)) {
|
||||||
|
return new TranslatableMessage('activity.deleted');
|
||||||
|
}
|
||||||
|
|
||||||
|
return new TranslatableMessage('activity.title', [
|
||||||
|
'date' => $activity->getDate(),
|
||||||
|
'type' => $this->translatableStringHelper->localize($activity->getActivityType()->getName()),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getAssociatedPersons(Notification $notification, array $options = []): array
|
||||||
|
{
|
||||||
|
if (null === $activity = $this->getRelatedEntity($notification)) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $activity->getPersonsAssociated();
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getRelatedEntity(Notification $notification): ?Activity
|
||||||
|
{
|
||||||
|
return $this->activityRepository->find($notification->getRelatedEntityId());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -120,3 +120,34 @@ li.document-list-item {
|
|||||||
vertical-align: baseline;
|
vertical-align: baseline;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.badge-activity-type-simple {
|
||||||
|
@extend .badge;
|
||||||
|
display: inline-block;
|
||||||
|
margin: 0.2rem 0;
|
||||||
|
padding-left: 0;
|
||||||
|
padding-right: 0.5rem;
|
||||||
|
|
||||||
|
border-left: 20px groove #9acd32;
|
||||||
|
border-radius: $badge-border-radius;
|
||||||
|
|
||||||
|
color: black;
|
||||||
|
font-weight: normal;
|
||||||
|
font-size: unset;
|
||||||
|
max-width: 100%;
|
||||||
|
background-color: $gray-100;
|
||||||
|
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
text-indent: 5px hanging;
|
||||||
|
text-align: left;
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
margin-right: 3px;
|
||||||
|
position: relative;
|
||||||
|
left: -0.5px;
|
||||||
|
font-family: ForkAwesome;
|
||||||
|
content: '\f04b';
|
||||||
|
color: #9acd32;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@@ -1 +1 @@
|
|||||||
require('./chillactivity.scss');
|
require("./chillactivity.scss");
|
||||||
|
@@ -1,21 +1,21 @@
|
|||||||
<template>
|
<template>
|
||||||
<concerned-groups v-if="hasPerson"></concerned-groups>
|
<concerned-groups v-if="hasPerson" />
|
||||||
<social-issues-acc v-if="hasSocialIssues"></social-issues-acc>
|
<social-issues-acc v-if="hasSocialIssues" />
|
||||||
<location v-if="hasLocation"></location>
|
<location v-if="hasLocation" />
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import ConcernedGroups from './components/ConcernedGroups.vue';
|
import ConcernedGroups from "./components/ConcernedGroups.vue";
|
||||||
import SocialIssuesAcc from './components/SocialIssuesAcc.vue';
|
import SocialIssuesAcc from "./components/SocialIssuesAcc.vue";
|
||||||
import Location from './components/Location.vue';
|
import Location from "./components/Location.vue";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "App",
|
name: "App",
|
||||||
props: ['hasSocialIssues', 'hasLocation', 'hasPerson'],
|
props: ["hasSocialIssues", "hasLocation", "hasPerson", "isSimpleEditor"],
|
||||||
components: {
|
components: {
|
||||||
ConcernedGroups,
|
ConcernedGroups,
|
||||||
SocialIssuesAcc,
|
SocialIssuesAcc,
|
||||||
Location
|
Location,
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
@@ -1,38 +1,39 @@
|
|||||||
import { getSocialIssues } from 'ChillPersonAssets/vuejs/AccompanyingCourse/api.js';
|
import { getSocialIssues } from "ChillPersonAssets/vuejs/AccompanyingCourse/api.js";
|
||||||
import { fetchResults } from 'ChillMainAssets/lib/api/apiMethods';
|
import { fetchResults } from "ChillMainAssets/lib/api/apiMethods";
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Load socialActions by socialIssue (id)
|
* Load socialActions by socialIssue (id)
|
||||||
*/
|
*/
|
||||||
const getSocialActionByIssue = (id) => {
|
const getSocialActionByIssue = (id) => {
|
||||||
const url = `/api/1.0/person/social/social-action/by-social-issue/${id}.json`;
|
const url = `/api/1.0/person/social/social-action/by-social-issue/${id}.json`;
|
||||||
return fetch(url)
|
return fetch(url).then((response) => {
|
||||||
.then(response => {
|
if (response.ok) {
|
||||||
if (response.ok) { return response.json(); }
|
return response.json();
|
||||||
throw Error('Error with request resource response');
|
}
|
||||||
});
|
throw Error("Error with request resource response");
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const getLocations = () => fetchResults('/api/1.0/main/location.json');
|
const getLocations = () => fetchResults("/api/1.0/main/location.json");
|
||||||
|
|
||||||
const getLocationTypes = () => fetchResults('/api/1.0/main/location-type.json');
|
const getLocationTypes = () => fetchResults("/api/1.0/main/location-type.json");
|
||||||
|
|
||||||
const getUserCurrentLocation =
|
|
||||||
() => fetch('/api/1.0/main/user-current-location.json')
|
|
||||||
.then(response => {
|
|
||||||
if (response.ok) { return response.json(); }
|
|
||||||
throw Error('Error with request resource response');
|
|
||||||
});
|
|
||||||
|
|
||||||
|
const getUserCurrentLocation = () =>
|
||||||
|
fetch("/api/1.0/main/user-current-location.json").then((response) => {
|
||||||
|
if (response.ok) {
|
||||||
|
return response.json();
|
||||||
|
}
|
||||||
|
throw Error("Error with request resource response");
|
||||||
|
});
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Load Location Type by defaultFor
|
* Load Location Type by defaultFor
|
||||||
* @param {string} entity - can be "person" or "thirdparty"
|
* @param {string} entity - can be "person" or "thirdparty"
|
||||||
*/
|
*/
|
||||||
const getLocationTypeByDefaultFor = (entity) => {
|
const getLocationTypeByDefaultFor = (entity) => {
|
||||||
return getLocationTypes().then(results =>
|
return getLocationTypes().then(
|
||||||
results.filter(t => t.defaultFor === entity)[0]
|
(results) => results.filter((t) => t.defaultFor === entity)[0],
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -43,26 +44,27 @@ const getLocationTypeByDefaultFor = (entity) => {
|
|||||||
* @returns {Promise<T>}
|
* @returns {Promise<T>}
|
||||||
*/
|
*/
|
||||||
const postLocation = (body) => {
|
const postLocation = (body) => {
|
||||||
const url = `/api/1.0/main/location.json`;
|
const url = `/api/1.0/main/location.json`;
|
||||||
return fetch(url, {
|
return fetch(url, {
|
||||||
method: 'POST',
|
method: "POST",
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json;charset=utf-8'
|
"Content-Type": "application/json;charset=utf-8",
|
||||||
},
|
},
|
||||||
body: JSON.stringify(body)
|
body: JSON.stringify(body),
|
||||||
})
|
}).then((response) => {
|
||||||
.then(response => {
|
if (response.ok) {
|
||||||
if (response.ok) { return response.json(); }
|
return response.json();
|
||||||
throw Error('Error with request resource response');
|
}
|
||||||
});
|
throw Error("Error with request resource response");
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
export {
|
export {
|
||||||
getSocialIssues,
|
getSocialIssues,
|
||||||
getSocialActionByIssue,
|
getSocialActionByIssue,
|
||||||
getLocations,
|
getLocations,
|
||||||
getLocationTypes,
|
getLocationTypes,
|
||||||
getLocationTypeByDefaultFor,
|
getLocationTypeByDefaultFor,
|
||||||
postLocation,
|
postLocation,
|
||||||
getUserCurrentLocation
|
getUserCurrentLocation,
|
||||||
};
|
};
|
||||||
|
@@ -1,219 +1,257 @@
|
|||||||
<template>
|
<template>
|
||||||
<teleport to="#add-persons" v-if="isComponentVisible">
|
<teleport to="#add-persons" v-if="isComponentVisible">
|
||||||
|
<div class="flex-bloc concerned-groups" :class="getContext">
|
||||||
|
<persons-bloc
|
||||||
|
v-for="bloc in contextPersonsBlocs"
|
||||||
|
:key="bloc.key"
|
||||||
|
:bloc="bloc"
|
||||||
|
:bloc-width="getBlocWidth"
|
||||||
|
:set-persons-in-bloc="setPersonsInBloc"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-if="
|
||||||
|
getContext === 'accompanyingCourse' &&
|
||||||
|
suggestedEntities.length > 0
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<ul class="list-suggest add-items inline">
|
||||||
|
<li
|
||||||
|
v-for="(p, i) in suggestedEntities"
|
||||||
|
@click="addSuggestedEntity(p)"
|
||||||
|
:key="`suggestedEntities-${i}`"
|
||||||
|
>
|
||||||
|
<person-text v-if="p.type === 'person'" :person="p" />
|
||||||
|
<span v-else>{{ p.text }}</span>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="flex-bloc concerned-groups" :class="getContext">
|
<ul class="record_actions">
|
||||||
<persons-bloc
|
<li class="add-persons">
|
||||||
v-for="bloc in contextPersonsBlocs"
|
<add-persons
|
||||||
v-bind:key="bloc.key"
|
:buttonTitle="trans(ACTIVITY_ADD_PERSONS)"
|
||||||
v-bind:bloc="bloc"
|
:modalTitle="trans(ACTIVITY_ADD_PERSONS)"
|
||||||
v-bind:blocWidth="getBlocWidth"
|
v-bind:key="addPersons.key"
|
||||||
v-bind:setPersonsInBloc="setPersonsInBloc">
|
v-bind:options="addPersonsOptions"
|
||||||
</persons-bloc>
|
@addNewPersons="addNewPersons"
|
||||||
</div>
|
ref="addPersons"
|
||||||
<div v-if="getContext === 'accompanyingCourse' && suggestedEntities.length > 0">
|
>
|
||||||
<ul class="list-suggest add-items inline">
|
</add-persons>
|
||||||
<li v-for="(p, i) in suggestedEntities" @click="addSuggestedEntity(p)" :key="`suggestedEntities-${i}`">
|
</li>
|
||||||
<person-text v-if="p.type === 'person'" :person="p"></person-text>
|
</ul>
|
||||||
<span v-else>{{ p.text }}</span>
|
</teleport>
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<ul class="record_actions">
|
|
||||||
<li class="add-persons">
|
|
||||||
<add-persons
|
|
||||||
buttonTitle="activity.add_persons"
|
|
||||||
modalTitle="activity.add_persons"
|
|
||||||
v-bind:key="addPersons.key"
|
|
||||||
v-bind:options="addPersonsOptions"
|
|
||||||
@addNewPersons="addNewPersons"
|
|
||||||
ref="addPersons">
|
|
||||||
</add-persons>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
</teleport>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { mapState, mapGetters } from 'vuex';
|
import { mapState, mapGetters } from "vuex";
|
||||||
import AddPersons from 'ChillPersonAssets/vuejs/_components/AddPersons.vue';
|
import AddPersons from "ChillPersonAssets/vuejs/_components/AddPersons.vue";
|
||||||
import PersonsBloc from './ConcernedGroups/PersonsBloc.vue';
|
import PersonsBloc from "./ConcernedGroups/PersonsBloc.vue";
|
||||||
import PersonText from 'ChillPersonAssets/vuejs/_components/Entity/PersonText.vue';
|
import PersonText from "ChillPersonAssets/vuejs/_components/Entity/PersonText.vue";
|
||||||
|
import {
|
||||||
|
ACTIVITY_BLOC_PERSONS,
|
||||||
|
ACTIVITY_BLOC_PERSONS_ASSOCIATED,
|
||||||
|
ACTIVITY_BLOC_THIRDPARTY,
|
||||||
|
ACTIVITY_BLOC_USERS,
|
||||||
|
ACTIVITY_ADD_PERSONS,
|
||||||
|
trans,
|
||||||
|
} from "translator";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "ConcernedGroups",
|
name: "ConcernedGroups",
|
||||||
components: {
|
components: {
|
||||||
AddPersons,
|
AddPersons,
|
||||||
PersonsBloc,
|
PersonsBloc,
|
||||||
PersonText
|
PersonText,
|
||||||
},
|
},
|
||||||
data() {
|
setup() {
|
||||||
return {
|
return {
|
||||||
personsBlocs: [
|
trans,
|
||||||
{ key: 'persons',
|
ACTIVITY_ADD_PERSONS,
|
||||||
title: 'activity.bloc_persons',
|
};
|
||||||
persons: [],
|
},
|
||||||
included: false
|
data() {
|
||||||
|
return {
|
||||||
|
personsBlocs: [
|
||||||
|
{
|
||||||
|
key: "persons",
|
||||||
|
title: trans(ACTIVITY_BLOC_PERSONS),
|
||||||
|
persons: [],
|
||||||
|
included: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "personsAssociated",
|
||||||
|
title: trans(ACTIVITY_BLOC_PERSONS_ASSOCIATED),
|
||||||
|
persons: [],
|
||||||
|
included: window.activity
|
||||||
|
? window.activity.activityType.personsVisible !== 0
|
||||||
|
: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "personsNotAssociated",
|
||||||
|
title: "activity.bloc_persons_not_associated",
|
||||||
|
persons: [],
|
||||||
|
included: window.activity
|
||||||
|
? window.activity.activityType.personsVisible !== 0
|
||||||
|
: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "thirdparty",
|
||||||
|
title: trans(ACTIVITY_BLOC_THIRDPARTY),
|
||||||
|
persons: [],
|
||||||
|
included: window.activity
|
||||||
|
? window.activity.activityType.thirdPartiesVisible !== 0
|
||||||
|
: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
key: "users",
|
||||||
|
title: trans(ACTIVITY_BLOC_USERS),
|
||||||
|
persons: [],
|
||||||
|
included: window.activity
|
||||||
|
? window.activity.activityType.usersVisible !== 0
|
||||||
|
: true,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
addPersons: {
|
||||||
|
key: "activity",
|
||||||
},
|
},
|
||||||
{ key: 'personsAssociated',
|
};
|
||||||
title: 'activity.bloc_persons_associated',
|
},
|
||||||
persons: [],
|
computed: {
|
||||||
included: window.activity ? window.activity.activityType.personsVisible !== 0 : true
|
isComponentVisible() {
|
||||||
},
|
return window.activity
|
||||||
{ key: 'personsNotAssociated',
|
? window.activity.activityType.personsVisible !== 0 ||
|
||||||
title: 'activity.bloc_persons_not_associated',
|
window.activity.activityType.thirdPartiesVisible !== 0 ||
|
||||||
persons: [],
|
window.activity.activityType.usersVisible !== 0
|
||||||
included: window.activity ? window.activity.activityType.personsVisible !== 0 : true
|
: true;
|
||||||
},
|
},
|
||||||
{ key: 'thirdparty',
|
...mapState({
|
||||||
title: 'activity.bloc_thirdparty',
|
persons: (state) => state.activity.persons,
|
||||||
persons: [],
|
thirdParties: (state) => state.activity.thirdParties,
|
||||||
included: window.activity ? window.activity.activityType.thirdPartiesVisible !== 0 : true
|
users: (state) => state.activity.users,
|
||||||
},
|
accompanyingCourse: (state) => state.activity.accompanyingPeriod,
|
||||||
{ key: 'users',
|
}),
|
||||||
title: 'activity.bloc_users',
|
...mapGetters(["suggestedEntities"]),
|
||||||
persons: [],
|
getContext() {
|
||||||
included: window.activity ? window.activity.activityType.usersVisible !== 0 : true
|
return this.accompanyingCourse ? "accompanyingCourse" : "person";
|
||||||
},
|
},
|
||||||
],
|
contextPersonsBlocs() {
|
||||||
addPersons: {
|
return this.personsBlocs.filter((bloc) => bloc.included !== false);
|
||||||
key: 'activity'
|
},
|
||||||
}
|
addPersonsOptions() {
|
||||||
}
|
let optionsType = [];
|
||||||
},
|
if (window.activity) {
|
||||||
computed: {
|
if (window.activity.activityType.personsVisible !== 0) {
|
||||||
isComponentVisible() {
|
optionsType.push("person");
|
||||||
return window.activity
|
}
|
||||||
? (window.activity.activityType.personsVisible !== 0 || window.activity.activityType.thirdPartiesVisible !== 0 || window.activity.activityType.usersVisible !== 0)
|
if (window.activity.activityType.thirdPartiesVisible !== 0) {
|
||||||
: true
|
optionsType.push("thirdparty");
|
||||||
},
|
}
|
||||||
...mapState({
|
if (window.activity.activityType.usersVisible !== 0) {
|
||||||
persons: state => state.activity.persons,
|
optionsType.push("user");
|
||||||
thirdParties: state => state.activity.thirdParties,
|
}
|
||||||
users: state => state.activity.users,
|
|
||||||
accompanyingCourse: state => state.activity.accompanyingPeriod
|
|
||||||
}),
|
|
||||||
...mapGetters([
|
|
||||||
'suggestedEntities'
|
|
||||||
]),
|
|
||||||
getContext() {
|
|
||||||
return (this.accompanyingCourse) ? "accompanyingCourse" : "person";
|
|
||||||
},
|
|
||||||
contextPersonsBlocs() {
|
|
||||||
return this.personsBlocs.filter(bloc => bloc.included !== false);
|
|
||||||
},
|
|
||||||
addPersonsOptions() {
|
|
||||||
let optionsType = [];
|
|
||||||
if (window.activity) {
|
|
||||||
if (window.activity.activityType.personsVisible !== 0) {
|
|
||||||
optionsType.push('person')
|
|
||||||
}
|
|
||||||
if (window.activity.activityType.thirdPartiesVisible !== 0) {
|
|
||||||
optionsType.push('thirdparty')
|
|
||||||
}
|
|
||||||
if (window.activity.activityType.usersVisible !== 0) {
|
|
||||||
optionsType.push('user')
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
optionsType = ['person', 'thirdparty', 'user'];
|
|
||||||
}
|
|
||||||
return {
|
|
||||||
type: optionsType,
|
|
||||||
priority: null,
|
|
||||||
uniq: false,
|
|
||||||
button: {
|
|
||||||
size: 'btn-sm'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
getBlocWidth() {
|
|
||||||
return Math.round(100/(this.contextPersonsBlocs.length)) + '%';
|
|
||||||
}
|
|
||||||
},
|
|
||||||
mounted() {
|
|
||||||
this.setPersonsInBloc();
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
setPersonsInBloc() {
|
|
||||||
let groups;
|
|
||||||
if (this.accompanyingCourse) {
|
|
||||||
groups = this.splitPersonsInGroups();
|
|
||||||
}
|
|
||||||
this.personsBlocs.forEach(bloc => {
|
|
||||||
if (this.accompanyingCourse) {
|
|
||||||
switch (bloc.key) {
|
|
||||||
case 'personsAssociated':
|
|
||||||
bloc.persons = groups.personsAssociated;
|
|
||||||
bloc.included = true;
|
|
||||||
break;
|
|
||||||
case 'personsNotAssociated':
|
|
||||||
bloc.persons = groups.personsNotAssociated;
|
|
||||||
bloc.included = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
switch (bloc.key) {
|
optionsType = ["person", "thirdparty", "user"];
|
||||||
case 'persons':
|
|
||||||
bloc.persons = this.persons;
|
|
||||||
bloc.included = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
switch (bloc.key) {
|
return {
|
||||||
case 'thirdparty':
|
type: optionsType,
|
||||||
bloc.persons = this.thirdParties;
|
priority: null,
|
||||||
break;
|
uniq: false,
|
||||||
case 'users':
|
button: {
|
||||||
bloc.persons = this.users;
|
size: "btn-sm",
|
||||||
break;
|
},
|
||||||
}
|
};
|
||||||
}, groups);
|
},
|
||||||
},
|
getBlocWidth() {
|
||||||
splitPersonsInGroups() {
|
return Math.round(100 / this.contextPersonsBlocs.length) + "%";
|
||||||
let personsAssociated = [];
|
},
|
||||||
let personsNotAssociated = this.persons;
|
},
|
||||||
let participations = this.getCourseParticipations();
|
mounted() {
|
||||||
this.persons.forEach(person => {
|
|
||||||
participations.forEach(participation => {
|
|
||||||
if (person.id === participation.id) {
|
|
||||||
//console.log(person.id);
|
|
||||||
personsAssociated.push(person);
|
|
||||||
personsNotAssociated = personsNotAssociated.filter(p => p !== person);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
return {
|
|
||||||
'personsAssociated': personsAssociated,
|
|
||||||
'personsNotAssociated': personsNotAssociated
|
|
||||||
};
|
|
||||||
},
|
|
||||||
getCourseParticipations() {
|
|
||||||
let participations = [];
|
|
||||||
this.accompanyingCourse.participations.forEach(participation => {
|
|
||||||
if (!participation.endDate) {
|
|
||||||
participations.push(participation.person);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return participations;
|
|
||||||
},
|
|
||||||
addNewPersons({ selected, modal }) {
|
|
||||||
console.log('@@@ CLICK button addNewPersons', selected);
|
|
||||||
selected.forEach((item) => {
|
|
||||||
this.$store.dispatch('addPersonsInvolved', item);
|
|
||||||
}, this
|
|
||||||
);
|
|
||||||
this.$refs.addPersons.resetSearch(); // to cast child method
|
|
||||||
modal.showModal = false;
|
|
||||||
this.setPersonsInBloc();
|
|
||||||
},
|
|
||||||
addSuggestedEntity(person) {
|
|
||||||
this.$store.dispatch('addPersonsInvolved', { result: person, type: 'person' });
|
|
||||||
this.setPersonsInBloc();
|
this.setPersonsInBloc();
|
||||||
},
|
},
|
||||||
}
|
methods: {
|
||||||
}
|
setPersonsInBloc() {
|
||||||
|
let groups;
|
||||||
|
if (this.accompanyingCourse) {
|
||||||
|
groups = this.splitPersonsInGroups();
|
||||||
|
}
|
||||||
|
this.personsBlocs.forEach((bloc) => {
|
||||||
|
if (this.accompanyingCourse) {
|
||||||
|
switch (bloc.key) {
|
||||||
|
case "personsAssociated":
|
||||||
|
bloc.persons = groups.personsAssociated;
|
||||||
|
bloc.included = true;
|
||||||
|
break;
|
||||||
|
case "personsNotAssociated":
|
||||||
|
bloc.persons = groups.personsNotAssociated;
|
||||||
|
bloc.included = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
switch (bloc.key) {
|
||||||
|
case "persons":
|
||||||
|
bloc.persons = this.persons;
|
||||||
|
bloc.included = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
switch (bloc.key) {
|
||||||
|
case "thirdparty":
|
||||||
|
bloc.persons = this.thirdParties;
|
||||||
|
break;
|
||||||
|
case "users":
|
||||||
|
bloc.persons = this.users;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}, groups);
|
||||||
|
},
|
||||||
|
splitPersonsInGroups() {
|
||||||
|
let personsAssociated = [];
|
||||||
|
let personsNotAssociated = this.persons;
|
||||||
|
let participations = this.getCourseParticipations();
|
||||||
|
this.persons.forEach((person) => {
|
||||||
|
participations.forEach((participation) => {
|
||||||
|
if (person.id === participation.id) {
|
||||||
|
//console.log(person.id);
|
||||||
|
personsAssociated.push(person);
|
||||||
|
personsNotAssociated = personsNotAssociated.filter(
|
||||||
|
(p) => p !== person,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
return {
|
||||||
|
personsAssociated: personsAssociated,
|
||||||
|
personsNotAssociated: personsNotAssociated,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
getCourseParticipations() {
|
||||||
|
let participations = [];
|
||||||
|
this.accompanyingCourse.participations.forEach((participation) => {
|
||||||
|
if (!participation.endDate) {
|
||||||
|
participations.push(participation.person);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return participations;
|
||||||
|
},
|
||||||
|
addNewPersons({ selected, modal }) {
|
||||||
|
console.log("@@@ CLICK button addNewPersons", selected);
|
||||||
|
selected.forEach((item) => {
|
||||||
|
this.$store.dispatch("addPersonsInvolved", item);
|
||||||
|
}, this);
|
||||||
|
this.$refs.addPersons.resetSearch(); // to cast child method
|
||||||
|
modal.showModal = false;
|
||||||
|
this.setPersonsInBloc();
|
||||||
|
},
|
||||||
|
addSuggestedEntity(person) {
|
||||||
|
this.$store.dispatch("addPersonsInvolved", {
|
||||||
|
result: person,
|
||||||
|
type: "person",
|
||||||
|
});
|
||||||
|
this.setPersonsInBloc();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped></style>
|
||||||
</style>
|
|
||||||
|
@@ -1,32 +1,30 @@
|
|||||||
<template>
|
<template>
|
||||||
<li>
|
<li>
|
||||||
<span :title="person.text" @click.prevent="$emit('remove', person)">
|
<span :title="person.text" @click.prevent="$emit('remove', person)">
|
||||||
<span class="chill_denomination">
|
<span class="chill_denomination">
|
||||||
<person-text :person="person" :isCut="true"></person-text>
|
<person-text :person="person" :is-cut="true" />
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
</li>
|
</li>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import PersonText from 'ChillPersonAssets/vuejs/_components/Entity/PersonText.vue';
|
import PersonText from "ChillPersonAssets/vuejs/_components/Entity/PersonText.vue";
|
||||||
|
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "PersonBadge",
|
name: "PersonBadge",
|
||||||
props: ['person'],
|
props: ["person"],
|
||||||
components: {
|
components: {
|
||||||
PersonText
|
PersonText,
|
||||||
},
|
},
|
||||||
// computed: {
|
// computed: {
|
||||||
// textCutted() {
|
// textCutted() {
|
||||||
// let more = (this.person.text.length > 15) ?'…' : '';
|
// let more = (this.person.text.length > 15) ?'…' : '';
|
||||||
// return this.person.text.slice(0,15) + more;
|
// return this.person.text.slice(0,15) + more;
|
||||||
// }
|
// }
|
||||||
// },
|
// },
|
||||||
emits: ['remove'],
|
emits: ["remove"],
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="css" scoped>
|
<style lang="css" scoped></style>
|
||||||
</style>
|
|
||||||
|
@@ -1,41 +1,39 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="item-bloc" :style="{ 'flex-basis': blocWidth }">
|
<div class="item-bloc" :style="{ 'flex-basis': blocWidth }">
|
||||||
<div class="item-row">
|
<div class="item-row">
|
||||||
<div class="item-col">
|
<div class="item-col">
|
||||||
<h4>{{ $t(bloc.title) }}</h4>
|
<h4>{{ $t(bloc.title) }}</h4>
|
||||||
</div>
|
</div>
|
||||||
<div class="item-col">
|
<div class="item-col">
|
||||||
<ul class="list-suggest remove-items">
|
<ul class="list-suggest remove-items">
|
||||||
<person-badge
|
<person-badge
|
||||||
v-for="person in bloc.persons"
|
v-for="person in bloc.persons"
|
||||||
v-bind:key="person.id"
|
:key="person.id"
|
||||||
v-bind:person="person"
|
:person="person"
|
||||||
@remove="removePerson">
|
@remove="removePerson"
|
||||||
</person-badge>
|
/>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import PersonBadge from './PersonBadge.vue';
|
import PersonBadge from "./PersonBadge.vue";
|
||||||
export default {
|
export default {
|
||||||
name:"PersonsBloc",
|
name: "PersonsBloc",
|
||||||
components: {
|
components: {
|
||||||
PersonBadge
|
PersonBadge,
|
||||||
},
|
},
|
||||||
props: ['bloc', 'setPersonsInBloc', 'blocWidth'],
|
props: ["bloc", "setPersonsInBloc", "blocWidth"],
|
||||||
methods: {
|
methods: {
|
||||||
removePerson(item) {
|
removePerson(item) {
|
||||||
console.log('@@ CLICK remove person: item', item);
|
console.log("@@ CLICK remove person: item", item);
|
||||||
this.$store.dispatch('removePersonInvolved', item);
|
this.$store.dispatch("removePersonInvolved", item);
|
||||||
this.setPersonsInBloc();
|
this.setPersonsInBloc();
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss">
|
<style lang="scss"></style>
|
||||||
|
|
||||||
</style>
|
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
<teleport to="#location">
|
<teleport to="#location">
|
||||||
<div class="mb-3 row">
|
<div class="mb-3 row">
|
||||||
<label :class="locationClassList">
|
<label :class="locationClassList">
|
||||||
{{ $t("activity.location") }}
|
{{ trans(ACTIVITY_LOCATION) }}
|
||||||
</label>
|
</label>
|
||||||
<div class="col-sm-8">
|
<div class="col-sm-8">
|
||||||
<VueMultiselect
|
<VueMultiselect
|
||||||
@@ -13,18 +13,17 @@
|
|||||||
open-direction="top"
|
open-direction="top"
|
||||||
:multiple="false"
|
:multiple="false"
|
||||||
:searchable="true"
|
:searchable="true"
|
||||||
:placeholder="$t('activity.choose_location')"
|
:placeholder="trans(ACTIVITY_CHOOSE_LOCATION)"
|
||||||
:custom-label="customLabel"
|
:custom-label="customLabel"
|
||||||
:select-label="$t('multiselect.select_label')"
|
:select-label="trans(MULTISELECT_SELECT_LABEL)"
|
||||||
:deselect-label="$t('multiselect.deselect_label')"
|
:deselect-label="trans(MULTISELECT_DESELECT_LABEL)"
|
||||||
:selected-label="$t('multiselect.selected_label')"
|
:selected-label="trans(MULTISELECT_SELECTED_LABEL)"
|
||||||
:options="availableLocations"
|
:options="availableLocations"
|
||||||
group-values="locations"
|
group-values="locations"
|
||||||
group-label="locationGroup"
|
group-label="locationGroup"
|
||||||
v-model="location"
|
v-model="location"
|
||||||
>
|
/>
|
||||||
</VueMultiselect>
|
<new-location v-bind:available-locations="availableLocations" />
|
||||||
<new-location v-bind:availableLocations="availableLocations"></new-location>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</teleport>
|
</teleport>
|
||||||
@@ -34,6 +33,14 @@
|
|||||||
import { mapState, mapGetters } from "vuex";
|
import { mapState, mapGetters } from "vuex";
|
||||||
import VueMultiselect from "vue-multiselect";
|
import VueMultiselect from "vue-multiselect";
|
||||||
import NewLocation from "./Location/NewLocation.vue";
|
import NewLocation from "./Location/NewLocation.vue";
|
||||||
|
import {
|
||||||
|
trans,
|
||||||
|
ACTIVITY_LOCATION,
|
||||||
|
ACTIVITY_CHOOSE_LOCATION,
|
||||||
|
MULTISELECT_SELECT_LABEL,
|
||||||
|
MULTISELECT_DESELECT_LABEL,
|
||||||
|
MULTISELECT_SELECTED_LABEL,
|
||||||
|
} from "translator";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "Location",
|
name: "Location",
|
||||||
@@ -41,11 +48,20 @@ export default {
|
|||||||
NewLocation,
|
NewLocation,
|
||||||
VueMultiselect,
|
VueMultiselect,
|
||||||
},
|
},
|
||||||
|
setup() {
|
||||||
|
return {
|
||||||
|
trans,
|
||||||
|
ACTIVITY_LOCATION,
|
||||||
|
ACTIVITY_CHOOSE_LOCATION,
|
||||||
|
MULTISELECT_SELECT_LABEL,
|
||||||
|
MULTISELECT_DESELECT_LABEL,
|
||||||
|
MULTISELECT_SELECTED_LABEL,
|
||||||
|
};
|
||||||
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
locationClassList:
|
locationClassList: `col-form-label col-sm-4 ${document.querySelector("input#chill_activitybundle_activity_location").getAttribute("required") ? "required" : ""}`,
|
||||||
`col-form-label col-sm-4 ${document.querySelector('input#chill_activitybundle_activity_location').getAttribute('required') ? 'required' : ''}`,
|
};
|
||||||
}
|
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(["activity", "availableLocations"]),
|
...mapState(["activity", "availableLocations"]),
|
||||||
@@ -61,16 +77,16 @@ export default {
|
|||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
labelAccompanyingCourseLocation(value) {
|
labelAccompanyingCourseLocation(value) {
|
||||||
return `${value.address.text} (${value.locationType.title.fr})`
|
return `${value.address.text} (${value.locationType.title.fr})`;
|
||||||
},
|
},
|
||||||
customLabel(value) {
|
customLabel(value) {
|
||||||
return value.locationType
|
return value.locationType
|
||||||
? value.name
|
? value.name
|
||||||
? value.name === '__AccompanyingCourseLocation__'
|
? value.name === "__AccompanyingCourseLocation__"
|
||||||
? this.labelAccompanyingCourseLocation(value)
|
? this.labelAccompanyingCourseLocation(value)
|
||||||
: `${value.name} (${value.locationType.title.fr})`
|
: `${value.name} (${value.locationType.title.fr})`
|
||||||
: value.locationType.title.fr
|
: value.locationType.title.fr
|
||||||
: '';
|
: "";
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@@ -3,40 +3,65 @@
|
|||||||
<ul class="record_actions">
|
<ul class="record_actions">
|
||||||
<li>
|
<li>
|
||||||
<a class="btn btn-sm btn-create" @click="openModal">
|
<a class="btn btn-sm btn-create" @click="openModal">
|
||||||
{{ $t('activity.create_new_location') }}
|
{{ trans(ACTIVITY_CREATE_NEW_LOCATION) }}
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
<teleport to="body">
|
<teleport to="body">
|
||||||
<modal v-if="modal.showModal"
|
<modal
|
||||||
|
v-if="modal.showModal"
|
||||||
:modalDialogClass="modal.modalDialogClass"
|
:modalDialogClass="modal.modalDialogClass"
|
||||||
@close="modal.showModal = false">
|
@close="modal.showModal = false"
|
||||||
|
>
|
||||||
<template v-slot:header>
|
<template #header>
|
||||||
<h3 class="modal-title">{{ $t('activity.create_new_location') }}</h3>
|
<h3 class="modal-title">
|
||||||
|
{{ trans(ACTIVITY_CREATE_NEW_LOCATION) }}
|
||||||
|
</h3>
|
||||||
</template>
|
</template>
|
||||||
<template v-slot:body>
|
<template #body>
|
||||||
<form>
|
<form>
|
||||||
<div class="alert alert-warning" v-if="errors.length">
|
<div class="alert alert-warning" v-if="errors.length">
|
||||||
<ul>
|
<ul>
|
||||||
<li v-for="(e, i) in errors" :key="i">{{ e }}</li>
|
<li v-for="(e, i) in errors" :key="i">
|
||||||
|
{{ e }}
|
||||||
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-floating mb-3">
|
<div class="form-floating mb-3">
|
||||||
<select class="form-select form-select-lg" id="type" required v-model="selectType">
|
<select
|
||||||
<option selected disabled value="">{{ $t('activity.choose_location_type') }}</option>
|
class="form-select form-select-lg"
|
||||||
<option v-for="t in locationTypes" :value="t" :key="t.id">
|
id="type"
|
||||||
|
required
|
||||||
|
v-model="selectType"
|
||||||
|
>
|
||||||
|
<option selected disabled value="">
|
||||||
|
{{ trans(ACTIVITY_CHOOSE_LOCATION_TYPE) }}
|
||||||
|
</option>
|
||||||
|
<option
|
||||||
|
v-for="t in locationTypes"
|
||||||
|
:value="t"
|
||||||
|
:key="t.id"
|
||||||
|
>
|
||||||
{{ t.title.fr }}
|
{{ t.title.fr }}
|
||||||
</option>
|
</option>
|
||||||
</select>
|
</select>
|
||||||
<label>{{ $t('activity.location_fields.type') }}</label>
|
<label>{{
|
||||||
|
trans(ACTIVITY_LOCATION_FIELDS_TYPE)
|
||||||
|
}}</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="form-floating mb-3">
|
<div class="form-floating mb-3">
|
||||||
<input class="form-control form-control-lg" id="name" v-model="inputName" placeholder />
|
<input
|
||||||
<label for="name">{{ $t('activity.location_fields.name') }}</label>
|
class="form-control form-control-lg"
|
||||||
|
id="name"
|
||||||
|
v-model="inputName"
|
||||||
|
placeholder
|
||||||
|
/>
|
||||||
|
<label for="name">{{
|
||||||
|
trans(ACTIVITY_LOCATION_FIELDS_NAME)
|
||||||
|
}}</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<add-address
|
<add-address
|
||||||
@@ -44,43 +69,74 @@
|
|||||||
:options="addAddress.options"
|
:options="addAddress.options"
|
||||||
:addressChangedCallback="submitNewAddress"
|
:addressChangedCallback="submitNewAddress"
|
||||||
v-if="showAddAddress"
|
v-if="showAddAddress"
|
||||||
ref="addAddress">
|
ref="addAddress"
|
||||||
</add-address>
|
/>
|
||||||
|
|
||||||
<div class="form-floating mb-3" v-if="showContactData">
|
<div class="form-floating mb-3" v-if="showContactData">
|
||||||
<input class="form-control form-control-lg" id="phonenumber1" v-model="inputPhonenumber1" placeholder />
|
<input
|
||||||
<label for="phonenumber1">{{ $t('activity.location_fields.phonenumber1') }}</label>
|
class="form-control form-control-lg"
|
||||||
|
id="phonenumber1"
|
||||||
|
v-model="inputPhonenumber1"
|
||||||
|
placeholder
|
||||||
|
/>
|
||||||
|
<label for="phonenumber1">{{
|
||||||
|
trans(ACTIVITY_LOCATION_FIELDS_PHONENUMBER1)
|
||||||
|
}}</label>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-floating mb-3" v-if="hasPhonenumber1">
|
<div class="form-floating mb-3" v-if="hasPhonenumber1">
|
||||||
<input class="form-control form-control-lg" id="phonenumber2" v-model="inputPhonenumber2" placeholder />
|
<input
|
||||||
<label for="phonenumber2">{{ $t('activity.location_fields.phonenumber2') }}</label>
|
class="form-control form-control-lg"
|
||||||
|
id="phonenumber2"
|
||||||
|
v-model="inputPhonenumber2"
|
||||||
|
placeholder
|
||||||
|
/>
|
||||||
|
<label for="phonenumber2">{{
|
||||||
|
trans(ACTIVITY_LOCATION_FIELDS_PHONENUMBER2)
|
||||||
|
}}</label>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-floating mb-3" v-if="showContactData">
|
<div class="form-floating mb-3" v-if="showContactData">
|
||||||
<input class="form-control form-control-lg" id="email" v-model="inputEmail" placeholder />
|
<input
|
||||||
<label for="email">{{ $t('activity.location_fields.email') }}</label>
|
class="form-control form-control-lg"
|
||||||
|
id="email"
|
||||||
|
v-model="inputEmail"
|
||||||
|
placeholder
|
||||||
|
/>
|
||||||
|
<label for="email">{{
|
||||||
|
trans(ACTIVITY_LOCATION_FIELDS_EMAIL)
|
||||||
|
}}</label>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</form>
|
</form>
|
||||||
</template>
|
</template>
|
||||||
<template v-slot:footer>
|
<template #footer>
|
||||||
<button class="btn btn-save"
|
<button
|
||||||
|
class="btn btn-save"
|
||||||
@click.prevent="saveNewLocation"
|
@click.prevent="saveNewLocation"
|
||||||
>
|
>
|
||||||
{{ $t('action.save') }}
|
{{ trans(SAVE) }}
|
||||||
</button>
|
</button>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
</modal>
|
</modal>
|
||||||
</teleport>
|
</teleport>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import Modal from 'ChillMainAssets/vuejs/_components/Modal.vue';
|
import Modal from "ChillMainAssets/vuejs/_components/Modal.vue";
|
||||||
import AddAddress from "ChillMainAssets/vuejs/Address/components/AddAddress.vue";
|
import AddAddress from "ChillMainAssets/vuejs/Address/components/AddAddress.vue";
|
||||||
import { mapState } from "vuex";
|
import { mapState } from "vuex";
|
||||||
import { getLocationTypes } from "../../api";
|
import { getLocationTypes } from "../../api";
|
||||||
import { makeFetch } from 'ChillMainAssets/lib/api/apiMethods';
|
import { makeFetch } from "ChillMainAssets/lib/api/apiMethods";
|
||||||
|
import {
|
||||||
|
SAVE,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_EMAIL,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_PHONENUMBER1,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_PHONENUMBER2,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_NAME,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_TYPE,
|
||||||
|
ACTIVITY_CHOOSE_LOCATION_TYPE,
|
||||||
|
ACTIVITY_CREATE_NEW_LOCATION,
|
||||||
|
trans,
|
||||||
|
} from "translator";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "NewLocation",
|
name: "NewLocation",
|
||||||
@@ -88,7 +144,20 @@ export default {
|
|||||||
Modal,
|
Modal,
|
||||||
AddAddress,
|
AddAddress,
|
||||||
},
|
},
|
||||||
props: ['availableLocations'],
|
setup() {
|
||||||
|
return {
|
||||||
|
trans,
|
||||||
|
SAVE,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_EMAIL,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_PHONENUMBER1,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_PHONENUMBER2,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_NAME,
|
||||||
|
ACTIVITY_LOCATION_FIELDS_TYPE,
|
||||||
|
ACTIVITY_CHOOSE_LOCATION_TYPE,
|
||||||
|
ACTIVITY_CREATE_NEW_LOCATION,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
props: ["availableLocations"],
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
errors: [],
|
errors: [],
|
||||||
@@ -103,35 +172,42 @@ export default {
|
|||||||
locationTypes: [],
|
locationTypes: [],
|
||||||
modal: {
|
modal: {
|
||||||
showModal: false,
|
showModal: false,
|
||||||
modalDialogClass: "modal-dialog-scrollable modal-xl"
|
modalDialogClass: "modal-dialog-scrollable modal-xl",
|
||||||
},
|
},
|
||||||
addAddress: {
|
addAddress: {
|
||||||
options: {
|
options: {
|
||||||
button: {
|
button: {
|
||||||
text: { create: 'activity.create_address', edit: 'activity.edit_address' },
|
text: {
|
||||||
size: 'btn-sm'
|
create: "activity.create_address",
|
||||||
|
edit: "activity.edit_address",
|
||||||
|
},
|
||||||
|
size: "btn-sm",
|
||||||
|
},
|
||||||
|
title: {
|
||||||
|
create: "activity.create_address",
|
||||||
|
edit: "activity.edit_address",
|
||||||
},
|
},
|
||||||
title: { create: 'activity.create_address', edit: 'activity.edit_address' },
|
|
||||||
},
|
},
|
||||||
context: {
|
context: {
|
||||||
target: { //name, id
|
target: {
|
||||||
},
|
//name, id
|
||||||
|
},
|
||||||
edit: false,
|
edit: false,
|
||||||
addressId: null,
|
addressId: null,
|
||||||
defaults: window.addaddress
|
defaults: window.addaddress,
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
},
|
},
|
||||||
computed: {
|
computed: {
|
||||||
...mapState(['activity']),
|
...mapState(["activity"]),
|
||||||
selectType: {
|
selectType: {
|
||||||
get() {
|
get() {
|
||||||
return this.selected.type;
|
return this.selected.type;
|
||||||
},
|
},
|
||||||
set(value) {
|
set(value) {
|
||||||
this.selected.type = value;
|
this.selected.type = value;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
inputName: {
|
inputName: {
|
||||||
get() {
|
get() {
|
||||||
@@ -139,7 +215,7 @@ export default {
|
|||||||
},
|
},
|
||||||
set(value) {
|
set(value) {
|
||||||
this.selected.name = value;
|
this.selected.name = value;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
inputEmail: {
|
inputEmail: {
|
||||||
get() {
|
get() {
|
||||||
@@ -147,7 +223,7 @@ export default {
|
|||||||
},
|
},
|
||||||
set(value) {
|
set(value) {
|
||||||
this.selected.email = value;
|
this.selected.email = value;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
inputPhonenumber1: {
|
inputPhonenumber1: {
|
||||||
get() {
|
get() {
|
||||||
@@ -155,7 +231,7 @@ export default {
|
|||||||
},
|
},
|
||||||
set(value) {
|
set(value) {
|
||||||
this.selected.phonenumber1 = value;
|
this.selected.phonenumber1 = value;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
inputPhonenumber2: {
|
inputPhonenumber2: {
|
||||||
get() {
|
get() {
|
||||||
@@ -163,15 +239,18 @@ export default {
|
|||||||
},
|
},
|
||||||
set(value) {
|
set(value) {
|
||||||
this.selected.phonenumber2 = value;
|
this.selected.phonenumber2 = value;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
hasPhonenumber1() {
|
hasPhonenumber1() {
|
||||||
return this.selected.phonenumber1 !== null && this.selected.phonenumber1 !== "";
|
return (
|
||||||
|
this.selected.phonenumber1 !== null &&
|
||||||
|
this.selected.phonenumber1 !== ""
|
||||||
|
);
|
||||||
},
|
},
|
||||||
showAddAddress() {
|
showAddAddress() {
|
||||||
let cond = false;
|
let cond = false;
|
||||||
if (this.selected.type) {
|
if (this.selected.type) {
|
||||||
if (this.selected.type.addressRequired !== 'never') {
|
if (this.selected.type.addressRequired !== "never") {
|
||||||
cond = true;
|
cond = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -180,7 +259,7 @@ export default {
|
|||||||
showContactData() {
|
showContactData() {
|
||||||
let cond = false;
|
let cond = false;
|
||||||
if (this.selected.type) {
|
if (this.selected.type) {
|
||||||
if (this.selected.type.contactData !== 'never') {
|
if (this.selected.type.contactData !== "never") {
|
||||||
cond = true;
|
cond = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -195,28 +274,39 @@ export default {
|
|||||||
let cond = true;
|
let cond = true;
|
||||||
this.errors = [];
|
this.errors = [];
|
||||||
if (!this.selected.type) {
|
if (!this.selected.type) {
|
||||||
this.errors.push('Type de localisation requis');
|
this.errors.push("Type de localisation requis");
|
||||||
cond = false;
|
cond = false;
|
||||||
} else {
|
} else {
|
||||||
if (this.selected.type.addressRequired === 'required' && !this.selected.addressId) {
|
if (
|
||||||
this.errors.push('Adresse requise');
|
this.selected.type.addressRequired === "required" &&
|
||||||
|
!this.selected.addressId
|
||||||
|
) {
|
||||||
|
this.errors.push("Adresse requise");
|
||||||
cond = false;
|
cond = false;
|
||||||
}
|
}
|
||||||
if (this.selected.type.contactData === 'required' && !this.selected.phonenumber1) {
|
if (
|
||||||
this.errors.push('Numéro de téléphone requis');
|
this.selected.type.contactData === "required" &&
|
||||||
|
!this.selected.phonenumber1
|
||||||
|
) {
|
||||||
|
this.errors.push("Numéro de téléphone requis");
|
||||||
cond = false;
|
cond = false;
|
||||||
}
|
}
|
||||||
if (this.selected.type.contactData === 'required' && !this.selected.email) {
|
if (
|
||||||
this.errors.push('Adresse email requise');
|
this.selected.type.contactData === "required" &&
|
||||||
|
!this.selected.email
|
||||||
|
) {
|
||||||
|
this.errors.push("Adresse email requise");
|
||||||
cond = false;
|
cond = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return cond;
|
return cond;
|
||||||
},
|
},
|
||||||
getLocationTypesList() {
|
getLocationTypesList() {
|
||||||
getLocationTypes().then(results => {
|
getLocationTypes().then((results) => {
|
||||||
this.locationTypes = results.filter(t => t.availableForUsers === true);
|
this.locationTypes = results.filter(
|
||||||
})
|
(t) => t.availableForUsers === true,
|
||||||
|
);
|
||||||
|
});
|
||||||
},
|
},
|
||||||
openModal() {
|
openModal() {
|
||||||
this.modal.showModal = true;
|
this.modal.showModal = true;
|
||||||
@@ -224,11 +314,11 @@ export default {
|
|||||||
saveNewLocation() {
|
saveNewLocation() {
|
||||||
if (this.checkForm()) {
|
if (this.checkForm()) {
|
||||||
let body = {
|
let body = {
|
||||||
type: 'location',
|
type: "location",
|
||||||
name: this.selected.name,
|
name: this.selected.name,
|
||||||
locationType: {
|
locationType: {
|
||||||
id: this.selected.type.id,
|
id: this.selected.type.id,
|
||||||
type: 'location-type'
|
type: "location-type",
|
||||||
},
|
},
|
||||||
phonenumber1: this.selected.phonenumber1,
|
phonenumber1: this.selected.phonenumber1,
|
||||||
phonenumber2: this.selected.phonenumber2,
|
phonenumber2: this.selected.phonenumber2,
|
||||||
@@ -237,36 +327,36 @@ export default {
|
|||||||
if (this.selected.addressId) {
|
if (this.selected.addressId) {
|
||||||
body = Object.assign(body, {
|
body = Object.assign(body, {
|
||||||
address: {
|
address: {
|
||||||
id: this.selected.addressId
|
id: this.selected.addressId,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
makeFetch('POST', '/api/1.0/main/location.json', body)
|
makeFetch("POST", "/api/1.0/main/location.json", body)
|
||||||
.then(response => {
|
.then((response) => {
|
||||||
this.$store.dispatch('addAvailableLocationGroup', {
|
this.$store.dispatch("addAvailableLocationGroup", {
|
||||||
locationGroup: 'Localisations nouvellement créées',
|
locationGroup: "Localisations nouvellement créées",
|
||||||
locations: [response]
|
locations: [response],
|
||||||
});
|
});
|
||||||
this.$store.dispatch('updateLocation', response);
|
this.$store.dispatch("updateLocation", response);
|
||||||
this.modal.showModal = false;
|
this.modal.showModal = false;
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
if (error.name === 'ValidationException') {
|
if (error.name === "ValidationException") {
|
||||||
for (let v of error.violations) {
|
for (let v of error.violations) {
|
||||||
this.errors.push(v);
|
this.errors.push(v);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
this.errors.push('An error occurred');
|
this.errors.push("An error occurred");
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
};
|
}
|
||||||
},
|
},
|
||||||
submitNewAddress(payload) {
|
submitNewAddress(payload) {
|
||||||
this.selected.addressId = payload.addressId;
|
this.selected.addressId = payload.addressId;
|
||||||
this.addAddress.context.addressId = payload.addressId;
|
this.addAddress.context.addressId = payload.addressId;
|
||||||
this.addAddress.context.edit = true;
|
this.addAddress.context.edit = true;
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
@@ -1,228 +1,280 @@
|
|||||||
<template>
|
<template>
|
||||||
<teleport to="#social-issues-acc">
|
<teleport to="#social-issues-acc">
|
||||||
|
<div class="mb-3 row">
|
||||||
<div class="mb-3 row">
|
<div class="col-4">
|
||||||
<div class="col-4">
|
<label :class="socialIssuesClassList">{{
|
||||||
<label :class="socialIssuesClassList">{{ $t('activity.social_issues') }}</label>
|
trans(ACTIVITY_SOCIAL_ISSUES)
|
||||||
</div>
|
}}</label>
|
||||||
<div class="col-8">
|
|
||||||
|
|
||||||
<check-social-issue
|
|
||||||
v-for="issue in socialIssuesList"
|
|
||||||
:key="issue.id"
|
|
||||||
:issue="issue"
|
|
||||||
:selection="socialIssuesSelected"
|
|
||||||
@updateSelected="updateIssuesSelected">
|
|
||||||
</check-social-issue>
|
|
||||||
|
|
||||||
<div class="my-3">
|
|
||||||
<VueMultiselect
|
|
||||||
name="otherIssues"
|
|
||||||
label="text"
|
|
||||||
track-by="id"
|
|
||||||
open-direction="bottom"
|
|
||||||
:close-on-select="true"
|
|
||||||
:preserve-search="false"
|
|
||||||
:reset-after="true"
|
|
||||||
:hide-selected="true"
|
|
||||||
:taggable="false"
|
|
||||||
:multiple="false"
|
|
||||||
:searchable="true"
|
|
||||||
:allow-empty="true"
|
|
||||||
:show-labels="false"
|
|
||||||
:loading="issueIsLoading"
|
|
||||||
:placeholder="$t('activity.choose_other_social_issue')"
|
|
||||||
:options="socialIssuesOther"
|
|
||||||
@select="addIssueInList">
|
|
||||||
</VueMultiselect>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div class="col-8">
|
||||||
|
<check-social-issue
|
||||||
|
v-for="issue in socialIssuesList"
|
||||||
|
:key="issue.id"
|
||||||
|
:issue="issue"
|
||||||
|
:selection="socialIssuesSelected"
|
||||||
|
@updateSelected="updateIssuesSelected"
|
||||||
|
>
|
||||||
|
</check-social-issue>
|
||||||
|
|
||||||
</div>
|
<div class="my-3">
|
||||||
</div>
|
<VueMultiselect
|
||||||
|
name="otherIssues"
|
||||||
<div class="mb-3 row">
|
label="text"
|
||||||
<div class="col-4">
|
track-by="id"
|
||||||
<label :class="socialActionsClassList">{{ $t('activity.social_actions') }}</label>
|
open-direction="bottom"
|
||||||
</div>
|
:close-on-select="true"
|
||||||
<div class="col-8">
|
:preserve-search="false"
|
||||||
|
:reset-after="true"
|
||||||
<div v-if="actionIsLoading === true">
|
:hide-selected="true"
|
||||||
<i class="chill-green fa fa-circle-o-notch fa-spin fa-lg"></i>
|
:taggable="false"
|
||||||
|
:multiple="false"
|
||||||
|
:searchable="true"
|
||||||
|
:allow-empty="true"
|
||||||
|
:show-labels="false"
|
||||||
|
:loading="issueIsLoading"
|
||||||
|
:placeholder="trans(ACTIVITY_CHOOSE_OTHER_SOCIAL_ISSUE)"
|
||||||
|
:options="socialIssuesOther"
|
||||||
|
@select="addIssueInList"
|
||||||
|
>
|
||||||
|
</VueMultiselect>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<span v-else-if="socialIssuesSelected.length === 0" class="inline-choice chill-no-data-statement mt-3">
|
<div class="mb-3 row">
|
||||||
{{ $t('activity.select_first_a_social_issue') }}
|
<div class="col-4">
|
||||||
</span>
|
<label :class="socialActionsClassList">{{
|
||||||
|
trans(ACTIVITY_SOCIAL_ACTIONS)
|
||||||
|
}}</label>
|
||||||
|
</div>
|
||||||
|
<div class="col-8">
|
||||||
|
<div v-if="actionIsLoading === true">
|
||||||
|
<i
|
||||||
|
class="chill-green fa fa-circle-o-notch fa-spin fa-lg"
|
||||||
|
></i>
|
||||||
|
</div>
|
||||||
|
|
||||||
<template v-else-if="socialActionsList.length > 0">
|
<span
|
||||||
<check-social-action
|
v-else-if="socialIssuesSelected.length === 0"
|
||||||
v-if="socialIssuesSelected.length || socialActionsSelected.length"
|
class="inline-choice chill-no-data-statement mt-3"
|
||||||
v-for="action in socialActionsList"
|
>
|
||||||
:key="action.id"
|
{{ trans(ACTIVITY_SELECT_FIRST_A_SOCIAL_ISSUE) }}
|
||||||
:action="action"
|
</span>
|
||||||
:selection="socialActionsSelected"
|
|
||||||
@updateSelected="updateActionsSelected">
|
|
||||||
</check-social-action>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<span v-else-if="actionAreLoaded && socialActionsList.length === 0" class="inline-choice chill-no-data-statement mt-3">
|
<template
|
||||||
{{ $t('activity.social_action_list_empty') }}
|
v-else-if="
|
||||||
</span>
|
socialActionsList.length > 0 &&
|
||||||
|
(socialIssuesSelected.length ||
|
||||||
|
socialActionsSelected.length)
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
id="actionsList"
|
||||||
|
v-for="group in socialActionsList"
|
||||||
|
:key="group.issue"
|
||||||
|
>
|
||||||
|
<span class="badge bg-chill-l-gray text-dark">{{
|
||||||
|
group.issue
|
||||||
|
}}</span>
|
||||||
|
<check-social-action
|
||||||
|
v-for="action in group.actions"
|
||||||
|
:key="action.id"
|
||||||
|
:action="action"
|
||||||
|
:selection="socialActionsSelected"
|
||||||
|
@updateSelected="updateActionsSelected"
|
||||||
|
>
|
||||||
|
</check-social-action>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<span
|
||||||
</div>
|
v-else-if="
|
||||||
</div>
|
actionAreLoaded && socialActionsList.length === 0
|
||||||
|
"
|
||||||
</teleport>
|
class="inline-choice chill-no-data-statement mt-3"
|
||||||
|
>
|
||||||
|
{{ trans(ACTIVITY_SOCIAL_ACTION_LIST_EMPTY) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</teleport>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import VueMultiselect from 'vue-multiselect';
|
import VueMultiselect from "vue-multiselect";
|
||||||
import CheckSocialIssue from './SocialIssuesAcc/CheckSocialIssue.vue';
|
import CheckSocialIssue from "./SocialIssuesAcc/CheckSocialIssue.vue";
|
||||||
import CheckSocialAction from './SocialIssuesAcc/CheckSocialAction.vue';
|
import CheckSocialAction from "./SocialIssuesAcc/CheckSocialAction.vue";
|
||||||
import { getSocialIssues, getSocialActionByIssue } from '../api.js';
|
import { getSocialIssues, getSocialActionByIssue } from "../api.js";
|
||||||
|
import {
|
||||||
|
ACTIVITY_SOCIAL_ACTION_LIST_EMPTY,
|
||||||
|
ACTIVITY_SELECT_FIRST_A_SOCIAL_ISSUE,
|
||||||
|
ACTIVITY_SOCIAL_ACTIONS,
|
||||||
|
ACTIVITY_SOCIAL_ISSUES,
|
||||||
|
ACTIVITY_CHOOSE_OTHER_SOCIAL_ISSUE,
|
||||||
|
trans,
|
||||||
|
} from "translator";
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: "SocialIssuesAcc",
|
name: "SocialIssuesAcc",
|
||||||
components: {
|
components: {
|
||||||
CheckSocialIssue,
|
CheckSocialIssue,
|
||||||
CheckSocialAction,
|
CheckSocialAction,
|
||||||
VueMultiselect
|
VueMultiselect,
|
||||||
},
|
},
|
||||||
data() {
|
setup() {
|
||||||
return {
|
return {
|
||||||
issueIsLoading: false,
|
trans,
|
||||||
actionIsLoading: false,
|
ACTIVITY_SOCIAL_ACTION_LIST_EMPTY,
|
||||||
actionAreLoaded: false,
|
ACTIVITY_SELECT_FIRST_A_SOCIAL_ISSUE,
|
||||||
socialIssuesClassList:
|
ACTIVITY_SOCIAL_ACTIONS,
|
||||||
`col-form-label ${document.querySelector('input#chill_activitybundle_activity_socialIssues').getAttribute('required') ? 'required' : ''}`,
|
ACTIVITY_SOCIAL_ISSUES,
|
||||||
socialActionsClassList:
|
ACTIVITY_CHOOSE_OTHER_SOCIAL_ISSUE,
|
||||||
`col-form-label ${document.querySelector('input#chill_activitybundle_activity_socialActions').getAttribute('required') ? 'required' : ''}`,
|
};
|
||||||
}
|
},
|
||||||
},
|
data() {
|
||||||
computed: {
|
return {
|
||||||
socialIssuesList() {
|
issueIsLoading: false,
|
||||||
return this.$store.state.activity.accompanyingPeriod.socialIssues;
|
actionIsLoading: false,
|
||||||
},
|
actionAreLoaded: false,
|
||||||
socialIssuesSelected() {
|
socialIssuesClassList: `col-form-label ${document.querySelector("input#chill_activitybundle_activity_socialIssues").getAttribute("required") ? "required" : ""}`,
|
||||||
return this.$store.state.activity.socialIssues;
|
socialActionsClassList: `col-form-label ${document.querySelector("input#chill_activitybundle_activity_socialActions").getAttribute("required") ? "required" : ""}`,
|
||||||
},
|
};
|
||||||
socialIssuesOther() {
|
},
|
||||||
return this.$store.state.socialIssuesOther;
|
computed: {
|
||||||
},
|
socialIssuesList() {
|
||||||
socialActionsList() {
|
return this.$store.state.activity.accompanyingPeriod.socialIssues;
|
||||||
return this.$store.getters.socialActionsListSorted;
|
},
|
||||||
},
|
socialIssuesSelected() {
|
||||||
socialActionsSelected() {
|
return this.$store.state.activity.socialIssues;
|
||||||
return this.$store.state.activity.socialActions;
|
},
|
||||||
}
|
socialIssuesOther() {
|
||||||
},
|
return this.$store.state.socialIssuesOther;
|
||||||
mounted() {
|
},
|
||||||
/* Load others issues in multiselect
|
socialActionsList() {
|
||||||
*/
|
return this.$store.getters.socialActionsListSorted;
|
||||||
this.issueIsLoading = true;
|
},
|
||||||
this.actionAreLoaded = false;
|
socialActionsSelected() {
|
||||||
getSocialIssues().then(response => new Promise((resolve, reject) => {
|
return this.$store.state.activity.socialActions;
|
||||||
this.$store.commit('updateIssuesOther', response.results);
|
},
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
/* Load other issues in multiselect */
|
||||||
|
this.issueIsLoading = true;
|
||||||
|
this.actionAreLoaded = false;
|
||||||
|
|
||||||
/* Add in list the issues already associated (if not yet listed)
|
getSocialIssues().then((response) => {
|
||||||
*/
|
/* Add issues to the store */
|
||||||
this.socialIssuesSelected.forEach(issue => {
|
this.$store.commit("updateIssuesOther", response);
|
||||||
if (this.socialIssuesList.filter(i => i.id === issue.id).length !== 1) {
|
|
||||||
this.$store.commit('addIssueInList', issue);
|
|
||||||
}
|
|
||||||
}, this);
|
|
||||||
|
|
||||||
/* Remove from multiselect the issues that are not yet in checkbox list
|
/* Add in list the issues already associated (if not yet listed) */
|
||||||
*/
|
this.socialIssuesSelected.forEach((issue) => {
|
||||||
this.socialIssuesList.forEach(issue => {
|
if (
|
||||||
this.$store.commit('removeIssueInOther', issue);
|
this.socialIssuesList.filter((i) => i.id === issue.id)
|
||||||
}, this);
|
.length !== 1
|
||||||
|
) {
|
||||||
|
this.$store.commit("addIssueInList", issue);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
/* Filter issues
|
/* Remove from multiselect the issues that are not yet in the checkbox list */
|
||||||
*/
|
this.socialIssuesList.forEach((issue) => {
|
||||||
this.$store.commit('filterList', 'issues');
|
this.$store.commit("removeIssueInOther", issue);
|
||||||
|
});
|
||||||
|
|
||||||
/* Add in list the actions already associated (if not yet listed)
|
/* Filter issues */
|
||||||
*/
|
this.$store.commit("filterList", "issues");
|
||||||
this.socialActionsSelected.forEach(action => {
|
|
||||||
this.$store.commit('addActionInList', action);
|
|
||||||
}, this);
|
|
||||||
|
|
||||||
/* Filter issues
|
/* Add in list the actions already associated (if not yet listed) */
|
||||||
*/
|
this.socialActionsSelected.forEach((action) => {
|
||||||
this.$store.commit('filterList', 'actions');
|
this.$store.commit("addActionInList", action);
|
||||||
|
});
|
||||||
|
|
||||||
this.issueIsLoading = false;
|
/* Filter actions */
|
||||||
this.actionAreLoaded = true;
|
this.$store.commit("filterList", "actions");
|
||||||
this.updateActionsList();
|
|
||||||
resolve();
|
|
||||||
}));
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
|
|
||||||
/* When choosing an issue in multiselect, add it in checkboxes (as selected),
|
this.issueIsLoading = false;
|
||||||
|
this.actionAreLoaded = true;
|
||||||
|
this.updateActionsList();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
/* When choosing an issue in multiselect, add it in checkboxes (as selected),
|
||||||
remove it from multiselect, and add socialActions concerned
|
remove it from multiselect, and add socialActions concerned
|
||||||
*/
|
*/
|
||||||
addIssueInList(value) {
|
addIssueInList(value) {
|
||||||
//console.log('addIssueInList', value);
|
//console.log('addIssueInList', value);
|
||||||
this.$store.commit('addIssueInList', value);
|
this.$store.commit("addIssueInList", value);
|
||||||
this.$store.commit('removeIssueInOther', value);
|
this.$store.commit("removeIssueInOther", value);
|
||||||
this.$store.dispatch('addIssueSelected', value);
|
this.$store.dispatch("addIssueSelected", value);
|
||||||
this.updateActionsList();
|
this.updateActionsList();
|
||||||
},
|
},
|
||||||
/* Update value for selected issues checkboxes
|
/* Update value for selected issues checkboxes
|
||||||
*/
|
*/
|
||||||
updateIssuesSelected(issues) {
|
updateIssuesSelected(issues) {
|
||||||
//console.log('updateIssuesSelected', issues);
|
//console.log('updateIssuesSelected', issues);
|
||||||
this.$store.dispatch('updateIssuesSelected', issues);
|
this.$store.dispatch("updateIssuesSelected", issues);
|
||||||
this.updateActionsList();
|
this.updateActionsList();
|
||||||
},
|
},
|
||||||
/* Update value for selected actions checkboxes
|
/* Update value for selected actions checkboxes
|
||||||
*/
|
*/
|
||||||
updateActionsSelected(actions) {
|
updateActionsSelected(actions) {
|
||||||
//console.log('updateActionsSelected', actions);
|
//console.log('updateActionsSelected', actions);
|
||||||
this.$store.dispatch('updateActionsSelected', actions);
|
this.$store.dispatch("updateActionsSelected", actions);
|
||||||
},
|
},
|
||||||
/* Add socialActions concerned: after reset, loop on each issue selected
|
/* Add socialActions concerned: after reset, loop on each issue selected
|
||||||
to get social actions concerned
|
to get social actions concerned
|
||||||
*/
|
*/
|
||||||
updateActionsList() {
|
updateActionsList() {
|
||||||
this.resetActionsList();
|
this.resetActionsList();
|
||||||
this.socialIssuesSelected.forEach(item => {
|
this.socialIssuesSelected.forEach((item) => {
|
||||||
|
this.actionIsLoading = true;
|
||||||
|
getSocialActionByIssue(item.id).then(
|
||||||
|
(actions) =>
|
||||||
|
new Promise((resolve) => {
|
||||||
|
actions.results.forEach((action) => {
|
||||||
|
this.$store.commit("addActionInList", action);
|
||||||
|
}, this);
|
||||||
|
|
||||||
this.actionIsLoading = true;
|
this.$store.commit("filterList", "actions");
|
||||||
getSocialActionByIssue(item.id)
|
|
||||||
.then(actions => new Promise((resolve, reject) => {
|
|
||||||
|
|
||||||
actions.results.forEach(action => {
|
this.actionIsLoading = false;
|
||||||
this.$store.commit('addActionInList', action);
|
this.actionAreLoaded = true;
|
||||||
}, this);
|
resolve();
|
||||||
|
}),
|
||||||
this.$store.commit('filterList', 'actions');
|
);
|
||||||
|
}, this);
|
||||||
this.actionIsLoading = false;
|
},
|
||||||
this.actionAreLoaded = true;
|
/* Reset socialActions List: flush list and restore selected actions
|
||||||
resolve();
|
*/
|
||||||
}));
|
resetActionsList() {
|
||||||
}, this);
|
this.$store.commit("resetActionsList");
|
||||||
},
|
this.actionAreLoaded = false;
|
||||||
/* Reset socialActions List: flush list and restore selected actions
|
this.socialActionsSelected.forEach((item) => {
|
||||||
*/
|
this.$store.commit("addActionInList", item);
|
||||||
resetActionsList() {
|
}, this);
|
||||||
this.$store.commit('resetActionsList');
|
},
|
||||||
this.actionAreLoaded = false;
|
},
|
||||||
this.socialActionsSelected.forEach(item => {
|
};
|
||||||
this.$store.commit('addActionInList', item);
|
|
||||||
}, this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style src="vue-multiselect/dist/vue-multiselect.css"></style>
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
span.multiselect__single {
|
@import "ChillMainAssets/module/bootstrap/shared";
|
||||||
display: none !important;
|
@import "ChillPersonAssets/chill/scss/mixins";
|
||||||
}
|
@import "ChillMainAssets/chill/scss/chill_variables";
|
||||||
|
|
||||||
|
span.multiselect__single {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
#actionsList {
|
||||||
|
border-radius: 0.5rem;
|
||||||
|
padding: 1rem;
|
||||||
|
margin: 0.5rem;
|
||||||
|
background-color: whitesmoke;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.badge {
|
||||||
|
margin-bottom: 0.5rem;
|
||||||
|
@include badge_social($social-issue-color);
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@@ -1,48 +1,53 @@
|
|||||||
<template>
|
<template>
|
||||||
<span class="inline-choice">
|
<span class="inline-choice">
|
||||||
<div class="form-check">
|
<div class="form-check">
|
||||||
|
<input
|
||||||
<input class="form-check-input"
|
class="form-check-input"
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
v-model="selected"
|
v-model="selected"
|
||||||
name="action"
|
name="action"
|
||||||
v-bind:id="action.id"
|
:id="action.id"
|
||||||
v-bind:value="action"
|
:value="action"
|
||||||
/>
|
/>
|
||||||
<label class="form-check-label" v-bind:for="action.id">
|
<label class="form-check-label" :for="action.id">
|
||||||
<span class="badge bg-light text-dark">{{ action.text }}</span>
|
<span class="badge bg-light text-dark" :title="action.text">{{
|
||||||
</label>
|
action.text
|
||||||
|
}}</span>
|
||||||
</div>
|
</label>
|
||||||
</span>
|
</div>
|
||||||
|
</span>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
name: "CheckSocialAction",
|
name: "CheckSocialAction",
|
||||||
props: [ 'action', 'selection' ],
|
props: ["action", "selection"],
|
||||||
emits: [ 'updateSelected' ],
|
emits: ["updateSelected"],
|
||||||
computed: {
|
computed: {
|
||||||
selected: {
|
selected: {
|
||||||
set(value) {
|
set(value) {
|
||||||
this.$emit('updateSelected', value);
|
this.$emit("updateSelected", value);
|
||||||
},
|
},
|
||||||
get() {
|
get() {
|
||||||
return this.selection;
|
return this.selection;
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import 'ChillMainAssets/module/bootstrap/shared';
|
@import "ChillMainAssets/module/bootstrap/shared";
|
||||||
@import 'ChillPersonAssets/chill/scss/mixins';
|
@import "ChillPersonAssets/chill/scss/mixins";
|
||||||
@import 'ChillMainAssets/chill/scss/chill_variables';
|
@import "ChillMainAssets/chill/scss/chill_variables";
|
||||||
span.badge {
|
span.badge {
|
||||||
@include badge_social($social-action-color);
|
@include badge_social($social-action-color);
|
||||||
font-size: 95%;
|
font-size: 95%;
|
||||||
margin-bottom: 5px;
|
margin-bottom: 5px;
|
||||||
margin-right: 1em;
|
margin-right: 1em;
|
||||||
|
max-width: 100%; /* Adjust as needed */
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@@ -1,44 +1,45 @@
|
|||||||
<template>
|
<template>
|
||||||
<span class="inline-choice">
|
<span class="inline-choice">
|
||||||
<div class="form-check">
|
<div class="form-check">
|
||||||
|
<input
|
||||||
<input class="form-check-input"
|
class="form-check-input"
|
||||||
type="checkbox"
|
type="checkbox"
|
||||||
v-model="selected"
|
v-model="selected"
|
||||||
name="issue"
|
name="issue"
|
||||||
v-bind:id="issue.id"
|
:id="issue.id"
|
||||||
v-bind:value="issue"
|
:value="issue"
|
||||||
/>
|
/>
|
||||||
<label class="form-check-label" v-bind:for="issue.id">
|
<label class="form-check-label" :for="issue.id">
|
||||||
<span class="badge bg-chill-l-gray text-dark">{{ issue.text }}</span>
|
<span class="badge bg-chill-l-gray text-dark">{{
|
||||||
</label>
|
issue.text
|
||||||
|
}}</span>
|
||||||
</div>
|
</label>
|
||||||
</span>
|
</div>
|
||||||
|
</span>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
export default {
|
export default {
|
||||||
name: "CheckSocialIssue",
|
name: "CheckSocialIssue",
|
||||||
props: [ 'issue', 'selection' ],
|
props: ["issue", "selection"],
|
||||||
emits: [ 'updateSelected' ],
|
emits: ["updateSelected"],
|
||||||
computed: {
|
computed: {
|
||||||
selected: {
|
selected: {
|
||||||
set(value) {
|
set(value) {
|
||||||
this.$emit('updateSelected', value);
|
this.$emit("updateSelected", value);
|
||||||
},
|
},
|
||||||
get() {
|
get() {
|
||||||
return this.selection;
|
return this.selection;
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
@import 'ChillMainAssets/module/bootstrap/shared';
|
@import "ChillMainAssets/module/bootstrap/shared";
|
||||||
@import 'ChillPersonAssets/chill/scss/mixins';
|
@import "ChillPersonAssets/chill/scss/mixins";
|
||||||
@import 'ChillMainAssets/chill/scss/chill_variables';
|
@import "ChillMainAssets/chill/scss/chill_variables";
|
||||||
span.badge {
|
span.badge {
|
||||||
@include badge_social($social-issue-color);
|
@include badge_social($social-issue-color);
|
||||||
font-size: 95%;
|
font-size: 95%;
|
||||||
|
@@ -1,45 +1,44 @@
|
|||||||
import { personMessages } from 'ChillPersonAssets/vuejs/_js/i18n'
|
import { personMessages } from "ChillPersonAssets/vuejs/_js/i18n";
|
||||||
import { multiSelectMessages } from 'ChillMainAssets/vuejs/_js/i18n'
|
import { multiSelectMessages } from "ChillMainAssets/vuejs/_js/i18n";
|
||||||
|
|
||||||
const activityMessages = {
|
const activityMessages = {
|
||||||
fr: {
|
fr: {
|
||||||
activity: {
|
activity: {
|
||||||
//
|
//
|
||||||
errors: "Le formulaire contient des erreurs",
|
errors: "Le formulaire contient des erreurs",
|
||||||
social_issues: "Problématiques sociales",
|
social_issues: "Problématiques sociales",
|
||||||
choose_other_social_issue: "Ajouter une autre problématique sociale...",
|
choose_other_social_issue: "Ajouter une autre problématique sociale...",
|
||||||
social_actions: "Actions d'accompagnement",
|
social_actions: "Actions d'accompagnement",
|
||||||
select_first_a_social_issue: "Sélectionnez d'abord une problématique sociale",
|
select_first_a_social_issue:
|
||||||
social_action_list_empty: "Aucune action sociale disponible",
|
"Sélectionnez d'abord une problématique sociale",
|
||||||
|
social_action_list_empty: "Aucune action sociale disponible",
|
||||||
|
|
||||||
//
|
//
|
||||||
add_persons: "Ajouter des personnes concernées",
|
add_persons: "Ajouter des personnes concernées",
|
||||||
bloc_persons: "Usagers",
|
bloc_persons: "Usagers",
|
||||||
bloc_persons_associated: "Usagers du parcours",
|
bloc_persons_associated: "Usagers du parcours",
|
||||||
bloc_persons_not_associated: "Tiers non-pro.",
|
bloc_persons_not_associated: "Tiers non-pro.",
|
||||||
bloc_thirdparty: "Tiers professionnels",
|
bloc_thirdparty: "Tiers professionnels",
|
||||||
bloc_users: "T(M)S",
|
bloc_users: "T(M)S",
|
||||||
|
|
||||||
//
|
//
|
||||||
location: "Localisation",
|
location: "Localisation",
|
||||||
choose_location: "Choisissez une localisation",
|
choose_location: "Choisissez une localisation",
|
||||||
choose_location_type: "Choisissez un type de localisation",
|
choose_location_type: "Choisissez un type de localisation",
|
||||||
create_new_location: "Créer une nouvelle localisation",
|
create_new_location: "Créer une nouvelle localisation",
|
||||||
location_fields: {
|
location_fields: {
|
||||||
name: "Nom",
|
name: "Nom",
|
||||||
type: "Type",
|
type: "Type",
|
||||||
phonenumber1: "Téléphone",
|
phonenumber1: "Téléphone",
|
||||||
phonenumber2: "Autre téléphone",
|
phonenumber2: "Autre téléphone",
|
||||||
email: "Adresse courriel",
|
email: "Adresse courriel",
|
||||||
},
|
},
|
||||||
create_address: 'Créer une adresse',
|
create_address: "Créer une adresse",
|
||||||
edit_address: "Modifier l'adresse"
|
edit_address: "Modifier l'adresse",
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
|
|
||||||
Object.assign(activityMessages.fr, personMessages.fr, multiSelectMessages.fr);
|
Object.assign(activityMessages.fr, personMessages.fr, multiSelectMessages.fr);
|
||||||
|
|
||||||
export {
|
export { activityMessages };
|
||||||
activityMessages
|
|
||||||
};
|
|
||||||
|
@@ -1,86 +1,89 @@
|
|||||||
import { createApp } from 'vue';
|
import { createApp } from "vue";
|
||||||
import { _createI18n } from 'ChillMainAssets/vuejs/_js/i18n'
|
import { _createI18n } from "ChillMainAssets/vuejs/_js/i18n";
|
||||||
import { activityMessages } from './i18n'
|
import { activityMessages } from "./i18n";
|
||||||
import store from './store'
|
import store from "./store";
|
||||||
import PickTemplate from 'ChillDocGeneratorAssets/vuejs/_components/PickTemplate.vue';
|
import PickTemplate from "ChillDocGeneratorAssets/vuejs/_components/PickTemplate.vue";
|
||||||
import {fetchTemplates} from 'ChillDocGeneratorAssets/api/pickTemplate.js';
|
import { fetchTemplates } from "ChillDocGeneratorAssets/api/pickTemplate.js";
|
||||||
|
|
||||||
import App from './App.vue';
|
import App from "./App.vue";
|
||||||
|
|
||||||
const i18n = _createI18n(activityMessages);
|
const i18n = _createI18n(activityMessages);
|
||||||
|
|
||||||
// app for activity
|
// app for activity
|
||||||
|
|
||||||
const hasSocialIssues = document.querySelector('#social-issues-acc') !== null;
|
const hasSocialIssues = document.querySelector("#social-issues-acc") !== null;
|
||||||
const hasLocation = document.querySelector('#location') !== null;
|
const hasLocation = document.querySelector("#location") !== null;
|
||||||
const hasPerson = document.querySelector('#add-persons') !== null;
|
const hasPerson = document.querySelector("#add-persons") !== null;
|
||||||
|
const isSimpleEditor = true;
|
||||||
|
|
||||||
const app = createApp({
|
const app = createApp({
|
||||||
template: `<app
|
template: `<app
|
||||||
:hasSocialIssues="hasSocialIssues"
|
:hasSocialIssues="hasSocialIssues"
|
||||||
:hasLocation="hasLocation"
|
:hasLocation="hasLocation"
|
||||||
:hasPerson="hasPerson"
|
:hasPerson="hasPerson"
|
||||||
|
:isSimpleEditor = "isSimpleEditor"
|
||||||
></app>`,
|
></app>`,
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
hasSocialIssues,
|
hasSocialIssues,
|
||||||
hasLocation,
|
hasLocation,
|
||||||
hasPerson,
|
hasPerson,
|
||||||
};
|
isSimpleEditor
|
||||||
}
|
};
|
||||||
|
},
|
||||||
})
|
})
|
||||||
.use(store)
|
.use(store)
|
||||||
.use(i18n)
|
.use(i18n)
|
||||||
.component('app', App)
|
.component("app", App)
|
||||||
.mount('#activity');
|
.mount("#activity");
|
||||||
|
|
||||||
|
|
||||||
// app for picking template
|
// app for picking template
|
||||||
|
|
||||||
const i18nGendoc = _createI18n({});
|
const i18nGendoc = _createI18n({});
|
||||||
|
|
||||||
document.querySelectorAll('div[data-docgen-template-picker]').forEach(el => {
|
document.querySelectorAll("div[data-docgen-template-picker]").forEach((el) => {
|
||||||
fetchTemplates(el.dataset.entityClass).then(templates => {
|
fetchTemplates(el.dataset.entityClass).then((templates) => {
|
||||||
const picker = {
|
const picker = {
|
||||||
template:
|
template:
|
||||||
'<pick-template :templates="this.templates" :preventDefaultMoveToGenerate="true" ' +
|
'<pick-template :templates="this.templates" :preventDefaultMoveToGenerate="true" ' +
|
||||||
':entityClass="faked" @go-to-generate-document="generateDoc"></pick-template>',
|
':entityClass="faked" @go-to-generate-document="generateDoc"></pick-template>',
|
||||||
components: {
|
components: {
|
||||||
PickTemplate,
|
PickTemplate,
|
||||||
},
|
},
|
||||||
data() {
|
data() {
|
||||||
return {
|
return {
|
||||||
templates: templates,
|
templates: templates,
|
||||||
entityId: el.dataset.entityId,
|
entityId: el.dataset.entityId,
|
||||||
}
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
generateDoc({event, link, template}) {
|
|
||||||
console.log('generateDoc');
|
|
||||||
console.log('link', link);
|
|
||||||
console.log('template', template);
|
|
||||||
|
|
||||||
let hiddenInput = document.querySelector("input[data-template-id]");
|
|
||||||
|
|
||||||
if (hiddenInput === null) {
|
|
||||||
console.error('hidden input not found');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
hiddenInput.value = template;
|
|
||||||
|
|
||||||
let form = document.querySelector('form[name="chill_activitybundle_activity"');
|
|
||||||
|
|
||||||
if (form === null) {
|
|
||||||
console.error('form not found');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
form.submit();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
createApp(picker).use(i18nGendoc).mount(el);
|
},
|
||||||
})
|
methods: {
|
||||||
|
generateDoc({ event, link, template }) {
|
||||||
|
console.log("generateDoc");
|
||||||
|
console.log("link", link);
|
||||||
|
console.log("template", template);
|
||||||
|
|
||||||
|
let hiddenInput = document.querySelector("input[data-template-id]");
|
||||||
|
|
||||||
|
if (hiddenInput === null) {
|
||||||
|
console.error("hidden input not found");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
hiddenInput.value = template;
|
||||||
|
|
||||||
|
let form = document.querySelector(
|
||||||
|
'form[name="chill_activitybundle_activity"',
|
||||||
|
);
|
||||||
|
|
||||||
|
if (form === null) {
|
||||||
|
console.error("form not found");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
form.submit();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
createApp(picker).use(i18nGendoc).mount(el);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@@ -1,350 +1,389 @@
|
|||||||
import 'es6-promise/auto';
|
import "es6-promise/auto";
|
||||||
import { createStore } from 'vuex';
|
import { createStore } from "vuex";
|
||||||
import { postLocation } from './api';
|
import { postLocation } from "./api";
|
||||||
import prepareLocations from './store.locations.js';
|
import prepareLocations from "./store.locations.js";
|
||||||
|
import { makeFetch } from "ChillMainAssets/lib/api/apiMethods";
|
||||||
|
|
||||||
const debug = process.env.NODE_ENV !== 'production';
|
const debug = process.env.NODE_ENV !== "production";
|
||||||
//console.log('window.activity', window.activity);
|
//console.log('window.activity', window.activity);
|
||||||
|
|
||||||
const addIdToValue = (string, id) => {
|
const addIdToValue = (string, id) => {
|
||||||
let array = string ? string.split(',') : [];
|
let array = string ? string.split(",") : [];
|
||||||
array.push(id.toString());
|
array.push(id.toString());
|
||||||
let str = array.join();
|
let str = array.join();
|
||||||
return str;
|
return str;
|
||||||
};
|
};
|
||||||
|
|
||||||
const removeIdFromValue = (string, id) => {
|
const removeIdFromValue = (string, id) => {
|
||||||
let array = string.split(',');
|
let array = string.split(",");
|
||||||
array = array.filter(el => el !== id.toString());
|
array = array.filter((el) => el !== id.toString());
|
||||||
let str = array.join();
|
let str = array.join();
|
||||||
return str;
|
return str;
|
||||||
};
|
};
|
||||||
|
|
||||||
const store = createStore({
|
const store = createStore({
|
||||||
strict: debug,
|
strict: debug,
|
||||||
state: {
|
state: {
|
||||||
activity: window.activity,
|
me: null,
|
||||||
socialIssuesOther: [],
|
activity: window.activity,
|
||||||
socialActionsList: [],
|
accompanyingPeriodWorks: [],
|
||||||
availableLocations: [],
|
socialIssuesOther: [],
|
||||||
|
socialActionsList: [],
|
||||||
|
availableLocations: [],
|
||||||
|
},
|
||||||
|
getters: {
|
||||||
|
suggestedEntities(state) {
|
||||||
|
if (
|
||||||
|
typeof state.activity.accompanyingPeriod === "undefined" ||
|
||||||
|
state.activity.accompanyingPeriod === null
|
||||||
|
) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
const allEntities = [
|
||||||
|
...store.getters.suggestedPersons,
|
||||||
|
...store.getters.suggestedRequestor,
|
||||||
|
...store.getters.suggestedUsers,
|
||||||
|
...store.getters.suggestedResources,
|
||||||
|
];
|
||||||
|
const uniqueIds = [
|
||||||
|
...new Set(allEntities.map((i) => `${i.type}-${i.id}`)),
|
||||||
|
];
|
||||||
|
return Array.from(
|
||||||
|
uniqueIds,
|
||||||
|
(id) => allEntities.filter((r) => `${r.type}-${r.id}` === id)[0],
|
||||||
|
);
|
||||||
},
|
},
|
||||||
getters: {
|
suggestedPersons(state) {
|
||||||
suggestedEntities(state) {
|
const existingPersonIds = state.activity.persons.map((p) => p.id);
|
||||||
if (typeof state.activity.accompanyingPeriod === "undefined" || state.activity.accompanyingPeriod === null) {
|
return state.activity.activityType.personsVisible === 0
|
||||||
return [];
|
? []
|
||||||
}
|
: state.activity.accompanyingPeriod.participations
|
||||||
const allEntities = [
|
.filter((p) => p.endDate === null)
|
||||||
...store.getters.suggestedPersons,
|
.map((p) => p.person)
|
||||||
...store.getters.suggestedRequestor,
|
.filter((p) => !existingPersonIds.includes(p.id));
|
||||||
...store.getters.suggestedUser,
|
},
|
||||||
...store.getters.suggestedResources,
|
suggestedRequestor(state) {
|
||||||
];
|
if (state.activity.accompanyingPeriod.requestor === null) {
|
||||||
const uniqueIds = [
|
return [];
|
||||||
...new Set(allEntities.map((i) => `${i.type}-${i.id}`)),
|
}
|
||||||
];
|
const existingPersonIds = state.activity.persons.map((p) => p.id);
|
||||||
return Array.from(
|
const existingThirdPartyIds = state.activity.thirdParties.map(
|
||||||
uniqueIds,
|
(p) => p.id,
|
||||||
(id) => allEntities.filter((r) => `${r.type}-${r.id}` === id)[0]
|
);
|
||||||
);
|
|
||||||
},
|
return [state.activity.accompanyingPeriod.requestor].filter(
|
||||||
suggestedPersons(state) {
|
(r) =>
|
||||||
const existingPersonIds = state.activity.persons.map((p) => p.id);
|
(r.type === "person" &&
|
||||||
return state.activity.activityType.personsVisible === 0
|
!existingPersonIds.includes(r.id) &&
|
||||||
? []
|
state.activity.activityType.personsVisible !== 0) ||
|
||||||
: state.activity.accompanyingPeriod.participations
|
(r.type === "thirdparty" &&
|
||||||
.filter((p) => p.endDate === null)
|
!existingThirdPartyIds.includes(r.id) &&
|
||||||
.map((p) => p.person)
|
state.activity.activityType.thirdPartiesVisible !== 0),
|
||||||
.filter((p) => !existingPersonIds.includes(p.id));
|
);
|
||||||
},
|
},
|
||||||
suggestedRequestor(state) {
|
suggestedUsers(state) {
|
||||||
if (state.activity.accompanyingPeriod.requestor === null) {
|
const existingUserIds = state.activity.users.map((p) => p.id);
|
||||||
return [];
|
let suggestedUsers =
|
||||||
}
|
state.activity.activityType.usersVisible === 0
|
||||||
const existingPersonIds = state.activity.persons.map((p) => p.id);
|
? []
|
||||||
const existingThirdPartyIds = state.activity.thirdParties.map(
|
: [state.activity.accompanyingPeriod.user].filter(
|
||||||
(p) => p.id
|
(u) => u !== null && !existingUserIds.includes(u.id),
|
||||||
);
|
);
|
||||||
|
|
||||||
return [state.activity.accompanyingPeriod.requestor].filter(
|
state.accompanyingPeriodWorks.forEach((work) => {
|
||||||
(r) =>
|
work.referrers.forEach((r) => {
|
||||||
(r.type === "person" &&
|
if (!existingUserIds.includes(r.id)) {
|
||||||
!existingPersonIds.includes(r.id) &&
|
suggestedUsers.push(r);
|
||||||
state.activity.activityType.personsVisible !== 0) ||
|
}
|
||||||
(r.type === "thirdparty" &&
|
});
|
||||||
!existingThirdPartyIds.includes(r.id) &&
|
});
|
||||||
state.activity.activityType.thirdPartiesVisible !== 0)
|
// Add the current user from the state
|
||||||
);
|
if (state.me && !existingUserIds.includes(state.me.id)) {
|
||||||
},
|
suggestedUsers.push(state.me);
|
||||||
suggestedUser(state) {
|
}
|
||||||
const existingUserIds = state.activity.users.map((p) => p.id);
|
// console.log("suggested users", suggestedUsers);
|
||||||
return state.activity.activityType.usersVisible === 0
|
|
||||||
? []
|
|
||||||
: [state.activity.accompanyingPeriod.user].filter(
|
|
||||||
(u) => u !== null && !existingUserIds.includes(u.id)
|
|
||||||
);
|
|
||||||
},
|
|
||||||
suggestedResources(state) {
|
|
||||||
const resources = state.activity.accompanyingPeriod.resources;
|
|
||||||
const existingPersonIds = state.activity.persons.map((p) => p.id);
|
|
||||||
const existingThirdPartyIds = state.activity.thirdParties.map(
|
|
||||||
(p) => p.id
|
|
||||||
);
|
|
||||||
return state.activity.accompanyingPeriod.resources
|
|
||||||
.map((r) => r.resource)
|
|
||||||
.filter(
|
|
||||||
(r) =>
|
|
||||||
(r.type === "person" &&
|
|
||||||
!existingPersonIds.includes(r.id) &&
|
|
||||||
state.activity.activityType.personsVisible !== 0) ||
|
|
||||||
(r.type === "thirdparty" &&
|
|
||||||
!existingThirdPartyIds.includes(r.id) &&
|
|
||||||
state.activity.activityType.thirdPartiesVisible !== 0)
|
|
||||||
);
|
|
||||||
},
|
|
||||||
socialActionsListSorted(state) {
|
|
||||||
return [ ...state.socialActionsList].sort((a, b) => a.ordering - b.ordering);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
mutations: {
|
|
||||||
// SocialIssueAcc
|
|
||||||
addIssueInList(state, issue) {
|
|
||||||
//console.log('add issue list', issue.id);
|
|
||||||
state.activity.accompanyingPeriod.socialIssues.push(issue);
|
|
||||||
},
|
|
||||||
addIssueSelected(state, issue) {
|
|
||||||
//console.log('add issue selected', issue.id);
|
|
||||||
state.activity.socialIssues.push(issue);
|
|
||||||
},
|
|
||||||
updateIssuesSelected(state, issues) {
|
|
||||||
//console.log('update issues selected', issues);
|
|
||||||
state.activity.socialIssues = issues;
|
|
||||||
},
|
|
||||||
updateIssuesOther(state, payload) {
|
|
||||||
//console.log('update issues other');
|
|
||||||
state.socialIssuesOther = payload;
|
|
||||||
},
|
|
||||||
removeIssueInOther(state, issue) {
|
|
||||||
//console.log('remove issue other', issue.id);
|
|
||||||
state.socialIssuesOther = state.socialIssuesOther.filter(
|
|
||||||
(i) => i.id !== issue.id
|
|
||||||
);
|
|
||||||
},
|
|
||||||
resetActionsList(state) {
|
|
||||||
//console.log('reset list actions');
|
|
||||||
state.socialActionsList = [];
|
|
||||||
},
|
|
||||||
addActionInList(state, action) {
|
|
||||||
state.socialActionsList.push(action);
|
|
||||||
},
|
|
||||||
updateActionsSelected(state, actions) {
|
|
||||||
//console.log('update actions selected', actions);
|
|
||||||
state.activity.socialActions = actions;
|
|
||||||
},
|
|
||||||
filterList(state, list) {
|
|
||||||
const filterList = (list) => {
|
|
||||||
// remove duplicates entries
|
|
||||||
list = list.filter(
|
|
||||||
(value, index) =>
|
|
||||||
list.findIndex((array) => array.id === value.id) ===
|
|
||||||
index
|
|
||||||
);
|
|
||||||
// alpha sort
|
|
||||||
list.sort((a, b) =>
|
|
||||||
a.text > b.text ? 1 : b.text > a.text ? -1 : 0
|
|
||||||
);
|
|
||||||
return list;
|
|
||||||
};
|
|
||||||
if (list === "issues") {
|
|
||||||
state.activity.accompanyingPeriod.socialIssues = filterList(
|
|
||||||
state.activity.accompanyingPeriod.socialIssues
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if (list === "actions") {
|
|
||||||
state.socialActionsList = filterList(state.socialActionsList);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// ConcernedGroups
|
return suggestedUsers;
|
||||||
addPersonsInvolved(state, payload) {
|
},
|
||||||
console.log("### mutation addPersonsInvolved", payload);
|
suggestedResources(state) {
|
||||||
switch (payload.result.type) {
|
// const resources = state.activity.accompanyingPeriod.resources;
|
||||||
case "person":
|
const existingPersonIds = state.activity.persons.map((p) => p.id);
|
||||||
state.activity.persons.push(payload.result);
|
const existingThirdPartyIds = state.activity.thirdParties.map(
|
||||||
break;
|
(p) => p.id,
|
||||||
case "thirdparty":
|
);
|
||||||
state.activity.thirdParties.push(payload.result);
|
return state.activity.accompanyingPeriod.resources
|
||||||
break;
|
.map((r) => r.resource)
|
||||||
case "user":
|
.filter(
|
||||||
state.activity.users.push(payload.result);
|
(r) =>
|
||||||
break;
|
(r.type === "person" &&
|
||||||
}
|
!existingPersonIds.includes(r.id) &&
|
||||||
},
|
state.activity.activityType.personsVisible !== 0) ||
|
||||||
removePersonInvolved(state, payload) {
|
(r.type === "thirdparty" &&
|
||||||
//console.log('### mutation removePersonInvolved', payload.type);
|
!existingThirdPartyIds.includes(r.id) &&
|
||||||
switch (payload.type) {
|
state.activity.activityType.thirdPartiesVisible !== 0),
|
||||||
case "person":
|
);
|
||||||
state.activity.persons = state.activity.persons.filter(
|
},
|
||||||
(person) => person !== payload
|
socialActionsListSorted(state) {
|
||||||
);
|
return [...state.socialActionsList]
|
||||||
break;
|
.sort((a, b) => a.ordering - b.ordering)
|
||||||
case "thirdparty":
|
.reduce((acc, action) => {
|
||||||
state.activity.thirdParties =
|
const issueText = action.issue?.text || "Uncategorized";
|
||||||
state.activity.thirdParties.filter(
|
// Find if the group for the issue already exists
|
||||||
(thirdparty) => thirdparty !== payload
|
let group = acc.find((item) => item.issue === issueText);
|
||||||
);
|
if (!group) {
|
||||||
break;
|
group = { issue: issueText, actions: [] };
|
||||||
case "user":
|
acc.push(group);
|
||||||
state.activity.users = state.activity.users.filter(
|
}
|
||||||
(user) => user !== payload
|
group.actions.push(action);
|
||||||
);
|
return acc;
|
||||||
break;
|
}, []);
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
updateLocation(state, value) {
|
mutations: {
|
||||||
console.log("### mutation: updateLocation", value);
|
setWhoAmI(state, me) {
|
||||||
state.activity.location = value;
|
state.me = me;
|
||||||
},
|
},
|
||||||
addAvailableLocationGroup(state, group) {
|
// SocialIssueAcc
|
||||||
state.availableLocations.push(group);
|
addIssueInList(state, issue) {
|
||||||
|
//console.log('add issue list', issue.id);
|
||||||
|
state.activity.accompanyingPeriod.socialIssues.push(issue);
|
||||||
|
},
|
||||||
|
addIssueSelected(state, issue) {
|
||||||
|
//console.log('add issue selected', issue.id);
|
||||||
|
state.activity.socialIssues.push(issue);
|
||||||
|
},
|
||||||
|
updateIssuesSelected(state, issues) {
|
||||||
|
//console.log('update issues selected', issues);
|
||||||
|
state.activity.socialIssues = issues;
|
||||||
|
},
|
||||||
|
updateIssuesOther(state, payload) {
|
||||||
|
//console.log('update issues other');
|
||||||
|
state.socialIssuesOther = payload;
|
||||||
|
},
|
||||||
|
removeIssueInOther(state, issue) {
|
||||||
|
//console.log('remove issue other', issue.id);
|
||||||
|
state.socialIssuesOther = state.socialIssuesOther.filter(
|
||||||
|
(i) => i.id !== issue.id,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
resetActionsList(state) {
|
||||||
|
//console.log('reset list actions');
|
||||||
|
state.socialActionsList = [];
|
||||||
|
},
|
||||||
|
addActionInList(state, action) {
|
||||||
|
state.socialActionsList.push(action);
|
||||||
|
},
|
||||||
|
updateActionsSelected(state, actions) {
|
||||||
|
//console.log('update actions selected', actions);
|
||||||
|
state.activity.socialActions = actions;
|
||||||
|
},
|
||||||
|
filterList(state, list) {
|
||||||
|
const filterList = (list) => {
|
||||||
|
// remove duplicates entries
|
||||||
|
list = list.filter(
|
||||||
|
(value, index) =>
|
||||||
|
list.findIndex((array) => array.id === value.id) === index,
|
||||||
|
);
|
||||||
|
// alpha sort
|
||||||
|
list.sort((a, b) => (a.text > b.text ? 1 : b.text > a.text ? -1 : 0));
|
||||||
|
return list;
|
||||||
|
};
|
||||||
|
if (list === "issues") {
|
||||||
|
state.activity.accompanyingPeriod.socialIssues = filterList(
|
||||||
|
state.activity.accompanyingPeriod.socialIssues,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if (list === "actions") {
|
||||||
|
state.socialActionsList = filterList(state.socialActionsList);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// ConcernedGroups
|
||||||
|
addPersonsInvolved(state, payload) {
|
||||||
|
console.log("### mutation addPersonsInvolved", payload);
|
||||||
|
switch (payload.result.type) {
|
||||||
|
case "person":
|
||||||
|
state.activity.persons.push(payload.result);
|
||||||
|
break;
|
||||||
|
case "thirdparty":
|
||||||
|
state.activity.thirdParties.push(payload.result);
|
||||||
|
break;
|
||||||
|
case "user":
|
||||||
|
state.activity.users.push(payload.result);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
removePersonInvolved(state, payload) {
|
||||||
|
//console.log('### mutation removePersonInvolved', payload.type);
|
||||||
|
switch (payload.type) {
|
||||||
|
case "person":
|
||||||
|
state.activity.persons = state.activity.persons.filter(
|
||||||
|
(person) => person !== payload,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case "thirdparty":
|
||||||
|
state.activity.thirdParties = state.activity.thirdParties.filter(
|
||||||
|
(thirdparty) => thirdparty !== payload,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case "user":
|
||||||
|
state.activity.users = state.activity.users.filter(
|
||||||
|
(user) => user !== payload,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
updateLocation(state, value) {
|
||||||
|
console.log("### mutation: updateLocation", value);
|
||||||
|
state.activity.location = value;
|
||||||
|
},
|
||||||
|
addAvailableLocationGroup(state, group) {
|
||||||
|
state.availableLocations.push(group);
|
||||||
|
},
|
||||||
|
setAccompanyingPeriodWorks(state, works) {
|
||||||
|
state.accompanyingPeriodWorks = works;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
actions: {
|
||||||
|
addIssueSelected({ commit }, issue) {
|
||||||
|
let aSocialIssues = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_socialIssues",
|
||||||
|
);
|
||||||
|
aSocialIssues.value = addIdToValue(aSocialIssues.value, issue.id);
|
||||||
|
commit("addIssueSelected", issue);
|
||||||
|
},
|
||||||
|
updateIssuesSelected({ commit }, payload) {
|
||||||
|
let aSocialIssues = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_socialIssues",
|
||||||
|
);
|
||||||
|
aSocialIssues.value = "";
|
||||||
|
payload.forEach((item) => {
|
||||||
|
aSocialIssues.value = addIdToValue(aSocialIssues.value, item.id);
|
||||||
|
});
|
||||||
|
commit("updateIssuesSelected", payload);
|
||||||
|
},
|
||||||
|
updateActionsSelected({ commit }, payload) {
|
||||||
|
let aSocialActions = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_socialActions",
|
||||||
|
);
|
||||||
|
aSocialActions.value = "";
|
||||||
|
payload.forEach((item) => {
|
||||||
|
aSocialActions.value = addIdToValue(aSocialActions.value, item.id);
|
||||||
|
});
|
||||||
|
commit("updateActionsSelected", payload);
|
||||||
|
},
|
||||||
|
addAvailableLocationGroup({ commit }, payload) {
|
||||||
|
commit("addAvailableLocationGroup", payload);
|
||||||
|
},
|
||||||
|
addPersonsInvolved({ commit }, payload) {
|
||||||
|
//console.log('### action addPersonsInvolved', payload.result.type);
|
||||||
|
switch (payload.result.type) {
|
||||||
|
case "person":
|
||||||
|
let aPersons = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_persons",
|
||||||
|
);
|
||||||
|
aPersons.value = addIdToValue(aPersons.value, payload.result.id);
|
||||||
|
break;
|
||||||
|
case "thirdparty":
|
||||||
|
let aThirdParties = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_thirdParties",
|
||||||
|
);
|
||||||
|
aThirdParties.value = addIdToValue(
|
||||||
|
aThirdParties.value,
|
||||||
|
payload.result.id,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case "user":
|
||||||
|
let aUsers = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_users",
|
||||||
|
);
|
||||||
|
aUsers.value = addIdToValue(aUsers.value, payload.result.id);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
commit("addPersonsInvolved", payload);
|
||||||
|
},
|
||||||
|
removePersonInvolved({ commit }, payload) {
|
||||||
|
//console.log('### action removePersonInvolved', payload);
|
||||||
|
switch (payload.type) {
|
||||||
|
case "person":
|
||||||
|
let aPersons = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_persons",
|
||||||
|
);
|
||||||
|
aPersons.value = removeIdFromValue(aPersons.value, payload.id);
|
||||||
|
break;
|
||||||
|
case "thirdparty":
|
||||||
|
let aThirdParties = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_thirdParties",
|
||||||
|
);
|
||||||
|
aThirdParties.value = removeIdFromValue(
|
||||||
|
aThirdParties.value,
|
||||||
|
payload.id,
|
||||||
|
);
|
||||||
|
break;
|
||||||
|
case "user":
|
||||||
|
let aUsers = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_users",
|
||||||
|
);
|
||||||
|
aUsers.value = removeIdFromValue(aUsers.value, payload.id);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
commit("removePersonInvolved", payload);
|
||||||
|
},
|
||||||
|
updateLocation({ commit }, value) {
|
||||||
|
console.log("### action: updateLocation", value);
|
||||||
|
let hiddenLocation = document.getElementById(
|
||||||
|
"chill_activitybundle_activity_location",
|
||||||
|
);
|
||||||
|
if (value.onthefly) {
|
||||||
|
const body = {
|
||||||
|
type: "location",
|
||||||
|
name:
|
||||||
|
value.name === "__AccompanyingCourseLocation__" ? null : value.name,
|
||||||
|
locationType: {
|
||||||
|
id: value.locationType.id,
|
||||||
|
type: "location-type",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
if (value.address.id) {
|
||||||
|
Object.assign(body, {
|
||||||
|
address: {
|
||||||
|
id: value.address.id,
|
||||||
|
},
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
postLocation(body)
|
||||||
|
.then((location) => (hiddenLocation.value = location.id))
|
||||||
|
.catch((err) => {
|
||||||
|
console.log(err.message);
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
hiddenLocation.value = value.id;
|
||||||
|
}
|
||||||
|
commit("updateLocation", value);
|
||||||
},
|
},
|
||||||
actions: {
|
async fetchAccompanyingPeriodWorks({ state, commit }) {
|
||||||
addIssueSelected({ commit }, issue) {
|
const accompanyingPeriodId = state.activity.accompanyingPeriod.id;
|
||||||
let aSocialIssues = document.getElementById(
|
const url = `/api/1.0/person/accompanying-course/${accompanyingPeriodId}/works.json`;
|
||||||
"chill_activitybundle_activity_socialIssues"
|
try {
|
||||||
);
|
const works = await makeFetch("GET", url);
|
||||||
aSocialIssues.value = addIdToValue(aSocialIssues.value, issue.id);
|
// console.log("works", works);
|
||||||
commit("addIssueSelected", issue);
|
commit("setAccompanyingPeriodWorks", works);
|
||||||
},
|
} catch (error) {
|
||||||
updateIssuesSelected({ commit }, payload) {
|
console.error("Failed to fetch accompanying period works:", error);
|
||||||
let aSocialIssues = document.getElementById(
|
}
|
||||||
"chill_activitybundle_activity_socialIssues"
|
|
||||||
);
|
|
||||||
aSocialIssues.value = "";
|
|
||||||
payload.forEach((item) => {
|
|
||||||
aSocialIssues.value = addIdToValue(
|
|
||||||
aSocialIssues.value,
|
|
||||||
item.id
|
|
||||||
);
|
|
||||||
});
|
|
||||||
commit("updateIssuesSelected", payload);
|
|
||||||
},
|
|
||||||
updateActionsSelected({ commit }, payload) {
|
|
||||||
let aSocialActions = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_socialActions"
|
|
||||||
);
|
|
||||||
aSocialActions.value = "";
|
|
||||||
payload.forEach((item) => {
|
|
||||||
aSocialActions.value = addIdToValue(
|
|
||||||
aSocialActions.value,
|
|
||||||
item.id
|
|
||||||
);
|
|
||||||
});
|
|
||||||
commit("updateActionsSelected", payload);
|
|
||||||
},
|
|
||||||
addAvailableLocationGroup({ commit }, payload) {
|
|
||||||
commit("addAvailableLocationGroup", payload);
|
|
||||||
},
|
|
||||||
addPersonsInvolved({ commit }, payload) {
|
|
||||||
//console.log('### action addPersonsInvolved', payload.result.type);
|
|
||||||
switch (payload.result.type) {
|
|
||||||
case "person":
|
|
||||||
let aPersons = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_persons"
|
|
||||||
);
|
|
||||||
aPersons.value = addIdToValue(
|
|
||||||
aPersons.value,
|
|
||||||
payload.result.id
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case "thirdparty":
|
|
||||||
let aThirdParties = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_thirdParties"
|
|
||||||
);
|
|
||||||
aThirdParties.value = addIdToValue(
|
|
||||||
aThirdParties.value,
|
|
||||||
payload.result.id
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case "user":
|
|
||||||
let aUsers = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_users"
|
|
||||||
);
|
|
||||||
aUsers.value = addIdToValue(
|
|
||||||
aUsers.value,
|
|
||||||
payload.result.id
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
commit("addPersonsInvolved", payload);
|
|
||||||
},
|
|
||||||
removePersonInvolved({ commit }, payload) {
|
|
||||||
//console.log('### action removePersonInvolved', payload);
|
|
||||||
switch (payload.type) {
|
|
||||||
case "person":
|
|
||||||
let aPersons = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_persons"
|
|
||||||
);
|
|
||||||
aPersons.value = removeIdFromValue(
|
|
||||||
aPersons.value,
|
|
||||||
payload.id
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case "thirdparty":
|
|
||||||
let aThirdParties = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_thirdParties"
|
|
||||||
);
|
|
||||||
aThirdParties.value = removeIdFromValue(
|
|
||||||
aThirdParties.value,
|
|
||||||
payload.id
|
|
||||||
);
|
|
||||||
break;
|
|
||||||
case "user":
|
|
||||||
let aUsers = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_users"
|
|
||||||
);
|
|
||||||
aUsers.value = removeIdFromValue(aUsers.value, payload.id);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
commit("removePersonInvolved", payload);
|
|
||||||
},
|
|
||||||
updateLocation({ commit }, value) {
|
|
||||||
console.log("### action: updateLocation", value);
|
|
||||||
let hiddenLocation = document.getElementById(
|
|
||||||
"chill_activitybundle_activity_location"
|
|
||||||
);
|
|
||||||
if (value.onthefly) {
|
|
||||||
const body = {
|
|
||||||
"type": "location",
|
|
||||||
"name": value.name === '__AccompanyingCourseLocation__' ? null : value.name,
|
|
||||||
"locationType": {
|
|
||||||
"id": value.locationType.id,
|
|
||||||
"type": "location-type"
|
|
||||||
}
|
|
||||||
};
|
|
||||||
if (value.address.id) {
|
|
||||||
Object.assign(body, {
|
|
||||||
"address": {
|
|
||||||
"id": value.address.id
|
|
||||||
},
|
|
||||||
})
|
|
||||||
}
|
|
||||||
postLocation(body)
|
|
||||||
.then(
|
|
||||||
location => hiddenLocation.value = location.id
|
|
||||||
).catch(
|
|
||||||
err => {
|
|
||||||
console.log(err.message);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
hiddenLocation.value = value.id;
|
|
||||||
}
|
|
||||||
commit("updateLocation", value);
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
getWhoAmI({ commit }) {
|
||||||
|
const url = `/api/1.0/main/whoami.json`;
|
||||||
|
makeFetch("GET", url).then((user) => {
|
||||||
|
commit("setWhoAmI", user);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
store.dispatch("getWhoAmI");
|
||||||
|
|
||||||
prepareLocations(store);
|
prepareLocations(store);
|
||||||
|
store.dispatch("fetchAccompanyingPeriodWorks");
|
||||||
|
|
||||||
export default store;
|
export default store;
|
||||||
|
@@ -1,124 +1,132 @@
|
|||||||
import {getLocations, getLocationTypeByDefaultFor, getUserCurrentLocation} from "./api";
|
import {
|
||||||
|
getLocations,
|
||||||
|
getLocationTypeByDefaultFor,
|
||||||
|
getUserCurrentLocation,
|
||||||
|
} from "./api";
|
||||||
|
|
||||||
const makeConcernedPersonsLocation = (locationType, store) => {
|
const makeConcernedPersonsLocation = (locationType, store) => {
|
||||||
let locations = [];
|
let locations = [];
|
||||||
store.getters.suggestedEntities.forEach(
|
store.getters.suggestedEntities.forEach((e) => {
|
||||||
(e) => {
|
if (e.type === "person" && e.current_household_address !== null) {
|
||||||
if (e.type === 'person' && e.current_household_address !== null){
|
locations.push({
|
||||||
locations.push({
|
type: "location",
|
||||||
type: 'location',
|
id: -store.getters.suggestedEntities.indexOf(e) * 10,
|
||||||
id: -store.getters.suggestedEntities.indexOf(e)*10,
|
onthefly: true,
|
||||||
onthefly: true,
|
name: e.text,
|
||||||
name: e.text,
|
address: {
|
||||||
address: {
|
id: e.current_household_address.address_id,
|
||||||
id: e.current_household_address.address_id,
|
},
|
||||||
},
|
locationType: locationType,
|
||||||
locationType: locationType
|
});
|
||||||
});
|
}
|
||||||
}
|
});
|
||||||
}
|
return locations;
|
||||||
)
|
|
||||||
return locations;
|
|
||||||
};
|
};
|
||||||
const makeConcernedThirdPartiesLocation = (locationType, store) => {
|
const makeConcernedThirdPartiesLocation = (locationType, store) => {
|
||||||
let locations = [];
|
let locations = [];
|
||||||
store.getters.suggestedEntities.forEach(
|
store.getters.suggestedEntities.forEach((e) => {
|
||||||
(e) => {
|
if (e.type === "thirdparty" && e.address !== null) {
|
||||||
if (e.type === 'thirdparty' && e.address !== null){
|
locations.push({
|
||||||
locations.push({
|
type: "location",
|
||||||
type: 'location',
|
id: -store.getters.suggestedEntities.indexOf(e) * 10,
|
||||||
id: -store.getters.suggestedEntities.indexOf(e)*10,
|
onthefly: true,
|
||||||
onthefly: true,
|
name: e.text,
|
||||||
name: e.text,
|
address: { id: e.address.address_id },
|
||||||
address: { id: e.address.address_id },
|
locationType: locationType,
|
||||||
locationType: locationType
|
});
|
||||||
});
|
}
|
||||||
}
|
});
|
||||||
}
|
return locations;
|
||||||
)
|
|
||||||
return locations;
|
|
||||||
};
|
};
|
||||||
const makeAccompanyingPeriodLocation = (locationType, store) => {
|
const makeAccompanyingPeriodLocation = (locationType, store) => {
|
||||||
if (store.state.activity.accompanyingPeriod === null) {
|
if (store.state.activity.accompanyingPeriod === null) {
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
const accPeriodLocation = store.state.activity.accompanyingPeriod.location;
|
const accPeriodLocation = store.state.activity.accompanyingPeriod.location;
|
||||||
return {
|
return {
|
||||||
type: 'location',
|
type: "location",
|
||||||
id: -1,
|
id: -1,
|
||||||
onthefly: true,
|
onthefly: true,
|
||||||
name: '__AccompanyingCourseLocation__',
|
name: "__AccompanyingCourseLocation__",
|
||||||
address: {
|
address: {
|
||||||
id: accPeriodLocation.address_id,
|
id: accPeriodLocation.address_id,
|
||||||
text: `${accPeriodLocation.text} - ${accPeriodLocation.postcode.code} ${accPeriodLocation.postcode.name}`
|
text: `${accPeriodLocation.text} - ${accPeriodLocation.postcode.code} ${accPeriodLocation.postcode.name}`,
|
||||||
},
|
},
|
||||||
locationType: locationType
|
locationType: locationType,
|
||||||
}
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function prepareLocations(store) {
|
export default function prepareLocations(store) {
|
||||||
|
|
||||||
// find the locations
|
// find the locations
|
||||||
let allLocations = getLocations().then(
|
let allLocations = getLocations().then((results) => {
|
||||||
(results) => {
|
store.commit("addAvailableLocationGroup", {
|
||||||
store.commit('addAvailableLocationGroup', {
|
locationGroup: "Autres localisations",
|
||||||
locationGroup: 'Autres localisations',
|
locations: results,
|
||||||
locations: results
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
let currentLocation = getUserCurrentLocation().then((userCurrentLocation) => {
|
||||||
|
if (null !== userCurrentLocation) {
|
||||||
|
store.commit("addAvailableLocationGroup", {
|
||||||
|
locationGroup: "Ma localisation",
|
||||||
|
locations: [userCurrentLocation],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
let partiesLocations = [],
|
||||||
|
partyPromise;
|
||||||
|
["person", "thirdparty"].forEach((kind) => {
|
||||||
|
partyPromise = getLocationTypeByDefaultFor(kind).then(
|
||||||
|
(kindLocationType) => {
|
||||||
|
if (kindLocationType) {
|
||||||
|
let concernedKindLocations;
|
||||||
|
if (kind === "person") {
|
||||||
|
concernedKindLocations = makeConcernedPersonsLocation(
|
||||||
|
kindLocationType,
|
||||||
|
store,
|
||||||
|
);
|
||||||
|
// add location for the parcours into suggestions
|
||||||
|
const personLocation = makeAccompanyingPeriodLocation(
|
||||||
|
kindLocationType,
|
||||||
|
store,
|
||||||
|
);
|
||||||
|
store.commit("addAvailableLocationGroup", {
|
||||||
|
locationGroup: "Localisation du parcours",
|
||||||
|
locations: [personLocation],
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
concernedKindLocations = makeConcernedThirdPartiesLocation(
|
||||||
|
kindLocationType,
|
||||||
|
store,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
store.commit("addAvailableLocationGroup", {
|
||||||
|
locationGroup:
|
||||||
|
kind === "person" ? "Usagers concernés" : "Tiers concernés",
|
||||||
|
locations: concernedKindLocations,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
},
|
||||||
);
|
);
|
||||||
|
partiesLocations.push(partyPromise);
|
||||||
|
});
|
||||||
|
|
||||||
let currentLocation = getUserCurrentLocation().then(
|
// when all location are loaded
|
||||||
userCurrentLocation => {
|
Promise.all([allLocations, currentLocation, ...partiesLocations]).then(() => {
|
||||||
if (null !== userCurrentLocation) {
|
console.log("current location in activity", store.state.activity.location);
|
||||||
store.commit('addAvailableLocationGroup', {
|
console.log("default loation id", window.default_location_id);
|
||||||
locationGroup: 'Ma localisation',
|
if (window.default_location_id) {
|
||||||
locations: [userCurrentLocation]
|
for (let group of store.state.availableLocations) {
|
||||||
});
|
let location = group.locations.find(
|
||||||
}
|
(l) => l.id === window.default_location_id,
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
let partiesLocations = [], partyPromise;
|
|
||||||
['person', 'thirdparty'].forEach(kind => {
|
|
||||||
partyPromise = getLocationTypeByDefaultFor(kind).then(
|
|
||||||
(kindLocationType) => {
|
|
||||||
if (kindLocationType) {
|
|
||||||
let concernedKindLocations;
|
|
||||||
if (kind === 'person') {
|
|
||||||
concernedKindLocations = makeConcernedPersonsLocation(kindLocationType, store);
|
|
||||||
// add location for the parcours into suggestions
|
|
||||||
const personLocation = makeAccompanyingPeriodLocation(kindLocationType, store);
|
|
||||||
store.commit('addAvailableLocationGroup', {
|
|
||||||
locationGroup: 'Localisation du parcours',
|
|
||||||
locations: [personLocation]
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
concernedKindLocations = makeConcernedThirdPartiesLocation(kindLocationType, store);
|
|
||||||
}
|
|
||||||
|
|
||||||
store.commit('addAvailableLocationGroup', {
|
|
||||||
locationGroup: kind === 'person' ? 'Usagers concernés' : 'Tiers concernés',
|
|
||||||
locations: concernedKindLocations,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
);
|
||||||
partiesLocations.push(partyPromise);
|
if (location !== undefined && store.state.activity.location === null) {
|
||||||
});
|
store.dispatch("updateLocation", location);
|
||||||
|
break;
|
||||||
// when all location are loaded
|
|
||||||
Promise.all([allLocations, currentLocation, ...partiesLocations]).then(() => {
|
|
||||||
console.log('current location in activity', store.state.activity.location);
|
|
||||||
console.log('default loation id', window.default_location_id);
|
|
||||||
if (window.default_location_id) {
|
|
||||||
for (let group of store.state.availableLocations) {
|
|
||||||
let location = group.locations.find((l) => l.id === window.default_location_id);
|
|
||||||
if (location !== undefined && store.state.activity.location === null) {
|
|
||||||
store.dispatch('updateLocation', location);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
@@ -126,4 +126,4 @@
|
|||||||
|
|
||||||
{% block css %}
|
{% block css %}
|
||||||
{{ encore_entry_link_tags('mod_pickentity_type') }}
|
{{ encore_entry_link_tags('mod_pickentity_type') }}
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
@@ -1,83 +1,3 @@
|
|||||||
{% import "@ChillDocStore/Macro/macro.html.twig" as m %}
|
|
||||||
{% import "@ChillDocStore/Macro/macro_mimeicon.html.twig" as mm %}
|
|
||||||
{% import '@ChillPerson/Macro/updatedBy.html.twig' as mmm %}
|
|
||||||
|
|
||||||
{% set person_id = null %}
|
|
||||||
{% if activity.person %}
|
|
||||||
{% set person_id = activity.person.id %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% set accompanying_course_id = null %}
|
|
||||||
{% if activity.accompanyingPeriod %}
|
|
||||||
{% set accompanying_course_id = activity.accompanyingPeriod.id %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
<div class="item-bloc activity-item{% if itemBlocClass is defined %} {{ itemBlocClass }}{% endif %}">
|
<div class="item-bloc activity-item{% if itemBlocClass is defined %} {{ itemBlocClass }}{% endif %}">
|
||||||
<div class="item-row">
|
{{ include('@ChillActivity/GenericDoc/activity_document_row.html.twig') }}
|
||||||
<div class="item-col" style="width: unset">
|
|
||||||
{% if document.isPending %}
|
|
||||||
<div class="badge text-bg-info" data-docgen-is-pending="{{ document.id }}">{{ 'docgen.Doc generation is pending'|trans }}</div>
|
|
||||||
{% elseif document.isFailure %}
|
|
||||||
<div class="badge text-bg-warning">{{ 'docgen.Doc generation failed'|trans }}</div>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
<div>
|
|
||||||
{% if activity.accompanyingPeriod is not null and context == 'person' %}
|
|
||||||
<span class="badge bg-primary">
|
|
||||||
<i class="fa fa-random"></i> {{ activity.accompanyingPeriod.id }}
|
|
||||||
</span>
|
|
||||||
{% endif %}
|
|
||||||
<div class="badge-activity-type">
|
|
||||||
<span class="title_label"></span>
|
|
||||||
<span class="title_action">
|
|
||||||
{{ activity.type.name | localize_translatable_string }}
|
|
||||||
{% if activity.emergency %}
|
|
||||||
<span class="badge bg-danger rounded-pill fs-6 float-end">{{ 'Emergency'|trans|upper }}</span>
|
|
||||||
{% endif %}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="denomination h2">
|
|
||||||
{{ document.title|chill_print_or_message("No title") }}
|
|
||||||
</div>
|
|
||||||
{% if document.hasTemplate %}
|
|
||||||
<div>
|
|
||||||
<p>{{ document.template.name|localize_translatable_string }}</p>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="item-col">
|
|
||||||
<div class="container">
|
|
||||||
<div class="dates row text-end">
|
|
||||||
<span>{{ document.createdAt|format_date('short') }}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
<div class="item-row separator">
|
|
||||||
<div class="item-col item-meta">
|
|
||||||
{{ mmm.createdBy(document) }}
|
|
||||||
</div>
|
|
||||||
<ul class="item-col record_actions flex-shrink-1">
|
|
||||||
{% if is_granted('CHILL_ACTIVITY_SEE_DETAILS', activity) %}
|
|
||||||
<li>
|
|
||||||
{{ document|chill_document_button_group(document.title, is_granted('CHILL_ACTIVITY_UPDATE', activity), {small: false}) }}
|
|
||||||
</li>
|
|
||||||
{% endif %}
|
|
||||||
{% if is_granted('CHILL_ACTIVITY_SEE', activity)%}
|
|
||||||
<li>
|
|
||||||
<a href="{{ chill_path_add_return_path('chill_activity_activity_show', {'id': activity.id, 'person_id': person_id, 'accompanying_period_id': accompanying_course_id}) }}" class="btn btn-show"></a>
|
|
||||||
</li>
|
|
||||||
{% endif %}
|
|
||||||
{% if is_granted('CHILL_ACTIVITY_UPDATE', activity) %}
|
|
||||||
<li>
|
|
||||||
<a href="{{ chill_path_add_return_path('chill_activity_activity_edit', {'id': activity.id, 'person_id': person_id, 'accompanying_period_id': accompanying_course_id }) }}" class="btn btn-edit"></a>
|
|
||||||
</li>
|
|
||||||
{% endif %}
|
|
||||||
</ul>
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
@@ -0,0 +1,81 @@
|
|||||||
|
{% import "@ChillDocStore/Macro/macro.html.twig" as m %}
|
||||||
|
{% import "@ChillDocStore/Macro/macro_mimeicon.html.twig" as mm %}
|
||||||
|
{% import '@ChillPerson/Macro/updatedBy.html.twig' as mmm %}
|
||||||
|
|
||||||
|
{% set person_id = null %}
|
||||||
|
{% if activity.person %}
|
||||||
|
{% set person_id = activity.person.id %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% set accompanying_course_id = null %}
|
||||||
|
{% if activity.accompanyingPeriod %}
|
||||||
|
{% set accompanying_course_id = activity.accompanyingPeriod.id %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="item-row">
|
||||||
|
<div class="item-two-col-grid">
|
||||||
|
<div class="title">
|
||||||
|
{% if document.isPending %}
|
||||||
|
<div class="badge text-bg-info" data-docgen-is-pending="{{ document.id }}">{{ 'docgen.Doc generation is pending'|trans }}</div>
|
||||||
|
{% elseif document.isFailure %}
|
||||||
|
<div class="badge text-bg-warning">{{ 'docgen.Doc generation failed'|trans }}</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div>
|
||||||
|
<div class="badge-activity-type-simple">
|
||||||
|
{{ activity.type.name | localize_translatable_string }}
|
||||||
|
</div>
|
||||||
|
{% if activity.emergency %}
|
||||||
|
<span class="badge bg-danger rounded-pill fs-6 float-end">{{ 'Emergency'|trans|upper }}</span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="denomination h2">
|
||||||
|
{{ document.title|chill_print_or_message("No title") }}
|
||||||
|
</div>
|
||||||
|
{% if document.hasTemplate %}
|
||||||
|
<div>
|
||||||
|
<p>{{ document.template.name|localize_translatable_string }}</p>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<div class="aside">
|
||||||
|
<div class="dates row text-end">
|
||||||
|
<span>{{ document.createdAt|format_date('short') }}</span>
|
||||||
|
</div>
|
||||||
|
{% if activity.accompanyingPeriod is not null and context == 'person' %}
|
||||||
|
<div class="text-end">
|
||||||
|
<span class="badge bg-primary">
|
||||||
|
<i class="fa fa-random"></i> {{ activity.accompanyingPeriod.id }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if show_actions %}
|
||||||
|
<div class="item-row separator">
|
||||||
|
<div class="item-col item-meta">
|
||||||
|
{{ mmm.createdBy(document) }}
|
||||||
|
</div>
|
||||||
|
<ul class="item-col record_actions flex-shrink-1">
|
||||||
|
{% if is_granted('CHILL_ACTIVITY_SEE_DETAILS', activity) %}
|
||||||
|
<li>
|
||||||
|
{{ document|chill_document_button_group(document.title, is_granted('CHILL_ACTIVITY_UPDATE', activity), {small: false}) }}
|
||||||
|
</li>
|
||||||
|
{% endif %}
|
||||||
|
{% if is_granted('CHILL_ACTIVITY_SEE', activity)%}
|
||||||
|
<li>
|
||||||
|
<a href="{{ chill_path_add_return_path('chill_activity_activity_show', {'id': activity.id, 'person_id': person_id, 'accompanying_period_id': accompanying_course_id}) }}" class="btn btn-show"></a>
|
||||||
|
</li>
|
||||||
|
{% endif %}
|
||||||
|
{% if is_granted('CHILL_ACTIVITY_UPDATE', activity) %}
|
||||||
|
<li>
|
||||||
|
<a href="{{ chill_path_add_return_path('chill_activity_activity_edit', {'id': activity.id, 'person_id': person_id, 'accompanying_period_id': accompanying_course_id }) }}" class="btn btn-edit"></a>
|
||||||
|
</li>
|
||||||
|
{% endif %}
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
@@ -143,7 +143,10 @@ class ListActivitiesByAccompanyingPeriodContext implements
|
|||||||
array_filter(
|
array_filter(
|
||||||
$works,
|
$works,
|
||||||
function ($work) use ($user) {
|
function ($work) use ($user) {
|
||||||
$workUsernames = array_map(static fn (User $user) => $user['username'], $work['referrers'] ?? []);
|
$workUsernames = [];
|
||||||
|
foreach ($work['referrers'] as $referrer) {
|
||||||
|
$workUsernames[] = $referrer['username'];
|
||||||
|
}
|
||||||
|
|
||||||
return \in_array($user->getUserIdentifier(), $workUsernames, true);
|
return \in_array($user->getUserIdentifier(), $workUsernames, true);
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,54 @@
|
|||||||
|
<?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\Service\GenericDoc\Normalizer;
|
||||||
|
|
||||||
|
use Chill\ActivityBundle\Service\GenericDoc\Providers\AccompanyingPeriodActivityGenericDocProvider;
|
||||||
|
use Chill\ActivityBundle\Service\GenericDoc\Renderers\AccompanyingPeriodActivityGenericDocRenderer;
|
||||||
|
use Chill\DocStoreBundle\GenericDoc\GenericDocDTO;
|
||||||
|
use Chill\DocStoreBundle\GenericDoc\GenericDocNormalizerInterface;
|
||||||
|
use Chill\DocStoreBundle\Repository\StoredObjectRepositoryInterface;
|
||||||
|
use Symfony\Contracts\Translation\TranslatorInterface;
|
||||||
|
use Twig\Environment;
|
||||||
|
|
||||||
|
final readonly class AccompanyingPeriodActivityGenericDocNormalizer implements GenericDocNormalizerInterface
|
||||||
|
{
|
||||||
|
public function __construct(
|
||||||
|
private StoredObjectRepositoryInterface $storedObjectRepository,
|
||||||
|
private AccompanyingPeriodActivityGenericDocRenderer $renderer,
|
||||||
|
private Environment $twig,
|
||||||
|
private TranslatorInterface $translator,
|
||||||
|
) {}
|
||||||
|
|
||||||
|
public function supportsNormalization(GenericDocDTO $genericDocDTO, string $format, array $context = []): bool
|
||||||
|
{
|
||||||
|
return AccompanyingPeriodActivityGenericDocProvider::KEY === $genericDocDTO->key
|
||||||
|
&& 'json' == $format;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function normalize(GenericDocDTO $genericDocDTO, string $format, array $context = []): array
|
||||||
|
{
|
||||||
|
$storedObject = $this->storedObjectRepository->find($genericDocDTO->identifiers['id']);
|
||||||
|
|
||||||
|
if (null === $storedObject) {
|
||||||
|
return ['title' => $this->translator->trans('generic_doc.document removed'), 'isPresent' => false];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'isPresent' => true,
|
||||||
|
'title' => $storedObject->getTitle(),
|
||||||
|
'html' => $this->twig->render(
|
||||||
|
$this->renderer->getTemplate($genericDocDTO, ['show-actions' => false, 'row-only' => true]),
|
||||||
|
$this->renderer->getTemplateData($genericDocDTO, ['show-actions' => false, 'row-only' => true]),
|
||||||
|
),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
@@ -13,10 +13,12 @@ namespace Chill\ActivityBundle\Service\GenericDoc\Providers;
|
|||||||
|
|
||||||
use Chill\ActivityBundle\Entity\Activity;
|
use Chill\ActivityBundle\Entity\Activity;
|
||||||
use Chill\ActivityBundle\Repository\ActivityDocumentACLAwareRepositoryInterface;
|
use Chill\ActivityBundle\Repository\ActivityDocumentACLAwareRepositoryInterface;
|
||||||
|
use Chill\ActivityBundle\Repository\ActivityRepository;
|
||||||
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
use Chill\ActivityBundle\Security\Authorization\ActivityVoter;
|
||||||
use Chill\DocStoreBundle\Entity\StoredObject;
|
use Chill\DocStoreBundle\Entity\StoredObject;
|
||||||
use Chill\DocStoreBundle\GenericDoc\FetchQuery;
|
use Chill\DocStoreBundle\GenericDoc\FetchQuery;
|
||||||
use Chill\DocStoreBundle\GenericDoc\FetchQueryInterface;
|
use Chill\DocStoreBundle\GenericDoc\FetchQueryInterface;
|
||||||
|
use Chill\DocStoreBundle\GenericDoc\GenericDocDTO;
|
||||||
use Chill\DocStoreBundle\GenericDoc\GenericDocForAccompanyingPeriodProviderInterface;
|
use Chill\DocStoreBundle\GenericDoc\GenericDocForAccompanyingPeriodProviderInterface;
|
||||||
use Chill\DocStoreBundle\GenericDoc\GenericDocForPersonProviderInterface;
|
use Chill\DocStoreBundle\GenericDoc\GenericDocForPersonProviderInterface;
|
||||||
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
use Chill\PersonBundle\Entity\AccompanyingPeriod;
|
||||||
@@ -34,8 +36,47 @@ final readonly class AccompanyingPeriodActivityGenericDocProvider implements Gen
|
|||||||
private EntityManagerInterface $em,
|
private EntityManagerInterface $em,
|
||||||
private Security $security,
|
private Security $security,
|
||||||
private ActivityDocumentACLAwareRepositoryInterface $activityDocumentACLAwareRepository,
|
private ActivityDocumentACLAwareRepositoryInterface $activityDocumentACLAwareRepository,
|
||||||
|
private ActivityRepository $activityRepository,
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
|
public function fetchAssociatedStoredObject(GenericDocDTO $genericDocDTO): ?StoredObject
|
||||||
|
{
|
||||||
|
if (null === $activity = $this->getRelatedEntity($genericDocDTO->key, $genericDocDTO->identifiers)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $activity->getDocuments()->findFirst(fn (int $key, StoredObject $storedObject) => $storedObject->getId() === $genericDocDTO->identifiers['id']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function supportsGenericDoc(GenericDocDTO $genericDocDTO): bool
|
||||||
|
{
|
||||||
|
return $this->supportsKeyAndIdentifiers($genericDocDTO->key, $genericDocDTO->identifiers);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function supportsKeyAndIdentifiers(string $key, array $identifiers): bool
|
||||||
|
{
|
||||||
|
return self::KEY === $key && array_key_exists('activity_id', $identifiers);
|
||||||
|
}
|
||||||
|
|
||||||
|
private function getRelatedEntity(string $key, array $identifiers): ?Activity
|
||||||
|
{
|
||||||
|
return $this->activityRepository->find($identifiers['activity_id']);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function buildOneGenericDoc(string $key, array $identifiers): ?GenericDocDTO
|
||||||
|
{
|
||||||
|
if (null === $activity = $this->getRelatedEntity($key, $identifiers)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return new GenericDocDTO(
|
||||||
|
self::KEY,
|
||||||
|
$identifiers,
|
||||||
|
\DateTimeImmutable::createFromInterface($activity->getDate()),
|
||||||
|
$activity->getAccompanyingPeriod(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
public function buildFetchQueryForAccompanyingPeriod(AccompanyingPeriod $accompanyingPeriod, ?\DateTimeImmutable $startDate = null, ?\DateTimeImmutable $endDate = null, ?string $content = null, ?string $origin = null): FetchQueryInterface
|
public function buildFetchQueryForAccompanyingPeriod(AccompanyingPeriod $accompanyingPeriod, ?\DateTimeImmutable $startDate = null, ?\DateTimeImmutable $endDate = null, ?string $content = null, ?string $origin = null): FetchQueryInterface
|
||||||
{
|
{
|
||||||
$storedObjectMetadata = $this->em->getClassMetadata(StoredObject::class);
|
$storedObjectMetadata = $this->em->getClassMetadata(StoredObject::class);
|
||||||
|
@@ -18,6 +18,9 @@ use Chill\DocStoreBundle\GenericDoc\GenericDocDTO;
|
|||||||
use Chill\DocStoreBundle\GenericDoc\Twig\GenericDocRendererInterface;
|
use Chill\DocStoreBundle\GenericDoc\Twig\GenericDocRendererInterface;
|
||||||
use Chill\DocStoreBundle\Repository\StoredObjectRepository;
|
use Chill\DocStoreBundle\Repository\StoredObjectRepository;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @implements GenericDocRendererInterface<array{row-only?: bool, show-actions?: bool}>
|
||||||
|
*/
|
||||||
final readonly class AccompanyingPeriodActivityGenericDocRenderer implements GenericDocRendererInterface
|
final readonly class AccompanyingPeriodActivityGenericDocRenderer implements GenericDocRendererInterface
|
||||||
{
|
{
|
||||||
public function __construct(private StoredObjectRepository $objectRepository, private ActivityRepository $activityRepository) {}
|
public function __construct(private StoredObjectRepository $objectRepository, private ActivityRepository $activityRepository) {}
|
||||||
@@ -29,7 +32,8 @@ final readonly class AccompanyingPeriodActivityGenericDocRenderer implements Gen
|
|||||||
|
|
||||||
public function getTemplate(GenericDocDTO $genericDocDTO, $options = []): string
|
public function getTemplate(GenericDocDTO $genericDocDTO, $options = []): string
|
||||||
{
|
{
|
||||||
return '@ChillActivity/GenericDoc/activity_document.html.twig';
|
return ($options['row-only'] ?? false) ? '@ChillActivity/GenericDoc/activity_document_row.html.twig' :
|
||||||
|
'@ChillActivity/GenericDoc/activity_document.html.twig';
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getTemplateData(GenericDocDTO $genericDocDTO, $options = []): array
|
public function getTemplateData(GenericDocDTO $genericDocDTO, $options = []): array
|
||||||
@@ -38,6 +42,7 @@ final readonly class AccompanyingPeriodActivityGenericDocRenderer implements Gen
|
|||||||
'activity' => $this->activityRepository->find($genericDocDTO->identifiers['activity_id']),
|
'activity' => $this->activityRepository->find($genericDocDTO->identifiers['activity_id']),
|
||||||
'document' => $this->objectRepository->find($genericDocDTO->identifiers['id']),
|
'document' => $this->objectRepository->find($genericDocDTO->identifiers['id']),
|
||||||
'context' => $genericDocDTO->getContext(),
|
'context' => $genericDocDTO->getContext(),
|
||||||
|
'show_actions' => $options['show-actions'] ?? true,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,13 +1,18 @@
|
|||||||
// this file loads all assets from the Chill person bundle
|
// this file loads all assets from the Chill person bundle
|
||||||
module.exports = function(encore, entries)
|
module.exports = function (encore, entries) {
|
||||||
{
|
entries.push(__dirname + "/Resources/public/chill/index.js");
|
||||||
entries.push(__dirname + '/Resources/public/chill/index.js');
|
|
||||||
|
|
||||||
encore.addAliases({
|
encore.addAliases({
|
||||||
ChillActivityAssets: __dirname + '/Resources/public'
|
ChillActivityAssets: __dirname + "/Resources/public",
|
||||||
});
|
});
|
||||||
|
|
||||||
encore.addEntry('page_edit_activity', __dirname + '/Resources/public/page/edit_activity/index.scss');
|
encore.addEntry(
|
||||||
|
"page_edit_activity",
|
||||||
|
__dirname + "/Resources/public/page/edit_activity/index.scss",
|
||||||
|
);
|
||||||
|
|
||||||
encore.addEntry('vue_activity', __dirname + '/Resources/public/vuejs/Activity/index.js');
|
encore.addEntry(
|
||||||
|
"vue_activity",
|
||||||
|
__dirname + "/Resources/public/vuejs/Activity/index.js",
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
@@ -14,3 +14,5 @@ export:
|
|||||||
describe_action_with_subject: >-
|
describe_action_with_subject: >-
|
||||||
Filtré par personne ayant eu un échange entre le {date_from, date} et le {date_to, date}, et un de ces sujets choisis: {reasons}
|
Filtré par personne ayant eu un échange entre le {date_from, date} et le {date_to, date}, et un de ces sujets choisis: {reasons}
|
||||||
|
|
||||||
|
activity:
|
||||||
|
title: Échange du {date, date, long} - {type}
|
||||||
|
@@ -101,6 +101,33 @@ activity:
|
|||||||
Insert a document: Insérer un document
|
Insert a document: Insérer un document
|
||||||
Remove a document: Supprimer le document
|
Remove a document: Supprimer le document
|
||||||
comment: Commentaire
|
comment: Commentaire
|
||||||
|
deleted: Échange supprimé
|
||||||
|
|
||||||
|
errors: Le formulaire contient des erreurs
|
||||||
|
social_issues: Problématiques sociales
|
||||||
|
choose_other_social_issue: Ajouter une autre problématique sociale...
|
||||||
|
social_actions: Actions d'accompagnement
|
||||||
|
select_first_a_social_issue: Sélectionnez d'abord une problématique sociale
|
||||||
|
social_action_list_empty: Aucune action sociale disponible
|
||||||
|
add_persons: Ajouter des personnes concernées
|
||||||
|
bloc_persons: Usagers
|
||||||
|
bloc_persons_associated: Usagers du parcours
|
||||||
|
bloc_persons_not_associated: Tiers non-pro.
|
||||||
|
bloc_thirdparty: Tiers professionnels
|
||||||
|
bloc_users: T(M)S
|
||||||
|
location: Localisation
|
||||||
|
choose_location: Choisissez une localisation
|
||||||
|
choose_location_type: Choisissez un type de localisation
|
||||||
|
create_new_location: Créer une nouvelle localisation
|
||||||
|
location_fields:
|
||||||
|
name: Nom
|
||||||
|
type: Type
|
||||||
|
phonenumber1: Téléphone
|
||||||
|
phonenumber2: Autre téléphone
|
||||||
|
email: Adresse courriel
|
||||||
|
create_address: Créer une adresse
|
||||||
|
edit_address: Modifier l'adresse
|
||||||
|
|
||||||
No documents: Aucun document
|
No documents: Aucun document
|
||||||
|
|
||||||
# activity filter in list page
|
# activity filter in list page
|
||||||
|
@@ -1 +1 @@
|
|||||||
require('./chillbudget.scss');
|
require("./chillbudget.scss");
|
||||||
|
@@ -1,9 +1,8 @@
|
|||||||
// this file loads all assets from the Chill budget bundle
|
// this file loads all assets from the Chill budget bundle
|
||||||
module.exports = function(encore, entries)
|
module.exports = function (encore, entries) {
|
||||||
{
|
encore.addAliases({
|
||||||
encore.addAliases({
|
ChillBudgetAssets: __dirname + "/Resources/public",
|
||||||
ChillBudgetAssets: __dirname + '/Resources/public'
|
});
|
||||||
});
|
|
||||||
|
|
||||||
encore.addEntry('page_budget', __dirname + '/Resources/public/page/index.js');
|
encore.addEntry("page_budget", __dirname + "/Resources/public/page/index.js");
|
||||||
};
|
};
|
||||||
|
@@ -21,9 +21,7 @@ namespace Chill\CalendarBundle\Command;
|
|||||||
use Chill\CalendarBundle\Entity\Calendar;
|
use Chill\CalendarBundle\Entity\Calendar;
|
||||||
use Chill\CalendarBundle\Service\ShortMessageNotification\ShortMessageForCalendarBuilderInterface;
|
use Chill\CalendarBundle\Service\ShortMessageNotification\ShortMessageForCalendarBuilderInterface;
|
||||||
use Chill\MainBundle\Entity\User;
|
use Chill\MainBundle\Entity\User;
|
||||||
use Chill\MainBundle\Phonenumber\PhoneNumberHelperInterface;
|
|
||||||
use Chill\MainBundle\Repository\UserRepositoryInterface;
|
use Chill\MainBundle\Repository\UserRepositoryInterface;
|
||||||
use Chill\MainBundle\Service\ShortMessage\ShortMessageTransporterInterface;
|
|
||||||
use Chill\PersonBundle\Entity\Person;
|
use Chill\PersonBundle\Entity\Person;
|
||||||
use Chill\PersonBundle\Repository\PersonRepository;
|
use Chill\PersonBundle\Repository\PersonRepository;
|
||||||
use libphonenumber\PhoneNumber;
|
use libphonenumber\PhoneNumber;
|
||||||
@@ -36,6 +34,7 @@ use Symfony\Component\Console\Input\InputInterface;
|
|||||||
use Symfony\Component\Console\Output\OutputInterface;
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
use Symfony\Component\Console\Question\ConfirmationQuestion;
|
use Symfony\Component\Console\Question\ConfirmationQuestion;
|
||||||
use Symfony\Component\Console\Question\Question;
|
use Symfony\Component\Console\Question\Question;
|
||||||
|
use Symfony\Component\Notifier\TexterInterface;
|
||||||
|
|
||||||
class SendTestShortMessageOnCalendarCommand extends Command
|
class SendTestShortMessageOnCalendarCommand extends Command
|
||||||
{
|
{
|
||||||
@@ -44,9 +43,8 @@ class SendTestShortMessageOnCalendarCommand extends Command
|
|||||||
public function __construct(
|
public function __construct(
|
||||||
private readonly PersonRepository $personRepository,
|
private readonly PersonRepository $personRepository,
|
||||||
private readonly PhoneNumberUtil $phoneNumberUtil,
|
private readonly PhoneNumberUtil $phoneNumberUtil,
|
||||||
private readonly PhoneNumberHelperInterface $phoneNumberHelper,
|
|
||||||
private readonly ShortMessageForCalendarBuilderInterface $messageForCalendarBuilder,
|
private readonly ShortMessageForCalendarBuilderInterface $messageForCalendarBuilder,
|
||||||
private readonly ShortMessageTransporterInterface $transporter,
|
private readonly TexterInterface $transporter,
|
||||||
private readonly UserRepositoryInterface $userRepository,
|
private readonly UserRepositoryInterface $userRepository,
|
||||||
) {
|
) {
|
||||||
parent::__construct('chill:calendar:test-send-short-message');
|
parent::__construct('chill:calendar:test-send-short-message');
|
||||||
@@ -152,10 +150,6 @@ class SendTestShortMessageOnCalendarCommand extends Command
|
|||||||
return $phone;
|
return $phone;
|
||||||
});
|
});
|
||||||
|
|
||||||
$phone = $helper->ask($input, $output, $question);
|
|
||||||
|
|
||||||
$question = new ConfirmationQuestion('really send the message to the phone ?');
|
|
||||||
$reallySend = (bool) $helper->ask($input, $output, $question);
|
|
||||||
|
|
||||||
$messages = $this->messageForCalendarBuilder->buildMessageForCalendar($calendar);
|
$messages = $this->messageForCalendarBuilder->buildMessageForCalendar($calendar);
|
||||||
|
|
||||||
@@ -165,8 +159,12 @@ class SendTestShortMessageOnCalendarCommand extends Command
|
|||||||
|
|
||||||
foreach ($messages as $key => $message) {
|
foreach ($messages as $key => $message) {
|
||||||
$output->writeln("The short message for SMS {$key} will be: ");
|
$output->writeln("The short message for SMS {$key} will be: ");
|
||||||
$output->writeln($message->getContent());
|
$output->writeln($message->getSubject());
|
||||||
$message->setPhoneNumber($phone);
|
$output->writeln('The destination number will be:');
|
||||||
|
$output->writeln($message->getPhone());
|
||||||
|
|
||||||
|
$question = new ConfirmationQuestion('really send the message to the phone ?');
|
||||||
|
$reallySend = (bool) $helper->ask($input, $output, $question);
|
||||||
|
|
||||||
if ($reallySend) {
|
if ($reallySend) {
|
||||||
$this->transporter->send($message);
|
$this->transporter->send($message);
|
||||||
|
@@ -12,6 +12,7 @@ declare(strict_types=1);
|
|||||||
namespace Chill\CalendarBundle\Repository;
|
namespace Chill\CalendarBundle\Repository;
|
||||||
|
|
||||||
use Chill\CalendarBundle\Entity\CalendarDoc;
|
use Chill\CalendarBundle\Entity\CalendarDoc;
|
||||||
|
use Chill\DocStoreBundle\Entity\StoredObject;
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Doctrine\ORM\EntityRepository;
|
use Doctrine\ORM\EntityRepository;
|
||||||
use Doctrine\Persistence\ObjectRepository;
|
use Doctrine\Persistence\ObjectRepository;
|
||||||
@@ -49,4 +50,21 @@ class CalendarDocRepository implements ObjectRepository, CalendarDocRepositoryIn
|
|||||||
{
|
{
|
||||||
return CalendarDoc::class;
|
return CalendarDoc::class;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param StoredObject|int $storedObject the StoredObject instance, or the id of the stored object
|
||||||
|
*/
|
||||||
|
public function findOneByStoredObject(StoredObject|int $storedObject): ?CalendarDoc
|
||||||
|
{
|
||||||
|
$storedObjectId = $storedObject instanceof StoredObject ? $storedObject->getId() : $storedObject;
|
||||||
|
|
||||||
|
$qb = $this->repository->createQueryBuilder('c');
|
||||||
|
$qb->where(
|
||||||
|
$qb->expr()->eq(':storedObject', 'c.storedObject')
|
||||||
|
);
|
||||||
|
|
||||||
|
$qb->setParameter('storedObject', $storedObjectId);
|
||||||
|
|
||||||
|
return $qb->getQuery()->getOneOrNullResult();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -12,6 +12,7 @@ declare(strict_types=1);
|
|||||||
namespace Chill\CalendarBundle\Repository;
|
namespace Chill\CalendarBundle\Repository;
|
||||||
|
|
||||||
use Chill\CalendarBundle\Entity\CalendarDoc;
|
use Chill\CalendarBundle\Entity\CalendarDoc;
|
||||||
|
use Chill\DocStoreBundle\Entity\StoredObject;
|
||||||
|
|
||||||
interface CalendarDocRepositoryInterface
|
interface CalendarDocRepositoryInterface
|
||||||
{
|
{
|
||||||
@@ -29,5 +30,7 @@ interface CalendarDocRepositoryInterface
|
|||||||
|
|
||||||
public function findOneBy(array $criteria): ?CalendarDoc;
|
public function findOneBy(array $criteria): ?CalendarDoc;
|
||||||
|
|
||||||
|
public function findOneByStoredObject(StoredObject|int $storedObject): ?CalendarDoc;
|
||||||
|
|
||||||
public function getClassName();
|
public function getClassName();
|
||||||
}
|
}
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user