Merge branch 'issue318_internal_addIssue_fromAction' into 'master'

Adding an unrelated to issue to an accompanyingCourseWork

See merge request Chill-Projet/chill-bundles!246
This commit is contained in:
Julien Fastré 2021-12-12 13:55:23 +00:00
commit 1823a1b031
12 changed files with 687 additions and 578 deletions

View File

@ -13,8 +13,8 @@ and this project adheres to
<!-- write down unreleased development here --> <!-- write down unreleased development here -->
* [person] add validator for accompanying period with a test on social issues (https://gitlab.com/champs-libres/departement-de-la-vendee/accent-suivi-developpement/-/issues/76) * [person] add validator for accompanying period with a test on social issues (https://gitlab.com/champs-libres/departement-de-la-vendee/accent-suivi-developpement/-/issues/76)
* [activity] fix visibility for location * [activity] fix visibility for location
* [action] add an unrelated issue within action creation.
* [origin] fix origin: use correctly the translatable strings * [origin] fix origin: use correctly the translatable strings
* /!\ everyone must update the origin table. As there is only one row, execute `update chill_person_accompanying_period_origin set label = jsonb_build_object('fr', 'appel téléphonique');` * /!\ everyone must update the origin table. As there is only one row, execute `update chill_person_accompanying_period_origin set label = jsonb_build_object('fr', 'appel téléphonique');`
## Test releases ## Test releases

View File

@ -34,6 +34,8 @@ use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\Routing\Annotation\Route; use Symfony\Component\Routing\Annotation\Route;
use Symfony\Component\Serializer\Exception\RuntimeException; use Symfony\Component\Serializer\Exception\RuntimeException;
use Symfony\Component\Serializer\Normalizer\AbstractNormalizer; use Symfony\Component\Serializer\Normalizer\AbstractNormalizer;
use Symfony\Component\Validator\ConstraintViolationList;
use Symfony\Component\Validator\ConstraintViolationListInterface;
use Symfony\Component\Validator\Validator\ValidatorInterface; use Symfony\Component\Validator\Validator\ValidatorInterface;
use Symfony\Component\Workflow\Registry; use Symfony\Component\Workflow\Registry;
use function array_values; use function array_values;
@ -294,4 +296,17 @@ final class AccompanyingCourseApiController extends ApiController
return null; return null;
} }
protected function validate(string $action, Request $request, string $_format, $entity, array $more = []): ConstraintViolationListInterface
{
if ('work' !== $action) {
return parent::validate($action, $request, $_format, $entity, $more);
}
if (Request::METHOD_POST === $request->getMethod()) {
return $this->getValidator()->validate($more[0], null);
}
return new ConstraintViolationList([]);
}
} }

View File

@ -79,6 +79,7 @@ class ChillPersonExtension extends Extension implements PrependExtensionInterfac
$loader->load('services/serializer.yaml'); $loader->load('services/serializer.yaml');
$loader->load('services/security.yaml'); $loader->load('services/security.yaml');
$loader->load('services/doctrineEventListener.yaml'); $loader->load('services/doctrineEventListener.yaml');
$loader->load('services/accompanyingPeriodConsistency.yaml');
if ($container->getParameter('chill_person.accompanying_period') !== 'hidden') { if ($container->getParameter('chill_person.accompanying_period') !== 'hidden') {
$loader->load('services/exports_accompanying_period.yaml'); $loader->load('services/exports_accompanying_period.yaml');

View File

@ -14,10 +14,12 @@ namespace Chill\PersonBundle\Entity\AccompanyingPeriod;
use Chill\MainBundle\Doctrine\Model\TrackCreationInterface; use Chill\MainBundle\Doctrine\Model\TrackCreationInterface;
use Chill\MainBundle\Doctrine\Model\TrackUpdateInterface; use Chill\MainBundle\Doctrine\Model\TrackUpdateInterface;
use Chill\MainBundle\Entity\User; use Chill\MainBundle\Entity\User;
use Chill\PersonBundle\AccompanyingPeriod\SocialIssueConsistency\AccompanyingPeriodLinkedWithSocialIssuesEntityInterface;
use Chill\PersonBundle\Entity\AccompanyingPeriod; use Chill\PersonBundle\Entity\AccompanyingPeriod;
use Chill\PersonBundle\Entity\Person; use Chill\PersonBundle\Entity\Person;
use Chill\PersonBundle\Entity\SocialWork\Result; use Chill\PersonBundle\Entity\SocialWork\Result;
use Chill\PersonBundle\Entity\SocialWork\SocialAction; use Chill\PersonBundle\Entity\SocialWork\SocialAction;
use Chill\PersonBundle\Entity\SocialWork\SocialIssue;
use Chill\ThirdPartyBundle\Entity\ThirdParty; use Chill\ThirdPartyBundle\Entity\ThirdParty;
use DateTimeImmutable; use DateTimeImmutable;
use DateTimeInterface; use DateTimeInterface;
@ -38,7 +40,7 @@ use Symfony\Component\Validator\Constraints as Assert;
* } * }
* ) * )
*/ */
class AccompanyingPeriodWork implements TrackCreationInterface, TrackUpdateInterface class AccompanyingPeriodWork implements AccompanyingPeriodLinkedWithSocialIssuesEntityInterface, TrackCreationInterface, TrackUpdateInterface
{ {
/** /**
* @ORM\ManyToOne(targetEntity=AccompanyingPeriod::class) * @ORM\ManyToOne(targetEntity=AccompanyingPeriod::class)
@ -320,6 +322,11 @@ use Symfony\Component\Validator\Constraints as Assert;
return $this->socialAction; return $this->socialAction;
} }
public function getSocialIssues(): Collection
{
return new ArrayCollection([$this->getSocialAction()->getIssue()]);
}
public function getStartDate(): ?DateTimeInterface public function getStartDate(): ?DateTimeInterface
{ {
return $this->startDate; return $this->startDate;
@ -383,6 +390,13 @@ use Symfony\Component\Validator\Constraints as Assert;
return $this; return $this;
} }
public function removeSocialIssue(SocialIssue $issue): AccompanyingPeriodLinkedWithSocialIssuesEntityInterface
{
$this->getSocialIssues()->removeElement($issue);
return $this;
}
public function removeThirdParty(ThirdParty $thirdParty): self public function removeThirdParty(ThirdParty $thirdParty): self
{ {
$this->thirdParties->removeElement($thirdParty); $this->thirdParties->removeElement($thirdParty);

View File

@ -132,6 +132,10 @@ const appMessages = {
sure_description: "Une fois le changement confirmé, il ne sera plus possible de le remettre à l'état de brouillon !", sure_description: "Une fois le changement confirmé, il ne sera plus possible de le remettre à l'état de brouillon !",
ok: "Confirmer le parcours" ok: "Confirmer le parcours"
}, },
action: {
choose_other_social_issue: "Veuillez choisir un autre problématique",
cancel: "Annuler",
},
// catch errors // catch errors
'Error while updating AccompanyingPeriod Course.': "Erreur du serveur lors de la mise à jour du parcours d'accompagnement.", 'Error while updating AccompanyingPeriod Course.': "Erreur du serveur lors de la mise à jour du parcours d'accompagnement.",
'Error while retriving AccompanyingPeriod Course.': "Erreur du serveur lors du chargement du parcours d'accompagnement.", 'Error while retriving AccompanyingPeriod Course.': "Erreur du serveur lors du chargement du parcours d'accompagnement.",

View File

@ -7,13 +7,35 @@
<div id="picking"> <div id="picking">
<p>{{ $t('pick_social_issue_linked_with_action') }}</p> <p>{{ $t('pick_social_issue_linked_with_action') }}</p>
<div v-for="si in socialIssues"> <div v-for="si in socialIssues">
<input type="radio" v-bind:value="si.id" name="socialIssue" v-model="socialIssuePicked"><span class="badge bg-chill-l-gray text-dark">{{ si.text }}</span> <input type="radio" checked v-bind:value="si.id" name="socialIssue" v-model="socialIssuePicked"><span class="badge bg-chill-l-gray text-dark">{{ si.text }}</span>
</div>
<div class="my-3">
<div class="col-8">
<vue-multiselect
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('action.choose_other_social_issue')"
:options="socialIssuesOther"
@select="addIssueInList">
</vue-multiselect>
</div>
</div> </div>
<div v-if="hasSocialIssuePicked"> <div v-if="hasSocialIssuePicked">
<h2>{{ $t('pick_an_action') }}</h2> <h2>{{ $t('pick_an_action') }}</h2>
<div class="col-8">
<vue-multiselect <vue-multiselect
v-model="socialActionPicked" v-model="socialActionPicked"
label="text" label="text"
@ -24,6 +46,7 @@
track-by="id" track-by="id"
></vue-multiselect> ></vue-multiselect>
</div> </div>
</div>
<div v-if="isLoadingSocialActions"> <div v-if="isLoadingSocialActions">
<i class="fa fa-circle-o-notch fa-spin fa-fw"></i> <i class="fa fa-circle-o-notch fa-spin fa-fw"></i>
@ -63,9 +86,9 @@
<div> <div>
<ul class="record_actions"> <ul class="record_actions">
<li class="cancel"> <li class="cancel">
<a href="#" class="btn btn-cancel"> <button class="btn btn-cancel" @click="goToPrevious">
{{ $t('action.cancel') }} {{ $t('action.cancel') }}
</a> </button>
</li> </li>
<li v-if="hasSocialActionPicked"> <li v-if="hasSocialActionPicked">
<button class="btn btn-save" v-show="!isPostingWork" @click="submit"> <button class="btn btn-save" v-show="!isPostingWork" @click="submit">
@ -82,43 +105,7 @@
</template> </template>
<style lang="scss">
#awc_create_form {
display: grid;
grid-template-areas:
"picking picking"
"start_date end_date"
"confirm confirm"
;
grid-template-columns: 50% 50%;
column-gap: 1.5rem;
#picking {
grid-area: picking;
#persons {
ul {
padding: 0;
list-style-type: none;
}
}
}
#start_date {
grid-area: start_date;
}
#end_date {
grid-area: end_date;
}
#confirm {
grid-area: confirm;
}
}
</style>
<script> <script>
import { mapState, mapActions, mapGetters } from 'vuex'; import { mapState, mapActions, mapGetters } from 'vuex';
@ -149,13 +136,34 @@ export default {
}, },
methods: { methods: {
submit() { submit() {
this.$store.dispatch('submit'); this.$store.dispatch('submit')
.catch(({name, violations}) => {
if (name === 'ValidationException' || name === 'AccessException') {
violations.forEach((violation) => this.$toast.open({message: violation}));
} else {
this.$toast.open({message: 'An error occurred'})
} }
});
},
addIssueInList(value) {
this.$store.commit('addIssueInList', value);
this.$store.commit('removeIssueInOther', value);
this.$store.dispatch('pickSocialIssue', value.id);
},
goToPrevious() {
let params = new URLSearchParams(window.location.search);
if (params.has('returnPath')) {
window.location.replace(params.get('returnPath'));
} else {
return;
}
},
}, },
i18n, i18n,
computed: { computed: {
...mapState([ ...mapState([
'socialIssues', 'socialIssues',
'socialIssuesOther',
'socialActionsReachables', 'socialActionsReachables',
'errors', 'errors',
'personsReachables', 'personsReachables',
@ -170,12 +178,12 @@ export default {
personsPicked: { personsPicked: {
get() { get() {
let s = this.$store.state.personsPicked.map(p => p.id); let s = this.$store.state.personsPicked.map(p => p.id);
console.log('persons picked', s); // console.log('persons picked', s);
return s; return s;
}, },
set(v) { set(v) {
console.log('persons picked', v); // console.log('persons picked', v);
this.$store.commit('setPersonsPickedIds', v); this.$store.commit('setPersonsPickedIds', v);
} }
}, },
@ -235,3 +243,41 @@ span.badge {
margin-left: 1em; margin-left: 1em;
} }
</style> </style>
<style lang="scss">
#awc_create_form {
display: grid;
grid-template-areas:
"picking picking"
"start_date end_date"
"confirm confirm"
;
grid-template-columns: 50% 50%;
column-gap: 1.5rem;
#picking {
grid-area: picking;
#persons {
ul {
padding: 0;
list-style-type: none;
}
}
}
#start_date {
grid-area: start_date;
}
#end_date {
grid-area: end_date;
}
#confirm {
grid-area: confirm;
}
}
</style>

View File

@ -2,7 +2,8 @@
import { createStore } from 'vuex'; import { createStore } from 'vuex';
import { datetimeToISO } from 'ChillMainAssets/chill/js/date.js'; import { datetimeToISO } from 'ChillMainAssets/chill/js/date.js';
import { findSocialActionsBySocialIssue } from 'ChillPersonAssets/vuejs/_api/SocialWorkSocialAction.js'; import { findSocialActionsBySocialIssue } from 'ChillPersonAssets/vuejs/_api/SocialWorkSocialAction.js';
import { create } from 'ChillPersonAssets/vuejs/_api/AccompanyingCourseWork.js'; // import { create } from 'ChillPersonAssets/vuejs/_api/AccompanyingCourseWork.js';
import { makeFetch } from 'ChillMainAssets/lib/api/apiMethods';
const debug = process.env.NODE_ENV !== 'production'; const debug = process.env.NODE_ENV !== 'production';
@ -12,6 +13,7 @@ const store = createStore({
accompanyingCourse: window.accompanyingCourse, accompanyingCourse: window.accompanyingCourse,
socialIssues: window.accompanyingCourse.socialIssues, socialIssues: window.accompanyingCourse.socialIssues,
socialIssuePicked: null, socialIssuePicked: null,
socialIssuesOther: [],
socialActionsReachables: [], socialActionsReachables: [],
socialActionPicked: null, socialActionPicked: null,
personsPicked: window.accompanyingCourse.participations.filter(p => p.endDate == null) personsPicked: window.accompanyingCourse.participations.filter(p => p.endDate == null)
@ -26,7 +28,6 @@ const store = createStore({
}, },
getters: { getters: {
hasSocialActionPicked(state) { hasSocialActionPicked(state) {
console.log(state.socialActionPicked);
return null !== state.socialActionPicked; return null !== state.socialActionPicked;
}, },
hasSocialIssuePicked(state) { hasSocialIssuePicked(state) {
@ -73,27 +74,43 @@ const store = createStore({
}, },
mutations: { mutations: {
setSocialActionsReachables(state, actions) { setSocialActionsReachables(state, actions) {
console.log('set social action reachables'); // console.log('set social action reachables');
console.log(actions); // console.log(actions);
state.socialActionsReachables = actions; state.socialActionsReachables = actions;
}, },
setSocialAction(state, socialAction) { setSocialAction(state, socialAction) {
console.log('socialAction', socialAction); // console.log('socialAction', socialAction);
state.socialActionPicked = socialAction; state.socialActionPicked = socialAction;
}, },
setSocialIssue(state, socialIssueId) { setSocialIssue(state, socialIssueId) {
console.log('set social issue', socialIssueId); // console.log('set social issue', socialIssueId);
if (socialIssueId === null) { if (socialIssueId === null) {
state.socialIssuePicked = null; state.socialIssuePicked = null;
} else { } else {
let mapped = state.socialIssues let mapped = state.socialIssues
.find(e => e.id === socialIssueId); .find(e => e.id === socialIssueId);
console.log('mapped', mapped);
state.socialIssuePicked = mapped; state.socialIssuePicked = mapped;
console.log('social issue setted', state.socialIssuePicked); // console.log('social issue setted', state.socialIssuePicked);
} }
}, },
addIssueInList(state, issue) {
//console.log('add issue list', issue.id);
state.socialIssues.push(issue);
},
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
);
},
updateSelected(state, payload) {
state.socialIssueSelected = payload;
},
setIsLoadingSocialActions(state, s) { setIsLoadingSocialActions(state, s) {
state.isLoadingSocialActions = s; state.isLoadingSocialActions = s;
}, },
@ -131,8 +148,6 @@ const store = createStore({
findSocialActionsBySocialIssue(socialIssueId).then( findSocialActionsBySocialIssue(socialIssueId).then(
(response) => { (response) => {
console.log(response);
console.log(response.results);
commit('setSocialIssue', socialIssueId); commit('setSocialIssue', socialIssueId);
commit('setSocialActionsReachables', response.results); commit('setSocialActionsReachables', response.results);
commit('setIsLoadingSocialActions', false); commit('setIsLoadingSocialActions', false);
@ -142,34 +157,34 @@ const store = createStore({
}); });
}, },
submit({ commit, getters, state }) { submit({ commit, getters, state }) {
console.log('submit'); let payload = getters.buildPayloadCreate;
let const url = `/api/1.0/person/accompanying-course/${state.accompanyingCourse.id}/work.json`;
payload = getters.buildPayloadCreate,
errors = [];
commit('setPostingWork'); commit('setPostingWork');
create(state.accompanyingCourse.id, payload) makeFetch('POST', url, payload)
.then( ({status, data}) => { .then((response) => {
console.log('created return', { status, data}); window.location.assign(`/fr/person/accompanying-period/work/${response.id}/edit`)
if (status === 200) { })
console.log('created, nothing to do here any more. Bye-bye!'); .catch((error) => {
window.location.assign(`/fr/person/accompanying-period/work/${data.id}/edit`); throw error;
} else if (status === 422) {
console.log(data);
for (let i in data.violations) {
console.log(i);
console.log(data.violations[i].title);
errors.push(data.violations[i].title);
}
console.log('errors after reseponse handling', errors);
console.log({errors, cancel_posting: true});
commit('addErrors', { errors, cancel_posting: true });
}
}); });
}, },
fetchOtherSocialIssues({commit}) {
const url = `/api/1.0/person/social-work/social-issue.json`;
return makeFetch('GET', url)
.then((response) => {
commit('updateIssuesOther', response.results);
})
.catch((error) => {
throw error;
})
}
}, },
}); });
store.dispatch('fetchOtherSocialIssues');
export { store }; export { store };

View File

@ -1,30 +1,30 @@
const create = (accompanying_period_id, payload) => { // const create = (accompanying_period_id, payload) => {
const url = `/api/1.0/person/accompanying-course/${accompanying_period_id}/work.json`; // const url = `/api/1.0/person/accompanying-course/${accompanying_period_id}/work.json`;
let status; // let status;
console.log('create', payload); // console.log('create', payload);
return fetch(url, { // return fetch(url, {
method: 'POST', // method: 'POST',
headers: { // headers: {
'Content-Type': 'application/json', // 'Content-Type': 'application/json',
}, // },
body: JSON.stringify(payload), // body: JSON.stringify(payload),
}) // })
.then(response => { // .then(response => {
if (response.ok || response.status === 422) { // if (response.ok || response.status === 422) {
status = response.status; // status = response.status;
return response.json(); // return response.json();
} // }
throw new Error("Error while retrieving social actions: " + response.status + // throw new Error("Error while retrieving social actions: " + response.status +
" " + response.statusText); // " " + response.statusText);
}) // })
.then(data => { // .then(data => {
return new Promise((resolve, reject) => { // return new Promise((resolve, reject) => {
resolve({ status, data }); // resolve({ status, data });
}); // });
}); // });
}; // };
export { create }; // export { create };

View File

@ -0,0 +1,14 @@
services:
accompanying_period_social_issue_consistency_with_action:
class: 'Chill\PersonBundle\AccompanyingPeriod\SocialIssueConsistency\AccompanyingPeriodSocialIssueConsistencyEntityListener'
tags:
-
name: 'doctrine.orm.entity_listener'
event: 'prePersist'
entity: 'Chill\PersonBundle\Entity\AccompanyingPeriod\AccompanyingPeriodWork'
lazy: true
-
name: 'doctrine.orm.entity_listener'
event: 'preUpdate'
entity: 'Chill\PersonBundle\Entity\AccompanyingPeriod\AccompanyingPeriodWork'
lazy: true

@ -1 +1 @@
Subproject commit 5952eda44831896991989c2e4881adc26329140e Subproject commit bd95d3c96a437757b7e8f35cdfd30da9aeac1a01