From 6efa50d3ef3caaa2975e9aebdb284d934f4bc958 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Mon, 14 Oct 2024 11:35:23 +0300 Subject: [PATCH 01/56] #13159 Automatically (Soft-)Delete Samples & Pathogen Tests with Negative Test Results for COVID-19 --- .../de/symeda/sormas/api/ConfigFacade.java | 2 + .../backend/common/ConfigFacadeEjb.java | 6 +++ .../sormas/backend/common/CronService.java | 8 ++++ .../sormas/backend/sample/SampleService.java | 38 +++++++++++++++++++ .../backend/sample/SampleServiceTest.java | 32 ++++++++++++++++ 5 files changed, 86 insertions(+) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java index a27d409517e..496c696ef02 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java @@ -161,4 +161,6 @@ public interface ConfigFacade { CaseClassificationCalculationMode getCaseClassificationCalculationMode(Disease disease); boolean isAnyCaseClassificationCalculationEnabled(); + + Integer getNegaiveCovidSamplesMaxAgeDays(); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java index affeadc8726..082182554ce 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java @@ -187,6 +187,7 @@ public class ConfigFacadeEjb implements ConfigFacade { public static final int DEFAULT_DOCUMENT_UPLOAD_SIZE_LIMIT_MB = 20; public static final String IMPORT_FILE_SIZE_LIMIT_MB = "importFileSizeLimitMb"; public static final int DEFAULT_IMPOR_FILE_SIZE_LIMIT_MB = 20; + public static final String NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS = "negaiveCovidSamplesMaxAgeDays"; private final Logger logger = LoggerFactory.getLogger(getClass()); @@ -838,6 +839,11 @@ public void resetRequestContext() { RequestContextHolder.reset(); } + @Override + public Integer getNegaiveCovidSamplesMaxAgeDays() { + return parseProperty(NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS, null, Integer::parseInt); + } + @LocalBean @Stateless public static class ConfigFacadeEjbLocal extends ConfigFacadeEjb { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java index ec8e92357c2..6bb6d7dda83 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java @@ -48,6 +48,7 @@ import de.symeda.sormas.backend.immunization.ImmunizationFacadeEjb; import de.symeda.sormas.backend.infrastructure.central.CentralInfraSyncFacade; import de.symeda.sormas.backend.report.WeeklyReportFacadeEjb.WeeklyReportFacadeEjbLocal; +import de.symeda.sormas.backend.sample.SampleService; import de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccessFacadeEjb.SpecialCaseAccessFacadeEjbLocal; import de.symeda.sormas.backend.systemevent.SystemEventFacadeEjb.SystemEventFacadeEjbLocal; import de.symeda.sormas.backend.task.TaskFacadeEjb.TaskFacadeEjbLocal; @@ -96,6 +97,8 @@ public class CronService { private SpecialCaseAccessFacadeEjbLocal specialCaseAccessFacade; @EJB private UserFacadeEjbLocal userFacade; + @EJB + private SampleService sampleService; @Schedule(hour = "*", minute = "*/" + TASK_UPDATE_INTERVAL, second = "0", persistent = false) public void sendNewAndDueTaskMessages() { @@ -297,4 +300,9 @@ public void syncUsersFromAuthenticationProvider() { userFacade.syncUsersFromAuthenticationProvider(); } } + + @Schedule(hour = "2", minute = "40", persistent = false) + public void sofDeleteOldNegativeSamples() { + sampleService.deleteOldNegativeSamples(); + } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java index 1295c31c7e8..95e58b65ed8 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java @@ -56,17 +56,21 @@ import org.apache.commons.collections4.CollectionUtils; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.EntityRelevanceStatus; import de.symeda.sormas.api.RequestContextHolder; import de.symeda.sormas.api.caze.IsCase; import de.symeda.sormas.api.common.DeletableEntityType; import de.symeda.sormas.api.common.DeletionDetails; +import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.common.progress.ProcessedEntity; import de.symeda.sormas.api.common.progress.ProcessedEntityStatus; import de.symeda.sormas.api.contact.ContactReferenceDto; import de.symeda.sormas.api.disease.DiseaseVariant; import de.symeda.sormas.api.event.EventParticipantReferenceDto; import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.sample.IsSample; import de.symeda.sormas.api.sample.PathogenTestResultType; import de.symeda.sormas.api.sample.PathogenTestType; @@ -87,6 +91,7 @@ import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractDeletableAdoService; import de.symeda.sormas.backend.common.AbstractDomainObject; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.common.DeletableAdo; import de.symeda.sormas.backend.common.JurisdictionFlagsService; @@ -145,6 +150,8 @@ public class SampleService extends AbstractDeletableAdoService protected FeatureConfigurationFacadeEjbLocal featureConfigurationFacade; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; public SampleService() { super(Sample.class, DeletableEntityType.SAMPLE); @@ -1276,4 +1283,35 @@ public List getAssociatedDiseaseVariants(String sampleUuid) { cq.select(pathogenTestJoin.get(PathogenTest.TESTED_DISEASE_VARIANT)); return em.createQuery(cq).getResultList(); } + + public void deleteOldNegativeSamples() { + final Integer maxAgeDays = configFacade.getNegaiveCovidSamplesMaxAgeDays(); + if (maxAgeDays == null) { + return; + } + + CriteriaBuilder cb = em.getCriteriaBuilder(); + CriteriaQuery cq = cb.createQuery(Sample.class); + Root from = cq.from(Sample.class); + SampleJoins sampleJoins = new SampleJoins(from); + + Subquery covidTestQuery = cq.subquery(PathogenTest.class); + Root covidTestRoot = covidTestQuery.from(PathogenTest.class); + covidTestQuery.where( + cb.equal(covidTestRoot.get(PathogenTest.SAMPLE), from), + cb.equal(covidTestRoot.get(PathogenTest.TESTED_DISEASE), Disease.CORONAVIRUS)); + + cq.where( + cb.or( + cb.equal(sampleJoins.getCaze().get(Case.DISEASE), Disease.CORONAVIRUS), + cb.equal(sampleJoins.getContact().get(Contact.DISEASE), Disease.CORONAVIRUS), + cb.equal(sampleJoins.getContactCase().get(Case.DISEASE), Disease.CORONAVIRUS), + cb.equal(sampleJoins.getEventParticipantJoins().getEvent().get(Event.DISEASE), Disease.CORONAVIRUS)), + cb.exists(covidTestQuery), + cb.equal(from.get(Sample.PATHOGEN_TEST_RESULT), PathogenTestResultType.NEGATIVE), + cb.greaterThan(from.get(Sample.SAMPLE_DATE_TIME), DateHelper.subtractDays(new Date(), maxAgeDays))); + em.createQuery(cq) + .getResultList() + .forEach(s -> delete(s, new DeletionDetails(DeletionReason.OTHER_REASON, I18nProperties.getString(Strings.entityAutomaticSoftDeletion)))); + } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java index 27cba075c46..942585a5c7a 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java @@ -1,10 +1,13 @@ package de.symeda.sormas.backend.sample; +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertNotNull; import static org.junit.jupiter.api.Assertions.assertNull; import static org.junit.jupiter.api.Assertions.assertTrue; +import java.util.Date; import java.util.List; import org.junit.jupiter.api.Test; @@ -13,10 +16,16 @@ import de.symeda.sormas.api.common.DeletionDetails; import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.externalmessage.ExternalMessageDto; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.sample.PathogenTestResultType; +import de.symeda.sormas.api.sample.SampleCriteria; import de.symeda.sormas.api.sample.SampleDto; import de.symeda.sormas.api.user.DefaultUserRole; import de.symeda.sormas.api.user.UserDto; +import de.symeda.sormas.api.user.UserReferenceDto; +import de.symeda.sormas.api.utils.DateHelper; import de.symeda.sormas.backend.AbstractBeanTest; import de.symeda.sormas.backend.TestDataCreator; @@ -64,4 +73,27 @@ public void testSamplePermanentDeletion() { assertNull(getSampleService().getByUuid(referralSample.getUuid()).getReferredTo()); assertNull(getSampleReportService().getByUuid(labMessage.getSampleReports().get(0).getUuid()).getSample()); } + + @Test + public void testDeleteOldNegativeSamples() { + TestDataCreator.RDCF rdcf = creator.createRDCF(); + UserReferenceDto user = creator.createUser(rdcf).toReference(); + CaseDataDto caze = creator.createCase(user, creator.createPerson().toReference(), rdcf); + SampleDto newSample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setSampleDateTime(new Date()); + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + SampleDto oldSample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setSampleDateTime(DateHelper.subtractDays(new Date(), 14)); + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().deleteOldNegativeSamples(); + + assertThat(getSampleFacade().count(new SampleCriteria()), is(1L)); + SampleDto deletedSample = getSampleFacade().getSampleByUuid(newSample.getUuid()); + assertThat(deletedSample.isDeleted(), is(true)); + assertThat(deletedSample.getDeletionReason(), is(DeletionReason.OTHER_REASON)); + assertThat(deletedSample.getOtherDeletionReason(), is(I18nProperties.getString(Strings.entityAutomaticSoftDeletion))); + } } From 8d143e1a0c855e123a0d85b2a0166463ffe67edd Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Mon, 14 Oct 2024 13:00:40 +0300 Subject: [PATCH 02/56] #13156 Hide Case Classification for RSV and Automatically Set Default Values --- .../symeda/sormas/api/caze/CaseDataDto.java | 15 ++++++++++ .../sormas/backend/caze/CaseFacadeEjb.java | 16 +++++++--- .../symeda/sormas/ui/caze/CaseDataForm.java | 29 ++++++++++--------- 3 files changed, 43 insertions(+), 17 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java index 8500490afa3..1fbfea11ef4 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java @@ -283,16 +283,20 @@ public class CaseDataDto extends SormasToSormasShareableDto implements IsCase { COUNTRY_CODE_SWITZERLAND }) private Date districtLevelDate; @Outbreaks + @Diseases(value = Disease.RESPIRATORY_SYNCYTIAL_VIRUS, hide = true) private CaseClassification caseClassification; @HideForCountriesExcept private CaseIdentificationSource caseIdentificationSource; @HideForCountriesExcept private ScreeningType screeningType; @Outbreaks + @Diseases(value = Disease.RESPIRATORY_SYNCYTIAL_VIRUS, hide = true) private UserReferenceDto classificationUser; @Outbreaks + @Diseases(value = Disease.RESPIRATORY_SYNCYTIAL_VIRUS, hide = true) private Date classificationDate; @Outbreaks + @Diseases(value = Disease.RESPIRATORY_SYNCYTIAL_VIRUS, hide = true) @SensitiveData @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String classificationComment; @@ -567,21 +571,32 @@ public class CaseDataDto extends SormasToSormasShareableDto implements IsCase { private YesNoUnknown bloodOrganOrTissueDonated; @HideForCountriesExcept + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) private boolean notACaseReasonNegativeTest; @HideForCountriesExcept + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) private boolean notACaseReasonPhysicianInformation; @HideForCountriesExcept + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) private boolean notACaseReasonDifferentPathogen; @HideForCountriesExcept + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) private boolean notACaseReasonOther; @HideForCountriesExcept @SensitiveData + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) private String notACaseReasonDetails; + private Date followUpStatusChangeDate; private UserReferenceDto followUpStatusChangeUser; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java index a698d2c3ba4..9583ee83690 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java @@ -2066,7 +2066,8 @@ private void updateTasksOnCaseChanged(Case newCase, CaseDataDto existingCase) { @PermitAll public void onCaseSampleChanged(Case associatedCase) { // Update case classification if the feature is enabled - if (configFacade.getCaseClassificationCalculationMode(associatedCase.getDisease()).isAutomaticEnabled()) { + if (configFacade.getCaseClassificationCalculationMode(associatedCase.getDisease()).isAutomaticEnabled() + & associatedCase.getDisease() != Disease.RESPIRATORY_SYNCYTIAL_VIRUS) { if (associatedCase.getCaseClassification() != CaseClassification.NO_CASE) { Long pathogenTestsCount = pathogenTestService.countByCase(associatedCase); if (pathogenTestsCount == 0) { @@ -2200,8 +2201,14 @@ public void onCaseChanged(CaseDataDto existingCase, Case newCase, boolean syncSh service.updateFollowUpDetails(newCase, existingCase != null && newCase.getFollowUpStatus() != existingCase.getFollowUpStatus()); updateTasksOnCaseChanged(newCase, existingCase); - - handleClassificationOnCaseChange(existingCase, newCase); + if ((existingCase == null || existingCase.getDisease() != newCase.getDisease()) + && newCase.getDisease() == Disease.RESPIRATORY_SYNCYTIAL_VIRUS) { + newCase.setCaseClassification(CaseClassification.CONFIRMED); + newCase.setClassificationDate(newCase.getReportDate()); + newCase.setClassificationUser(newCase.getReportingUser()); + } else { + handleClassificationOnCaseChange(existingCase, newCase); + } // Set Yes/No/Unknown fields associated with embedded lists to Yes if the lists // are not empty @@ -2263,7 +2270,8 @@ private void handleClassificationOnCaseChange(CaseDataDto existingDto, Case save // Update case classification if the feature is enabled CaseClassification classification = null; boolean setClassificationInfo = true; - if (configFacade.getCaseClassificationCalculationMode(savedCase.getDisease()).isAutomaticEnabled()) { + if (configFacade.getCaseClassificationCalculationMode(savedCase.getDisease()).isAutomaticEnabled() + & savedCase.getDisease() != Disease.RESPIRATORY_SYNCYTIAL_VIRUS) { if (savedCase.getCaseClassification() != CaseClassification.NO_CASE || configFacade.isConfiguredCountry(CountryHelper.COUNTRY_CODE_LUXEMBOURG)) { // calculate classification diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java index 5fc2d4a525b..b7d82324f84 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java @@ -523,19 +523,6 @@ protected void addFields() { caseClassificationGroup.removeItem(CaseClassification.CONFIRMED_UNKNOWN_SYMPTOMS); } - if (diseaseClassificationExists() && FacadeProvider.getConfigFacade().getCaseClassificationCalculationMode(disease).isManualEnabled()) { - Button caseClassificationCalculationButton = ButtonHelper.createButton(Captions.caseClassificationCalculationButton, e -> { - CaseClassification classification = FacadeProvider.getCaseClassificationFacade().getClassification(getValue()); - ((Field) getField(CaseDataDto.CASE_CLASSIFICATION)).setValue(classification); - }, ValoTheme.BUTTON_PRIMARY, FORCE_CAPTION); - - getContent().addComponent(caseClassificationCalculationButton, CASE_CLASSIFICATION_CALCULATE_BTN_LOC); - - if (!UiUtil.permitted(UserRight.CASE_CLASSIFY)) { - caseClassificationCalculationButton.setEnabled(false); - } - } - boolean extendedClassification = FacadeProvider.getDiseaseConfigurationFacade().usesExtendedClassification(disease); if (extendedClassification) { @@ -1033,6 +1020,22 @@ protected void addFields() { CaseDataDto.OUTCOME, CaseDataDto.DISEASE); setSoftRequired(true, CaseDataDto.INVESTIGATED_DATE, CaseDataDto.OUTCOME_DATE, CaseDataDto.PLAGUE_TYPE, CaseDataDto.SURVEILLANCE_OFFICER); + + if (diseaseClassificationExists() + && FacadeProvider.getConfigFacade().getCaseClassificationCalculationMode(disease).isManualEnabled() + && isVisibleAllowed(CaseDataDto.CASE_CLASSIFICATION)) { + Button caseClassificationCalculationButton = ButtonHelper.createButton(Captions.caseClassificationCalculationButton, e -> { + CaseClassification classification = FacadeProvider.getCaseClassificationFacade().getClassification(getValue()); + ((Field) getField(CaseDataDto.CASE_CLASSIFICATION)).setValue(classification); + }, ValoTheme.BUTTON_PRIMARY, FORCE_CAPTION); + + getContent().addComponent(caseClassificationCalculationButton, CASE_CLASSIFICATION_CALCULATE_BTN_LOC); + + if (!UiUtil.permitted(UserRight.CASE_CLASSIFY)) { + caseClassificationCalculationButton.setEnabled(false); + } + } + if (isEditableAllowed(CaseDataDto.INVESTIGATED_DATE)) { FieldHelper.setVisibleWhen( getFieldGroup(), From 13b5a4b0bbc30f86a74a9a6af712002f7f9a0fca Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Mat=C3=A9=20Strysewske?= <23701005+MateStrysewske@users.noreply.github.com> Date: Wed, 16 Oct 2024 10:39:25 +0200 Subject: [PATCH 03/56] New Crowdin updates (#13129) * New translations captions.properties (Urdu (Pakistan)) * New translations captions.properties (German, Switzerland) * New translations captions.properties (French, Switzerland) * New translations captions.properties (Italian, Switzerland) * New translations captions.properties (Dari) * New translations captions.properties (Pashto) * New translations captions.properties (Spanish, Cuba) * New translations captions.properties (English, Afghanistan) * New translations captions.properties (English, Nigeria) * New translations captions.properties (English, Ghana) * New translations captions.properties (French, Tunisia) * New translations strings.properties (Czech) * New translations captions.properties (English, Kenya) * New translations enum.properties (English, Kenya) * New translations strings.properties (English, Kenya) * New translations captions.properties (English, Liberia) * New translations enum.properties (English, Liberia) * New translations strings.properties (English, Liberia) * New translations captions.properties (English, Gambia) * New translations enum.properties (English, Gambia) * New translations strings.properties (English, Gambia) * New translations captions.properties (Portuguese, Cape Verde) * New translations enum.properties (Portuguese, Cape Verde) * New translations strings.properties (Portuguese, Cape Verde) * New translations strings.properties (French) * New translations strings.properties (Arabic) * New translations strings.properties (German) * New translations strings.properties (Finnish) * New translations strings.properties (Italian) * New translations strings.properties (Polish) * New translations strings.properties (Russian) * New translations strings.properties (Chinese Simplified) * New translations strings.properties (Urdu (Pakistan)) * New translations strings.properties (German, Switzerland) * New translations strings.properties (French, Switzerland) * New translations strings.properties (Italian, Switzerland) * New translations strings.properties (Dari) * New translations strings.properties (Pashto) * New translations strings.properties (Spanish, Cuba) * New translations strings.properties (English, Afghanistan) * New translations strings.properties (English, Nigeria) * New translations strings.properties (English, Ghana) * New translations strings.properties (French, Tunisia) * New translations enum.properties (French) * New translations enum.properties (Arabic) * New translations enum.properties (German) * New translations enum.properties (Finnish) * New translations enum.properties (Italian) * New translations enum.properties (Polish) * New translations enum.properties (Russian) * New translations enum.properties (Chinese Simplified) * New translations enum.properties (Urdu (Pakistan)) * New translations enum.properties (German, Switzerland) * New translations enum.properties (Spanish, Bolivia) * New translations enum.properties (French, Switzerland) * New translations enum.properties (Italian, Switzerland) * New translations enum.properties (Dari) * New translations enum.properties (Pashto) * New translations enum.properties (Spanish, Cuba) * New translations enum.properties (English, Afghanistan) * New translations enum.properties (English, Nigeria) * New translations enum.properties (English, Ghana) * New translations enum.properties (French, Tunisia) * New translations validations.properties (Spanish, Bolivia) * New translations validations.properties (French) * New translations validations.properties (Arabic) * New translations validations.properties (Czech) * New translations validations.properties (German) * New translations validations.properties (Finnish) * New translations validations.properties (Italian) * New translations validations.properties (Polish) * New translations validations.properties (Russian) * New translations validations.properties (Chinese Simplified) * New translations validations.properties (Urdu (Pakistan)) * New translations validations.properties (German, Switzerland) * New translations validations.properties (French, Switzerland) * New translations validations.properties (Italian, Switzerland) * New translations validations.properties (Dari) * New translations validations.properties (Pashto) * New translations validations.properties (Spanish, Cuba) * New translations validations.properties (English, Afghanistan) * New translations validations.properties (English, Nigeria) * New translations validations.properties (English, Ghana) * New translations validations.properties (French, Tunisia) * New translations validations.properties (English, Kenya) * New translations validations.properties (English, Liberia) * New translations validations.properties (English, Gambia) * New translations validations.properties (Portuguese, Cape Verde) * New translations captions.properties (Spanish, Cuba) * New translations strings.properties (Spanish, Cuba) * New translations enum.properties (Spanish, Cuba) * New translations strings.properties (Spanish, Cuba) * New translations validations.properties (Spanish, Cuba) * New translations captions.properties (English, Kenya) * New translations enum.properties (English, Kenya) * New translations enum.properties (English, Kenya) * New translations enum.properties (English, Kenya) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations enum.properties (French) * New translations enum.properties (Arabic) * New translations enum.properties (Czech) * New translations enum.properties (German) * New translations enum.properties (Finnish) * New translations enum.properties (Italian) * New translations enum.properties (Polish) * New translations enum.properties (Russian) * New translations enum.properties (Chinese Simplified) * New translations enum.properties (Urdu (Pakistan)) * New translations enum.properties (German, Switzerland) * New translations enum.properties (Spanish, Bolivia) * New translations enum.properties (French, Switzerland) * New translations enum.properties (Italian, Switzerland) * New translations enum.properties (Dari) * New translations enum.properties (Pashto) * New translations enum.properties (Spanish, Cuba) * New translations enum.properties (English, Afghanistan) * New translations enum.properties (English, Nigeria) * New translations enum.properties (English, Ghana) * New translations enum.properties (French, Tunisia) * New translations enum.properties (English, Kenya) * New translations enum.properties (English, Liberia) * New translations enum.properties (English, Gambia) * New translations enum.properties (Portuguese, Cape Verde) * New translations enum.properties (French) * New translations enum.properties (Arabic) * New translations enum.properties (Czech) * New translations enum.properties (German) * New translations enum.properties (Finnish) * New translations enum.properties (Italian) * New translations enum.properties (Polish) * New translations enum.properties (Russian) * New translations enum.properties (Chinese Simplified) * New translations enum.properties (Urdu (Pakistan)) * New translations enum.properties (German, Switzerland) * New translations enum.properties (Spanish, Bolivia) * New translations enum.properties (French, Switzerland) * New translations enum.properties (Italian, Switzerland) * New translations enum.properties (Dari) * New translations enum.properties (Pashto) * New translations enum.properties (Spanish, Cuba) * New translations enum.properties (English, Afghanistan) * New translations enum.properties (English, Nigeria) * New translations enum.properties (English, Ghana) * New translations enum.properties (French, Tunisia) * New translations enum.properties (English, Kenya) * New translations enum.properties (English, Liberia) * New translations enum.properties (English, Gambia) * New translations enum.properties (Portuguese, Cape Verde) * New translations enum.properties (Spanish, Cuba) * New translations enum.properties (Spanish, Cuba) * New translations enum.properties (English, Kenya) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations strings.properties (French) * New translations strings.properties (Arabic) * New translations strings.properties (Czech) * New translations strings.properties (German) * New translations strings.properties (Finnish) * New translations strings.properties (Italian) * New translations strings.properties (Polish) * New translations strings.properties (Russian) * New translations strings.properties (Chinese Simplified) * New translations strings.properties (Urdu (Pakistan)) * New translations strings.properties (German, Switzerland) * New translations strings.properties (Spanish, Bolivia) * New translations strings.properties (French, Switzerland) * New translations strings.properties (Italian, Switzerland) * New translations strings.properties (Dari) * New translations strings.properties (Pashto) * New translations strings.properties (Spanish, Cuba) * New translations strings.properties (English, Afghanistan) * New translations strings.properties (English, Nigeria) * New translations strings.properties (English, Ghana) * New translations strings.properties (French, Tunisia) * New translations strings.properties (English, Kenya) * New translations strings.properties (English, Liberia) * New translations strings.properties (English, Gambia) * New translations strings.properties (Portuguese, Cape Verde) * New translations strings.properties (French) * New translations strings.properties (Spanish, Cuba) * New translations captions.properties (Czech) * New translations enum.properties (Czech) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (French, Tunisia) * New translations enum.properties (French, Tunisia) * New translations captions.properties (Czech) * New translations enum.properties (Czech) * New translations strings.properties (Czech) * New translations strings.xml (Czech) * New translations strings.properties (Czech) * New translations captions.properties (French, Tunisia) * New translations strings.xml (French, Tunisia) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (Arabic) * New translations captions.properties (French, Tunisia) * New translations captions.properties (Czech) * New translations captions.properties (French) * New translations captions.properties (German) * New translations captions.properties (Finnish) * New translations captions.properties (Italian) * New translations captions.properties (Polish) * New translations captions.properties (Russian) * New translations captions.properties (Chinese Simplified) * New translations captions.properties (Urdu (Pakistan)) * New translations captions.properties (German, Switzerland) * New translations captions.properties (Spanish, Bolivia) * New translations captions.properties (French, Switzerland) * New translations captions.properties (Italian, Switzerland) * New translations captions.properties (Dari) * New translations captions.properties (Pashto) * New translations captions.properties (Spanish, Cuba) * New translations captions.properties (English, Afghanistan) * New translations captions.properties (English, Nigeria) * New translations captions.properties (English, Ghana) * New translations captions.properties (English, Kenya) * New translations captions.properties (English, Liberia) * New translations captions.properties (English, Gambia) * New translations captions.properties (Portuguese, Cape Verde) * New translations captions.properties (French) * New translations captions.properties (Spanish, Cuba) --------- Co-authored-by: sergiupacurariu <62688603+sergiupacurariu@users.noreply.github.com> Co-authored-by: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> --- .../main/resources/captions_ar-SA.properties | 1105 +++++++++-------- .../main/resources/captions_cs-CZ.properties | 33 +- .../main/resources/captions_de-CH.properties | 5 +- .../main/resources/captions_de-DE.properties | 5 +- .../main/resources/captions_en-AF.properties | 5 +- .../main/resources/captions_en-GH.properties | 5 +- .../main/resources/captions_en-GM.properties | 5 +- .../main/resources/captions_en-KE.properties | 1 + .../main/resources/captions_en-LR.properties | 5 +- .../main/resources/captions_en-NG.properties | 5 +- .../main/resources/captions_es-BO.properties | 1 + .../main/resources/captions_es-CU.properties | 1 + .../main/resources/captions_fa-AF.properties | 5 +- .../main/resources/captions_fi-FI.properties | 5 +- .../main/resources/captions_fr-CH.properties | 5 +- .../main/resources/captions_fr-FR.properties | 9 +- .../main/resources/captions_fr-TN.properties | 9 +- .../main/resources/captions_it-CH.properties | 5 +- .../main/resources/captions_it-IT.properties | 5 +- .../main/resources/captions_pl-PL.properties | 5 +- .../main/resources/captions_ps-AF.properties | 5 +- .../main/resources/captions_pt-CV.properties | 5 +- .../main/resources/captions_ru-RU.properties | 5 +- .../main/resources/captions_ur-PK.properties | 5 +- .../main/resources/captions_zh-CN.properties | 5 +- .../src/main/resources/enum_ar-SA.properties | 8 +- .../src/main/resources/enum_cs-CZ.properties | 38 +- .../src/main/resources/enum_de-CH.properties | 8 +- .../src/main/resources/enum_de-DE.properties | 8 +- .../src/main/resources/enum_en-AF.properties | 8 +- .../src/main/resources/enum_en-GH.properties | 8 +- .../src/main/resources/enum_en-GM.properties | 8 +- .../src/main/resources/enum_en-KE.properties | 4 + .../src/main/resources/enum_en-LR.properties | 8 +- .../src/main/resources/enum_en-NG.properties | 8 +- .../src/main/resources/enum_es-BO.properties | 8 +- .../src/main/resources/enum_es-CU.properties | 4 + .../src/main/resources/enum_fa-AF.properties | 8 +- .../src/main/resources/enum_fi-FI.properties | 8 +- .../src/main/resources/enum_fr-CH.properties | 8 +- .../src/main/resources/enum_fr-FR.properties | 8 +- .../src/main/resources/enum_fr-TN.properties | 10 +- .../src/main/resources/enum_it-CH.properties | 8 +- .../src/main/resources/enum_it-IT.properties | 8 +- .../src/main/resources/enum_pl-PL.properties | 8 +- .../src/main/resources/enum_ps-AF.properties | 8 +- .../src/main/resources/enum_pt-CV.properties | 8 +- .../src/main/resources/enum_ru-RU.properties | 8 +- .../src/main/resources/enum_ur-PK.properties | 8 +- .../src/main/resources/enum_zh-CN.properties | 8 +- .../main/resources/strings_ar-SA.properties | 4 +- .../main/resources/strings_cs-CZ.properties | 16 +- .../main/resources/strings_de-CH.properties | 2 + .../main/resources/strings_de-DE.properties | 2 + .../main/resources/strings_en-AF.properties | 2 + .../main/resources/strings_en-GH.properties | 2 + .../main/resources/strings_en-GM.properties | 2 + .../main/resources/strings_en-KE.properties | 2 + .../main/resources/strings_en-LR.properties | 2 + .../main/resources/strings_en-NG.properties | 2 + .../main/resources/strings_es-BO.properties | 3 + .../main/resources/strings_es-CU.properties | 2 + .../main/resources/strings_fa-AF.properties | 2 + .../main/resources/strings_fi-FI.properties | 2 + .../main/resources/strings_fr-CH.properties | 2 + .../main/resources/strings_fr-FR.properties | 2 + .../main/resources/strings_fr-TN.properties | 2 + .../main/resources/strings_it-CH.properties | 2 + .../main/resources/strings_it-IT.properties | 2 + .../main/resources/strings_pl-PL.properties | 2 + .../main/resources/strings_ps-AF.properties | 2 + .../main/resources/strings_pt-CV.properties | 2 + .../main/resources/strings_ru-RU.properties | 2 + .../main/resources/strings_ur-PK.properties | 2 + .../main/resources/strings_zh-CN.properties | 2 + .../src/main/res/values-cs-rCZ/strings.xml | 10 +- .../src/main/res/values-fr-rTN/strings.xml | 4 +- 77 files changed, 865 insertions(+), 689 deletions(-) diff --git a/sormas-api/src/main/resources/captions_ar-SA.properties b/sormas-api/src/main/resources/captions_ar-SA.properties index e06ddf3ee62..004418877eb 100644 --- a/sormas-api/src/main/resources/captions_ar-SA.properties +++ b/sormas-api/src/main/resources/captions_ar-SA.properties @@ -428,56 +428,56 @@ CaseData.classificationUser=المستخدم للتصنيف CaseData.classifiedBy=صنف من قبل CaseData.clinicalCourse=الدورة الإكلينيكية CaseData.clinicianName=إسم الاكلينيكي المسؤول -CaseData.clinicianPhone=Phone number of responsible clinician -CaseData.clinicianEmail=Email address of responsible clinician -CaseData.contactOfficer=Contact officer -CaseData.dengueFeverType=Dengue fever type -CaseData.diseaseVariant=Disease variant -CaseData.diseaseDetails=Disease name -CaseData.district=District -CaseData.districtLevelDate=Date received at district level -CaseData.doses=How many doses -CaseData.epiData=Epidemiological data -CaseData.epidNumber=EPID number -CaseData.externalID=External ID -CaseData.externalToken=External Token -CaseData.internalToken=Internal Token -CaseData.caseReferenceNumber=Case Reference Number -CaseData.facilityType=Facility type -CaseData.healthFacility=Facility -CaseData.healthFacilityDetails=Facility name & description -CaseData.hospitalization=Hospitalization -CaseData.investigatedDate=Date of investigation -CaseData.investigationStatus=Investigation status -CaseData.maternalHistory=Maternal history -CaseData.nationalLevelDate=Date received at national level -CaseData.noneHealthFacilityDetails=Place description -CaseData.notifyingClinic=Notifying clinic -CaseData.notifyingClinicDetails=Notifying clinic details -CaseData.numberOfVisits=Number of visits -CaseData.outcome=Outcome of case -CaseData.outcomeDate=Date of outcome -CaseData.person=Case person -CaseData.personUuid=Person ID -CaseData.personFirstName=First name -CaseData.personLastName=Last name -CaseData.plagueType=Plague type -CaseData.pointOfEntry=Point of entry -CaseData.pointOfEntryDetails=Point of entry name & description -CaseData.pointOfEntryName=Point of entry -CaseData.portHealthInfo=Port health -CaseData.postpartum=Postpartum -CaseData.pregnant=Pregnancy -CaseData.previousQuarantineTo=Previous quarantine end -CaseData.quarantineChangeComment=Quarantine change comment -CaseData.region=Region -CaseData.regionLevelDate=Date received at region level -CaseData.reportDate=Date of report -CaseData.reportingUser=Reporting user -CaseData.reportLat=Report GPS latitude -CaseData.reportLon=Report GPS longitude -CaseData.reportLatLonAccuracy=Report GPS accuracy in m -CaseData.sequelae=Sequelae +CaseData.clinicianPhone=رقم هاتف الطبيب المسؤول +CaseData.clinicianEmail=عنوان البريد الإلكتروني للطبيب المسؤول +CaseData.contactOfficer=مسؤول الاتصال +CaseData.dengueFeverType=نوع حمى الضنك +CaseData.diseaseVariant=متغير المرض +CaseData.diseaseDetails=اسم المرض +CaseData.district=المنطقة +CaseData.districtLevelDate=تاريخ الاستلام على مستوى المنطقة +CaseData.doses=كم عدد الجرعات +CaseData.epiData=البيانات الوبائية +CaseData.epidNumber=رقم EPID +CaseData.externalID=المعرف الخارجي +CaseData.externalToken=رمز خارجي +CaseData.internalToken=رمز داخلي +CaseData.caseReferenceNumber=الرقم المرجعي للحالة +CaseData.facilityType=نوع المنشأة +CaseData.healthFacility=منشأة +CaseData.healthFacilityDetails=اسم المنشأة والوصف +CaseData.hospitalization=العلاج في المستشفي +CaseData.investigatedDate=تاريخ التحقيق +CaseData.investigationStatus=حالة التحقيق +CaseData.maternalHistory=التاريخ الأمومي +CaseData.nationalLevelDate=تاريخ الاستلام على المستوى الوطني +CaseData.noneHealthFacilityDetails=وصف المكان +CaseData.notifyingClinic=عيادة الإخطار +CaseData.notifyingClinicDetails=تفاصيل العيادة المخطرة +CaseData.numberOfVisits=عدد الزيارات +CaseData.outcome=حصيلة الحالة +CaseData.outcomeDate=تاريخ النتيجة +CaseData.person=الشخص المسؤول عن الحالة +CaseData.personUuid=معرف الشخص +CaseData.personFirstName=الاسم الأول +CaseData.personLastName=إسم العائلة +CaseData.plagueType=نوع الطاعون +CaseData.pointOfEntry=نقطة الدخول +CaseData.pointOfEntryDetails=اسم ووصف نقطة الإدخال +CaseData.pointOfEntryName=نقطة الدخول +CaseData.portHealthInfo=صحة الميناء +CaseData.postpartum=ما بعد الولادة +CaseData.pregnant=الحمل +CaseData.previousQuarantineTo=نهاية الحجر الصحي السابق +CaseData.quarantineChangeComment=تعليق حول تغيير الحجر الصحي +CaseData.region=إقليم +CaseData.regionLevelDate=تاريخ الاستلام على مستوى الاقليم +CaseData.reportDate=تاريخ التقرير +CaseData.reportingUser=المستخدم المبلغ +CaseData.reportLat=تقرير خط العرض GPS +CaseData.reportLon=تقرير خط الطول GPS +CaseData.reportLatLonAccuracy=تقرير دقة نظام تحديد المواقع العالمي (GPS) بالمتر +CaseData.sequelae=العواقب CaseData.sequelaeDetails=وصف العواقب CaseData.smallpoxVaccinationReceived=هل تم التطعيم ضد الجدزي سابقا؟ CaseData.smallpoxVaccinationScar=هل يوجد ندب التطعيم ضد الجدري؟ @@ -836,60 +836,60 @@ Contact.prohibitionToWorkUntil=حظر العمل حتى Contact.reportingDistrict=المنطقة المبلِّغة Contact.followUpStatusChangeDate=تاريخ تعديل حالة المتابعة Contact.followUpStatusChangeUser=المستخدم المسؤول -Contact.expectedFollowUpUntil=Expected follow-up until -Contact.vaccinationStatus=Vaccination status -Contact.changeDate=Date of last change -Contact.creationDate=Creation date +Contact.expectedFollowUpUntil=المتابعة المتوقعة حتى +Contact.vaccinationStatus=حالة التطعيم +Contact.changeDate=تاريخ التغيير الاخير +Contact.creationDate=تاريخ الانشاء # ContactExport -ContactExport.address=Address -ContactExport.addressDistrict=Address District -ContactExport.addressRegion=Address Region -ContactExport.addressCommunity=Address Community -ContactExport.burialAttended=Visited a burial -ContactExport.contactWithRodent=Contact with rodent? -ContactExport.directContactConfirmedCase=Direct contact with a confirmed case -ContactExport.directContactProbableCase=Direct contact with a probable or confirmed case -ContactExport.firstName=First name of contact person -ContactExport.lastCooperativeVisitDate=Date of last cooperative visit -ContactExport.lastCooperativeVisitSymptomatic=Symptomatic at last cooperative visit? -ContactExport.lastCooperativeVisitSymptoms=Symptoms at last cooperative visit -ContactExport.lastName=Last name of contact person -ContactExport.sourceCaseUuid=Source case ID -ContactExport.traveled=Traveled outside of district -ContactExport.travelHistory=Travel history -ContactExport.quarantineInformation=Quarantine information -ContactExport.reportingUserName=Reporting user -ContactExport.reportingUserRoles=Reporting user roles -ContactExport.followUpStatusChangeUserName=Responsible user -ContactExport.followUpStatusChangeUserRoles=Responsible user roles +ContactExport.address=العنوان +ContactExport.addressDistrict=منطقة العنوان +ContactExport.addressRegion=إقليم العنوان +ContactExport.addressCommunity=مجتمع العنوان +ContactExport.burialAttended=قام بزيارة دفنه +ContactExport.contactWithRodent=التواصل مع القوارض؟ +ContactExport.directContactConfirmedCase=الاتصال المباشر مع حالة مؤكدة +ContactExport.directContactProbableCase=اتصال مباشر مع حالة محتملة أو مؤكدة +ContactExport.firstName=الاسم الأول لشخص الاتصال +ContactExport.lastCooperativeVisitDate=تاريخ الزيارة التعاونية الاخيرة +ContactExport.lastCooperativeVisitSymptomatic=هل ظهرت عليك أعراض في الزيارة التعاونية الأخيرة؟ +ContactExport.lastCooperativeVisitSymptoms=الأعراض في الزيارة التعاونية الأخيرة +ContactExport.lastName=الاسم الأخير لشخص الاتصال +ContactExport.sourceCaseUuid=معرف حالة المصدر +ContactExport.traveled=سافرت خارج المنطقة +ContactExport.travelHistory=تواريخ السفريات +ContactExport.quarantineInformation=معلومات الحجر الصحي +ContactExport.reportingUserName=المستخدم المبلغ +ContactExport.reportingUserRoles=أدوار المستخدم المبلغ +ContactExport.followUpStatusChangeUserName=المستخدم المسؤول +ContactExport.followUpStatusChangeUserRoles=أدوار المستخدم المسؤول # Customizable enums -CustomizableEnum.hasDetails=Show an additional details text field when this value is selected -CustomizableEnum.hasDetails.short=Has details +CustomizableEnum.hasDetails=إظهار حقل نص تفاصيل إضافي عند تحديد هذه القيمة +CustomizableEnum.hasDetails.short=لديه تفاصيل # Customizable enum values -customizableEnumValueAllDiseases=All diseases -customizableEnumValueDiseaseCount=%d diseases -customizableEnumValueNoProperties=No properties -customizableEnumValueActiveValues=Active values -customizableEnumValueInactiveValues=Inactive values -CustomizableEnumValue.dataType=Data type -CustomizableEnumValue.value=Value -CustomizableEnumValue.caption=Caption -CustomizableEnumValue.diseases=Diseases -CustomizableEnumValue.properties=Properties -CustomizableEnumValue.active=Active +customizableEnumValueAllDiseases=جميع الأمراض +customizableEnumValueDiseaseCount=%d أمراض +customizableEnumValueNoProperties=لا يوجد خصائص +customizableEnumValueActiveValues=القيم النشطة +customizableEnumValueInactiveValues=القيم الغير النشطة +CustomizableEnumValue.dataType=نوع البيانات +CustomizableEnumValue.value=القيمة +CustomizableEnumValue.caption=التسمية التوضيحية +CustomizableEnumValue.diseases=أمراض +CustomizableEnumValue.properties=خصائص +CustomizableEnumValue.active=نشط # Dashboard -dashboardAlive=Alive -dashboardApplyCustomFilter=Apply custom filter -dashboardCanceledFollowUp=Canceled follow-up -dashboardCanceledFollowUpShort=Canceled F/U +dashboardAlive=على قيد الحياة +dashboardApplyCustomFilter=تطبيق مرشح مخصص +dashboardCanceledFollowUp=إلغاء المتابعة +dashboardCanceledFollowUpShort=إلغاء F/U dashboardCaseFatalityRateShort=CFR -dashboardCasesIn=Cases in -dashboardComparedTo=compared to -dashboardComparedToPreviousPeriod=%s compared to %s -dashboardCompletedFollowUp=Completed follow-up -dashboardCompletedFollowUpShort=Completed F/U -dashboardConfirmed=Confirmed -dashboardConfirmedContact=Confirmed contact +dashboardCasesIn=حالات في +dashboardComparedTo=مقارنة مع +dashboardComparedToPreviousPeriod=%s مقارنة بـ %s +dashboardCompletedFollowUp=تم الانتهاء من المتابعة +dashboardCompletedFollowUpShort=مكتمل F/U +dashboardConfirmed=مؤكد +dashboardConfirmedContact=جهة اتصال مؤكدة dashboardConfirmedNoSymptoms=تأكيد عدم وجود أعراض dashboardConfirmedUnknownSymptoms=تأكيد أعراض غير معروفة dashboardConvertedToCase=تم تحويلها إلى حالة @@ -1041,8 +1041,8 @@ devModeContactCreateWithoutSourceCases=إنشاء جهات اتصال بدون devModeContactCreateWithResultingCases=إنشاء بعض جهات الاتصال مع الحالات الناتجة devModeContactCreateMultipleContactsPerPerson=إنشاء جهات اتصال متعددة لكل شخص devModeContactCreateWithVisits=إنشاء زيارات لجهات الاتصال -devModeEventCasePercentage=Percentage of cases among participants -devModeEventCount=Number of generated Events +devModeEventCasePercentage=نسبة الحالات بين المشاركين +devModeEventCount=عدد الأحداث المولدة devModeEventDisease=Event Disease devModeEventDistrict=District of the events devModeEventEndDate=Latest event start date @@ -1208,54 +1208,54 @@ eventEvolutionDateWithStatus=تاريخ التطور %s eventEvolutionCommentWithStatus=طبيعة التطور %s eventNewEvent=New event eventNewEventGroup=New event group -eventSearchEvent=Search event -eventSearchSpecificEvent=Search specific Event -linkEvent=Link event -linkEventGroup=Link event group -eventSelect=Select event -eventSelectGroup=Select event group -eventDefaultView=Events -eventActionsView=Actions -eventGroupsView=Groups -eventNoEventLinkedToCase=No event linked to case -eventNoEventLinkedToEventGroup=No event linked to event group -eventNoEventLinkedToContact=No event linked to contact -eventOnlyWithContactSourceCaseInvolved=Only events in which the contact's source case is involved -eventLinkToCases=See event cases -eventLinkToContacts=See event contacts -eventSubordinateEvents=Subordinate Events -eventSuperordinateEvent=Superordinate Event -eventUnlinkEvent=Unlink event -eventUnlinkEventGroup=Unlink event group -eventGroupListEvents=Navigate to event directory filtered on this event group -eventOpenSuperordinateEvent=Open event -eventEditEvent=Edit event -eventEditEventGroup=Edit event group -eventLinkToEventsWithinTheSameFacility=See events within the same facility -eventNoDisease=No disease -eventGroups=Event groups -eventGroupsMultiple=This event is related to %s event groups -eventFilterOnlyEventsNotSharedWithExternalSurvTool=Only events not yet shared with reporting tool -eventFilterOnlyEventsSharedWithExternalSurvTool=Only events already shared with reporting tool -eventFilterOnlyEventsChangedSinceLastSharedWithExternalSurvTool=Only events changed since last shared with reporting tool -Event=Event -Event.caseCount=Cases -Event.contactCount=Contacts -Event.contactCountMethod=Contact Count Method -Event.contactCountSourceInEvent=Contacts with source case in event -Event.deathCount=Fatalities -Event.deletionReason=Reason for deletion -Event.otherDeletionReason=Reason for deletion details -Event.diseaseDetails=Disease name -Event.diseaseShort=Disease -Event.diseaseVariant=Disease variant -singleDayEventDate=Date of event -singleDayEventEvolutionDate=Evolution date of event -Event.startDate=Start date -Event.eventActions=Event actions -Event.endDate=End date -Event.multiDayEvent=Multi-day event -Event.externalId=External ID +eventSearchEvent=البحث عن الحدث +eventSearchSpecificEvent=البحث عن حدث محدد +linkEvent=رابط الحدث +linkEventGroup=رابط مجموعة الأحداث +eventSelect=اختر الحدث +eventSelectGroup=اختر مجموعة أحداث +eventDefaultView=الاحداث +eventActionsView=الإجراءات +eventGroupsView=المجموعات +eventNoEventLinkedToCase=لا يوجد حدث مرتبط بحالة +eventNoEventLinkedToEventGroup=لا يوجد حدث مرتبط بمجموعة الأحداث +eventNoEventLinkedToContact=لا يوجد حدث مرتبط بجهة الاتصال +eventOnlyWithContactSourceCaseInvolved=الأحداث التي تتعلق بحالة مصدر جهة الاتصال فقط +eventLinkToCases=مشاهدة حالات الحدث +eventLinkToContacts=مشاهدة جهات اتصال الحدث +eventSubordinateEvents=الأحداث التابعة +eventSuperordinateEvent=الحدث الأعظم +eventUnlinkEvent=إلغاء ربط الحدث +eventUnlinkEventGroup=إلغاء ربط مجموعة الأحداث +eventGroupListEvents=انتقل إلى دليل الأحداث المفلتر في مجموعة الأحداث هذه +eventOpenSuperordinateEvent=فتح الحدث +eventEditEvent=تعديل الحدث +eventEditEventGroup=تحرير مجموعة الحدث +eventLinkToEventsWithinTheSameFacility=شاهد الأحداث داخل نفس المنشأة +eventNoDisease=لا يوجد مرض +eventGroups=مجموعات الأحداث +eventGroupsMultiple=يرتبط هذا الحدث بمجموعات أحداث %s +eventFilterOnlyEventsNotSharedWithExternalSurvTool=فقط الأحداث التي لم تتم مشاركتها بعد باستخدام أداة إعداد التقارير +eventFilterOnlyEventsSharedWithExternalSurvTool=الأحداث التي تمت مشاركتها بالفعل باستخدام أداة إعداد التقارير فقط +eventFilterOnlyEventsChangedSinceLastSharedWithExternalSurvTool=تم تغيير الأحداث فقط منذ آخر مشاركة باستخدام أداة إعداد التقارير +Event=الحدث +Event.caseCount=الحالات +Event.contactCount=جهات الاتصال +Event.contactCountMethod=طريقة حساب عدد جهات الاتصال +Event.contactCountSourceInEvent=جهات الاتصال مع حالة المصدر في الحدث +Event.deathCount=الوفيات +Event.deletionReason=سبب الحذف +Event.otherDeletionReason=تفاصيل سبب الحذف +Event.diseaseDetails=اسم المرض +Event.diseaseShort=المرض +Event.diseaseVariant=متغير المرض +singleDayEventDate=تاريخ الحدث +singleDayEventEvolutionDate=تاريخ تطور الحدث +Event.startDate=تاريخ البدء +Event.eventActions=إجراءات الحدث +Event.endDate=تاريخ الانتهاء +Event.multiDayEvent=حدث يستمر لعدة أيام +Event.externalId=المعرف الخارجي Event.externalToken=رمز خارجي Event.eventTitle=العنوان Event.eventDesc=التفاصيل @@ -1307,8 +1307,8 @@ Event.epidemiologicalEvidence=الأدلة الوبائية Event.laboratoryDiagnosticEvidence=الأدلة التشخيصية المختبرية Event.internalToken=رمز داخلي Event.eventGroups=المجموعات -Event.latestEventGroup=Latest Event Group -Event.eventGroupCount=Event Group Count +Event.latestEventGroup=أحدث مجموعة الأحداث +Event.eventGroupCount=عدد مجموعات الأحداث Event.region=Region Event.district=District Event.community=Community @@ -1418,54 +1418,54 @@ exportEditExportConfiguration=تعديل إعدادات التصدير exportConfigurationData=بيانات التكوين exportExternalData=External data ExportConfiguration.NAME=Configuration name -ExportConfiguration.myExports=My exports -ExportConfiguration.sharedExports=Shared exports -ExportConfiguration.sharedToPublic=Shared to public -exposureFlightNumber=Flight number -exposureTimePeriod=Time period -exposureSourceCaseName=Name of source case -Exposure=Exposure -Exposure.probableInfectionEnvironment=Probable infection environment -Exposure.startDate=Start of exposure -Exposure.endDate=End of exposure -Exposure.exposureType=Type of activity -Exposure.exposureTypeDetails=Type of activity details -Exposure.location=Location -Exposure.typeOfPlace=Type of place -Exposure.typeOfPlaceDetails=Type of place details -Exposure.meansOfTransport=Means of transport -Exposure.meansOfTransportDetails=Means of transport details -Exposure.connectionNumber=Connection number -Exposure.seatNumber=Seat number -Exposure.workEnvironment=Work environment -Exposure.indoors=Indoors -Exposure.outdoors=Outdoors -Exposure.wearingMask=Wearing mask -Exposure.wearingPpe=Wearing appropriate PPE -Exposure.otherProtectiveMeasures=Other protective measures -Exposure.protectiveMeasuresDetails=Protective measures details -Exposure.shortDistance=< 1.5 m distance -Exposure.longFaceToFaceContact=> 15 min face-to-face contact -Exposure.animalMarket=Animal market -Exposure.percutaneous=Percutaneous exposure -Exposure.contactToBodyFluids=Contact to blood or body fluids -Exposure.handlingSamples=Handling samples (animal or human) -Exposure.eatingRawAnimalProducts=Eating raw or undercooked animal products -Exposure.handlingAnimals=Handling or butchering animals (or their remains) -Exposure.animalCondition=Condition of the animal -Exposure.animalVaccinated=Animal vaccinated -Exposure.animalContactType=Kind of contact to the animal -Exposure.animalContactTypeDetails=Kind of contact details -Exposure.contactToCase=Contact to source case -Exposure.gatheringType=Type of gathering -Exposure.gatheringDetails=Type of gathering details -Exposure.habitationType=Type of habitation -Exposure.habitationDetails=Type of habitation details -Exposure.typeOfAnimal=Type of animal -Exposure.typeOfAnimalDetails=Type of animal details -Exposure.physicalContactDuringPreparation=Has had direct physical contact during burial preparation ritual -Exposure.physicalContactWithBody=Has had direct physical contact with the (deceased) case at a funeral -Exposure.deceasedPersonIll=Was the deceased person ill? +ExportConfiguration.myExports=صادراتي +ExportConfiguration.sharedExports=الصادرات المشتركة +ExportConfiguration.sharedToPublic=تمت مشاركتها للعامة +exposureFlightNumber=رقم رحلة الطيران +exposureTimePeriod=الفترة الزمنية +exposureSourceCaseName=اسم حالة المصدر +Exposure=التعرض +Exposure.probableInfectionEnvironment=بيئة العدوى المحتملة +Exposure.startDate=بداية التعرض +Exposure.endDate=نهاية التعرض +Exposure.exposureType=نوع النشاط +Exposure.exposureTypeDetails=تفاصيل نوع النشاط +Exposure.location=المكان +Exposure.typeOfPlace=نوع المكان +Exposure.typeOfPlaceDetails=تفاصيل نوع المكان +Exposure.meansOfTransport=وسيلة النقل +Exposure.meansOfTransportDetails=تفاصيل وسيلة النقل +Exposure.connectionNumber=رقم الاتصال +Exposure.seatNumber=رقم المقعد +Exposure.workEnvironment=بيئة العمل +Exposure.indoors=في الداخل +Exposure.outdoors=في الخارج +Exposure.wearingMask=مرتدي القناع +Exposure.wearingPpe=ارتداء معدات الحماية الشخصية المناسبة +Exposure.otherProtectiveMeasures=تدابير الحماية الأخرى +Exposure.protectiveMeasuresDetails=تفاصيل التدابير الوقائية +Exposure.shortDistance=مسافة < ١. ٥ متر +Exposure.longFaceToFaceContact=> ١٥ دقيقة من الاتصال وجهاً لوجه +Exposure.animalMarket=سوق الحيوانات +Exposure.percutaneous=التعرض الجلدي +Exposure.contactToBodyFluids=الاتصال بالدم أو سوائل الجسم +Exposure.handlingSamples=التعامل مع العينات (الحيوانية أو البشرية) +Exposure.eatingRawAnimalProducts=تناول المنتجات الحيوانية النيئة أو غير المطبوخة جيدا +Exposure.handlingAnimals=التعامل مع الحيوانات أو ذبحها (أو بقاياها) +Exposure.animalCondition=حالة الحيوان +Exposure.animalVaccinated=تم تطعيم الحيوان +Exposure.animalContactType=نوع الاتصال بالحيوان +Exposure.animalContactTypeDetails=نوع تفاصيل الاتصال +Exposure.contactToCase=الاتصال بمصدر الحالة +Exposure.gatheringType=نوع التجمع +Exposure.gatheringDetails=تفاصيل نوع التجمع +Exposure.habitationType=نوع السكن +Exposure.habitationDetails=تفاصيل نوع السكن +Exposure.typeOfAnimal=نوع الحيوان +Exposure.typeOfAnimalDetails=تفاصيل نوع الحيوان +Exposure.physicalContactDuringPreparation=كان له اتصال جسدي مباشر أثناء طقوس تحضير الدفن +Exposure.physicalContactWithBody=كان له اتصال جسدي مباشر مع الحالة (المتوفاة) في الجنازة +Exposure.deceasedPersonIll=هل كان الشخص المتوفى مريضا؟ Exposure.deceasedPersonName=اسم الشخص المتوفى Exposure.deceasedPersonRelation=العلاقة بالشخص المتوفى Exposure.bodyOfWater=الاتصال مع الجسم المائي @@ -1520,8 +1520,8 @@ FollowUp.reportDate=تاريخ التقرير FollowUp.followUpUntil=متابعة حتى # HealthConditions HealthConditions=الحالة الصحية -HealthConditions.tuberculosis=Tuberculosis -HealthConditions.asplenia=Asplenia +HealthConditions.tuberculosis=مرض الدرن +HealthConditions.asplenia=انعدام الطحال HealthConditions.hepatitis=Hepatitis HealthConditions.diabetes=Diabetes HealthConditions.hiv=HIV @@ -1627,58 +1627,58 @@ lineListingInfrastructureData=تولي بيانات البنية التحتية lineListingNewCasesList=قائمة الحالات الجديدة lineListingNewContactsList=List of new contacts lineListingNewEventParticipantsList = List of new event participants -lineListingSharedInformation=Shared information -lineListingEdit=Edit line listing -lineListingDisableAll=Disable all line listing -lineListingEnableForDisease=Enable line listing for disease -lineListingEnableAll=Enable all -lineListingDisableAllShort=Disable all -lineListingEndDate=End date -lineListingSetEndDateForAll=Set end date for all +lineListingSharedInformation=معلومات مشتركة +lineListingEdit=تعديل قائمة الخطوط +lineListingDisableAll=تعطيل قائمة جميع الخطوط +lineListingEnableForDisease=تمكين قائمة الخطوط للأمراض +lineListingEnableAll=تمكين الكل +lineListingDisableAllShort=تعطيل الكل +lineListingEndDate=تاريخ الانتهاء +lineListingSetEndDateForAll=تعيين تاريخ النهاية للجميع # Location -Location=Location -Location.additionalInformation=Additional information -Location.addressType=Address Type -Location.addressTypeDetails=Address name / description -Location.areaType=Area type (urban/rural) -Location.details=Community contact person -Location.facility=Facility -Location.facilityDetails=Facility name & description -Location.facilityType=Facility type -Location.houseNumber=House number -Location.latitude=GPS latitude -Location.latLon=GPS lat and lon -Location.latLonAccuracy=GPS accuracy in m -Location.longitude=GPS longitude -Location.postalCode=Postal code -Location.continent=Continent -Location.subcontinent=Subcontinent -Location.country=Country -Location.region=Region -Location.district=District -Location.community=Community -Location.street=Street -Location.contactPersonFirstName=Contact person first name -Location.contactPersonLastName=Contact person last name -Location.contactPersonPhone=Contact person phone number -Location.contactPersonEmail=Contact person email address +Location=المكان +Location.additionalInformation=معلومات إضافية +Location.addressType=نوع العنوان +Location.addressTypeDetails=اسم العنوان / الوصف +Location.areaType=نوع المنطقة (حضرية/ريفية) +Location.details=جهة الاتصال المجتمعية +Location.facility=منشأة +Location.facilityDetails=اسم المنشأة والوصف +Location.facilityType=نوع المنشأة +Location.houseNumber=رقم المنزل +Location.latitude=خط عرض لنظام تحديد المواقع العالمي (GPS) +Location.latLon=خطوط العرض والطول لنظام تحديد المواقع العالمي GPS +Location.latLonAccuracy=تقرير دقة نظام تحديد المواقع العالمي (GPS) بالمتر +Location.longitude=خط طول GPS +Location.postalCode=الرمز البريدي +Location.continent=القارة +Location.subcontinent=شبه القارة +Location.country=الدولة +Location.region=إقليم +Location.district=المقاطعة +Location.community=المجتمع +Location.street=الشارع +Location.contactPersonFirstName=الاسم الأول لشخص الاتصال +Location.contactPersonLastName=اسم العائلة لشخص الاتصال +Location.contactPersonPhone=رقم هاتف شخص الاتصال +Location.contactPersonEmail=عنوان البريد الإلكتروني لشخص الاتصال # Login -Login.doLogIn=Log in -Login.login=Login -Login.password=password -Login.username=username +Login.doLogIn=تسجيل الدخول +Login.login=تسجيل دخول +Login.password=كلمة المرور +Login.username=اسم المستخدم #LoginSidebar -LoginSidebar.diseaseDetection=Disease Detection -LoginSidebar.diseasePrevention=Disease Prevention -LoginSidebar.outbreakResponse=Outbreak Response -LoginSidebar.poweredBy=Powered By +LoginSidebar.diseaseDetection=الكشف عن المرض +LoginSidebar.diseasePrevention=الوقاية من المرض +LoginSidebar.outbreakResponse=الاستجابة لتفشي المرض +LoginSidebar.poweredBy=مدعوم بواسطة # Messaging -messagesSendSMS=Send SMS -messagesSentBy=Sent by -messagesNoSmsSentForCase=No SMS sent to case person -messagesNoPhoneNumberForCasePerson=Case person has no phone number -messagesSms=SMS -messagesEmail=Email +messagesSendSMS=إرسال رسالة قصيرة +messagesSentBy=أرسلت بواسطة +messagesNoSmsSentForCase=لا توجد رسائل قصيرة مرسلة لشخص الحالة +messagesNoPhoneNumberForCasePerson=لا يوجد رقم هاتف لشخص الحالة +messagesSms=الرسائل النصية القصيرة +messagesEmail=البريد الإلكتروني messagesEmails=البريد الإلكتروني messagesSendEmail=إرسال بريد إلكتروني messagesSendingSms=إرسال الرسائل القصيرة الجديدة @@ -1730,8 +1730,8 @@ MaternalHistory.rashExposureDistrict=المقاطعة MaternalHistory.rashExposureCommunity=المجتمع MaternalHistory.otherComplications=تعقيدات أخرى MaternalHistory.otherComplicationsOnset=تاريخ بداية -MaternalHistory.otherComplicationsMonth=Month of pregnancy -MaternalHistory.otherComplicationsDetails=Complication details +MaternalHistory.otherComplicationsMonth=شهر الحمل +MaternalHistory.otherComplicationsDetails=تفاصيل المضاعفات # Outbreak outbreakAffectedDistricts=Affected districts outbreakNoOutbreak=No outbreak @@ -1835,54 +1835,54 @@ Person.districtName=المقاطعة Person.educationType=تعليم Person.educationDetails=Details Person.fathersName=Father's name -Person.namesOfGuardians=Names of guardians -Person.gestationAgeAtBirth=Gestation age at birth (weeks) -Person.lastDisease=Last disease -Person.matchingCase=Matching case -Person.mothersMaidenName=Mother's maiden name -Person.mothersName=Mother's name -Person.nickname=Nickname -Person.occupationCommunity=Facility community -Person.occupationDetails=Please specify occupation -Person.occupationDistrict=Facility district -Person.occupationFacility=Healthcare facility -Person.occupationFacilityDetails=Facility name & description -Person.occupationFacilityType=Facility type -Person.occupationRegion=Facility region -Person.occupationType=Type of occupation -Person.armedForcesRelationType=Staff of armed forces -Person.phone=Primary phone number -Person.phoneOwner=Owner of phone -Person.placeOfBirthRegion=Region of birth -Person.placeOfBirthDistrict=District of birth -Person.placeOfBirthCommunity=Community of birth -Person.placeOfBirthFacility=Facility of birth -Person.placeOfBirthFacilityDetails=Facility name & description -Person.placeOfBirthFacilityType=Facility type -Person.presentCondition=Present condition of person -Person.sex=Sex -Person.generalPractitionerDetails=General practitioner name and contact details -Person.emailAddress=Primary email address -Person.otherContactDetails=Other contact details -Person.passportNumber=Passport number -Person.nationalHealthId=National health ID -Person.uuid=Person ID -Person.hasCovidApp=Has COVID app -Person.covidCodeDelivered=COVID code was generated and delivered -Person.externalId=External ID -Person.externalToken=External Token -Person.internalToken=Internal Token -Person.symptomJournalStatus=Symptom journal status -Person.salutation=Salutation -Person.otherSalutation=Other salutation -Person.birthName=Birth name -Person.birthCountry=Country of birth -Person.citizenship=Citizenship -personContactDetailOwner=Owner -personContactDetailOwnerName=Owner name -personContactDetailThisPerson=This person -personContactDetailThirdParty=Collect contact details of another person or facility -PersonContactDetail=Person contact detail +Person.namesOfGuardians=أسماء الأوصياء +Person.gestationAgeAtBirth=عمر الحمل عند الولادة (أسابيع) +Person.lastDisease=آخر مرض +Person.matchingCase=حالة مطابقة +Person.mothersMaidenName=اسم عائلة الأم +Person.mothersName=اسم الأم +Person.nickname=كنية +Person.occupationCommunity=مجتمع المرافق +Person.occupationDetails=الرجاء تحديد المهنة +Person.occupationDistrict=منطقة المرفق +Person.occupationFacility=المنشأة الصحية +Person.occupationFacilityDetails=اسم المنشأة والوصف +Person.occupationFacilityType=نوع المنشأة +Person.occupationRegion=منطقة المنشأة +Person.occupationType=نوع المهنة +Person.armedForcesRelationType=أفراد القوات المسلحة +Person.phone=رقم الهاتف الأساسي +Person.phoneOwner=مالك الهاتف +Person.placeOfBirthRegion=إقليم الميلاد +Person.placeOfBirthDistrict=منطقة الميلاد +Person.placeOfBirthCommunity=مجتمع الميلاد +Person.placeOfBirthFacility=مرفق الولادة +Person.placeOfBirthFacilityDetails=اسم المنشأة والوصف +Person.placeOfBirthFacilityType=نوع المنشأة +Person.presentCondition=الحالة الراهنة للشخص +Person.sex=الجنس +Person.generalPractitionerDetails=اسم الممارس العام وتفاصيل الاتصال +Person.emailAddress=عنوان البريد الإلكتروني الأساسي +Person.otherContactDetails=تفاصيل الاتصال الأخرى +Person.passportNumber=رقم جواز السفر +Person.nationalHealthId=بطاقة التأمين الصحي +Person.uuid=معرف الشخص +Person.hasCovidApp=هل يوجد تطبيق COVID +Person.covidCodeDelivered=تم إنشاء رمز COVID وتسليمه +Person.externalId=المعرف الخارجي +Person.externalToken=الرمز الخارجي +Person.internalToken=الرمز الداخلي +Person.symptomJournalStatus=حالة مجلة الاعراض +Person.salutation=التحية +Person.otherSalutation=تحية أخرى +Person.birthName=اسم الميلاد +Person.birthCountry=بلد الميلاد +Person.citizenship=المواطنة +personContactDetailOwner=المالك +personContactDetailOwnerName=اسم المالك +personContactDetailThisPerson=هذا الشخص +personContactDetailThirdParty=جمع تفاصيل الاتصال بشخص آخر أو منشأة أخرى +PersonContactDetail=تفاصيل الاتصال بالشخص PersonContactDetail.person=شخص PersonContactDetail.primaryContact=تفاصيل الاتصال الأساسية PersonContactDetail.personContactDetailType=نوع تفاصيل الاتصال @@ -1933,8 +1933,8 @@ PortHealthInfo.vesselName=اسم السفينة PortHealthInfo.vesselDetails=تفاصيل السفينة PortHealthInfo.portOfDeparture=ميناء المغادرة PortHealthInfo.lastPortOfCall=ميناء الاتصال الأخير -PortHealthInfo.conveyanceType=Conveyance type -PortHealthInfo.conveyanceTypeDetails=Specify the conveyance type +PortHealthInfo.conveyanceType=نوع النقل +PortHealthInfo.conveyanceTypeDetails=تحديد نوع النقل PortHealthInfo.departureLocation=Start location of travel PortHealthInfo.finalDestination=Final destination PortHealthInfo.details=Point of entry details @@ -2042,54 +2042,54 @@ Sample=عينة Sample.additionalTestingRequested=طلب إجراء اختبارات إضافية؟ Sample.additionalTestingStatus=Additional testing status Sample.associatedCase=Associated case -Sample.associatedContact=Associated contact -Sample.associatedEventParticipant=Associated event participant -Sample.caseClassification=Case classification -Sample.caseDistrict=District -Sample.casePersonName=Corresponding person -Sample.caseRegion=Region -Sample.comment=Comment -Sample.diseaseShort=Disease -Sample.lab=Laboratory -Sample.labDetails=Laboratory name & description -Sample.labSampleID=Lab sample ID -Sample.fieldSampleID=Field sample ID -Sample.labUser=Lab user -Sample.noTestPossibleReason=Reason -Sample.otherLab=Referral laboratory -Sample.pathogenTestingRequested=Request pathogen tests to be performed? -Sample.pathogenTestCount=Number of tests -Sample.pathogenTestResult=Final laboratory result -Sample.received=Received -Sample.receivedDate=Date sample received at lab -Sample.referredToUuid=Sample referred to -Sample.reportDateTime=Date of report -Sample.reportInfo=Report date & user -Sample.reportingUser=Reporting user -Sample.requestedAdditionalTests=If you wish to request specific pathogen tests, tick each of them in the list below -Sample.requestedAdditionalTestsTags=Requested additional tests\: -Sample.requestedOtherAdditionalTests=Other requested additional tests -Sample.requestedOtherPathogenTests=Other requested pathogen tests -Sample.requestedPathogenTests=If you wish to request specific additional tests, tick each of them in the list below -Sample.requestedPathogenTestsTags=Requested pathogen tests\: -Sample.sampleCode=Sample code -Sample.sampleDateTime=Date sample was collected -Sample.sampleMaterial=Type of sample -Sample.sampleMaterialText=Specify other type -Sample.sampleSource=Sample source -Sample.shipmentDate=Date sample was sent -Sample.shipmentDetails=Dispatch details -Sample.shipped=Sent/dispatched -Sample.specimenCondition=Specimen condition -Sample.suggestedTypeOfTest=Suggested type of test -Sample.testResult=Test result -Sample.testStatusGen=Test status -Sample.testType=Type of test -Sample.typeOfTest=Type of test -Sample.uuid=Sample ID -Sample.samplePurpose=Purpose of the Sample -Sample.samplingReason=Reason for sampling/testing -Sample.samplingReasonDetails=Sampling reason details +Sample.associatedContact=جهة الاتصال المرتبطة +Sample.associatedEventParticipant=المشارك في الحدث المرتبط +Sample.caseClassification=تصنيف الحالة +Sample.caseDistrict=المنطقة +Sample.casePersonName=الشخص المقابل +Sample.caseRegion=إقليم +Sample.comment=تعليق +Sample.diseaseShort=المرض +Sample.lab=المختبر +Sample.labDetails=اسم المختبر ووصفه +Sample.labSampleID=معرف عينة المختبر +Sample.fieldSampleID=معرف عينة الحقل +Sample.labUser=مستخدم المختبر +Sample.noTestPossibleReason=السبب +Sample.otherLab=مختبر الإحالة +Sample.pathogenTestingRequested=طلب إجراء اختبارات مسببات الأمراض؟ +Sample.pathogenTestCount=عدد الاختبارات +Sample.pathogenTestResult=النتائج المخبرية النهائية +Sample.received=تم الإستلام +Sample.receivedDate=تاريخ استلام العينة في المختبر +Sample.referredToUuid=العينة المشار إليها +Sample.reportDateTime=تاريخ التقرير +Sample.reportInfo=تاريخ التقرير والمستخدم +Sample.reportingUser=المستخدم المبلغ +Sample.requestedAdditionalTests=إذا كنت ترغب في طلب اختبارات مسببات الأمراض المحددة، فحدد كل منها في القائمة أدناه +Sample.requestedAdditionalTestsTags=الاختبارات الإضافية المطلوبة\: +Sample.requestedOtherAdditionalTests=الاختبارات الإضافية المطلوبة +Sample.requestedOtherPathogenTests=اختبارات مسببات الأمراض الأخرى المطلوبة +Sample.requestedPathogenTests=إذا كنت ترغب في طلب اختبارات إضافية محددة، فحدد كل منها في القائمة أدناه +Sample.requestedPathogenTestsTags=اختبارات مسببات الأمراض المطلوبة\: +Sample.sampleCode=كود العينة +Sample.sampleDateTime=تاريخ جمع العينة +Sample.sampleMaterial=نوع العينة +Sample.sampleMaterialText=تحديد نوع آخر +Sample.sampleSource=مصدر العينة +Sample.shipmentDate=تاريخ إرسال العينة +Sample.shipmentDetails=تفاصيل الإرسال +Sample.shipped=تم الإرسال/الإرسال +Sample.specimenCondition=حالة العينة +Sample.suggestedTypeOfTest=نوع الاختبار المقترح +Sample.testResult=نتيجة الاختبار +Sample.testStatusGen=حاله الاختبار +Sample.testType=نوع الاختبار +Sample.typeOfTest=نوع الاختبار +Sample.uuid=معرف العينة +Sample.samplePurpose=الغرض من العينة +Sample.samplingReason=سبب أخذ العينات/الاختبار +Sample.samplingReasonDetails=تفاصيل سبب أخذ العينات Sample.region=إقليم Sample.district=المقاطعة Sample.community=المجتمع @@ -2141,8 +2141,8 @@ SampleExport.pathogenTestResult3=ثالث أحدث نتيجة اختبار مس SampleExport.pathogenTestVerified3=هل تم التحقق من ثالث أحدث اختبار لمسببات الأمراض؟ SampleExport.otherPathogenTestsDetails=اختبارات مسببات الأمراض الأخرى SampleExport.otherAdditionalTestsDetails=هل هناك اختبارات إضافية أخرى؟ -SampleExport.pathogenTestingRequested=Have pathogen tests been requested? -SampleExport.referredToUuid=Referred sample +SampleExport.pathogenTestingRequested=هل تم طلب اختبارات مسببات الأمراض؟ +SampleExport.referredToUuid=العينة المشار إليها SampleExport.requestedAdditionalTests=Requested additional tests SampleExport.requestedPathogenTests=Requested pathogen tests SampleExport.shipped=Sent/dispatched? @@ -2245,55 +2245,55 @@ statisticsRemoveFilter=إزالة عامل التصفية statisticsResetFilters=إعادة تعيين المرشحات statisticsShowZeroValues=Show zero values statisticsShowCaseIncidence=Show case incidence -statisticsSpecifySelection=Specify your selection -statisticsStatistics=Statistics -statisticsVisualizationType=Type -statisticsIncidenceDivisor=Incidence divisor -statisticsDataDisplayed=Data displayed -statisticsOpenSormasStats=Open Sormas-Stats +statisticsSpecifySelection=حدد إختيارك +statisticsStatistics=الإحصائيات +statisticsVisualizationType=النوع +statisticsIncidenceDivisor=قاسم الحدوث +statisticsDataDisplayed=البيانات المعروضة +statisticsOpenSormasStats=افتح إحصائيات-سورماس # Symptoms -symptomsLesionsLocations=Localization of the lesions -symptomsMaxTemperature=Maximum body temperature in ° C -symptomsSetClearedToNo=Set cleared to No -symptomsSetClearedToUnknown=Set cleared to Unknown -Symptoms=Symptoms -Symptoms.abdominalPain=Abdominal pain -Symptoms.alteredConsciousness=Altered level of consciousness -Symptoms.anorexiaAppetiteLoss=Anorexia/loss of appetite -Symptoms.backache=Backache -Symptoms.bedridden=Is the patient bedridden? -Symptoms.bilateralCataracts=Bilateral cataracts -Symptoms.blackeningDeathOfTissue=Blackening and death of tissue in extremities -Symptoms.bleedingVagina=Bleeding from vagina, other than menstruation -Symptoms.bloodInStool=Blood in stool -Symptoms.bloodPressureDiastolic=Blood pressure (diastolic) -Symptoms.bloodPressureSystolic=Blood pressure (systolic) -Symptoms.bloodUrine=Blood in urine (hematuria) -Symptoms.bloodyBlackStool=Bloody or black stools (melena) -Symptoms.buboesGroinArmpitNeck=Buboes in the groin, armpit or neck -Symptoms.bulgingFontanelle=Bulging fontanelle -Symptoms.chestPain=Chest pain -Symptoms.chillsSweats=Chills or sweats -Symptoms.confusedDisoriented=Confused or disoriented -Symptoms.congenitalGlaucoma=Congenital glaucoma -Symptoms.congenitalHeartDisease=Congenital heart disease -Symptoms.congenitalHeartDiseaseType=Heart disease type -Symptoms.congenitalHeartDiseaseDetails=Specify -Symptoms.conjunctivitis=Conjunctivitis (red eyes) -Symptoms.cough=Cough -Symptoms.coughWithSputum=Cough with sputum -Symptoms.coughWithHeamoptysis=Cough with heamoptysis -Symptoms.coughingBlood=Coughing up blood (haemoptysis) -Symptoms.darkUrine=Dark Urine -Symptoms.dehydration=Dehydration -Symptoms.developmentalDelay=Developmental delay -Symptoms.diarrhea=Diarrhea -Symptoms.difficultyBreathing=Difficulty breathing/Dyspnea -Symptoms.digestedBloodVomit=Digested blood/"coffee grounds" in vomit -Symptoms.eyePainLightSensitive=Pain behind eyes/Sensitivity to light -Symptoms.eyesBleeding=Bleeding from the eyes -Symptoms.fatigueWeakness=Fatigue/general weakness -Symptoms.fever=Fever +symptomsLesionsLocations=تحديد موقع الآفات +symptomsMaxTemperature=درجة حرارة الجسم القصوى ب الدرجة المئوية +symptomsSetClearedToNo=تم التعيين على لا +symptomsSetClearedToUnknown=تم التعيين على غير معروف +Symptoms=الأعراض +Symptoms.abdominalPain=ألم في البطن +Symptoms.alteredConsciousness=تغير مستوى الوعي +Symptoms.anorexiaAppetiteLoss=فقدان الشهية +Symptoms.backache=آلام الظهر +Symptoms.bedridden=هل المريض طريح الفراش؟ +Symptoms.bilateralCataracts=إعتام عدسة العين الثنائي +Symptoms.blackeningDeathOfTissue=اسوداد وموت الأنسجة في الأطراف +Symptoms.bleedingVagina=نزيف من المهبل، غير الحيض +Symptoms.bloodInStool=دم في البراز +Symptoms.bloodPressureDiastolic=ضغط الدم (الانبساطي) +Symptoms.bloodPressureSystolic=ضغط الدم (الانقباضي) +Symptoms.bloodUrine=دم في البول (بيلة دموية) +Symptoms.bloodyBlackStool=براز دموي أو أسود (ميلينا) +Symptoms.buboesGroinArmpitNeck=ظهور بثور في الفخذ أو الإبط أو الرقبة +Symptoms.bulgingFontanelle=اليافوخ المنتفخ +Symptoms.chestPain=ألم في الصدر +Symptoms.chillsSweats=قشعريرة أو تعرق +Symptoms.confusedDisoriented=مشوشة أو مضللة +Symptoms.congenitalGlaucoma=الجلوكوما الخلقية +Symptoms.congenitalHeartDisease=مرض القلب الخلقي +Symptoms.congenitalHeartDiseaseType=نوع مرض القلب +Symptoms.congenitalHeartDiseaseDetails=حدد +Symptoms.conjunctivitis=التهاب الملتحمة (العيون الحمراء) +Symptoms.cough=السعال +Symptoms.coughWithSputum=السعال مع البلغم +Symptoms.coughWithHeamoptysis=السعال مع نفث الدم +Symptoms.coughingBlood=سعال الدم (نفث الدم) +Symptoms.darkUrine=البول الداكن +Symptoms.dehydration=جفاف +Symptoms.developmentalDelay=تأخير النمو +Symptoms.diarrhea=إسهال +Symptoms.difficultyBreathing=صعوبة التنفس/ضيق التنفس +Symptoms.digestedBloodVomit=دم مهضوم/"بقايا قهوة" في القيء +Symptoms.eyePainLightSensitive=ألم خلف العين/حساسية للضوء +Symptoms.eyesBleeding=نزيف من العين +Symptoms.fatigueWeakness=الإجهاد/الضعف العام +Symptoms.fever=حمى Symptoms.firstSymptom=الأعراض الأولى Symptoms.fluidInLungCavity=سائل في تجويف الرئة Symptoms.glasgowComaScale=مقياس غلاسكو للغيبوبة @@ -2344,8 +2344,8 @@ Symptoms.lymphadenopathyInguinal=تضخم الغدد الليمفاوية الإ Symptoms.malaise=توعك Symptoms.meningealSigns=علامات السحايا Symptoms.meningoencephalitis=التهاب السحايا والدماغ -Symptoms.microcephaly=Microcephaly -Symptoms.midUpperArmCircumference=Mid-upper arm circumf. (cm) +Symptoms.microcephaly=صغر الرأس +Symptoms.midUpperArmCircumference=محيط منتصف الذراع العلوي (سم) Symptoms.musclePain=Muscle pain Symptoms.nausea=Nausea Symptoms.neckStiffness=Stiff neck @@ -2447,55 +2447,55 @@ Symptoms.urinaryRetention=احتباس البول taskMyTasks=مهامي taskNewTask=New task taskNoTasks=There are no tasks for this %s -taskOfficerTasks=Officer tasks -taskActiveTasks=Active tasks -taskArchivedTasks=Archived tasks -taskAllTasks=All tasks -Task=Task -Task.assigneeReply=Comments on execution -Task.assigneeUser=Assigned to -Task.assignedByUser=Assigned by -Task.caze=Associated case -Task.contact=Associated contact -Task.contextReference=Associated link -Task.creatorComment=Comments on task -Task.creatorUser=Created by -Task.dueDate=Due date -Task.environment = Associated environment -Task.event=Associated event -Task.observerUsers=Observed by -Task.perceivedStart=Perceived start -Task.priority=Priority -Task.statusChangeDate=Status change date -Task.suggestedStart=Suggested start -Task.taskContext=Task context -Task.taskStatus=Task status -Task.taskType=Task type -Task.taskAssignee=Task assignee -Task.taskPriority=Task priority -Task.travelEntry=Travel entry -Task.region=Region -Task.district=District +taskOfficerTasks=مهام الضابط +taskActiveTasks=المهام النشطة +taskArchivedTasks=المهام المؤرشفة +taskAllTasks=جميع المهام +Task=مهمة +Task.assigneeReply=تعليقات على التنفيذ +Task.assigneeUser=مُخصص لـ +Task.assignedByUser=تم تعيينه بواسطة +Task.caze=حالة مرتبطة +Task.contact=جهة الاتصال المرتبطة +Task.contextReference=رابط مرتبط +Task.creatorComment=تعليقات على المهمة +Task.creatorUser=أنشئ من قبل +Task.dueDate=تاريخ الاستحقاق +Task.environment = البيئة المرتبطة +Task.event=الاحداث المرتبطة +Task.observerUsers=تمت ملاحظته بواسطة +Task.perceivedStart=البداية المتصورة +Task.priority=الأولوية +Task.statusChangeDate=تاريخ تغيير الحالة +Task.suggestedStart=البداية المقترحة +Task.taskContext=سياق المهمة +Task.taskStatus=حالة المهمة +Task.taskType=نوع المهمة +Task.taskAssignee=المكلف بالمهمة +Task.taskPriority=أولوية المهمة +Task.travelEntry=دخول السفر +Task.region=إقليم +Task.district=المنطقة # TestReport -TestReport=Test report -TestReport.dateOfResult=Date of result -TestReport.testDateTime=Date and time of result -TestReport.testLabCity=Lab city -TestReport.testLabExternalId=Lab external ID -TestReport.testLabName=Lab name -TestReport.testLabPostalCode=Lab postal code -TestReport.testResult=Test result -TestReport.testType=Type of test +TestReport=تقرير الاختبار +TestReport.dateOfResult=تاريخ النتيجة +TestReport.testDateTime=تاريخ ووقت النتيجة +TestReport.testLabCity=مدينة المختبر +TestReport.testLabExternalId=معرف المختبر الخارجي +TestReport.testLabName=اسم المختبر +TestReport.testLabPostalCode=الرمز البريدي للمختبر +TestReport.testResult=نتيجة الاختبار +TestReport.testType=نوع الاختبار TestReport.ctValueE=Ct target E -TestReport.ctValueN=Ct target N +TestReport.ctValueN=Ct target E TestReport.ctValueRdrp=Ct target RDRP TestReport.ctValueS=Ct target S TestReport.ctValueOrf1=Ct target ORF1 TestReport.ctValueRdrpS=Ct target RDRP/S -TestReport.prescriberPhysicianCode=Physician code -TestReport.prescriberFirstName=First name -TestReport.prescriberLastName=Last name -TestReport.prescriberPhoneNumber=Phone number +TestReport.prescriberPhysicianCode=رمز الطبيب +TestReport.prescriberFirstName=الاسم الأول +TestReport.prescriberLastName=إسم العائلة +TestReport.prescriberPhoneNumber=رقم الهاتف TestReport.prescriberAddress=العنوان TestReport.prescriberPostalCode=الرمز البريدي TestReport.prescriberCity=المدينة @@ -2547,8 +2547,8 @@ TravelEntry.quarantineHomeSupplyEnsuredComment=تعليق TravelEntry.quarantineOrderedVerbally=هل طلب الحجر الصحي شفوياً؟ TravelEntry.quarantineOrderedVerballyDate=تاريخ صدور الأمر الشفوي TravelEntry.quarantineOrderedOfficialDocument=هل الحجر الحي أمر بوثيقة رسمية؟ -TravelEntry.quarantineOrderedOfficialDocumentDate=Date of the official document order -TravelEntry.quarantineExtended=Quarantine period extended? +TravelEntry.quarantineOrderedOfficialDocumentDate=تاريخ أمر الوثيقة الرسمية +TravelEntry.quarantineExtended=هل تم مد فترة الحجر الصحي؟ TravelEntry.quarantineReduced=Quarantine period reduced? TravelEntry.quarantineOfficialOrderSent=Official quarantine order sent? TravelEntry.quarantineOfficialOrderSentDate=Date official quarantine order was sent @@ -2601,60 +2601,61 @@ User.pointOfEntry=Assigned point of entry User.userEmail=Email User.userName=User name User.userRoles=User roles -User.address=Address -User.uuid=UUID -User.region=Region -User.district=District -User.community=Community -userRestrictDiseases=Restrict access to specific diseases +User.address=العنوان +User.uuid=معرف فريد عالميًا +User.region=إقليم +User.district=المنطقة +User.community=المجتمع +User.externalId=External ID +userRestrictDiseases=تقييد الوصول إلى أمراض محددة # Vaccination -vaccinationNewVaccination=New vaccination -vaccinationNoVaccinationsForPerson=There are no vaccinations for this person -vaccinationNoVaccinationsForPersonAndDisease=There are no vaccinations for this person and disease -Vaccination=Vaccination -Vaccination.uuid=Vaccination ID -Vaccination.reportDate=Report date -Vaccination.reportingUser=Reporting user -Vaccination.vaccinationDate=Vaccination date -Vaccination.vaccineName=Vaccine name -Vaccination.otherVaccineName=Vaccine name details -Vaccination.vaccineManufacturer=Vaccine manufacturer -Vaccination.otherVaccineManufacturer=Vaccine manufacturer details -Vaccination.vaccineType=Vaccine type -Vaccination.vaccineDose=Vaccine dose -Vaccination.vaccineInn=INN -Vaccination.vaccineBatchNumber=Batch number -Vaccination.vaccineUniiCode=UNII code -Vaccination.vaccineAtcCode=ATC code -Vaccination.vaccinationInfoSource=Vaccination info source -Vaccination.pregnant=Pregnant -Vaccination.trimester=Trimester +vaccinationNewVaccination=تطعيم جديد +vaccinationNoVaccinationsForPerson=لا يوجد تطعيمات لهذا الشخص +vaccinationNoVaccinationsForPersonAndDisease=لا يوجد تطعيمات لهذا الشخص والمرض +Vaccination=التطعيمات +Vaccination.uuid=معرف التطعيم +Vaccination.reportDate=تاريخ التقرير +Vaccination.reportingUser=المستخدم المبلغ +Vaccination.vaccinationDate=تاريخ التطعيم +Vaccination.vaccineName=اسم اللقاح +Vaccination.otherVaccineName=تفاصيل اسم اللقاح +Vaccination.vaccineManufacturer=الشركة المصنعة للقاح +Vaccination.otherVaccineManufacturer=تفاصيل الشركة المصنعة للقاحات +Vaccination.vaccineType=نوع اللقاح +Vaccination.vaccineDose=جرعة اللقاح +Vaccination.vaccineInn=الأسماء الدولية غير المسجلة الملكية +Vaccination.vaccineBatchNumber=رقم الدفعة +Vaccination.vaccineUniiCode=معرف المكون الفريد +Vaccination.vaccineAtcCode=مادة كيميائية علاجية تشريحية +Vaccination.vaccinationInfoSource=مصدر معلومات التطعيم +Vaccination.pregnant=حامل +Vaccination.trimester=الثلث الثالث من الحمل # Views -View.actions=Action Directory -View.groups=Group Directory -View.aggregatereports=Aggregate Reporting (mSERS) -View.aggregatereports.aggregatereporting=Aggregate reporting -View.aggregatereports.reportdata=Report data +View.actions=دليل الإجراءات +View.groups=دليل المجموعات +View.aggregatereports=التقارير المجمعة (mSERS) +View.aggregatereports.aggregatereporting=التقارير المجمعة +View.aggregatereports.reportdata=بيانات التقرير View.aggregatereports.sub= -View.campaign.campaigns=Campaign Directory -View.campaign.campaigns.short=Campaigns -View.campaign.campaigndata=Campaign Data -View.campaign.campaigndata.short=Campaign Data -View.campaign.campaigndata.dataform=Campaign Data Form -View.campaign.campaigndata.dataform.short=Data Form -View.campaign.campaignstatistics=Campaign statistics -View.campaign.campaignstatistics.short=Campaign statistics -View.cases=Case Directory -View.cases.merge=Merge Duplicate Cases -View.cases.archive=Case Archive -View.cases.contacts=Case Contacts -View.cases.data=Case Information -View.cases.epidata=Case Epidemiological Data -View.cases.hospitalization=Case Hospitalization -View.cases.person=Case Person +View.campaign.campaigns=دليل الحملة الإعلانية +View.campaign.campaigns.short=الحملات +View.campaign.campaigndata=بيانات الحملة +View.campaign.campaigndata.short=بيانات الحملة +View.campaign.campaigndata.dataform=نموذج بيانات الحملة +View.campaign.campaigndata.dataform.short=نموذج البيانات +View.campaign.campaignstatistics=إحصاءات الحملة +View.campaign.campaignstatistics.short=إحصاءات الحملة +View.cases=دليل الحالات +View.cases.merge=دمج الحالات المتكررة +View.cases.archive=أرشيف الحالات +View.cases.contacts=إتصلات الحالة +View.cases.data=معلومات الحالة +View.cases.epidata=البيانات الوبائية للحالة +View.cases.hospitalization=حالة دخول المستشفى +View.cases.person=الشخص المسؤول عن الحالة View.cases.sub= -View.cases.symptoms=Case Symptoms -View.cases.therapy=Case Therapy +View.cases.symptoms=أعراض الحالة +View.cases.therapy=علاج الحالة View.cases.clinicalcourse=Clinical Course View.cases.maternalhistory=Maternal History View.cases.porthealthinfo=Port Health Information @@ -2706,65 +2707,65 @@ View.contacts.data=Contact Information View.contacts.merge=Merge Duplicate Contacts View.contacts.person=Contact Person View.contacts.sub= -View.contacts.visits=Contact Visits -View.dashboard.contacts=Contacts Dashboard -View.dashboard.surveillance=Surveillance Dashboard -View.dashboard.campaigns=Campaigns Dashboard -View.dashboard.samples=Samples Dashboard -View.events=Event Directory -View.events.archive=Event Archive -View.events.data=Event Information -View.events.eventactions=Event Actions -View.events.eventparticipants=Event Participants +View.contacts.visits=زيارات الاتصال +View.dashboard.contacts=لوحة معلومات جهات الاتصال +View.dashboard.surveillance=لوحة معلومات المراقبة +View.dashboard.campaigns=لوحة تحكم الحملات +View.dashboard.samples=لوحة معلومات العينات +View.events=دليل الحدث +View.events.archive=أرشيف الحدث +View.events.data=معلومات الحدث +View.events.eventactions=إجراءات الحدث +View.events.eventparticipants=المشاركون في الحدث View.events.sub= -View.events.eventparticipants.data=Event Participant Information -View.messages=Message Directory -View.reports=Weekly Reports +View.events.eventparticipants.data=معلومات المشارك في الحدث +View.messages=دليل الرسائل +View.reports=التقارير الأسبوعية View.reports.sub= -View.samples=Sample Directory -View.samples.archive=Sample Archive -View.samples.data=Sample Information +View.samples=دليل العينة +View.samples.archive=آرشيف العينة +View.samples.data=معلومات العينة View.samples.sub= -View.travelEntries=Travel Entries Directory -View.immunizations=Immunization Directory -View.statistics=Statistics -View.statistics.database-export=Database export -View.tasks=Task Management -View.tasks.archive=Task Archive +View.travelEntries=دليل إدخالات السفر +View.immunizations=دليل التحصين +View.statistics=الإحصائيات +View.statistics.database-export=تصدير قاعدة البيانات +View.tasks=إدارة المهام +View.tasks.archive=أرشيف المهام View.tasks.sub= -View.user.users=User Management -View.user.userroles=User Roles +View.user.users=إدارة المستخدمين +View.user.userroles=أدوار المستخدم View.users.sub= -View.shareRequests=Share directory -View.environments=Environment directory -View.selfreports=Self reports directory +View.shareRequests=مشاركة الدليل +View.environments=دليل البيئة +View.selfreports=دليل التقارير الذاتية # Visit -visitNewVisit=New visit -Visit=Visit -Visit.person=Visited person -Visit.symptoms=Symptoms -Visit.visitDateTime=Date and time of visit -Visit.visitRemarks=Visit remarks -Visit.visitStatus=Person available and cooperative? -Visit.origin=Visit origin -Visit.visitUser=Visiting officer -Visit.disease=Disease -Visit.reportLat=Report latitude -Visit.reportLon=Report longitude +visitNewVisit=زيارة جديدة +Visit=زيارة +Visit.person=الشخص الذي زار +Visit.symptoms=الأعراض +Visit.visitDateTime=تاريخ الزيارة ووقتها +Visit.visitRemarks=ملاحظات الزيارة +Visit.visitStatus=الشخص متاح ومتعاون؟ +Visit.origin=أصل الزيارة +Visit.visitUser=ضابط زائر +Visit.disease=المرض +Visit.reportLat=تقرير خط العرض GPS +Visit.reportLon=تقرير خط الطول GPS # WeeklyReport -weeklyReportNoReport=Missing report -weeklyReportOfficerInformants=Informants -weeklyReportsInDistrict=Weekly Reports in %s -weeklyReportRegionOfficers=Officers -weeklyReportRegionInformants=Informants -WeeklyReport.epiWeek=Epi Week -WeeklyReport.year=Year +weeklyReportNoReport=تقرير مفقود +weeklyReportOfficerInformants=المخبرون +weeklyReportsInDistrict=التقارير الأسبوعية في %s +weeklyReportRegionOfficers=الضباط +weeklyReportRegionInformants=المخبرون +WeeklyReport.epiWeek=أسبوع Epi +WeeklyReport.year=السنة # WeeklyReportEntry -WeeklyReportEntry.numberOfCases=Cases reported +WeeklyReportEntry.numberOfCases=الحالات المبلغ عنها # WeeklyReportInformantSummary -WeeklyReportInformantSummary.informantReportDate=Informant report submission -WeeklyReportInformantSummary.totalCaseCount=Cases reported by informant -WeeklyReportInformantSummary.informant=Informant +WeeklyReportInformantSummary.informantReportDate=تقديم تقرير المخبر +WeeklyReportInformantSummary.totalCaseCount=الحالات التي أبلغ عنها المخبر +WeeklyReportInformantSummary.informant=المخبر WeeklyReportInformantSummary.community=Community WeeklyReportInformantSummary.facility=Facility # WeeklyReportOfficerSummary @@ -2819,12 +2820,12 @@ BAGExport=BAG Export # Survnet Gateway ExternalSurveillanceToolGateway.title=Reporting Tool ExternalSurveillanceToolGateway.send=Send to reporting tool -ExternalSurveillanceToolGateway.unableToSend=Unable to send -ExternalSurveillanceToolGateway.confirmSend=Confirm sending -ExternalSurveillanceToolGateway.notTransferred=Not yet sent to reporting tool -ExternalSurveillanceToolGateway.confirmDelete=Confirm delete -ExternalSurveillanceToolGateway.excludeAndSend=Send %d of %d -patientDiaryRegistrationError=Could not register person in the patient diary. +ExternalSurveillanceToolGateway.unableToSend=غير قادر على الإرسال +ExternalSurveillanceToolGateway.confirmSend=تأكيد الإرسال +ExternalSurveillanceToolGateway.notTransferred=لم يتم الإرسال بعد إلى أداة الإبلاغ +ExternalSurveillanceToolGateway.confirmDelete=تأكيد الحذف +ExternalSurveillanceToolGateway.excludeAndSend=إرسال %d من %d +patientDiaryRegistrationError=لا يمكن تسجيل الشخص في سجل المريض. patientDiaryCancelError=Could not cancel external journal follow-up patientDiaryPersonNotExportable=Cannot export the person to the patient diary. The person needs a valid birthdate and either a valid phone number or email address. showPlacesOnMap=Show diff --git a/sormas-api/src/main/resources/captions_cs-CZ.properties b/sormas-api/src/main/resources/captions_cs-CZ.properties index b64d607dd2a..ed23984cfb9 100644 --- a/sormas-api/src/main/resources/captions_cs-CZ.properties +++ b/sormas-api/src/main/resources/captions_cs-CZ.properties @@ -442,7 +442,7 @@ CaseData.epidNumber=Číslo EPID CaseData.externalID=Externí ID CaseData.externalToken=Externí token CaseData.internalToken=Interní token -CaseData.caseReferenceNumber=Case Reference Number +CaseData.caseReferenceNumber=Referenční číslo případu CaseData.facilityType=Typ zařízení CaseData.healthFacility=Zařízení CaseData.healthFacilityDetails=Název a popis zařízení @@ -768,7 +768,7 @@ Contact.epiData=Epidemiologické údaje Contact.externalID=Externí ID Contact.externalToken=Externí token Contact.internalToken=Interní token -Contact.caseReferenceNumber=Case Reference Number +Contact.caseReferenceNumber=Referenční číslo případu Contact.personUuid=Identifikační číslo osoby Contact.firstContactDate=Datum prvního kontaktu Contact.firstName=Jméno kontaktní osoby @@ -1626,7 +1626,7 @@ lineListingDiseaseOfSourceCase=Nákaza zdrojového případu lineListingInfrastructureData=Převzetí údajů o infrastruktuře z poslední trati lineListingNewCasesList=Seznam nových případů lineListingNewContactsList=Seznam nových kontaktů -lineListingNewEventParticipantsList = List of new event participants +lineListingNewEventParticipantsList = Seznam nových účastníků události lineListingSharedInformation=Sdílené informace lineListingEdit=Upravit výpis řádků lineListingDisableAll=Zakázat výpis všech řádků @@ -1760,7 +1760,7 @@ PathogenTest.testedDisease=Testovaná choroba PathogenTest.testedDiseaseVariant=Testovaná varianta choroby PathogenTest.testedDiseaseDetails=Název testované choroby PathogenTest.testedPathogen=Testovaný patogen -PathogenTest.testedPathogenDetails=Tested pathogen details +PathogenTest.testedPathogenDetails=Detaily testovaného patogenu PathogenTest.typingId=Psaní ID PathogenTest.serotype=Sérotyp PathogenTest.cqValue=Hodnota CQ/CT @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Oblast User.district=Okres User.community=Komunita +User.externalId=External ID userRestrictDiseases=Omezit přístup ke konkrétním chorobám # Vaccination vaccinationNewVaccination=Nové očkování @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Odstraněné vzorky prostředí humanSampleViewType=Lidský environmentSampleViewType=Prostředí environmentSampleNewSample=Nový vzorek -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Vyberte šablonu ExternalEmailOptions.recipientEmail=E-mailová adresa ExternalEmailOptions.attachedDocuments=Přiložit dokument(y) ExternalEmailOptionsWithAttachments.attachedDocuments=Přiložit dokument(y) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Vyberte šablonu externalEmailUsedTemplate=Šablona externalEmailSentBy=Odeslal externalEmailSentTo=Příjemce @@ -2956,21 +2957,21 @@ endDateTime = Přístup do assignmentDate = Datum přidělení specialCaseAccessNew=Nový -specailCaseAccessNoAccessGranted=No special access granted for this case +specailCaseAccessNoAccessGranted=Pro tento případ není udělen žádný zvláštní přístup # SelfReport SelfReport = Self report SelfReport.type = Typ SelfReport.reportDate = Datum zprávy -SelfReport.caseReference = Case reference number +SelfReport.caseReference = Referenční číslo případu SelfReport.disease = Nemoc -SelfReport.diseaseDetails = Disease details +SelfReport.diseaseDetails = Detaily choroby SelfReport.diseaseVariant = Varianta nemoci -SelfReport.diseaseVariantDetails = Disease variant details +SelfReport.diseaseVariantDetails = Podrobnosti o variantě choroby SelfReport.firstName = Jméno SelfReport.lastName = Příjmení SelfReport.sex = Pohlaví -SelfReport.birthDate = Birth date +SelfReport.birthDate = Datum narození SelfReport.birthdateDD = Den narození SelfReport.birthdateMM = Měsíc narození SelfReport.birthdateYYYY = Rok narození @@ -2978,10 +2979,10 @@ SelfReport.nationalHealthId = Číslo zdravotního pojištění SelfReport.email = E-mail SelfReport.phoneNumber = Telefonní číslo SelfReport.address = Adresa -SelfReport.dateOfTest = Date of test +SelfReport.dateOfTest = Datum testu SelfReport.dateOfSymptoms = Datum příznaků SelfReport.workplace = Pracoviště -SelfReport.dateWorkplace = Date workplace +SelfReport.dateWorkplace = Datum pracovního místa SelfReport.isolationDate = Datum izolace SelfReport.contactDate = Datum kontaktu SelfReport.comment = Komentář @@ -2990,8 +2991,8 @@ SelfReport.investigationStatus = Stav šetření SelfReport.processingStatus = Stav zpracování selfReportActiveEnvironments=Active self reports selfReportArchivedEnvironments=Archived self reports -selfReportAllActiveAndArchivedEnvironments=All active and archived self reports -selfReportDeletedEnvironments=Deleted self reports -selfReportSelfReportsList=Self reports list +selfReportAllActiveAndArchivedEnvironments=Všechna aktivní a archivovaná vlastní hlášení +selfReportDeletedEnvironments=Odstraněná vlastní hlášení +selfReportSelfReportsList=Seznam vlastních hlášení selfReportProcess=Process diff --git a/sormas-api/src/main/resources/captions_de-CH.properties b/sormas-api/src/main/resources/captions_de-CH.properties index 15df3a94920..576a0533718 100644 --- a/sormas-api/src/main/resources/captions_de-CH.properties +++ b/sormas-api/src/main/resources/captions_de-CH.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Kanton User.district=Bezirk User.community=Gemeinde +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=Neue Impfung @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_de-DE.properties b/sormas-api/src/main/resources/captions_de-DE.properties index f4ad9cb3f08..6831c0bcb9e 100644 --- a/sormas-api/src/main/resources/captions_de-DE.properties +++ b/sormas-api/src/main/resources/captions_de-DE.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Bundesland User.district=Landkreis/Kreisfreie Stadt User.community=Gemeinde +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=Neue Impfung @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Gelöschte Umgebungs Proben humanSampleViewType=Human environmentSampleViewType=Umgebung environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_en-AF.properties b/sormas-api/src/main/resources/captions_en-AF.properties index 02ec8cbca9a..00be356bc58 100644 --- a/sormas-api/src/main/resources/captions_en-AF.properties +++ b/sormas-api/src/main/resources/captions_en-AF.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_en-GH.properties b/sormas-api/src/main/resources/captions_en-GH.properties index ac00620a291..01bc8a64031 100644 --- a/sormas-api/src/main/resources/captions_en-GH.properties +++ b/sormas-api/src/main/resources/captions_en-GH.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_en-GM.properties b/sormas-api/src/main/resources/captions_en-GM.properties index 5e15bcdd56c..e66247c99d3 100644 --- a/sormas-api/src/main/resources/captions_en-GM.properties +++ b/sormas-api/src/main/resources/captions_en-GM.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_en-KE.properties b/sormas-api/src/main/resources/captions_en-KE.properties index 8cf1839fe71..946c22b0c39 100644 --- a/sormas-api/src/main/resources/captions_en-KE.properties +++ b/sormas-api/src/main/resources/captions_en-KE.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=County User.district=Sub County User.community=Ward +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination diff --git a/sormas-api/src/main/resources/captions_en-LR.properties b/sormas-api/src/main/resources/captions_en-LR.properties index 5e15bcdd56c..e66247c99d3 100644 --- a/sormas-api/src/main/resources/captions_en-LR.properties +++ b/sormas-api/src/main/resources/captions_en-LR.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_en-NG.properties b/sormas-api/src/main/resources/captions_en-NG.properties index 4698c883f92..55f38f74cbe 100644 --- a/sormas-api/src/main/resources/captions_en-NG.properties +++ b/sormas-api/src/main/resources/captions_en-NG.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_es-BO.properties b/sormas-api/src/main/resources/captions_es-BO.properties index d22ca1492ce..aac7da9cf41 100644 --- a/sormas-api/src/main/resources/captions_es-BO.properties +++ b/sormas-api/src/main/resources/captions_es-BO.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Departamento User.district=Red de salud User.community=Municipio +User.externalId=External ID userRestrictDiseases=Restringir el acceso a enfermedades específicas # Vaccination vaccinationNewVaccination=Nueva vacuna diff --git a/sormas-api/src/main/resources/captions_es-CU.properties b/sormas-api/src/main/resources/captions_es-CU.properties index 484227c4054..6fe3bbe9195 100644 --- a/sormas-api/src/main/resources/captions_es-CU.properties +++ b/sormas-api/src/main/resources/captions_es-CU.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Provincia User.district=Municipio User.community=Área de salud +User.externalId=ID externo userRestrictDiseases=Restringir el acceso a enfermedades específicas # Vaccination vaccinationNewVaccination=Nueva vacunación diff --git a/sormas-api/src/main/resources/captions_fa-AF.properties b/sormas-api/src/main/resources/captions_fa-AF.properties index 27587e1139d..2dd16f7386b 100644 --- a/sormas-api/src/main/resources/captions_fa-AF.properties +++ b/sormas-api/src/main/resources/captions_fa-AF.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_fi-FI.properties b/sormas-api/src/main/resources/captions_fi-FI.properties index 90051049edd..ac31a73e81c 100644 --- a/sormas-api/src/main/resources/captions_fi-FI.properties +++ b/sormas-api/src/main/resources/captions_fi-FI.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_fr-CH.properties b/sormas-api/src/main/resources/captions_fr-CH.properties index 0558f096cbb..f7dfca9530f 100644 --- a/sormas-api/src/main/resources/captions_fr-CH.properties +++ b/sormas-api/src/main/resources/captions_fr-CH.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_fr-FR.properties b/sormas-api/src/main/resources/captions_fr-FR.properties index 4cade11d410..fdc122c7a20 100644 --- a/sormas-api/src/main/resources/captions_fr-FR.properties +++ b/sormas-api/src/main/resources/captions_fr-FR.properties @@ -442,7 +442,7 @@ CaseData.epidNumber=Numéro EPID CaseData.externalID=ID externe CaseData.externalToken=Référence externe CaseData.internalToken=Token interne -CaseData.caseReferenceNumber=Case Reference Number +CaseData.caseReferenceNumber=Numéro de référence du cas CaseData.facilityType=Type d'établissement CaseData.healthFacility=Établissement CaseData.healthFacilityDetails=Nom et description de l'établissement de santé @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Région User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=Nouvelle vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to @@ -2980,7 +2981,7 @@ SelfReport.phoneNumber = Phone number SelfReport.address = Address SelfReport.dateOfTest = Date of test SelfReport.dateOfSymptoms = Date of symptoms -SelfReport.workplace = Workplace +SelfReport.workplace = Lieu de travail SelfReport.dateWorkplace = Date workplace SelfReport.isolationDate = Date of isolation SelfReport.contactDate = Date of contact diff --git a/sormas-api/src/main/resources/captions_fr-TN.properties b/sormas-api/src/main/resources/captions_fr-TN.properties index 0211e617f95..503c922a4c2 100644 --- a/sormas-api/src/main/resources/captions_fr-TN.properties +++ b/sormas-api/src/main/resources/captions_fr-TN.properties @@ -47,7 +47,7 @@ system=Système to=À total=Total notSpecified=Non spécifié -noUserSelected=No user selected +noUserSelected=Aucun utilisateur sélectionné creationDate=Date de création changeDate=Date de dernière modification notAvailableShort=ND @@ -1893,7 +1893,7 @@ PersonContactDetail.additionalInformation=Information Additionnelle PersonContactDetail.thirdParty=Tierce personne PersonContactDetail.thirdPartyRole=Rôle du tiers PersonContactDetail.thirdPartyName=Nom du tiers -personRegionPrompt=Region +personRegionPrompt=Délegation personDistrictPrompt=District personCommunityPrompt=Community pointOfEntryActivePointsOfEntry=Points d’entrée actifs @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Gouvernorat User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=Nouvelle vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_it-CH.properties b/sormas-api/src/main/resources/captions_it-CH.properties index b37f263a0f8..12c1de885b1 100644 --- a/sormas-api/src/main/resources/captions_it-CH.properties +++ b/sormas-api/src/main/resources/captions_it-CH.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_it-IT.properties b/sormas-api/src/main/resources/captions_it-IT.properties index 3996bc9ebdd..a1a6bce70dc 100644 --- a/sormas-api/src/main/resources/captions_it-IT.properties +++ b/sormas-api/src/main/resources/captions_it-IT.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_pl-PL.properties b/sormas-api/src/main/resources/captions_pl-PL.properties index 9abda377415..1111500a2dc 100644 --- a/sormas-api/src/main/resources/captions_pl-PL.properties +++ b/sormas-api/src/main/resources/captions_pl-PL.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Człowiek environmentSampleViewType=Środowisko environmentSampleNewSample=Nowa próbka -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Wybierz szablon ExternalEmailOptions.recipientEmail=Adres e-mail ExternalEmailOptions.attachedDocuments=Załącz dokumenty ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Szablon externalEmailSentBy=Wysłane przez externalEmailSentTo=Wysłane do diff --git a/sormas-api/src/main/resources/captions_ps-AF.properties b/sormas-api/src/main/resources/captions_ps-AF.properties index 520eda21892..86db71ab1b4 100644 --- a/sormas-api/src/main/resources/captions_ps-AF.properties +++ b/sormas-api/src/main/resources/captions_ps-AF.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_pt-CV.properties b/sormas-api/src/main/resources/captions_pt-CV.properties index 0096df9cacc..3588afbb9ec 100644 --- a/sormas-api/src/main/resources/captions_pt-CV.properties +++ b/sormas-api/src/main/resources/captions_pt-CV.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_ru-RU.properties b/sormas-api/src/main/resources/captions_ru-RU.properties index be21e31b250..791cfb22b79 100644 --- a/sormas-api/src/main/resources/captions_ru-RU.properties +++ b/sormas-api/src/main/resources/captions_ru-RU.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/captions_ur-PK.properties b/sormas-api/src/main/resources/captions_ur-PK.properties index a9d28d5eda7..28754c59df5 100644 --- a/sormas-api/src/main/resources/captions_ur-PK.properties +++ b/sormas-api/src/main/resources/captions_ur-PK.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=علاقہ User.district=ضلع User.community=کمیونیٹی +User.externalId=External ID userRestrictDiseases=مخصوص بیماریوں تک رسائی کو محدود کریں # Vaccination vaccinationNewVaccination=نئی ویکسینیشن @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=حذف شدہ ماحول کے نمونے humanSampleViewType=انسان environmentSampleViewType=ماحول environmentSampleNewSample=نیا نمونہ -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=ٹیمپلیٹ کا انتخاب کریں ExternalEmailOptions.recipientEmail=ای میل اڈریس ExternalEmailOptions.attachedDocuments=دستاویز منسلک کریں ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=ٹیمپلیٹ externalEmailSentBy=کی طرف سے بھیجا externalEmailSentTo=کو بھیجا گیا diff --git a/sormas-api/src/main/resources/captions_zh-CN.properties b/sormas-api/src/main/resources/captions_zh-CN.properties index 48a736946ea..fbaff445953 100644 --- a/sormas-api/src/main/resources/captions_zh-CN.properties +++ b/sormas-api/src/main/resources/captions_zh-CN.properties @@ -2606,6 +2606,7 @@ User.uuid=UUID User.region=Region User.district=District User.community=Community +User.externalId=External ID userRestrictDiseases=Restrict access to specific diseases # Vaccination vaccinationNewVaccination=New vaccination @@ -2937,11 +2938,11 @@ environmentSampleDeletedSamples=Deleted environment samples humanSampleViewType=Human environmentSampleViewType=Environment environmentSampleNewSample=New sample -ExternalEmailOptions.templateName=Choose email template +ExternalEmailOptions.templateName=Choose template ExternalEmailOptions.recipientEmail=Email address ExternalEmailOptions.attachedDocuments=Attach document(s) ExternalEmailOptionsWithAttachments.attachedDocuments=Attach document(s) -ExternalEmailOptionsWithAttachments.templateName=Choose email template +ExternalEmailOptionsWithAttachments.templateName=Choose template externalEmailUsedTemplate=Template externalEmailSentBy=Sent by externalEmailSentTo=Sent to diff --git a/sormas-api/src/main/resources/enum_ar-SA.properties b/sormas-api/src/main/resources/enum_ar-SA.properties index 19bae559a5d..2be43de7c44 100644 --- a/sormas-api/src/main/resources/enum_ar-SA.properties +++ b/sormas-api/src/main/resources/enum_ar-SA.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_cs-CZ.properties b/sormas-api/src/main/resources/enum_cs-CZ.properties index 4d45f58c6a9..b741debee97 100644 --- a/sormas-api/src/main/resources/enum_cs-CZ.properties +++ b/sormas-api/src/main/resources/enum_cs-CZ.properties @@ -463,7 +463,7 @@ Disease.EVD = Virus Ebola Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Spalničky -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Chřipka (Nový subtyp) Disease.UNDEFINED = Zatím není definováno Disease.OTHER = Jiná epidemická onemocnění @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Horečka Lassa Disease.Short.MEASLES = Spalničky -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nová chřipka Disease.Short.UNDEFINED = Nedefinováno Disease.Short.OTHER = Ostatní @@ -873,17 +873,21 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potenciálně vystaven nákaze KindOfInvolvement.POTENTIAL_INDEX_CASE = Seznam potenciálních případů Language.EN = Angličtina +Language.EN_GM = Angličtina (Gambie) +Language.EN_LR = Angličtina (Nigerie) Language.EN_AF = Angličtina (Afghánistán) Language.EN_NG = Angličtina (Nigerie) Language.EN_GH = Angličtina (Ghana) +Language.EN_KE = Angličtina (Kenya) Language.FR = Francouzština Language.DE = Němčina -Language.ES_BO = Español (Bolivia) +Language.ES_BO = Španělština (Bolívie) Language.ES_EC = Španělština (Ekvádor) Language.ES_CU = Španělština (Kuba) Language.FI = Finština Language.IT = Italština Language.DE_CH = Němčina (Švýcarsko) +Language.PT_CV = Portugalština (Kapverdy) Language.IT_CH = Italština (Švýcarsko) Language.FR_CH = Francouzština (Švýcarsko) Language.PS = Paštto @@ -1398,7 +1402,7 @@ UserRight.CASE_TRANSFER = Přenos případů do jiného regionu/okresu/zařízen UserRight.CASE_REFER_FROM_POE = Viz případ z místa vstupu UserRight.CASE_RESPONSIBLE = Může být zodpovědný za případ UserRight.CASE_VIEW = Zobrazit existující případy -UserRight.CASE_VIEW_ARCHIVED = View archived cases +UserRight.CASE_VIEW_ARCHIVED = Zobrazit archivované případy UserRight.CONTACT_ASSIGN = Přiřadit kontakty úředníkům UserRight.CONTACT_CLASSIFY = Upravit klasifikaci kontaktů UserRight.CONTACT_CONVERT = Vytvořit výsledné případy z kontaktů @@ -1409,7 +1413,7 @@ UserRight.CONTACT_EDIT = Upravit existující kontakty UserRight.CONTACT_EXPORT = Exportovat kontakty ze SORMAS UserRight.CONTACT_RESPONSIBLE = Může být zodpovědný za kontakt UserRight.CONTACT_VIEW = Zobrazit existující kontakty -UserRight.CONTACT_VIEW_ARCHIVED = View archived contacts +UserRight.CONTACT_VIEW_ARCHIVED = Zobrazit archivované kontakty UserRight.CONTACT_ARCHIVE = Archivovat kontakty UserRight.DASHBOARD_CONTACT_VIEW = Přístup k ovládacínu panelu kontaktního orgánu UserRight.DASHBOARD_SURVEILLANCE_VIEW = Přístup do ovládacího panelu dozorčího orgánu @@ -1421,16 +1425,16 @@ UserRight.EVENT_EDIT = Upravit existující události UserRight.EVENT_EXPORT = Exportovat události ze SORMAS UserRight.EVENT_RESPONSIBLE = Může být zodpovědný za událost UserRight.EVENT_VIEW = Zobrazit existující události -UserRight.EVENT_VIEW_ARCHIVED = View archived events +UserRight.EVENT_VIEW_ARCHIVED = Zobrazit archivované události UserRight.EVENTPARTICIPANT_CREATE = Vytvořit nové účastníky události UserRight.EVENTPARTICIPANT_EDIT = Upravit existující účastníky události UserRight.EVENTPARTICIPANT_ARCHIVE = Archiv účastníků událostí UserRight.EVENTPARTICIPANT_VIEW = Zobrazit existující účastníky události -UserRight.EVENTPARTICIPANT_VIEW_ARCHIVED = View archived event participants +UserRight.EVENTPARTICIPANT_VIEW_ARCHIVED = Zobrazit archivované účastníky události UserRight.INFRASTRUCTURE_CREATE = Vytvořit nové regiony/okresy/komunity/zařízení UserRight.INFRASTRUCTURE_EDIT = Upravit regiony/okresy/komunity/zařízení UserRight.INFRASTRUCTURE_VIEW = Zobrazit regiony/okresy/komunity/zařízení v systému -UserRight.INFRASTRUCTURE_VIEW_ARCHIVED = View archived infrastructure data +UserRight.INFRASTRUCTURE_VIEW_ARCHIVED = Zobrazit archivované údaje o infrastruktuře UserRight.PERFORM_BULK_OPERATIONS = Provést hromadné operace v seznamech UserRight.SAMPLE_CREATE = Vytvořit nové vzorky UserRight.SAMPLE_EDIT = Upravit existující vzorky @@ -1447,7 +1451,7 @@ UserRight.TASK_CREATE = Vytvořit nové úkoly UserRight.TASK_EDIT = Upravit existující úkoly UserRight.TASK_SEE_ARCHIVED = Zobrazit archivované úkoly UserRight.TASK_VIEW = Zobrazit existující úkoly -UserRight.TASK_VIEW_ARCHIVED = View archived tasks +UserRight.TASK_VIEW_ARCHIVED = Zobrazit archivované úkoly UserRight.USER_CREATE = Vytvořit nové uživatele UserRight.USER_EDIT = Upravit existující uživatele UserRight.USER_VIEW = Zobrazit existující uživatele @@ -1520,12 +1524,12 @@ UserRight.SEE_PERSONAL_DATA_OUTSIDE_JURISDICTION = Viz osobní údaje mimo juris UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION = Viz citlivé údaje v jurisdikci UserRight.SEE_SENSITIVE_DATA_OUTSIDE_JURISDICTION = Viz citlivé údaje mimo jurisdikci UserRight.CAMPAIGN_VIEW = Zobrazit existující kampaně -UserRight.CAMPAIGN_VIEW_ARCHIVED = View archived campaigns +UserRight.CAMPAIGN_VIEW_ARCHIVED = Zobrazit archivované kampaně UserRight.CAMPAIGN_EDIT = Upravit existující kampaně UserRight.CAMPAIGN_ARCHIVE = Archivovat kampaně UserRight.CAMPAIGN_DELETE = Odstranit kampaně ze systému UserRight.CAMPAIGN_FORM_DATA_VIEW = Zobrazit existující data formuláře kampaně -UserRight.CAMPAIGN_FORM_DATA_VIEW_ARCHIVED = View archived campaign form data +UserRight.CAMPAIGN_FORM_DATA_VIEW_ARCHIVED = Zobrazit archivovaná data formuláře kampaně UserRight.CAMPAIGN_FORM_DATA_EDIT = Upravit existující data formuláře kampaně UserRight.CAMPAIGN_FORM_DATA_ARCHIVE = Archivovat data formuláře kampaně UserRight.CAMPAIGN_FORM_DATA_DELETE = Odstranit data formuláře kampaně ze systému @@ -1541,7 +1545,7 @@ UserRight.EXTERNAL_MESSAGE_PUSH = Nabídnout odeslání externí zprávy do syst UserRight.EXTERNAL_MESSAGE_DELETE = Odstranit zprávy ze systému UserRight.CASE_SHARE = Sdílet případy s celou zemí UserRight.IMMUNIZATION_VIEW = Zobrazit existující imunizace a očkování -UserRight.IMMUNIZATION_VIEW_ARCHIVED = View archived immunizations and vaccinations +UserRight.IMMUNIZATION_VIEW_ARCHIVED = Zobrazit archivované imunizace a očkování UserRight.IMMUNIZATION_CREATE = Vytvořit novou imunizaci a očkování UserRight.IMMUNIZATION_EDIT = Upravit existující imunizaci a očkování UserRight.IMMUNIZATION_DELETE = Odstranit imunizaci a očkování ze systému @@ -1553,10 +1557,10 @@ UserRight.EVENTGROUP_EDIT = Upravit existující skupiny událostí UserRight.EVENTGROUP_LINK = Propojit události se skupinami událostí UserRight.EVENTGROUP_ARCHIVE = Archivovat skupiny událostí UserRight.EVENTGROUP_DELETE = Odstranit skupiny událostí ze systému -UserRight.EVENTGROUP_VIEW_ARCHIVED = View archived event groups from the system +UserRight.EVENTGROUP_VIEW_ARCHIVED = Zobrazit archivované skupiny událostí ze systému UserRight.TRAVEL_ENTRY_MANAGEMENT_ACCESS = Přístup k adresáři míst vstupu UserRight.TRAVEL_ENTRY_VIEW = Zobrazit existující místa vstupu -UserRight.TRAVEL_ENTRY_VIEW_ARCHIVED = View archived travel entries +UserRight.TRAVEL_ENTRY_VIEW_ARCHIVED = Zobrazit archivovaná místa vstupu UserRight.TRAVEL_ENTRY_CREATE = Vytvořit nové místo vstupu UserRight.TRAVEL_ENTRY_EDIT = Upravit stávající místa vstupu UserRight.TRAVEL_ENTRY_DELETE = Odstranit místa vstupu ze systému @@ -1579,7 +1583,7 @@ UserRight.ENVIRONMENT_VIEW = Zobrazit existující prostředí UserRight.ENVIRONMENT_CREATE = Vytvořit nové prostředí UserRight.ENVIRONMENT_EDIT = Upravit existující prostředí UserRight.ENVIRONMENT_ARCHIVE = Archivovat prostředí -UserRight.ENVIRONMENT_VIEW_ARCHIVED = View archived environments +UserRight.ENVIRONMENT_VIEW_ARCHIVED = Zobrazit archivované prostředí UserRight.ENVIRONMENT_DELETE = Odstranit prostředí ze systému UserRight.ENVIRONMENT_IMPORT = Importovat prostředí UserRight.ENVIRONMENT_EXPORT = Exportovat prostředí @@ -1668,7 +1672,7 @@ UserRight.Desc.TASK_ASSIGN = Může přiřadit úkoly uživatelům UserRight.Desc.TASK_CREATE = Může vytvářet nové úkoly UserRight.Desc.TASK_EDIT = Může upravit existující úkoly UserRight.Desc.TASK_VIEW = Může zobrazit existující úkoly -UserRight.Desc.TASK_VIEW_ARCHIVED = Able to view archived tasks +UserRight.Desc.TASK_VIEW_ARCHIVED = Může zobrazit archivované úkoly UserRight.Desc.TASK_ARCHIVE = Může archivovat úkoly UserRight.Desc.USER_CREATE = Může vytvářet nové uživatele UserRight.Desc.USER_EDIT = Může upravit existující uživatele @@ -1738,7 +1742,7 @@ UserRight.Desc.SEE_PERSONAL_DATA_OUTSIDE_JURISDICTION = Může vidět osobní ú UserRight.Desc.SEE_SENSITIVE_DATA_IN_JURISDICTION = Může vidět citlivé údaje v jurisdikci UserRight.Desc.SEE_SENSITIVE_DATA_OUTSIDE_JURISDICTION = Může vidět citlivé údaje mimo jurisdikci UserRight.Desc.CAMPAIGN_VIEW = Může zobrazit existující kampaně -UserRight.Desc.CAMPAIGN_VIEW_ARCHIVED = Able to view archived campaigns +UserRight.Desc.CAMPAIGN_VIEW_ARCHIVED = Může zobrazit archivované kampaně UserRight.Desc.CAMPAIGN_EDIT = Může upravit existující kampaně UserRight.Desc.CAMPAIGN_ARCHIVE = Může archivovat kampaně UserRight.Desc.CAMPAIGN_DELETE = Může odstranit kampaně ze systému diff --git a/sormas-api/src/main/resources/enum_de-CH.properties b/sormas-api/src/main/resources/enum_de-CH.properties index 4a1f5ded21d..17b24b2ed07 100644 --- a/sormas-api/src/main/resources/enum_de-CH.properties +++ b/sormas-api/src/main/resources/enum_de-CH.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Disease.GUINEA_WORM = Medinawurm Disease.LASSA = Lassa Disease.MEASLES = Masern -Disease.MONKEYPOX = Affenpocken +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (neuer Subtyp) Disease.UNDEFINED = Noch nicht definiert Disease.OTHER = Andere epidemische Krankheit @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Medinawurm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Masern -Disease.Short.MONKEYPOX = Affenpocken +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Neue Grippe Disease.Short.UNDEFINED = Nicht definiert Disease.Short.OTHER = Sonstiges @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potenziell exponiert KindOfInvolvement.POTENTIAL_INDEX_CASE = Potentieller Indexfall Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Englisch (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Spanisch (Kuba) Language.FI = Finnisch Language.IT = Italienisch Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Paschtunisch diff --git a/sormas-api/src/main/resources/enum_de-DE.properties b/sormas-api/src/main/resources/enum_de-DE.properties index 9d11565d7a2..5605c7d53d9 100644 --- a/sormas-api/src/main/resources/enum_de-DE.properties +++ b/sormas-api/src/main/resources/enum_de-DE.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Disease.GUINEA_WORM = Medinawurm Disease.LASSA = Lassa Disease.MEASLES = Masern -Disease.MONKEYPOX = Affenpocken +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (neuer Subtyp) Disease.UNDEFINED = Noch nicht definiert Disease.OTHER = Andere epidemische Krankheit @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Medinawurm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Masern -Disease.Short.MONKEYPOX = Affenpocken +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Neue Grippe Disease.Short.UNDEFINED = Nicht definiert Disease.Short.OTHER = Sonstiges @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potenziell exponiert KindOfInvolvement.POTENTIAL_INDEX_CASE = Potentieller Indexfall Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Englisch (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Französisch Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Spanisch (Kuba) Language.FI = Finnisch Language.IT = Italienisch Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Paschtunisch diff --git a/sormas-api/src/main/resources/enum_en-AF.properties b/sormas-api/src/main/resources/enum_en-AF.properties index 0e3ee25130b..4c3594f2f8b 100644 --- a/sormas-api/src/main/resources/enum_en-AF.properties +++ b/sormas-api/src/main/resources/enum_en-AF.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_en-GH.properties b/sormas-api/src/main/resources/enum_en-GH.properties index 93837b0b213..b1175f3ac1b 100644 --- a/sormas-api/src/main/resources/enum_en-GH.properties +++ b/sormas-api/src/main/resources/enum_en-GH.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_en-GM.properties b/sormas-api/src/main/resources/enum_en-GM.properties index 19bae559a5d..2be43de7c44 100644 --- a/sormas-api/src/main/resources/enum_en-GM.properties +++ b/sormas-api/src/main/resources/enum_en-GM.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_en-KE.properties b/sormas-api/src/main/resources/enum_en-KE.properties index 9748d9c26bf..5e875557fca 100644 --- a/sormas-api/src/main/resources/enum_en-KE.properties +++ b/sormas-api/src/main/resources/enum_en-KE.properties @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_en-LR.properties b/sormas-api/src/main/resources/enum_en-LR.properties index 19bae559a5d..2be43de7c44 100644 --- a/sormas-api/src/main/resources/enum_en-LR.properties +++ b/sormas-api/src/main/resources/enum_en-LR.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_en-NG.properties b/sormas-api/src/main/resources/enum_en-NG.properties index 19bae559a5d..2be43de7c44 100644 --- a/sormas-api/src/main/resources/enum_en-NG.properties +++ b/sormas-api/src/main/resources/enum_en-NG.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_es-BO.properties b/sormas-api/src/main/resources/enum_es-BO.properties index 03fa7579ebe..4f284d5d722 100644 --- a/sormas-api/src/main/resources/enum_es-BO.properties +++ b/sormas-api/src/main/resources/enum_es-BO.properties @@ -463,7 +463,7 @@ Disease.EVD = Enfermedad del virus del ébola Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Medidas -Disease.MONKEYPOX = Viruela +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (nuevo subtipo) Disease.UNDEFINED = Aún no definido Disease.OTHER = Otras enfermedades epidémicas @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Sarampión -Disease.Short.MONKEYPOX = Viruela del mono +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nueva gripe Disease.Short.UNDEFINED = Indefinido Disease.Short.OTHER = Otro @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentialmente Expuesta KindOfInvolvement.POTENTIAL_INDEX_CASE = caso Índice Potential Language.EN = Inglés +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Inglés (Afganistán) Language.EN_NG = Inglés (Nigeria) Language.EN_GH = Inglés (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Francés-ais Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Alemán (Suiza) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Suiza) Language.FR_CH = Francés\: ais (Suizo) Language.PS = Pastún diff --git a/sormas-api/src/main/resources/enum_es-CU.properties b/sormas-api/src/main/resources/enum_es-CU.properties index d907fad3781..b7f89132f1c 100644 --- a/sormas-api/src/main/resources/enum_es-CU.properties +++ b/sormas-api/src/main/resources/enum_es-CU.properties @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potencialmente expuesto KindOfInvolvement.POTENTIAL_INDEX_CASE = Caso índice potencial Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_fa-AF.properties b/sormas-api/src/main/resources/enum_fa-AF.properties index df16181bc78..46831078229 100644 --- a/sormas-api/src/main/resources/enum_fa-AF.properties +++ b/sormas-api/src/main/resources/enum_fa-AF.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_fi-FI.properties b/sormas-api/src/main/resources/enum_fi-FI.properties index 7708fc6a90e..441e078e5d2 100644 --- a/sormas-api/src/main/resources/enum_fi-FI.properties +++ b/sormas-api/src/main/resources/enum_fi-FI.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebolavirustauti Disease.GUINEA_WORM = Guinean matotauti (drakunkuliaasi) Disease.LASSA = Lassakuume Disease.MEASLES = Tuhkarokko -Disease.MONKEYPOX = Apinarokko +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenssa (uusi alamuoto) Disease.UNDEFINED = Ei vielä määritelty Disease.OTHER = Muu tartuntatauti @@ -524,7 +524,7 @@ Disease.Short.EVD = Ebola Disease.Short.GUINEA_WORM = Guinean matotauti Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Tuhkarokko -Disease.Short.MONKEYPOX = Apinarokko +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Uusi influenssa Disease.Short.UNDEFINED = Määrittelemätön Disease.Short.OTHER = Muu @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Mahdollisesti altistunut KindOfInvolvement.POTENTIAL_INDEX_CASE = Mahdollinen indeksitapaus Language.EN = Englanti +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = Englanti (Nigeria) Language.EN_GH = Englanti (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Ranska Language.DE = Saksa Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Espanja (Kuuba) Language.FI = Suomi Language.IT = Italia Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (France) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_fr-CH.properties b/sormas-api/src/main/resources/enum_fr-CH.properties index 572f1ee9f91..377dd1d3437 100644 --- a/sormas-api/src/main/resources/enum_fr-CH.properties +++ b/sormas-api/src/main/resources/enum_fr-CH.properties @@ -463,7 +463,7 @@ Disease.EVD = Maladie à virus Ebola Disease.GUINEA_WORM = Ver de Guinée Disease.LASSA = Fièvre de lassa Disease.MEASLES = Rougeole -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Grippe (nouveau sous-type) Disease.UNDEFINED = Pas encore défini Disease.OTHER = Autre maladie épidémique @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Ver de Guinée Disease.Short.LASSA = Lasse Disease.Short.MEASLES = Rougeole -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nouvelle grippe Disease.Short.UNDEFINED = Non défini Disease.Short.OTHER = Autres @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentiellement exposé KindOfInvolvement.POTENTIAL_INDEX_CASE = Cas d'index potentiel Language.EN = Anglais +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Anglais (Afghanistan) Language.EN_NG = Anglais (Nigeria) Language.EN_GH = Anglais (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Allemand Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Espagnol (Cuba) Language.FI = Finnois Language.IT = Italien Language.DE_CH = Allemand (Suisse) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italien (Suisse) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_fr-FR.properties b/sormas-api/src/main/resources/enum_fr-FR.properties index d2a4058c001..1cf1360da9d 100644 --- a/sormas-api/src/main/resources/enum_fr-FR.properties +++ b/sormas-api/src/main/resources/enum_fr-FR.properties @@ -463,7 +463,7 @@ Disease.EVD = Maladie à virus Ebola Disease.GUINEA_WORM = Ver de Guinée Disease.LASSA = Fièvre de Lassa Disease.MEASLES = Rougeole -Disease.MONKEYPOX = Variole du singe +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Grippe (nouveau sous-type) Disease.UNDEFINED = Pas encore défini Disease.OTHER = Autre maladie épidémique @@ -524,7 +524,7 @@ Disease.Short.EVD = Ebola Disease.Short.GUINEA_WORM = Ver de Guinée Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Rougeole -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nouvelle grippe Disease.Short.UNDEFINED = Non défini Disease.Short.OTHER = Autres @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentiellement exposé KindOfInvolvement.POTENTIAL_INDEX_CASE = Cas d'index potentiel Language.EN = Anglais +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Anglais (Afghanistan) Language.EN_NG = Anglais (Nigeria) Language.EN_GH = Anglais (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Allemand Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Espagnol (Cuba) Language.FI = Finlande Language.IT = Italien Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pachto diff --git a/sormas-api/src/main/resources/enum_fr-TN.properties b/sormas-api/src/main/resources/enum_fr-TN.properties index dbfa892f94b..2d4774f5a9f 100644 --- a/sormas-api/src/main/resources/enum_fr-TN.properties +++ b/sormas-api/src/main/resources/enum_fr-TN.properties @@ -463,7 +463,7 @@ Disease.EVD = Maladie à virus Ebola Disease.GUINEA_WORM = Ver de Guinée Disease.LASSA = Fièvre de Lassa Disease.MEASLES = Rougeole -Disease.MONKEYPOX = Variole du singe +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Grippe (nouveau sous-type) Disease.UNDEFINED = Pas encore défini Disease.OTHER = Autre maladie épidémique @@ -524,7 +524,7 @@ Disease.Short.EVD = Ebola Disease.Short.GUINEA_WORM = Ver de Guinée Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Rougeole -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nouvelle grippe Disease.Short.UNDEFINED = Non défini Disease.Short.OTHER = Autres @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentiellement exposé KindOfInvolvement.POTENTIAL_INDEX_CASE = Cas d'index potentiel Language.EN = Anglais +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Anglais (Afghanistan) Language.EN_NG = Anglais (Nigeria) Language.EN_GH = Anglais (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Allemand Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Espagnol (Cuba) Language.FI = Finlande Language.IT = Italien Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pachto @@ -1375,7 +1379,7 @@ TypeOfPlace.FACILITY_23_IFSG = Etablissement (§ 23 IfSG) TypeOfPlace.COMMUNITY_FACILITY = Etablissement communautaire (§ 33 IfSG) TypeOfPlace.FACILITY_36_IFSG = Etablissement (§ 36 IfSG) TypeOfPlace.FESTIVITIES = Fêtes -TypeOfPlace.HOME = Sphère privée +TypeOfPlace.HOME = Domicile TypeOfPlace.HOSPITAL = Hôpitaux TypeOfPlace.MEANS_OF_TRANSPORT = Moyens de transport TypeOfPlace.OTHER = Autre diff --git a/sormas-api/src/main/resources/enum_it-CH.properties b/sormas-api/src/main/resources/enum_it-CH.properties index cd18ddb492e..2d088f94e1e 100644 --- a/sormas-api/src/main/resources/enum_it-CH.properties +++ b/sormas-api/src/main/resources/enum_it-CH.properties @@ -463,7 +463,7 @@ Disease.EVD = Malattia da virus Ebola Disease.GUINEA_WORM = Dracunculiasi Disease.LASSA = Febbre di Lassa Disease.MEASLES = Morbillo -Disease.MONKEYPOX = Vaiolo delle scimmie +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (nuovo sottotipo) Disease.UNDEFINED = Non ancora definito Disease.OTHER = Altre Malattie Epidemiche @@ -524,7 +524,7 @@ Disease.Short.EVD = Ebola Disease.Short.GUINEA_WORM = Dracunculiasi Disease.Short.LASSA = Febbre di Lassa Disease.Short.MEASLES = Morbillo -Disease.Short.MONKEYPOX = Vaiolo delle scimmie +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nuovo Influenza Disease.Short.UNDEFINED = Non definito Disease.Short.OTHER = Altro @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potenzialmente esposto KindOfInvolvement.POTENTIAL_INDEX_CASE = Caso indice potenziale Language.EN = Inglese +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = Inglese (Afghanistan) Language.EN_NG = Inglese (Nigeria) Language.EN_GH = Inglese (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Francese Language.DE = Tedesco Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Spagnolo (Cuba) Language.FI = Finlandese Language.IT = Italiano Language.DE_CH = Tedesco (Svizzera) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Francese (Svizzera) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_it-IT.properties b/sormas-api/src/main/resources/enum_it-IT.properties index 8a42e618dce..ad062ed5c77 100644 --- a/sormas-api/src/main/resources/enum_it-IT.properties +++ b/sormas-api/src/main/resources/enum_it-IT.properties @@ -463,7 +463,7 @@ Disease.EVD = Malattia da virus Ebola Disease.GUINEA_WORM = Dracunculiasi Disease.LASSA = Febbre di Lassa Disease.MEASLES = Morbillo -Disease.MONKEYPOX = Vaiolo delle scimmie +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (nuovo sottotipo) Disease.UNDEFINED = Non ancora definito Disease.OTHER = Altre Malattie Epidemiche @@ -524,7 +524,7 @@ Disease.Short.EVD = Ebola Disease.Short.GUINEA_WORM = Dracunculiasi Disease.Short.LASSA = Febbre di Lassa Disease.Short.MEASLES = Morbillo -Disease.Short.MONKEYPOX = Vaiolo delle scimmie +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = Nuovo Influenza Disease.Short.UNDEFINED = Non definito Disease.Short.OTHER = Altro @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potenzialmente esposto KindOfInvolvement.POTENTIAL_INDEX_CASE = Caso indice potenziale Language.EN = Inglese +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = Inglese (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Francese Language.DE = Tedesco Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Spagnolo (Cuba) Language.FI = Finlandese Language.IT = Italiano Language.DE_CH = Tedesco (Svizzera) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Francese (Svizzera) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_pl-PL.properties b/sormas-api/src/main/resources/enum_pl-PL.properties index f2fa89bf188..5b17c31d54c 100644 --- a/sormas-api/src/main/resources/enum_pl-PL.properties +++ b/sormas-api/src/main/resources/enum_pl-PL.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_ps-AF.properties b/sormas-api/src/main/resources/enum_ps-AF.properties index df16181bc78..46831078229 100644 --- a/sormas-api/src/main/resources/enum_ps-AF.properties +++ b/sormas-api/src/main/resources/enum_ps-AF.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_pt-CV.properties b/sormas-api/src/main/resources/enum_pt-CV.properties index 4c04745499c..04f6056b9c7 100644 --- a/sormas-api/src/main/resources/enum_pt-CV.properties +++ b/sormas-api/src/main/resources/enum_pt-CV.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_ru-RU.properties b/sormas-api/src/main/resources/enum_ru-RU.properties index 4c162522722..545283ae28e 100644 --- a/sormas-api/src/main/resources/enum_ru-RU.properties +++ b/sormas-api/src/main/resources/enum_ru-RU.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/enum_ur-PK.properties b/sormas-api/src/main/resources/enum_ur-PK.properties index e0696a944f6..651253f8830 100644 --- a/sormas-api/src/main/resources/enum_ur-PK.properties +++ b/sormas-api/src/main/resources/enum_ur-PK.properties @@ -463,7 +463,7 @@ Disease.EVD = ایبولا وائرس کی بیماری Disease.GUINEA_WORM = گائنی ورم Disease.LASSA = لسا Disease.MEASLES = خسرہ -Disease.MONKEYPOX = مونکی پوکس +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = انفلوئنزا (نئی ذیلی قسم) Disease.UNDEFINED = ابھی تک متعین نہیں Disease.OTHER = دیگر وبائی بیماریاں @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = گائنی ورم Disease.Short.LASSA = لسا Disease.Short.MEASLES = خسرہ -Disease.Short.MONKEYPOX = مونکی پوکس +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = نیا فلو Disease.Short.UNDEFINED = غیر وضاحتی Disease.Short.OTHER = دیگر @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = ممکنہ طور پرظاہر KindOfInvolvement.POTENTIAL_INDEX_CASE = ممکنہ انڈیکس کیس Language.EN = انگریزی +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = انگریزی (افغانستان) Language.EN_NG = انگریزی (نائیجیریا) Language.EN_GH = انگریزی (گھانا) +Language.EN_KE = English (Kenya) Language.FR = فرانسیسی Language.DE = جرمن Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = ہسپانوی (کیوبا) Language.FI = سوومی Language.IT = اطالوی Language.DE_CH = جرمن (سوئٹزرلینڈ) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = اطالوی (سوئٹزرلینڈ) Language.FR_CH = فرانسیسی (سوئٹزرلینڈ) Language.PS = پشتو diff --git a/sormas-api/src/main/resources/enum_zh-CN.properties b/sormas-api/src/main/resources/enum_zh-CN.properties index 941983ad433..0b5cf79d7da 100644 --- a/sormas-api/src/main/resources/enum_zh-CN.properties +++ b/sormas-api/src/main/resources/enum_zh-CN.properties @@ -463,7 +463,7 @@ Disease.EVD = Ebola Virus Disease Disease.GUINEA_WORM = Guinea Worm Disease.LASSA = Lassa Disease.MEASLES = Measles -Disease.MONKEYPOX = Monkeypox +Disease.MONKEYPOX = Mpox Disease.NEW_INFLUENZA = Influenza (New subtype) Disease.UNDEFINED = Not Yet Defined Disease.OTHER = Other Epidemic Disease @@ -524,7 +524,7 @@ Disease.Short.EVD = EVD Disease.Short.GUINEA_WORM = Guinea Worm Disease.Short.LASSA = Lassa Disease.Short.MEASLES = Measles -Disease.Short.MONKEYPOX = Monkeypox +Disease.Short.MONKEYPOX = Mpox Disease.Short.NEW_INFLUENZA = New Flu Disease.Short.UNDEFINED = Undefined Disease.Short.OTHER = Other @@ -873,9 +873,12 @@ KindOfInvolvement.POTENTIALLY_EXPOSED = Potentially exposed KindOfInvolvement.POTENTIAL_INDEX_CASE = Potential index case Language.EN = English +Language.EN_GM = English (The Gambia) +Language.EN_LR = English (Liberia) Language.EN_AF = English (Afghanistan) Language.EN_NG = English (Nigeria) Language.EN_GH = English (Ghana) +Language.EN_KE = English (Kenya) Language.FR = Français Language.DE = Deutsch Language.ES_BO = Español (Bolivia) @@ -884,6 +887,7 @@ Language.ES_CU = Español (Cuba) Language.FI = Suomi Language.IT = Italiano Language.DE_CH = Deutsch (Schweiz) +Language.PT_CV = Português (Cabo Verde) Language.IT_CH = Italiano (Svizzera) Language.FR_CH = Français (Suisse) Language.PS = Pashto diff --git a/sormas-api/src/main/resources/strings_ar-SA.properties b/sormas-api/src/main/resources/strings_ar-SA.properties index cc545e46a9f..0c46b89e9f0 100644 --- a/sormas-api/src/main/resources/strings_ar-SA.properties +++ b/sormas-api/src/main/resources/strings_ar-SA.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. @@ -1821,4 +1823,4 @@ confirmationSelfReportLinkContactsByCaseReference=Some contacts without a case h headingSelfReportCaseReportWithSameReferenceFound=Case self report with same case reference found confirmationSelfReportCaseReportWithSameReferenceFound=There is a case self report with the same case reference number as the processed self report.
It is recommended to process case reports first.

Do you want to continue processing this self report? headingSelfReportCaseWithSameReferenceNumberFound=Case with same reference number found -confirmationSelfReportLinkContactToCaseWithSameReferenceNumber=There is a case with the same reference number as the contact found

Do you want to link this contact to that case? +confirmationSelfReportLinkContactToCaseWithSameReferenceNumber=There is a case with the same reference number as the contact found

Do you want to link this contact to that case? \ No newline at end of file diff --git a/sormas-api/src/main/resources/strings_cs-CZ.properties b/sormas-api/src/main/resources/strings_cs-CZ.properties index 647470e2901..66f2cfe678e 100644 --- a/sormas-api/src/main/resources/strings_cs-CZ.properties +++ b/sormas-api/src/main/resources/strings_cs-CZ.properties @@ -1489,7 +1489,7 @@ messageEnvironmentJurisdictionUpdated = Změna umístění tohoto prostředí by messageNoEnvironmentSamplesSelected = You have not selected any environment samples messageEnvironmentSamplesDeleted = All selected eligible environment samples have been deleted messageCountEnvironmentSamplesNotDeleted = %s environment samples not deleted\: %s -messageCountEnvironmentSamplesNotDeletedAccessDeniedReason = %s environment samples not deleted because they are not in jurisdiction or owned\: %s +messageCountEnvironmentSamplesNotDeletedAccessDeniedReason = %s vzorky prostředí nebyly odstraněny, protože nejsou v jurisdikci, nebo vlastněny\: %s messageEnvironmentSamplesRestored = Všechny vybrané vzorky prostředí byly obnoveny messageEnvironmentSampleSaved = Vzorek prostředí uložen messageRestoreNotPossibleAlreadyInEvent = Uživatel události nemůže být obnoven, protože osoba již má jiného účastníka aktivní události v této události @@ -1513,7 +1513,7 @@ messageContactPersonHasNoEmail=Kontaktní osoba nemá zadanou e-mailovou adresu messageEventParticipantPersonHasNoEmail=Uživatel události nemá zadanou e-mailovou adresu messageTravelEntryPersonHasNoEmail=Travel entry person has no email address specified messageNoExternalEmailToCaseSent=No email sent to case person -messageNoExternalEmailToContactSent=No email sent to contact person +messageNoExternalEmailToContactSent=Kontaktní osobě nebyl odeslán žádný e-mail messageNoExternalEmailToEventParticipantSent=No email sent to event participant person messageNoExternalEmailToTravelEntrySent=No email sent to travel entry person messageExternalEmailNoAttachments=Žádné přílohy @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Použijte toto heslo pro otevření dokumentů, které vám byly zaslány e-mailem od SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = Klasifikace případu %s se změnila na %s. notificationCaseInvestigationDone = Vyšetřování případu %s bylo dokončeno. @@ -1702,7 +1704,7 @@ promptTravelEntryDateTo = ... do promptTravelEntryEpiWeekFrom = Místo vstupu od týdne epi... promptTravelEntryEpiWeekTo = ... do epi týdne -promptEnvironmentFreeTextSearch = UUID, External ID, Name, Description +promptEnvironmentFreeTextSearch = UUID, externí ID, jméno, popis promptEnvironmentDateFrom = Datum hlášení od... promptEnvironmentDateTo = ... do promptEnvironmentEpiWeekFrom = Datum hlášení od týdne epi... @@ -1795,15 +1797,15 @@ headingSampleDashboardMap=Mapa stavu vzorků infoHeadingSampleDashboardMap=Vzorky se zobrazují pomocí souřadnic GPS adresy bydliště dané osoby infoHeadingEnvironmentSampleDashboardMap=Vzorky životního prostředí jsou uvedeny pomocí souřadnic GPS těchto vzorků, nebo pokud nejsou k dispozici, jejich přiřazeného prostředí. -headingSpecailCaseAccess = Grant special access +headingSpecailCaseAccess = Povolit speciální přístup headingCreateSpecailCaseAccess = Vytvořit nový speciální přístup headingEditSpecailCaseAccess = Upravit speciální přístup headingConfirmBulkGrantSpecialAccess = Potvrdit ukládání speciálního přístupu -confirmationBulkGrantSpecialAccess = The selected user already has special access to at least one of the selected cases. Saving will overwrite the existing special access. Do you want to continue? +confirmationBulkGrantSpecialAccess = Vybraný uživatel již má zvláštní přístup alespoň k jednomu z vybraných případů. Uložení přepíše stávající speciální přístup. Chcete pokračovat? headingBulkSpecialCaseAccessSomeNotProcessed= Granted special access to some of the selected cases headingBulkSpecialCaseAccessNoneProcessed = Not granted special access to any of the selected cases -messageBulkSpecialCaseAccessAllProcessed = Granted special access to all selected cases -messageCountAccessesNotGrantedDueToError=%s cases have not been processed\: %s +messageBulkSpecialCaseAccessAllProcessed = Poskytnutý zvláštní přístup všem vybraným případům +messageCountAccessesNotGrantedDueToError=%s případy nebyly zpracovány\: %s messageSelfReportSaved=Self report saved headingArchiveSelfReport = Archive self report confirmationArchiveSelfReport = Are you sure you want to archive this self report? This will not remove it from the system or any statistics, but only hide it from the normal self report directory. diff --git a/sormas-api/src/main/resources/strings_de-CH.properties b/sormas-api/src/main/resources/strings_de-CH.properties index 8a1273a1b7f..f82967e3ec5 100644 --- a/sormas-api/src/main/resources/strings_de-CH.properties +++ b/sormas-api/src/main/resources/strings_de-CH.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = Die Falldefinitionskategorie des Falls %s wurde auf %s geändert. notificationCaseInvestigationDone = Die Untersuchung des Falls %s wurde durchgeführt. diff --git a/sormas-api/src/main/resources/strings_de-DE.properties b/sormas-api/src/main/resources/strings_de-DE.properties index af90ea84e53..32e144b7cd8 100644 --- a/sormas-api/src/main/resources/strings_de-DE.properties +++ b/sormas-api/src/main/resources/strings_de-DE.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = Die Falldefinitionskategorie des Falls %s wurde auf %s geändert. notificationCaseInvestigationDone = Die Untersuchung des Falls %s wurde durchgeführt. diff --git a/sormas-api/src/main/resources/strings_en-AF.properties b/sormas-api/src/main/resources/strings_en-AF.properties index e0890cbba2b..5baecfeba9e 100644 --- a/sormas-api/src/main/resources/strings_en-AF.properties +++ b/sormas-api/src/main/resources/strings_en-AF.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_en-GH.properties b/sormas-api/src/main/resources/strings_en-GH.properties index fe568a14f15..986002f5ac3 100644 --- a/sormas-api/src/main/resources/strings_en-GH.properties +++ b/sormas-api/src/main/resources/strings_en-GH.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_en-GM.properties b/sormas-api/src/main/resources/strings_en-GM.properties index d88beac977c..c59cb632d3f 100644 --- a/sormas-api/src/main/resources/strings_en-GM.properties +++ b/sormas-api/src/main/resources/strings_en-GM.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_en-KE.properties b/sormas-api/src/main/resources/strings_en-KE.properties index d50c13913a7..bdec88b7127 100644 --- a/sormas-api/src/main/resources/strings_en-KE.properties +++ b/sormas-api/src/main/resources/strings_en-KE.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_en-LR.properties b/sormas-api/src/main/resources/strings_en-LR.properties index d88beac977c..c59cb632d3f 100644 --- a/sormas-api/src/main/resources/strings_en-LR.properties +++ b/sormas-api/src/main/resources/strings_en-LR.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_en-NG.properties b/sormas-api/src/main/resources/strings_en-NG.properties index 857a9cfdcaf..aa41e9be696 100644 --- a/sormas-api/src/main/resources/strings_en-NG.properties +++ b/sormas-api/src/main/resources/strings_en-NG.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_es-BO.properties b/sormas-api/src/main/resources/strings_es-BO.properties index 6a7d8004651..8004ef2eac6 100644 --- a/sormas-api/src/main/resources/strings_es-BO.properties +++ b/sormas-api/src/main/resources/strings_es-BO.properties @@ -336,6 +336,7 @@ entityTreatments = Tratamientos entityUser = Usuario entityUserRoles = Roles de usuario entityUsers = Usuarios +entityVaccinations = Vacunas entityVisits = Visitas entityWeeklyReports = Informes semanales @@ -1520,6 +1521,8 @@ messageCustomizableEnumValueSaved = Valor de enum personalizable guardado messageExternalEmailAttachmentPassword=Por favor, utilice esta contraseña para abrir los documentos que se le envían por correo electrónico desde SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Adjuntar documentos está deshabilitado porque el cifrado no sería posible. Para cifrar documentos, la persona necesita tener un identificador nacional de salud, o un número de teléfono móvil principal configurado con SMS de envío en este sistema. messagePersonNationalHealthIdInvalid=El identificador nacional de salud introducido no parece ser correcto +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = La clasificación del caso %s ha cambiado a %s. notificationCaseInvestigationDone = Se ha realizado la investigación del caso %s. diff --git a/sormas-api/src/main/resources/strings_es-CU.properties b/sormas-api/src/main/resources/strings_es-CU.properties index 0838ca1ac16..9dfda75b96f 100644 --- a/sormas-api/src/main/resources/strings_es-CU.properties +++ b/sormas-api/src/main/resources/strings_es-CU.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Valor de enumeración personalizable guardad messageExternalEmailAttachmentPassword=Por favor, utilice esta contraseña para abrir los documentos que se le envíen por correo electrónico desde SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=No se pueden adjuntar documentos porque el cifrado no sería posible. Para cifrar documentos, la persona debe tener un carnet de identidad, o un número de teléfono móvil con capacidad de enviar SMS especificado en este sistema. messagePersonNationalHealthIdInvalid=El carnet de identidad introducido no parece ser correcto +messageSyncUsersFromAuthProviderConfigurationError=La sincronización de usuarios del proveedor de autenticación no es posible porque la configuración es incorrecta. Por favor, póngase en contacto con un administrador e infórmele sobre este problema. + # Notifications notificationCaseClassificationChanged = La clasificación del caso %s se cambió a %s. notificationCaseInvestigationDone = La investigación del caso %s fue realizada. diff --git a/sormas-api/src/main/resources/strings_fa-AF.properties b/sormas-api/src/main/resources/strings_fa-AF.properties index cbd2539042f..b3d613f0523 100644 --- a/sormas-api/src/main/resources/strings_fa-AF.properties +++ b/sormas-api/src/main/resources/strings_fa-AF.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_fi-FI.properties b/sormas-api/src/main/resources/strings_fi-FI.properties index 3fd782721ef..3f9b16fe188 100644 --- a/sormas-api/src/main/resources/strings_fi-FI.properties +++ b/sormas-api/src/main/resources/strings_fi-FI.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = Potilaan %s luokitus on muuttunut luokkaan %s. notificationCaseInvestigationDone = Potilaan %s selvittely tehty. diff --git a/sormas-api/src/main/resources/strings_fr-CH.properties b/sormas-api/src/main/resources/strings_fr-CH.properties index fe019d88739..b25c8da1d63 100644 --- a/sormas-api/src/main/resources/strings_fr-CH.properties +++ b/sormas-api/src/main/resources/strings_fr-CH.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = La classification du cas %s a changé en %s. notificationCaseInvestigationDone = L'enquête du cas %s a été effectuée. diff --git a/sormas-api/src/main/resources/strings_fr-FR.properties b/sormas-api/src/main/resources/strings_fr-FR.properties index f43f92807f8..b08c70f4bf1 100644 --- a/sormas-api/src/main/resources/strings_fr-FR.properties +++ b/sormas-api/src/main/resources/strings_fr-FR.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Valeur de l'énumération personnalisable en messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=La synchronisation des utilisateurs du fournisseur d'authentification n'est pas possible car la configuration est incorrecte. Veuillez contacter un administrateur et l'informer de ce problème. + # Notifications notificationCaseClassificationChanged = La classification du cas %s a changé en %s. notificationCaseInvestigationDone = L'enquête du cas %s a été effectuée. diff --git a/sormas-api/src/main/resources/strings_fr-TN.properties b/sormas-api/src/main/resources/strings_fr-TN.properties index 1ade749641f..81fe9f8e416 100644 --- a/sormas-api/src/main/resources/strings_fr-TN.properties +++ b/sormas-api/src/main/resources/strings_fr-TN.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = La classification du cas %s a changé en %s. notificationCaseInvestigationDone = L'enquête du cas %s a été effectuée. diff --git a/sormas-api/src/main/resources/strings_it-CH.properties b/sormas-api/src/main/resources/strings_it-CH.properties index 253fec8db36..c6678fe8001 100644 --- a/sormas-api/src/main/resources/strings_it-CH.properties +++ b/sormas-api/src/main/resources/strings_it-CH.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = La classificazione del caso %s è cambiata in %s. notificationCaseInvestigationDone = La ricerca sul caso %s è stata effettuata. diff --git a/sormas-api/src/main/resources/strings_it-IT.properties b/sormas-api/src/main/resources/strings_it-IT.properties index 24d00ba6ca8..0f3a66e1ccf 100644 --- a/sormas-api/src/main/resources/strings_it-IT.properties +++ b/sormas-api/src/main/resources/strings_it-IT.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = La classificazione del caso %s è cambiata in %s. notificationCaseInvestigationDone = La ricerca sul caso %s è stata effettuata. diff --git a/sormas-api/src/main/resources/strings_pl-PL.properties b/sormas-api/src/main/resources/strings_pl-PL.properties index 8231ec015ef..701d95b5400 100644 --- a/sormas-api/src/main/resources/strings_pl-PL.properties +++ b/sormas-api/src/main/resources/strings_pl-PL.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_ps-AF.properties b/sormas-api/src/main/resources/strings_ps-AF.properties index e6b621455d6..c5d1e61f9ad 100644 --- a/sormas-api/src/main/resources/strings_ps-AF.properties +++ b/sormas-api/src/main/resources/strings_ps-AF.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_pt-CV.properties b/sormas-api/src/main/resources/strings_pt-CV.properties index 85e67206697..6a331737e15 100644 --- a/sormas-api/src/main/resources/strings_pt-CV.properties +++ b/sormas-api/src/main/resources/strings_pt-CV.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_ru-RU.properties b/sormas-api/src/main/resources/strings_ru-RU.properties index a9a7e5b592b..480e0f046f3 100644 --- a/sormas-api/src/main/resources/strings_ru-RU.properties +++ b/sormas-api/src/main/resources/strings_ru-RU.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-api/src/main/resources/strings_ur-PK.properties b/sormas-api/src/main/resources/strings_ur-PK.properties index 8e8a2aff70d..06082ebb60b 100644 --- a/sormas-api/src/main/resources/strings_ur-PK.properties +++ b/sormas-api/src/main/resources/strings_ur-PK.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = حسب ضرورت اینوم ویلیو مح messageExternalEmailAttachmentPassword=براہ کرم SORMAS سے ای میل کے ذریعے آپ کو بھیجے گئے دستاویزات کو کھولنے کے لیے اس پاس ورڈ کا استعمال کریں\: %s messageExternalEmailAttachmentNotAvailableInfo=دستاویزات کو منسلک کرنا غیر فعال ہے کیونکہ انکرپشن ممکن نہیں ہوگی۔ دستاویزات کو انکرپٹ کرنے کے لیے، اس شخص کے پاس یا تو قومی صحت کی شناخت کی ضرورت ہوتی ہے، یا اس سسٹم پر ایس ایم ایس بھیجنے کے ساتھ ایک بنیادی موبائل فون نمبر سیٹ کرنا ہوتا ہے۔ messagePersonNationalHealthIdInvalid=درج کردہ قومی صحت کی شناخت درست معلوم نہیں ہوتی +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = کیس %s کی درجہ بندی %s میں تبدیل ہو گئی ہے۔ notificationCaseInvestigationDone = کیس %s کی تفتیش ہو چکی ہے۔ diff --git a/sormas-api/src/main/resources/strings_zh-CN.properties b/sormas-api/src/main/resources/strings_zh-CN.properties index ddbb377a97a..fbf0c742236 100644 --- a/sormas-api/src/main/resources/strings_zh-CN.properties +++ b/sormas-api/src/main/resources/strings_zh-CN.properties @@ -1521,6 +1521,8 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS\: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct +messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. + # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. notificationCaseInvestigationDone = The investigation of case %s has been done. diff --git a/sormas-app/app/src/main/res/values-cs-rCZ/strings.xml b/sormas-app/app/src/main/res/values-cs-rCZ/strings.xml index 99a1fd7875c..da9582390b8 100644 --- a/sormas-app/app/src/main/res/values-cs-rCZ/strings.xml +++ b/sormas-app/app/src/main/res/values-cs-rCZ/strings.xml @@ -170,17 +170,17 @@ Epi týden Zpráva E-mail: - Environment Information + Informace o prostředí Vzorky prostředí Environment Tasks Environment Sample Information Popis Informace o události Poloha - Event Participant + Účastník události Immunizace účastníků akce Vakcíny účastníka události - Event Participants + Účastníci události Datum události Úkoly událostí Provedl @@ -200,7 +200,7 @@ Negativní Nový případ Nový kontakt - New Event Participant + Nový účastník události Nový vzorek Nový test patogenu Nový úkol @@ -533,7 +533,7 @@ Imunizace Nastavení Úkoly - Aggregate + Agregát Již se ukládá.. Upozornění: Pro výše uvedená kritéria byly nalezeny duplicitní zprávy. Nemoci označené červenou již mají hlášení. Věkové skupiny označené jako \"Vypršelo\" jsou založeny na existujících datech pro věkovou skupinu, která se již nepoužívá. diff --git a/sormas-app/app/src/main/res/values-fr-rTN/strings.xml b/sormas-app/app/src/main/res/values-fr-rTN/strings.xml index d78ffe5eb1f..6a5716aaa2d 100644 --- a/sormas-app/app/src/main/res/values-fr-rTN/strings.xml +++ b/sormas-app/app/src/main/res/values-fr-rTN/strings.xml @@ -336,7 +336,7 @@ Confirmer le code PIN Confirmation Modifier le contact - Contact information + Coordonnées du contact Nouveau Contact Lire le contact Résumé du contact @@ -353,7 +353,7 @@ Oups, une erreur s\'est produite. Modifier l\'événement Nouvel évènement - Event Participant + Participant à l\'événement New Event Participant Lire l\'événement Résumé de l\'événement From 24b5234ea4feb3fe711efc8673a551babf84975b Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Sat, 17 Feb 2024 18:33:34 +0100 Subject: [PATCH 04/56] #12634 - Started development of AEFI module - #12728 - Added a new AEFI entity, related entities, dto's and enumerations - #12772 - Added AEFI user rights - #12767 - Added an AEFI directory for the web app - #12769 - Added a create/edit form for AEFI in the web app - #12976 - Added a basic AEFI dashboard --- .../de/symeda/sormas/api/FacadeProvider.java | 10 + .../AdverseEventState.java | 33 + .../AdverseEventsDto.java | 186 ++++++ .../AefiAgeGroup.java | 35 + .../AefiCriteria.java | 232 +++++++ .../AefiDashboardFilterDateType.java | 32 + .../AefiDateType.java | 33 + .../AefiDto.java | 614 ++++++++++++++++++ .../AefiFacade.java | 31 + .../AefiHelper.java | 87 +++ .../AefiIndexDto.java | 277 ++++++++ .../AefiListCriteria.java | 48 ++ .../AefiListEntryDto.java | 97 +++ .../AefiOutcome.java | 36 + .../AefiReferenceDto.java | 45 ++ .../AefiType.java | 41 ++ .../SeizureType.java | 32 + .../SeriousAefiReason.java | 36 + .../api/common/DeletableEntityType.java | 1 + .../api/dashboard/AefiDashboardCriteria.java | 49 ++ .../AefiChartData.java | 58 ++ .../AefiChartSeries.java | 61 ++ .../AefiDashboardFacade.java | 44 ++ .../MapAefiDto.java | 78 +++ .../sormas/api/feature/FeatureType.java | 1 + .../de/symeda/sormas/api/i18n/Captions.java | 106 +++ .../symeda/sormas/api/i18n/Descriptions.java | 3 + .../de/symeda/sormas/api/i18n/Strings.java | 33 + .../symeda/sormas/api/i18n/Validations.java | 3 + .../de/symeda/sormas/api/user/UserRight.java | 13 + .../sormas/api/user/UserRightGroup.java | 1 + .../src/main/resources/captions.properties | 112 ++++ .../main/resources/descriptions.properties | 6 +- sormas-api/src/main/resources/enum.properties | 52 ++ .../src/main/resources/strings.properties | 42 +- .../src/main/resources/validations.properties | 3 + .../AdverseEventsMapper.java | 82 +++ .../AefiFacadeEjb.java | 455 +++++++++++++ .../AefiJurisdictionPredicateValidator.java | 108 +++ .../AefiQueryContext.java | 45 ++ .../AefiService.java | 521 +++++++++++++++ .../entity/AdverseEvents.java | 179 +++++ .../entity/Aefi.java | 606 +++++++++++++++++ .../entity/AefiJoins.java | 162 +++++ .../AefiIndexDtoResultTransformer.java | 86 +++ .../AefiListEntryDtoResultTransformer.java | 60 ++ .../AefiDashboardFacadeEjb.java | 76 +++ .../AefiDashboardService.java | 515 +++++++++++++++ .../immunization/ImmunizationService.java | 21 + .../main/resources/META-INF/persistence.xml | 3 + .../symeda/sormas/ui/ControllerProvider.java | 7 + .../java/de/symeda/sormas/ui/MainScreen.java | 24 +- .../AbstractAefiView.java | 115 ++++ .../AefiController.java | 195 ++++++ .../AefiDataView.java | 128 ++++ .../AefiView.java | 208 ++++++ .../aefilink/AefiList.java | 89 +++ .../aefilink/AefiListComponent.java | 56 ++ .../aefilink/AefiListEntry.java | 79 +++ .../components/directory/AefiDataLayout.java | 42 ++ .../components/directory/AefiFilterForm.java | 420 ++++++++++++ .../directory/AefiFilterFormLayout.java | 53 ++ .../components/directory/AefiGrid.java | 132 ++++ ...imarySuspectVaccinationSelectionField.java | 146 +++++ .../vaccines/AefiVaccinationsField.java | 200 ++++++ .../vaccines/AefiVaccinationsField_2.java | 198 ++++++ .../components/form/AdverseEventsForm.java | 149 +++++ .../components/form/AefiDataForm.java | 303 +++++++++ .../information/AefiImmunizationInfo.java | 109 ++++ .../information/AefiPersonInfo.java | 142 ++++ .../ui/dashboard/AbstractDashboardView.java | 8 + .../ui/dashboard/DashboardController.java | 5 + .../sormas/ui/dashboard/DashboardType.java | 3 +- .../AefiDashboardDataProvider.java | 81 +++ .../AefiDashboardFilterLayout.java | 112 ++++ .../AefiDashboardView.java | 230 +++++++ .../AefiEpiCurveComponent.java | 97 +++ .../components/AefiByVaccineDoseChart.java | 158 +++++ .../components/AefiCountTilesComponent.java | 204 ++++++ .../components/AefiDashboardMapComponent.java | 161 +++++ .../AefiReactionsByGenderChart.java | 166 +++++ .../AefiTypeStatisticsComponent.java | 100 +++ .../AefiTypeStatisticsGroupComponent.java | 64 ++ .../ui/immunization/ImmunizationDataView.java | 20 +- .../sormas/ui/utils/ArchiveHandlers.java | 6 + .../sormas/ui/utils/ArchiveMessages.java | 17 + .../de/symeda/sormas/ui/utils/CssStyles.java | 17 + .../utils/SormasFieldGroupFieldFactory.java | 18 +- .../webapp/VAADIN/themes/sormas/global.scss | 67 ++ 89 files changed, 9812 insertions(+), 7 deletions(-) create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventState.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventsDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiAgeGroup.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDashboardFilterDateType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDateType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListCriteria.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListEntryDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiOutcome.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeizureType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiReason.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/dashboard/AefiDashboardCriteria.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartSeries.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/MapAefiDto.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AdverseEventsMapper.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiQueryContext.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AdverseEvents.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiListEntryDtoResultTransformer.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListEntry.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterFormLayout.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiPrimarySuspectVaccinationSelectionField.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiImmunizationInfo.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiEpiCurveComponent.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiByVaccineDoseChart.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiDashboardMapComponent.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsComponent.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java b/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java index e731a9dd8e0..f7570d1c269 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java @@ -19,6 +19,7 @@ import javax.naming.NamingException; import de.symeda.sormas.api.action.ActionFacade; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiFacade; import de.symeda.sormas.api.audit.AuditLoggerFacade; import de.symeda.sormas.api.bagexport.BAGExportFacade; import de.symeda.sormas.api.campaign.CampaignFacade; @@ -38,6 +39,7 @@ import de.symeda.sormas.api.contact.ContactFacade; import de.symeda.sormas.api.customizableenum.CustomizableEnumFacade; import de.symeda.sormas.api.dashboard.DashboardFacade; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiDashboardFacade; import de.symeda.sormas.api.dashboard.sample.SampleDashboardFacade; import de.symeda.sormas.api.deletionconfiguration.DeletionConfigurationFacade; import de.symeda.sormas.api.disease.DiseaseConfigurationFacade; @@ -148,6 +150,10 @@ public static ImmunizationFacade getImmunizationFacade() { return get().lookupEjbRemote(ImmunizationFacade.class); } + public static AefiFacade getAefiFacade() { + return get().lookupEjbRemote(AefiFacade.class); + } + public static VaccinationFacade getVaccinationFacade() { return get().lookupEjbRemote(VaccinationFacade.class); } @@ -320,6 +326,10 @@ public static SampleDashboardFacade getSampleDashboardFacade() { return get().lookupEjbRemote(SampleDashboardFacade.class); } + public static AefiDashboardFacade getAefiDashboardFacade() { + return get().lookupEjbRemote(AefiDashboardFacade.class); + } + public static DiseaseConfigurationFacade getDiseaseConfigurationFacade() { return get().lookupEjbRemote(DiseaseConfigurationFacade.class); } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventState.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventState.java new file mode 100644 index 00000000000..73d64393393 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventState.java @@ -0,0 +1,33 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AdverseEventState { + + YES, + NO, + UNKNOWN; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventsDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventsDto.java new file mode 100644 index 00000000000..94551919c89 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AdverseEventsDto.java @@ -0,0 +1,186 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.DependingOnFeatureType; +import de.symeda.sormas.api.utils.Order; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableDto; + +@DependingOnFeatureType(featureType = { + FeatureType.IMMUNIZATION_MANAGEMENT, + FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT }) +public class AdverseEventsDto extends PseudonymizableDto { + + private static final long serialVersionUID = 8081578717541472008L; + + public static final String I18N_PREFIX = "AdverseEvents"; + + public static final String SEVERE_LOCAL_REACTION = "severeLocalReaction"; + public static final String SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS = "severeLocalReactionMoreThanThreeDays"; + public static final String SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT = "severeLocalReactionBeyondNearestJoint"; + public static final String SEIZURES = "seizures"; + public static final String SEIZURE_TYPE = "seizureType"; + public static final String ABSCESS = "abscess"; + public static final String SEPSIS = "sepsis"; + public static final String ENCEPHALOPATHY = "encephalopathy"; + public static final String TOXIC_SHOCK_SYNDROME = "toxicShockSyndrome"; + public static final String THROMBOCYTOPENIA = "thrombocytopenia"; + public static final String ANAPHYLAXIS = "anaphylaxis"; + public static final String FEVERISH_FEELING = "feverishFeeling"; + public static final String OTHER_ADVERSE_EVENT_DETAILS = "otherAdverseEventDetails"; + + private AdverseEventState severeLocalReaction; + private boolean severeLocalReactionMoreThanThreeDays; + private boolean severeLocalReactionBeyondNearestJoint; + private AdverseEventState seizures; + private SeizureType seizureType; + private AdverseEventState abscess; + private AdverseEventState sepsis; + private AdverseEventState encephalopathy; + private AdverseEventState toxicShockSyndrome; + private AdverseEventState thrombocytopenia; + private AdverseEventState anaphylaxis; + private AdverseEventState feverishFeeling; + private String otherAdverseEventDetails; + + public static AdverseEventsDto build() { + AdverseEventsDto adverseEvents = new AdverseEventsDto(); + adverseEvents.setUuid(DataHelper.createUuid()); + return adverseEvents; + } + + @Order(1) + public AdverseEventState getSevereLocalReaction() { + return severeLocalReaction; + } + + public void setSevereLocalReaction(AdverseEventState severeLocalReaction) { + this.severeLocalReaction = severeLocalReaction; + } + + @Order(2) + public boolean isSevereLocalReactionMoreThanThreeDays() { + return severeLocalReactionMoreThanThreeDays; + } + + public void setSevereLocalReactionMoreThanThreeDays(boolean severeLocalReactionMoreThanThreeDays) { + this.severeLocalReactionMoreThanThreeDays = severeLocalReactionMoreThanThreeDays; + } + + @Order(3) + public boolean isSevereLocalReactionBeyondNearestJoint() { + return severeLocalReactionBeyondNearestJoint; + } + + public void setSevereLocalReactionBeyondNearestJoint(boolean severeLocalReactionBeyondNearestJoint) { + this.severeLocalReactionBeyondNearestJoint = severeLocalReactionBeyondNearestJoint; + } + + @Order(4) + public AdverseEventState getSeizures() { + return seizures; + } + + public void setSeizures(AdverseEventState seizures) { + this.seizures = seizures; + } + + @Order(5) + public SeizureType getSeizureType() { + return seizureType; + } + + public void setSeizureType(SeizureType seizureType) { + this.seizureType = seizureType; + } + + @Order(6) + public AdverseEventState getAbscess() { + return abscess; + } + + public void setAbscess(AdverseEventState abscess) { + this.abscess = abscess; + } + + @Order(7) + public AdverseEventState getSepsis() { + return sepsis; + } + + public void setSepsis(AdverseEventState sepsis) { + this.sepsis = sepsis; + } + + @Order(7) + public AdverseEventState getEncephalopathy() { + return encephalopathy; + } + + public void setEncephalopathy(AdverseEventState encephalopathy) { + this.encephalopathy = encephalopathy; + } + + @Order(8) + public AdverseEventState getToxicShockSyndrome() { + return toxicShockSyndrome; + } + + public void setToxicShockSyndrome(AdverseEventState toxicShockSyndrome) { + this.toxicShockSyndrome = toxicShockSyndrome; + } + + @Order(9) + public AdverseEventState getThrombocytopenia() { + return thrombocytopenia; + } + + public void setThrombocytopenia(AdverseEventState thrombocytopenia) { + this.thrombocytopenia = thrombocytopenia; + } + + @Order(10) + public AdverseEventState getAnaphylaxis() { + return anaphylaxis; + } + + public void setAnaphylaxis(AdverseEventState anaphylaxis) { + this.anaphylaxis = anaphylaxis; + } + + @Order(11) + public AdverseEventState getFeverishFeeling() { + return feverishFeeling; + } + + public void setFeverishFeeling(AdverseEventState feverishFeeling) { + this.feverishFeeling = feverishFeeling; + } + + @Order(12) + public String getOtherAdverseEventDetails() { + return otherAdverseEventDetails; + } + + public void setOtherAdverseEventDetails(String otherAdverseEventDetails) { + this.otherAdverseEventDetails = otherAdverseEventDetails; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiAgeGroup.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiAgeGroup.java new file mode 100644 index 00000000000..4e57132ea70 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiAgeGroup.java @@ -0,0 +1,35 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiAgeGroup { + + ZERO_TO_ONE, + ONE_TO_FIVE, + FIVE_TO_EIGHTEEN, + EIGHTEEN_TO_SIXTY, + SIXY_AND_ABOVE; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java new file mode 100644 index 00000000000..742df524387 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java @@ -0,0 +1,232 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.EntityRelevanceStatus; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.caze.VaccineManufacturer; +import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityType; +import de.symeda.sormas.api.infrastructure.facility.FacilityTypeGroup; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.utils.DateFilterOption; +import de.symeda.sormas.api.utils.criteria.BaseCriteria; + +public class AefiCriteria extends BaseCriteria implements Serializable { + + public static final String I18N_PREFIX = "AefiCriteria"; + + public static final String DISEASE = "disease"; + public static final String NAME_ADDRESS_PHONE_EMAIL_LIKE = "nameAddressPhoneEmailLike"; + public static final String AEFI_TYPE = "aefiType"; + public static final String OUTCOME = "outcome"; + public static final String VACCINE_NAME = "vaccineName"; + public static final String VACCINE_MANUFACTURER = "vaccineManufacturer"; + public static final String REGION = "region"; + public static final String DISTRICT = "district"; + public static final String COMMUNITY = "community"; + public static final String FACILITY_TYPE_GROUP = "facilityTypeGroup"; + public static final String FACILITY_TYPE = "facilityType"; + public static final String HEALTH_FACILITY = "healthFacility"; + public static final String DATE_FILTER_OPTION = "dateFilterOption"; + public static final String AEFI_DATE_TYPE = "aefiDateType"; + public static final String FROM_DATE = "fromDate"; + public static final String TO_DATE = "toDate"; + public static final String RELEVANCE_STATUS = "relevanceStatus"; + + private Disease disease; + private String nameAddressPhoneEmailLike; + private AefiType aefiType; + private AefiOutcome outcome; + private Vaccine vaccineName; + private VaccineManufacturer vaccineManufacturer; + private RegionReferenceDto region; + private DistrictReferenceDto district; + private CommunityReferenceDto community; + private FacilityTypeGroup facilityTypeGroup; + private FacilityType facilityType; + private FacilityReferenceDto healthFacility; + private DateFilterOption dateFilterOption = DateFilterOption.DATE; + private AefiDateType aefiDateType; + private Date fromDate; + private Date toDate; + private EntityRelevanceStatus relevanceStatus; + + public Disease getDisease() { + return disease; + } + + public void setDisease(Disease disease) { + this.disease = disease; + } + + public String getNameAddressPhoneEmailLike() { + return nameAddressPhoneEmailLike; + } + + public void setNameAddressPhoneEmailLike(String nameAddressPhoneEmailLike) { + this.nameAddressPhoneEmailLike = nameAddressPhoneEmailLike; + } + + public AefiType getAefiType() { + return aefiType; + } + + public void setAefiType(AefiType aefiType) { + this.aefiType = aefiType; + } + + public AefiOutcome getOutcome() { + return outcome; + } + + public void setOutcome(AefiOutcome outcome) { + this.outcome = outcome; + } + + public Vaccine getVaccineName() { + return vaccineName; + } + + public void setVaccineName(Vaccine vaccineName) { + this.vaccineName = vaccineName; + } + + public VaccineManufacturer getVaccineManufacturer() { + return vaccineManufacturer; + } + + public void setVaccineManufacturer(VaccineManufacturer vaccineManufacturer) { + this.vaccineManufacturer = vaccineManufacturer; + } + + public RegionReferenceDto getRegion() { + return region; + } + + public void setRegion(RegionReferenceDto region) { + this.region = region; + } + + public DistrictReferenceDto getDistrict() { + return district; + } + + public void setDistrict(DistrictReferenceDto district) { + this.district = district; + } + + public CommunityReferenceDto getCommunity() { + return community; + } + + public void setCommunity(CommunityReferenceDto community) { + this.community = community; + } + + public FacilityTypeGroup getFacilityTypeGroup() { + return facilityTypeGroup; + } + + public void setFacilityTypeGroup(FacilityTypeGroup facilityTypeGroup) { + this.facilityTypeGroup = facilityTypeGroup; + } + + public FacilityType getFacilityType() { + return facilityType; + } + + public void setFacilityType(FacilityType facilityType) { + this.facilityType = facilityType; + } + + public FacilityReferenceDto getHealthFacility() { + return healthFacility; + } + + public void setHealthFacility(FacilityReferenceDto healthFacility) { + this.healthFacility = healthFacility; + } + + public DateFilterOption getDateFilterOption() { + return dateFilterOption; + } + + public void setDateFilterOption(DateFilterOption dateFilterOption) { + this.dateFilterOption = dateFilterOption; + } + + public AefiDateType getAefiDateType() { + return aefiDateType; + } + + public void setAefiDateType(AefiDateType aefiDateType) { + this.aefiDateType = aefiDateType; + } + + public Date getFromDate() { + return fromDate; + } + + public void setFromDate(Date fromDate) { + this.fromDate = fromDate; + } + + public Date getToDate() { + return toDate; + } + + public void setToDate(Date toDate) { + this.toDate = toDate; + } + + public EntityRelevanceStatus getRelevanceStatus() { + return relevanceStatus; + } + + public void setRelevanceStatus(EntityRelevanceStatus relevanceStatus) { + this.relevanceStatus = relevanceStatus; + } + + public AefiCriteria disease(Disease disease) { + this.disease = disease; + return this; + } + + public AefiCriteria region(RegionReferenceDto region) { + this.region = region; + return this; + } + + public AefiCriteria district(DistrictReferenceDto district) { + this.district = district; + return this; + } + + public AefiCriteria aefiType(AefiType aefiType) { + this.aefiType = aefiType; + return this; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDashboardFilterDateType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDashboardFilterDateType.java new file mode 100644 index 00000000000..984641a06e5 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDashboardFilterDateType.java @@ -0,0 +1,32 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiDashboardFilterDateType { + + REPORT_DATE, + START_DATE; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDateType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDateType.java new file mode 100644 index 00000000000..9eef385ce9c --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDateType.java @@ -0,0 +1,33 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiDateType { + + REPORT_DATE, + START_DATE, + VACCINATION_DATE; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java new file mode 100644 index 00000000000..7d32221109d --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java @@ -0,0 +1,614 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; + +import de.symeda.sormas.api.caze.Trimester; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.immunization.ImmunizationReferenceDto; +import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; +import de.symeda.sormas.api.infrastructure.country.CountryReferenceDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityReferenceDto; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.location.LocationDto; +import de.symeda.sormas.api.person.PersonReferenceDto; +import de.symeda.sormas.api.user.UserReferenceDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.DependingOnFeatureType; +import de.symeda.sormas.api.utils.FieldConstraints; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableDto; +import de.symeda.sormas.api.vaccination.VaccinationDto; + +@DependingOnFeatureType(featureType = { + FeatureType.IMMUNIZATION_MANAGEMENT, + FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT }) +public class AefiDto extends PseudonymizableDto { + + private static final long serialVersionUID = 5023410664970514090L; + + public static final long APPROXIMATE_JSON_SIZE_IN_BYTES = 25455; + + public static final String I18N_PREFIX = "Aefi"; + + public static final String IMMUNIZATION = "immunization"; + public static final String ADDRESS = "address"; + public static final String VACCINATIONS = "vaccinations"; + public static final String PRIMARY_SUSPECT_VACCINE = "primarySuspectVaccine"; + public static final String ADVERSE_EVENTS = "adverseEvents"; + public static final String PERSON = "person"; + public static final String REPORT_DATE = "reportDate"; + public static final String REPORTING_USER = "reportingUser"; + public static final String EXTERNAL_ID = "externalId"; + public static final String RESPONSIBLE_REGION = "responsibleRegion"; + public static final String RESPONSIBLE_DISTRICT = "responsibleDistrict"; + public static final String RESPONSIBLE_COMMUNITY = "responsibleCommunity"; + public static final String COUNTRY = "country"; + public static final String REPORTING_ID_NUMBER = "reportingIdNumber"; + public static final String PHONE_NUMBER = "phoneNumber"; + public static final String PREGNANT = "pregnant"; + public static final String TRIMESTER = "trimester"; + public static final String LACTATING = "lactating"; + public static final String ONSET_AGE_YEARS = "onsetAgeYears"; + public static final String ONSET_AGE_MONTHS = "onsetAgeMonths"; + public static final String ONSET_AGE_DAYS = "onsetAgeDays"; + public static final String AGE_GROUP = "ageGroup"; + public static final String HEALTH_FACILITY = "healthFacility"; + public static final String HEALTH_FACILITY_DETAILS = "healthFacilityDetails"; + public static final String REPORTER_NAME = "reporterName"; + public static final String REPORTER_INSTITUTION = "reporterInstitution"; + public static final String REPORTER_DESIGNATION = "reporterDesignation"; + public static final String REPORTER_DEPARTMENT = "reporterDepartment"; + public static final String REPORTER_ADDRESS = "reporterAddress"; + public static final String REPORTER_PHONE = "reporterPhone"; + public static final String REPORTER_EMAIL = "reporterEmail"; + public static final String TODAYS_DATE = "todaysDate"; + public static final String START_DATE_TIME = "startDateTime"; + public static final String AEFI_DESCRIPTION = "aefiDescription"; + public static final String SERIOUS = "serious"; + public static final String SERIOUS_REASON = "seriousReason"; + public static final String SERIOUS_REASON_DETAILS = "seriousReasonDetails"; + public static final String OUTCOME = "outcome"; + public static final String DEATH_DATE = "deathDate"; + public static final String AUTOPSY_DONE = "autopsyDone"; + public static final String PAST_MEDICAL_HISTORY = "pastMedicalHistory"; + public static final String INVESTIGATION_NEEDED = "investigationNeeded"; + public static final String INVESTIGATION_PLANNED_DATE = "investigationPlannedDate"; + public static final String RECEIVED_AT_NATIONAL_LEVEL_DATE = "receivedAtNationalLevelDate"; + public static final String WORLD_WIDE_ID = "worldwideId"; + public static final String NATIONAL_LEVEL_COMMENT = "nationalLevelComment"; + public static final String DELETION_REASON = "deletionReason"; + public static final String OTHER_DELETION_REASON = "otherDeletionReason"; + + @NotNull(message = Validations.validImmunization) + private ImmunizationReferenceDto immunization; + private PersonReferenceDto person; + private LocationDto address; + @NotEmpty(message = Validations.aefiWithoutSuspectVaccine) + private List vaccinations = new ArrayList<>(); + @NotNull(message = Validations.aefiWithoutPrimarySuspectVaccine) + private VaccinationDto primarySuspectVaccine; + @NotNull(message = Validations.aefiWithoutAdverseEvents) + private AdverseEventsDto adverseEvents; + @NotNull(message = Validations.validReportDateTime) + private Date reportDate; + private UserReferenceDto reportingUser; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String externalId; + private RegionReferenceDto responsibleRegion; + private DistrictReferenceDto responsibleDistrict; + private CommunityReferenceDto responsibleCommunity; + private CountryReferenceDto country; + @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) + private String reportingIdNumber; + private String phoneNumber; + private YesNoUnknown pregnant; + private Trimester trimester; + private YesNoUnknown lactating; + private Integer onsetAgeYears; + private Integer onsetAgeMonths; + private Integer onsetAgeDays; + private AefiAgeGroup ageGroup; + private FacilityReferenceDto healthFacility; + @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) + private String healthFacilityDetails; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String reporterName; + private FacilityReferenceDto reporterInstitution; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String reporterDesignation; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String reporterDepartment; + private LocationDto reporterAddress; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String reporterPhone; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String reporterEmail; + private Date todaysDate; + private Date startDateTime; + @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) + private String aefiDescription; + private YesNoUnknown serious; + private SeriousAefiReason seriousReason; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String seriousReasonDetails; + private AefiOutcome outcome; + private Date deathDate; + private YesNoUnknown autopsyDone; + @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) + private String pastMedicalHistory; + private YesNoUnknown investigationNeeded; + private Date investigationPlannedDate; + private Date receivedAtNationalLevelDate; + @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) + private String worldwideId; + @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) + private String nationalLevelComment; + private boolean archived; + private boolean deleted; + private DeletionReason deletionReason; + @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) + private String otherDeletionReason; + + public static AefiDto build(UserReferenceDto user) { + + final AefiDto aefiDto = new AefiDto(); + aefiDto.setUuid(DataHelper.createUuid()); + aefiDto.setReportingUser(user); + aefiDto.setReportDate(new Date()); + aefiDto.setAdverseEvents(AdverseEventsDto.build()); + + return aefiDto; + } + + public static AefiDto build(ImmunizationReferenceDto immunization) { + + final AefiDto aefiDto = new AefiDto(); + aefiDto.setUuid(DataHelper.createUuid()); + aefiDto.setImmunization(immunization); + aefiDto.setReportDate(new Date()); + aefiDto.setAdverseEvents(AdverseEventsDto.build()); + + return aefiDto; + } + + public static AefiDto build(AefiReferenceDto aefiReferenceDto) { + + final AefiDto aefiDto = new AefiDto(); + aefiDto.setUuid(aefiReferenceDto.getUuid()); + aefiDto.setReportDate(new Date()); + aefiDto.setAdverseEvents(AdverseEventsDto.build()); + + return aefiDto; + } + + public AefiReferenceDto toReference() { + return new AefiReferenceDto(getUuid(), getPerson().getCaption(), getExternalId()); + } + + public ImmunizationReferenceDto getImmunization() { + return immunization; + } + + public void setImmunization(ImmunizationReferenceDto immunization) { + this.immunization = immunization; + } + + public PersonReferenceDto getPerson() { + return person; + } + + public void setPerson(PersonReferenceDto person) { + this.person = person; + } + + public LocationDto getAddress() { + return address; + } + + public void setAddress(LocationDto address) { + this.address = address; + } + + public List getVaccinations() { + return vaccinations; + } + + public void setVaccinations(List vaccinations) { + this.vaccinations = vaccinations; + } + + public AdverseEventsDto getAdverseEvents() { + return adverseEvents; + } + + public VaccinationDto getPrimarySuspectVaccine() { + return primarySuspectVaccine; + } + + public void setPrimarySuspectVaccine(VaccinationDto primarySuspectVaccine) { + this.primarySuspectVaccine = primarySuspectVaccine; + } + + public void setAdverseEvents(AdverseEventsDto adverseEvents) { + this.adverseEvents = adverseEvents; + } + + public Date getReportDate() { + return reportDate; + } + + public void setReportDate(Date reportDate) { + this.reportDate = reportDate; + } + + public UserReferenceDto getReportingUser() { + return reportingUser; + } + + public void setReportingUser(UserReferenceDto reportingUser) { + this.reportingUser = reportingUser; + } + + public String getExternalId() { + return externalId; + } + + public void setExternalId(String externalId) { + this.externalId = externalId; + } + + public RegionReferenceDto getResponsibleRegion() { + return responsibleRegion; + } + + public void setResponsibleRegion(RegionReferenceDto responsibleRegion) { + this.responsibleRegion = responsibleRegion; + } + + public DistrictReferenceDto getResponsibleDistrict() { + return responsibleDistrict; + } + + public void setResponsibleDistrict(DistrictReferenceDto responsibleDistrict) { + this.responsibleDistrict = responsibleDistrict; + } + + public CommunityReferenceDto getResponsibleCommunity() { + return responsibleCommunity; + } + + public void setResponsibleCommunity(CommunityReferenceDto responsibleCommunity) { + this.responsibleCommunity = responsibleCommunity; + } + + public CountryReferenceDto getCountry() { + return country; + } + + public void setCountry(CountryReferenceDto country) { + this.country = country; + } + + public String getReportingIdNumber() { + return reportingIdNumber; + } + + public void setReportingIdNumber(String reportingIdNumber) { + this.reportingIdNumber = reportingIdNumber; + } + + public String getPhoneNumber() { + return phoneNumber; + } + + public void setPhoneNumber(String phoneNumber) { + this.phoneNumber = phoneNumber; + } + + public YesNoUnknown getPregnant() { + return pregnant; + } + + public void setPregnant(YesNoUnknown pregnant) { + this.pregnant = pregnant; + } + + public Trimester getTrimester() { + return trimester; + } + + public void setTrimester(Trimester trimester) { + this.trimester = trimester; + } + + public YesNoUnknown getLactating() { + return lactating; + } + + public void setLactating(YesNoUnknown lactating) { + this.lactating = lactating; + } + + public Integer getOnsetAgeYears() { + return onsetAgeYears; + } + + public void setOnsetAgeYears(Integer onsetAgeYears) { + this.onsetAgeYears = onsetAgeYears; + } + + public Integer getOnsetAgeMonths() { + return onsetAgeMonths; + } + + public void setOnsetAgeMonths(Integer onsetAgeMonths) { + this.onsetAgeMonths = onsetAgeMonths; + } + + public Integer getOnsetAgeDays() { + return onsetAgeDays; + } + + public void setOnsetAgeDays(Integer onsetAgeDays) { + this.onsetAgeDays = onsetAgeDays; + } + + public AefiAgeGroup getAgeGroup() { + return ageGroup; + } + + public void setAgeGroup(AefiAgeGroup ageGroup) { + this.ageGroup = ageGroup; + } + + public FacilityReferenceDto getHealthFacility() { + return healthFacility; + } + + public void setHealthFacility(FacilityReferenceDto healthFacility) { + this.healthFacility = healthFacility; + } + + public String getHealthFacilityDetails() { + return healthFacilityDetails; + } + + public void setHealthFacilityDetails(String healthFacilityDetails) { + this.healthFacilityDetails = healthFacilityDetails; + } + + public String getReporterName() { + return reporterName; + } + + public void setReporterName(String reporterName) { + this.reporterName = reporterName; + } + + public FacilityReferenceDto getReporterInstitution() { + return reporterInstitution; + } + + public void setReporterInstitution(FacilityReferenceDto reporterInstitution) { + this.reporterInstitution = reporterInstitution; + } + + public String getReporterDesignation() { + return reporterDesignation; + } + + public void setReporterDesignation(String reporterDesignation) { + this.reporterDesignation = reporterDesignation; + } + + public String getReporterDepartment() { + return reporterDepartment; + } + + public void setReporterDepartment(String reporterDepartment) { + this.reporterDepartment = reporterDepartment; + } + + public LocationDto getReporterAddress() { + return reporterAddress; + } + + public void setReporterAddress(LocationDto reporterAddress) { + this.reporterAddress = reporterAddress; + } + + public String getReporterPhone() { + return reporterPhone; + } + + public void setReporterPhone(String reporterPhone) { + this.reporterPhone = reporterPhone; + } + + public String getReporterEmail() { + return reporterEmail; + } + + public void setReporterEmail(String reporterEmail) { + this.reporterEmail = reporterEmail; + } + + public Date getTodaysDate() { + return todaysDate; + } + + public void setTodaysDate(Date todaysDate) { + this.todaysDate = todaysDate; + } + + public Date getStartDateTime() { + return startDateTime; + } + + public void setStartDateTime(Date startDateTime) { + this.startDateTime = startDateTime; + } + + public String getAefiDescription() { + return aefiDescription; + } + + public void setAefiDescription(String aefiDescription) { + this.aefiDescription = aefiDescription; + } + + public YesNoUnknown getSerious() { + return serious; + } + + public void setSerious(YesNoUnknown serious) { + this.serious = serious; + } + + public SeriousAefiReason getSeriousReason() { + return seriousReason; + } + + public void setSeriousReason(SeriousAefiReason seriousReason) { + this.seriousReason = seriousReason; + } + + public String getSeriousReasonDetails() { + return seriousReasonDetails; + } + + public void setSeriousReasonDetails(String seriousReasonDetails) { + this.seriousReasonDetails = seriousReasonDetails; + } + + public AefiOutcome getOutcome() { + return outcome; + } + + public void setOutcome(AefiOutcome outcome) { + this.outcome = outcome; + } + + public Date getDeathDate() { + return deathDate; + } + + public void setDeathDate(Date deathDate) { + this.deathDate = deathDate; + } + + public YesNoUnknown getAutopsyDone() { + return autopsyDone; + } + + public void setAutopsyDone(YesNoUnknown autopsyDone) { + this.autopsyDone = autopsyDone; + } + + public String getPastMedicalHistory() { + return pastMedicalHistory; + } + + public void setPastMedicalHistory(String pastMedicalHistory) { + this.pastMedicalHistory = pastMedicalHistory; + } + + public YesNoUnknown getInvestigationNeeded() { + return investigationNeeded; + } + + public void setInvestigationNeeded(YesNoUnknown investigationNeeded) { + this.investigationNeeded = investigationNeeded; + } + + public Date getInvestigationPlannedDate() { + return investigationPlannedDate; + } + + public void setInvestigationPlannedDate(Date investigationPlannedDate) { + this.investigationPlannedDate = investigationPlannedDate; + } + + public Date getReceivedAtNationalLevelDate() { + return receivedAtNationalLevelDate; + } + + public void setReceivedAtNationalLevelDate(Date receivedAtNationalLevelDate) { + this.receivedAtNationalLevelDate = receivedAtNationalLevelDate; + } + + public String getWorldwideId() { + return worldwideId; + } + + public void setWorldwideId(String worldwideId) { + this.worldwideId = worldwideId; + } + + public String getNationalLevelComment() { + return nationalLevelComment; + } + + public void setNationalLevelComment(String nationalLevelComment) { + this.nationalLevelComment = nationalLevelComment; + } + + public boolean isArchived() { + return archived; + } + + public void setArchived(boolean archived) { + this.archived = archived; + } + + public boolean isDeleted() { + return deleted; + } + + public void setDeleted(boolean deleted) { + this.deleted = deleted; + } + + public DeletionReason getDeletionReason() { + return deletionReason; + } + + public void setDeletionReason(DeletionReason deletionReason) { + this.deletionReason = deletionReason; + } + + public String getOtherDeletionReason() { + return otherDeletionReason; + } + + public void setOtherDeletionReason(String otherDeletionReason) { + this.otherDeletionReason = otherDeletionReason; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java new file mode 100644 index 00000000000..4267d27a9b5 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java @@ -0,0 +1,31 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.util.List; + +import javax.ejb.Remote; + +import de.symeda.sormas.api.CoreFacade; + +@Remote +public interface AefiFacade extends CoreFacade { + + List getEntriesList(AefiListCriteria criteria, Integer first, Integer max); +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java new file mode 100644 index 00000000000..815c46365dc --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java @@ -0,0 +1,87 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.util.ArrayList; +import java.util.List; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public final class AefiHelper { + + private AefiHelper() { + + } + + public static String buildAdverseEventsString( + AdverseEventState severeLocalReaction, + boolean severeLocalReactionMoreThanThreeDays, + boolean severeLocalReactionBeyondNearestJoint, + AdverseEventState seizures, + SeizureType seizureType, + AdverseEventState abscess, + AdverseEventState sepsis, + AdverseEventState encephalopathy, + AdverseEventState toxicShockSyndrome, + AdverseEventState thrombocytopenia, + AdverseEventState anaphylaxis, + AdverseEventState feverishFeeling, + String otherAdverseEventDetails) { + + List adverseEventsList = new ArrayList<>(); + + if (severeLocalReaction == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.SEVERE_LOCAL_REACTION)); + } + + if (seizures == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.SEIZURES)); + } + + if (abscess == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.ABSCESS)); + } + + if (sepsis == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.SEPSIS)); + } + + if (encephalopathy == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.ENCEPHALOPATHY)); + } + + if (toxicShockSyndrome == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.TOXIC_SHOCK_SYNDROME)); + } + + if (thrombocytopenia == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.THROMBOCYTOPENIA)); + } + + if (anaphylaxis == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.ANAPHYLAXIS)); + } + + if (feverishFeeling == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.FEVERISH_FEELING)); + } + + return String.join(", ", adverseEventsList); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java new file mode 100644 index 00000000000..e389093fc00 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java @@ -0,0 +1,277 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.utils.PersonalData; +import de.symeda.sormas.api.utils.SensitiveData; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableIndexDto; + +public class AefiIndexDto extends PseudonymizableIndexDto implements Serializable, Cloneable { + + public static final String I18N_PREFIX = "AefiIndex"; + + public static final String UUID = "uuid"; + public static final String IMMUNIZATION_UUID = "immunizationUuid"; + public static final String PERSON_UUID = "personUuid"; + public static final String PERSON_FIRST_NAME = "personFirstName"; + public static final String PERSON_LAST_NAME = "personLastName"; + public static final String REPORT_DATE = "reportDate"; + public static final String DISEASE = "disease"; + public static final String AGE_AND_BIRTH_DATE = "ageAndBirthDate"; + public static final String SEX = "sex"; + public static final String REGION = "region"; + public static final String DISTRICT = "district"; + public static final String SERIOUS = "serious"; + public static final String PRIMARY_VACCINE_NAME = "primaryVaccine"; + public static final String OUTCOME = "outcome"; + public static final String VACCINATION_DATE = "vaccinationDate"; + public static final String START_DATE_TIME = "startDateTime"; + public static final String ADVERSE_EVENTS = "adverseEvents"; + + private String immunizationUuid; + private String personUuid; + @PersonalData + @SensitiveData + private String personFirstName; + @PersonalData + @SensitiveData + private String personLastName; + private Date reportDate; + private Disease disease; + private AgeAndBirthDateDto ageAndBirthDate; + private Sex sex; + private String region; + private String district; + private YesNoUnknown serious; + private Vaccine primaryVaccine; + private AefiOutcome outcome; + private Date vaccinationDate; + private Date startDateTime; + private String adverseEvents; + private DeletionReason deletionReason; + private String otherDeletionReason; + private boolean isInJurisdiction; + + public AefiIndexDto( + String uuid, + String immunizationUuid, + String personUuid, + String personFirstName, + String personLastName, + Disease disease, + AgeAndBirthDateDto ageAndBirthDate, + Sex sex, + String region, + String district, + YesNoUnknown serious, + Vaccine primaryVaccine, + AefiOutcome outcome, + Date vaccinationDate, + Date reportDate, + Date startDateTime, + String adverseEvents, + DeletionReason deletionReason, + String otherDeletionReason, + boolean isInJurisdiction) { + + super(uuid); + this.immunizationUuid = immunizationUuid; + this.personUuid = personUuid; + this.personFirstName = personFirstName; + this.personLastName = personLastName; + this.disease = disease; + this.ageAndBirthDate = ageAndBirthDate; + this.sex = sex; + this.region = region; + this.district = district; + this.serious = serious; + this.primaryVaccine = primaryVaccine; + this.outcome = outcome; + this.vaccinationDate = vaccinationDate; + this.reportDate = reportDate; + this.startDateTime = startDateTime; + this.adverseEvents = adverseEvents; + this.deletionReason = deletionReason; + this.otherDeletionReason = otherDeletionReason; + this.isInJurisdiction = isInJurisdiction; + } + + public String getImmunizationUuid() { + return immunizationUuid; + } + + public void setImmunizationUuid(String immunizationUuid) { + this.immunizationUuid = immunizationUuid; + } + + public String getPersonUuid() { + return personUuid; + } + + public void setPersonUuid(String personUuid) { + this.personUuid = personUuid; + } + + public String getPersonFirstName() { + return personFirstName; + } + + public void setPersonFirstName(String personFirstName) { + this.personFirstName = personFirstName; + } + + public String getPersonLastName() { + return personLastName; + } + + public void setPersonLastName(String personLastName) { + this.personLastName = personLastName; + } + + public Date getReportDate() { + return reportDate; + } + + public void setReportDate(Date reportDate) { + this.reportDate = reportDate; + } + + public Disease getDisease() { + return disease; + } + + public void setDisease(Disease disease) { + this.disease = disease; + } + + public AgeAndBirthDateDto getAgeAndBirthDate() { + return ageAndBirthDate; + } + + public void setAgeAndBirthDate(AgeAndBirthDateDto ageAndBirthDate) { + this.ageAndBirthDate = ageAndBirthDate; + } + + public Sex getSex() { + return sex; + } + + public void setSex(Sex sex) { + this.sex = sex; + } + + public String getRegion() { + return region; + } + + public void setRegion(String region) { + this.region = region; + } + + public String getDistrict() { + return district; + } + + public void setDistrict(String district) { + this.district = district; + } + + public YesNoUnknown getSerious() { + return serious; + } + + public void setSerious(YesNoUnknown serious) { + this.serious = serious; + } + + public Vaccine getPrimaryVaccine() { + return primaryVaccine; + } + + public void setPrimaryVaccine(Vaccine primaryVaccine) { + this.primaryVaccine = primaryVaccine; + } + + public AefiOutcome getOutcome() { + return outcome; + } + + public void setOutcome(AefiOutcome outcome) { + this.outcome = outcome; + } + + public Date getVaccinationDate() { + return vaccinationDate; + } + + public void setVaccinationDate(Date vaccinationDate) { + this.vaccinationDate = vaccinationDate; + } + + public Date getStartDateTime() { + return startDateTime; + } + + public void setStartDateTime(Date startDateTime) { + this.startDateTime = startDateTime; + } + + public String getAdverseEvents() { + return adverseEvents; + } + + public void setAdverseEvents(String adverseEvents) { + this.adverseEvents = adverseEvents; + } + + public DeletionReason getDeletionReason() { + return deletionReason; + } + + public void setDeletionReason(DeletionReason deletionReason) { + this.deletionReason = deletionReason; + } + + public String getOtherDeletionReason() { + return otherDeletionReason; + } + + public void setOtherDeletionReason(String otherDeletionReason) { + this.otherDeletionReason = otherDeletionReason; + } + + @Override + public boolean isInJurisdiction() { + return isInJurisdiction; + } + + @Override + public void setInJurisdiction(boolean inJurisdiction) { + isInJurisdiction = inJurisdiction; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListCriteria.java new file mode 100644 index 00000000000..80ae6536c0a --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListCriteria.java @@ -0,0 +1,48 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.immunization.ImmunizationReferenceDto; +import de.symeda.sormas.api.utils.criteria.BaseCriteria; + +public class AefiListCriteria extends BaseCriteria { + + private final ImmunizationReferenceDto immunizationReferenceDto; + + public static class Builder { + + private final ImmunizationReferenceDto immunizationReferenceDto; + + public Builder(ImmunizationReferenceDto immunizationReferenceDto) { + this.immunizationReferenceDto = immunizationReferenceDto; + } + + public AefiListCriteria build() { + return new AefiListCriteria(this); + } + } + + private AefiListCriteria(Builder builder) { + this.immunizationReferenceDto = builder.immunizationReferenceDto; + } + + public ImmunizationReferenceDto getImmunization() { + return immunizationReferenceDto; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListEntryDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListEntryDto.java new file mode 100644 index 00000000000..706e97da2b2 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiListEntryDto.java @@ -0,0 +1,97 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableIndexDto; + +public class AefiListEntryDto extends PseudonymizableIndexDto implements Serializable, Cloneable { + + public static final String I18N_PREFIX = "Aefi"; + + public static final String UUID = "uuid"; + public static final String SERIOUS = "serious"; + public static final String PRIMARY_VACCINE_NAME = "primaryVaccineName"; + public static final String PRIMARY_VACCINE_DOSE = "primaryVaccineDose"; + public static final String PRIMARY_VACCINE_VACCINATION_DATE = "primaryVaccineVaccinationDate"; + private String ADVERSE_EVENTS = "adverseEvents"; + + private YesNoUnknown serious; + private Vaccine primaryVaccineName; + private String primaryVaccineDose; + private Date primaryVaccineVaccinationDate; + private String adverseEvents; + + public AefiListEntryDto( + String uuid, + YesNoUnknown serious, + Vaccine primaryVaccineName, + String primaryVaccineDose, + Date primaryVaccineVaccinationDate, + String adverseEvents) { + + super(uuid); + this.serious = serious; + this.primaryVaccineName = primaryVaccineName; + this.primaryVaccineDose = primaryVaccineDose; + this.primaryVaccineVaccinationDate = primaryVaccineVaccinationDate; + this.adverseEvents = adverseEvents; + } + + public YesNoUnknown getSerious() { + return serious; + } + + public void setSerious(YesNoUnknown serious) { + this.serious = serious; + } + + public Vaccine getPrimaryVaccineName() { + return primaryVaccineName; + } + + public void setPrimaryVaccineName(Vaccine primaryVaccineName) { + this.primaryVaccineName = primaryVaccineName; + } + + public String getPrimaryVaccineDose() { + return primaryVaccineDose; + } + + public void setPrimaryVaccineDose(String primaryVaccineDose) { + this.primaryVaccineDose = primaryVaccineDose; + } + + public Date getPrimaryVaccineVaccinationDate() { + return primaryVaccineVaccinationDate; + } + + public void setPrimaryVaccineVaccinationDate(Date primaryVaccineVaccinationDate) { + this.primaryVaccineVaccinationDate = primaryVaccineVaccinationDate; + } + + public String getAdverseEvents() { + return adverseEvents; + } + + public void setAdverseEvents(String adverseEvents) { + this.adverseEvents = adverseEvents; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiOutcome.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiOutcome.java new file mode 100644 index 00000000000..20d22beacbe --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiOutcome.java @@ -0,0 +1,36 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiOutcome { + + RECOVERING, + RECOVERED, + RECOVERED_WITH_SEQUELAE, + NOT_RECOVERED, + UNKNOWN, + DIED; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java new file mode 100644 index 00000000000..51acac53ec1 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java @@ -0,0 +1,45 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.ReferenceDto; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.utils.DependingOnFeatureType; + +@DependingOnFeatureType(featureType = FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT) +public class AefiReferenceDto extends ReferenceDto { + + private String externalId; + + public AefiReferenceDto() { + } + + public AefiReferenceDto(String uuid, String caption, String externalId) { + super(uuid, caption); + this.externalId = externalId; + } + + public String getExternalId() { + return externalId; + } + + public void setExternalId(String externalId) { + this.externalId = externalId; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiType.java new file mode 100644 index 00000000000..952eb1bc1cd --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiType.java @@ -0,0 +1,41 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.utils.YesNoUnknown; + +public enum AefiType { + + SERIOUS, + NON_SERIOUS; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } + + public static String toString(YesNoUnknown serious) { + if (serious == YesNoUnknown.YES) { + return AefiType.SERIOUS.toString(); + } else { + return AefiType.NON_SERIOUS.toString(); + } + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeizureType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeizureType.java new file mode 100644 index 00000000000..d5abb2d27ce --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeizureType.java @@ -0,0 +1,32 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum SeizureType { + + FEBRILE, + AFEBRILE; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiReason.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiReason.java new file mode 100644 index 00000000000..17156c90bbd --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiReason.java @@ -0,0 +1,36 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum SeriousAefiReason { + + DEATH, + LIFE_THREATENING, + DISABILITY, + HOSPITALIZATION, + CONGENITAL_ANOMALY, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/common/DeletableEntityType.java b/sormas-api/src/main/java/de/symeda/sormas/api/common/DeletableEntityType.java index 0d01e715da0..0e3a694ec31 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/common/DeletableEntityType.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/common/DeletableEntityType.java @@ -23,6 +23,7 @@ public enum DeletableEntityType { EVENT, EVENT_PARTICIPANT, IMMUNIZATION, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION, TRAVEL_ENTRY, CAMPAIGN, SAMPLE, diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/AefiDashboardCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/AefiDashboardCriteria.java new file mode 100644 index 00000000000..cf245b5a25a --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/AefiDashboardCriteria.java @@ -0,0 +1,49 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.dashboard; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDashboardFilterDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; + +public class AefiDashboardCriteria extends BaseDashboardCriteria { + + private AefiDashboardFilterDateType aefiDashboardFilterDateType; + private AefiType aefiType; + + public AefiDashboardCriteria() { + super(AefiDashboardCriteria.class); + } + + public AefiDashboardFilterDateType getAefiDashboardFilterDateType() { + return aefiDashboardFilterDateType; + } + + public AefiType getAefiType() { + return aefiType; + } + + public AefiDashboardCriteria aefiDashboardDateType(AefiDashboardFilterDateType aefiDashboardFilterDateType) { + this.aefiDashboardFilterDateType = aefiDashboardFilterDateType; + + return self; + } + + public AefiDashboardCriteria aefiType(AefiType aefiType) { + this.aefiType = aefiType; + + return self; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java new file mode 100644 index 00000000000..0ba1d2093bc --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java @@ -0,0 +1,58 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.List; + +public class AefiChartData implements Serializable { + + private static final long serialVersionUID = 3538219674050390425L; + + private List xAxisCategories = new ArrayList<>(); + private List series = new ArrayList<>(); + + public AefiChartData() { + } + + public List getxAxisCategories() { + return xAxisCategories; + } + + public void setxAxisCategories(List xAxisCategories) { + this.xAxisCategories = xAxisCategories; + } + + public List getSeries() { + return series; + } + + public void setSeries(List series) { + this.series = series; + } + + public void addXAxisCategory(Object category) { + xAxisCategories.add(category); + } + + public void addSeries(AefiChartSeries chartSeries) { + series.add(chartSeries); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartSeries.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartSeries.java new file mode 100644 index 00000000000..67824645c4e --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartSeries.java @@ -0,0 +1,61 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.ArrayList; +import java.util.List; + +public class AefiChartSeries implements Serializable { + + private static final long serialVersionUID = 8537929721515000783L; + + private Object name; + private String color; + private List seriesData = new ArrayList<>(); + + public AefiChartSeries(Object name) { + this.name = name; + } + + public Object getName() { + return name; + } + + public void setName(Object name) { + this.name = name; + } + + public String getColor() { + return color; + } + + public void setColor(String color) { + this.color = color; + } + + public List getSeriesData() { + return seriesData; + } + + public void setSeriesData(List seriesData) { + this.seriesData = seriesData; + } + + public void addData(String data) { + seriesData.add(data); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java new file mode 100644 index 00000000000..2e61f221bc5 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java @@ -0,0 +1,44 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization; + +import java.util.List; +import java.util.Map; + +import javax.ejb.Remote; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; + +@Remote +public interface AefiDashboardFacade { + + Map getAefiCountsByType(AefiDashboardCriteria dashboardCriteria); + + Map> getAefiCountsByVaccine(AefiDashboardCriteria dashboardCriteria); + + AefiChartData getAefiByVaccineDoseChartData(AefiDashboardCriteria dashboardCriteria); + + AefiChartData getAefiEventsByGenderChartData(AefiDashboardCriteria dashboardCriteria); + + Long countAefiForMap(AefiDashboardCriteria criteria); + + List getAefiForMap(AefiDashboardCriteria criteria); +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/MapAefiDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/MapAefiDto.java new file mode 100644 index 00000000000..66b1f573c05 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/MapAefiDto.java @@ -0,0 +1,78 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization; + +import java.io.Serializable; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.utils.YesNoUnknown; + +public class MapAefiDto implements Serializable { + + private static final long serialVersionUID = -7323648840592752250L; + + private Double latitude; + private Double longitude; + + private AefiType aefiType; + + public MapAefiDto(Double longitude, Double latitude, Double associatedEntityLongitude, Double associatedEntityLatitude) { + if (!setLatLonIfPresent(longitude, latitude, null)) + setLatLonIfPresent(associatedEntityLongitude, associatedEntityLatitude, null); + } + + public MapAefiDto( + Double aefiLon, + Double aefiLat, + Double immunizationFacilityLon, + Double immunizationFacilityLat, + Double immunizationPersonLon, + Double immunizationPersonLat, + YesNoUnknown serious) { + + aefiType = serious == YesNoUnknown.YES ? AefiType.SERIOUS : AefiType.NON_SERIOUS; + if (!setLatLonIfPresent(aefiLon, aefiLat, aefiType)) + if (!setLatLonIfPresent(immunizationFacilityLon, immunizationFacilityLat, aefiType)) + setLatLonIfPresent(immunizationPersonLon, immunizationPersonLat, aefiType); + } + + private boolean setLatLonIfPresent(Double longitude, Double latitude, AefiType aefiType) { + if (longitude != null && latitude != null) { + this.longitude = longitude; + this.latitude = latitude; + this.aefiType = aefiType; + + return true; + } + + return false; + } + + public Double getLatitude() { + return latitude; + } + + public Double getLongitude() { + return longitude; + } + + public AefiType getAefiType() { + return aefiType; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/feature/FeatureType.java b/sormas-api/src/main/java/de/symeda/sormas/api/feature/FeatureType.java index bd3180491d4..ca44b364a12 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/feature/FeatureType.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/feature/FeatureType.java @@ -69,6 +69,7 @@ public enum FeatureType { TASK_MANAGEMENT(true, true, null, null, ImmutableMap.of(FeatureTypeProperty.ALLOW_FREE_EDITING, Boolean.FALSE)), WEEKLY_REPORTING(true, true, null, null, null), IMMUNIZATION_MANAGEMENT(true, true, null, null, ImmutableMap.of(FeatureTypeProperty.REDUCED, Boolean.FALSE)), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT(true, false, null, null, ImmutableMap.of(FeatureTypeProperty.REDUCED, Boolean.FALSE)), TRAVEL_ENTRIES(true, false, null, null, null), DASHBOARD_SURVEILLANCE(true, true, null, null, null), diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index de514090097..ac9fd632af3 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -40,6 +40,8 @@ public interface Captions { String Action_title = "Action.title"; String actionAccept = "actionAccept"; String actionAdjustChanges = "actionAdjustChanges"; + String actionAefiAssignNewReportingIdNumber = "actionAefiAssignNewReportingIdNumber"; + String actionAefiSelectPrimarySuspectVaccination = "actionAefiSelectPrimarySuspectVaccination"; String actionApply = "actionApply"; String actionApplyDateFilter = "actionApplyDateFilter"; String actionApplyFilters = "actionApplyFilters"; @@ -177,6 +179,105 @@ public interface Captions { String address = "address"; String adoptHomeAddressOfCasePerson = "adoptHomeAddressOfCasePerson"; String adoptHomeAddressOfCasePersonIfRelationMatches = "adoptHomeAddressOfCasePersonIfRelationMatches"; + String AdverseEvents_abscess = "AdverseEvents.abscess"; + String AdverseEvents_anaphylaxis = "AdverseEvents.anaphylaxis"; + String AdverseEvents_encephalopathy = "AdverseEvents.encephalopathy"; + String AdverseEvents_feverishFeeling = "AdverseEvents.feverishFeeling"; + String AdverseEvents_otherAdverseEventDetails = "AdverseEvents.otherAdverseEventDetails"; + String AdverseEvents_seizures = "AdverseEvents.seizures"; + String AdverseEvents_seizureType = "AdverseEvents.seizureType"; + String AdverseEvents_sepsis = "AdverseEvents.sepsis"; + String AdverseEvents_severeLocalReaction = "AdverseEvents.severeLocalReaction"; + String AdverseEvents_severeLocalReactionBeyondNearestJoint = "AdverseEvents.severeLocalReactionBeyondNearestJoint"; + String AdverseEvents_severeLocalReactionMoreThanThreeDays = "AdverseEvents.severeLocalReactionMoreThanThreeDays"; + String AdverseEvents_thrombocytopenia = "AdverseEvents.thrombocytopenia"; + String AdverseEvents_toxicShockSyndrome = "AdverseEvents.toxicShockSyndrome"; + String Aefi_aefiDescription = "Aefi.aefiDescription"; + String Aefi_ageGroup = "Aefi.ageGroup"; + String Aefi_autopsyDone = "Aefi.autopsyDone"; + String Aefi_changeDate = "Aefi.changeDate"; + String Aefi_country = "Aefi.country"; + String Aefi_creationDate = "Aefi.creationDate"; + String Aefi_deathDate = "Aefi.deathDate"; + String Aefi_deletionReason = "Aefi.deletionReason"; + String Aefi_externalId = "Aefi.externalId"; + String Aefi_healthFacility = "Aefi.healthFacility"; + String Aefi_healthFacilityDetails = "Aefi.healthFacilityDetails"; + String Aefi_investigationNeeded = "Aefi.investigationNeeded"; + String Aefi_investigationPlannedDate = "Aefi.investigationPlannedDate"; + String Aefi_lactating = "Aefi.lactating"; + String Aefi_nationalLevelComment = "Aefi.nationalLevelComment"; + String Aefi_onsetAgeDays = "Aefi.onsetAgeDays"; + String Aefi_onsetAgeMonths = "Aefi.onsetAgeMonths"; + String Aefi_onsetAgeYears = "Aefi.onsetAgeYears"; + String Aefi_otherDeletionReason = "Aefi.otherDeletionReason"; + String Aefi_outcome = "Aefi.outcome"; + String Aefi_pastMedicalHistory = "Aefi.pastMedicalHistory"; + String Aefi_pregnant = "Aefi.pregnant"; + String Aefi_primaryVaccineVaccinationDate = "Aefi.primaryVaccineVaccinationDate"; + String Aefi_receivedAtNationalLevelDate = "Aefi.receivedAtNationalLevelDate"; + String Aefi_reportDate = "Aefi.reportDate"; + String Aefi_reporterDepartment = "Aefi.reporterDepartment"; + String Aefi_reporterDesignation = "Aefi.reporterDesignation"; + String Aefi_reporterEmail = "Aefi.reporterEmail"; + String Aefi_reporterInstitution = "Aefi.reporterInstitution"; + String Aefi_reporterName = "Aefi.reporterName"; + String Aefi_reporterPhone = "Aefi.reporterPhone"; + String Aefi_reportingIdNumber = "Aefi.reportingIdNumber"; + String Aefi_reportingUser = "Aefi.reportingUser"; + String Aefi_responsibleCommunity = "Aefi.responsibleCommunity"; + String Aefi_responsibleDistrict = "Aefi.responsibleDistrict"; + String Aefi_responsibleRegion = "Aefi.responsibleRegion"; + String Aefi_serious = "Aefi.serious"; + String Aefi_seriousReason = "Aefi.seriousReason"; + String Aefi_seriousReasonDetails = "Aefi.seriousReasonDetails"; + String Aefi_startDateTime = "Aefi.startDateTime"; + String Aefi_todaysDate = "Aefi.todaysDate"; + String Aefi_trimester = "Aefi.trimester"; + String Aefi_uuid = "Aefi.uuid"; + String Aefi_worldwideId = "Aefi.worldwideId"; + String aefiActiveAdverseEvents = "aefiActiveAdverseEvents"; + String aefiAefiDataView = "aefiAefiDataView"; + String aefiAefiList = "aefiAefiList"; + String aefiAllActiveAndArchivedAdverseEvents = "aefiAllActiveAndArchivedAdverseEvents"; + String aefiArchivedAdverseEvents = "aefiArchivedAdverseEvents"; + String AefiCriteria_aefiType = "AefiCriteria.aefiType"; + String AefiCriteria_outcome = "AefiCriteria.outcome"; + String AefiCriteria_vaccineManufacturer = "AefiCriteria.vaccineManufacturer"; + String AefiCriteria_vaccineName = "AefiCriteria.vaccineName"; + String aefiDashboardAllAefi = "aefiDashboardAllAefi"; + String aefiDashboardNonSerious = "aefiDashboardNonSerious"; + String aefiDashboardNonSeriousAefi = "aefiDashboardNonSeriousAefi"; + String aefiDashboardSerious = "aefiDashboardSerious"; + String aefiDashboardSeriousAefi = "aefiDashboardSeriousAefi"; + String aefiDashboardShowNonSeriousAefi = "aefiDashboardShowNonSeriousAefi"; + String aefiDashboardShowSeriousAefi = "aefiDashboardShowSeriousAefi"; + String aefiDeletedAdverseEvents = "aefiDeletedAdverseEvents"; + String AefiIndex_adverseEvents = "AefiIndex.adverseEvents"; + String AefiIndex_ageAndBirthDate = "AefiIndex.ageAndBirthDate"; + String AefiIndex_disease = "AefiIndex.disease"; + String AefiIndex_district = "AefiIndex.district"; + String AefiIndex_immunizationUuid = "AefiIndex.immunizationUuid"; + String AefiIndex_outcome = "AefiIndex.outcome"; + String AefiIndex_personFirstName = "AefiIndex.personFirstName"; + String AefiIndex_personLastName = "AefiIndex.personLastName"; + String AefiIndex_personUuid = "AefiIndex.personUuid"; + String AefiIndex_primaryVaccine = "AefiIndex.primaryVaccine"; + String AefiIndex_region = "AefiIndex.region"; + String AefiIndex_reportDate = "AefiIndex.reportDate"; + String AefiIndex_serious = "AefiIndex.serious"; + String AefiIndex_sex = "AefiIndex.sex"; + String AefiIndex_startDateTime = "AefiIndex.startDateTime"; + String AefiIndex_uuid = "AefiIndex.uuid"; + String AefiIndex_vaccinationDate = "AefiIndex.vaccinationDate"; + String aefiNewAdverseEvent = "aefiNewAdverseEvent"; + String aefiVaccinationsDiluentBatchLotNumber = "aefiVaccinationsDiluentBatchLotNumber"; + String aefiVaccinationsDiluentExpiryDate = "aefiVaccinationsDiluentExpiryDate"; + String aefiVaccinationsDiluentInformation = "aefiVaccinationsDiluentInformation"; + String aefiVaccinationsDiluentTimeOfReconstitution = "aefiVaccinationsDiluentTimeOfReconstitution"; + String aefiVaccinationsPrimaryVaccine = "aefiVaccinationsPrimaryVaccine"; + String aefiVaccinationsVaccineDetails = "aefiVaccinationsVaccineDetails"; + String aefiVaccinationsVaccineInformation = "aefiVaccinationsVaccineInformation"; String AggregateReport_deaths = "AggregateReport.deaths"; String AggregateReport_disease = "AggregateReport.disease"; String AggregateReport_grouping = "AggregateReport.grouping"; @@ -914,6 +1015,7 @@ public interface Captions { String dashboardNotVisitedFor = "dashboardNotVisitedFor"; String dashboardNotYetClassified = "dashboardNotYetClassified"; String dashboardNotYetClassifiedOnly = "dashboardNotYetClassifiedOnly"; + String dashboardNumberOfAdverseEvents = "dashboardNumberOfAdverseEvents"; String dashboardNumberOfCases = "dashboardNumberOfCases"; String dashboardNumberOfContacts = "dashboardNumberOfContacts"; String dashboardNumberOfSamples = "dashboardNumberOfSamples"; @@ -1690,6 +1792,7 @@ public interface Captions { String LoginSidebar_outbreakResponse = "LoginSidebar.outbreakResponse"; String LoginSidebar_poweredBy = "LoginSidebar.poweredBy"; String mainMenuAbout = "mainMenuAbout"; + String mainMenuAdverseEvents = "mainMenuAdverseEvents"; String mainMenuAggregateReports = "mainMenuAggregateReports"; String mainMenuCampaigns = "mainMenuCampaigns"; String mainMenuCases = "mainMenuCases"; @@ -1914,6 +2017,7 @@ public interface Captions { String personCreateNew = "personCreateNew"; String personDistrictPrompt = "personDistrictPrompt"; String personFindMatching = "personFindMatching"; + String personFullName = "personFullName"; String personLinkToCases = "personLinkToCases"; String personLinkToContacts = "personLinkToContacts"; String personLinkToEvents = "personLinkToEvents"; @@ -2748,6 +2852,7 @@ public interface Captions { String versionIsMissing = "versionIsMissing"; String view = "view"; String View_actions = "View.actions"; + String View_adverseevents = "View.adverseevents"; String View_aggregatereports = "View.aggregatereports"; String View_aggregatereports_aggregatereporting = "View.aggregatereports.aggregatereporting"; String View_aggregatereports_reportdata = "View.aggregatereports.reportdata"; @@ -2821,6 +2926,7 @@ public interface Captions { String View_contacts_person = "View.contacts.person"; String View_contacts_sub = "View.contacts.sub"; String View_contacts_visits = "View.contacts.visits"; + String View_dashboard_adverseevents = "View.dashboard.adverseevents"; String View_dashboard_campaigns = "View.dashboard.campaigns"; String View_dashboard_contacts = "View.dashboard.contacts"; String View_dashboard_samples = "View.dashboard.samples"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java index 2b093c11be0..e45b067dca1 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java @@ -10,6 +10,9 @@ public interface Descriptions { * 1. java:S115: Violation of name convention for constants of this class is accepted: Close as false positive. */ + String aefiDashboardDiseaseFilter = "aefiDashboardDiseaseFilter"; + String aefiDashboardDistrictFilter = "aefiDashboardDistrictFilter"; + String aefiDashboardRegionFilter = "aefiDashboardRegionFilter"; String Campaign_calculatedBasedOn = "Campaign.calculatedBasedOn"; String Campaign_campaignPhase = "Campaign.campaignPhase"; String CaseData_caseClassification = "CaseData.caseClassification"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index ed87fae2ee3..42c2dd68365 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -53,6 +53,7 @@ public interface Strings { String close = "close"; String comparedTo = "comparedTo"; String confirmationAlsoAdjustQuarantine = "confirmationAlsoAdjustQuarantine"; + String confirmationArchiveAdverseEvent = "confirmationArchiveAdverseEvent"; String confirmationArchiveArea = "confirmationArchiveArea"; String confirmationArchiveAreas = "confirmationArchiveAreas"; String confirmationArchiveCampaign = "confirmationArchiveCampaign"; @@ -96,6 +97,7 @@ public interface Strings { String confirmationCancelFollowUp = "confirmationCancelFollowUp"; String confirmationChangeCaseDisease = "confirmationChangeCaseDisease"; String confirmationContactSourceCaseDiscardUnsavedChanges = "confirmationContactSourceCaseDiscardUnsavedChanges"; + String confirmationDearchiveAdverseEvent = "confirmationDearchiveAdverseEvent"; String confirmationDearchiveArea = "confirmationDearchiveArea"; String confirmationDearchiveAreas = "confirmationDearchiveAreas"; String confirmationDearchiveCampaign = "confirmationDearchiveCampaign"; @@ -217,6 +219,8 @@ public interface Strings { String entityActivityAsCase = "entityActivityAsCase"; String entityAdditionalTest = "entityAdditionalTest"; String entityAdditionalTests = "entityAdditionalTests"; + String entityAdverseEvent = "entityAdverseEvent"; + String entityAdverseEvents = "entityAdverseEvents"; String entityAggregateReports = "entityAggregateReports"; String entityAreas = "entityAreas"; String entityAutomaticSoftDeletion = "entityAutomaticSoftDeletion"; @@ -298,6 +302,7 @@ public interface Strings { String entityWeeklyReports = "entityWeeklyReports"; String epiWeek = "epiWeek"; String errorAccessDenied = "errorAccessDenied"; + String errorAdverseEventNotEditable = "errorAdverseEventNotEditable"; String errorCampaignDiagramTotalsCalculationError = "errorCampaignDiagramTotalsCalculationError"; String errorCampaignNotEditable = "errorCampaignNotEditable"; String errorCaseDuplicateDeletion = "errorCaseDuplicateDeletion"; @@ -412,9 +417,21 @@ public interface Strings { String headingActivityAsCaseDetails = "headingActivityAsCaseDetails"; String headingAdditionalTests = "headingAdditionalTests"; String headingAdjustQuarantine = "headingAdjustQuarantine"; + String headingAefiAdverseEvents = "headingAefiAdverseEvents"; + String headingAefiDashboardEpiCurve = "headingAefiDashboardEpiCurve"; + String headingAefiDashboardMap = "headingAefiDashboardMap"; + String headingAefiFirstDecisionLevel = "headingAefiFirstDecisionLevel"; + String headingAefiNationalDecisionLevel = "headingAefiNationalDecisionLevel"; + String headingAefiPatientsAgeAtOnset = "headingAefiPatientsAgeAtOnset"; + String headingAefiPatientsIdentification = "headingAefiPatientsIdentification"; + String headingAefiPickPrimarySuspectVaccine = "headingAefiPickPrimarySuspectVaccine"; + String headingAefiReportersInformation = "headingAefiReportersInformation"; + String headingAefiReportingInformation = "headingAefiReportingInformation"; + String headingAefiVaccinations = "headingAefiVaccinations"; String headingAllContacts = "headingAllContacts"; String headingAnimalContactDetails = "headingAnimalContactDetails"; String headingAnimalContacts = "headingAnimalContacts"; + String headingArchiveAdverseEvent = "headingArchiveAdverseEvent"; String headingArchiveCampaign = "headingArchiveCampaign"; String headingArchiveCase = "headingArchiveCase"; String headingArchiveContact = "headingArchiveContact"; @@ -535,6 +552,7 @@ public interface Strings { String headingCustomizableEnumConfigurationInfo = "headingCustomizableEnumConfigurationInfo"; String headingDatabaseExportFailed = "headingDatabaseExportFailed"; String headingDataImport = "headingDataImport"; + String headingDearchiveAdverseEvent = "headingDearchiveAdverseEvent"; String headingDearchiveCampaign = "headingDearchiveCampaign"; String headingDearchiveCase = "headingDearchiveCase"; String headingDearchiveContact = "headingDearchiveContact"; @@ -644,6 +662,7 @@ public interface Strings { String headingHospitalization = "headingHospitalization"; String headingHowToMergeCases = "headingHowToMergeCases"; String headingHowToMergeContacts = "headingHowToMergeContacts"; + String headingImmunizationAdverseEvents = "headingImmunizationAdverseEvents"; String headingImmunizationsDeleted = "headingImmunizationsDeleted"; String headingImmunizationsNotDeleted = "headingImmunizationsNotDeleted"; String headingImmunizationsNotRestored = "headingImmunizationsNotRestored"; @@ -874,6 +893,8 @@ public interface Strings { String inColumn = "inColumn"; String infoActivityAsCaseInvestigation = "infoActivityAsCaseInvestigation"; String infoAddTestsToSample = "infoAddTestsToSample"; + String infoAefiSelectPrimarySuspectVaccine = "infoAefiSelectPrimarySuspectVaccine"; + String infoArchivedAefiEntries = "infoArchivedAefiEntries"; String infoArchivedCases = "infoArchivedCases"; String infoArchivedContacts = "infoArchivedContacts"; String infoArchivedEventParticipants = "infoArchivedEventParticipants"; @@ -967,6 +988,7 @@ public interface Strings { String infoExposuresRiskAreaHint = "infoExposuresRiskAreaHint"; String infoFacilityCsvImport = "infoFacilityCsvImport"; String infoFacilityNeedsDistrict = "infoFacilityNeedsDistrict"; + String infoHeadingAefiDashboardMap = "infoHeadingAefiDashboardMap"; String infoHeadingEnvironmentSampleDashboardMap = "infoHeadingEnvironmentSampleDashboardMap"; String infoHeadingSampleDashboardMap = "infoHeadingSampleDashboardMap"; String infoHowToMergeCases = "infoHowToMergeCases"; @@ -998,6 +1020,7 @@ public interface Strings { String infoNoDiseaseSelected = "infoNoDiseaseSelected"; String infoNoEnvironmentSamples = "infoNoEnvironmentSamples"; String infoNoEventGroups = "infoNoEventGroups"; + String infoNoImmunizationAdverseEvents = "infoNoImmunizationAdverseEvents"; String infoNoNetworkDiagram = "infoNoNetworkDiagram"; String infoNoPathogenTests = "infoNoPathogenTests"; String infoNoSourceCase = "infoNoSourceCase"; @@ -1095,6 +1118,9 @@ public interface Strings { String messageActivateAccount = "messageActivateAccount"; String messageAdditionalTestDeleted = "messageAdditionalTestDeleted"; String messageAdditionalTestSaved = "messageAdditionalTestSaved"; + String messageAdverseEventArchived = "messageAdverseEventArchived"; + String messageAdverseEventDearchived = "messageAdverseEventDearchived"; + String messageAdverseEventSaved = "messageAdverseEventSaved"; String messageAggregatedReportEpiWeekFilterNotFilled = "messageAggregatedReportEpiWeekFilterNotFilled"; String messageAggregateReportDelete = "messageAggregateReportDelete"; String messageAggregateReportExpiredAgeGroups = "messageAggregateReportExpiredAgeGroups"; @@ -1618,6 +1644,13 @@ public interface Strings { String promptActionDateTo = "promptActionDateTo"; String promptActionEpiWeekFrom = "promptActionEpiWeekFrom"; String promptActionEpiWeekTo = "promptActionEpiWeekTo"; + String promptAefiDashboardFilterDateType = "promptAefiDashboardFilterDateType"; + String promptAefiDateFrom = "promptAefiDateFrom"; + String promptAefiDateTo = "promptAefiDateTo"; + String promptAefiDateType = "promptAefiDateType"; + String promptAefiEpiWeekFrom = "promptAefiEpiWeekFrom"; + String promptAefiEpiWeekTo = "promptAefiEpiWeekTo"; + String promptAefiValidFrom = "promptAefiValidFrom"; String promptAllAreas = "promptAllAreas"; String promptAllCommunities = "promptAllCommunities"; String promptAllDistricts = "promptAllDistricts"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java index b4b553421e9..68a9fbe69a4 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java @@ -10,6 +10,9 @@ public interface Validations { * 1. java:S115: Violation of name convention for constants of this class is accepted: Close as false positive. */ + String aefiWithoutAdverseEvents = "aefiWithoutAdverseEvents"; + String aefiWithoutPrimarySuspectVaccine = "aefiWithoutPrimarySuspectVaccine"; + String aefiWithoutSuspectVaccine = "aefiWithoutSuspectVaccine"; String afterDate = "afterDate"; String afterDateSoft = "afterDateSoft"; String afterDateWithDate = "afterDateWithDate"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java b/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java index 9d43f73733f..64684f3a800 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java @@ -63,6 +63,12 @@ public enum UserRight { IMMUNIZATION_ARCHIVE(UserRightGroup.IMMUNIZATION, UserRight._IMMUNIZATION_VIEW), IMMUNIZATION_VIEW_ARCHIVED(UserRightGroup.IMMUNIZATION), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), + PERSON_VIEW(UserRightGroup.PERSON), PERSON_EDIT(UserRightGroup.PERSON, UserRight._PERSON_VIEW), PERSON_DELETE(UserRightGroup.PERSON, UserRight._PERSON_VIEW, UserRight._VISIT_DELETE), @@ -169,6 +175,7 @@ public enum UserRight { DASHBOARD_CONTACT_VIEW_TRANSMISSION_CHAINS(UserRightGroup.DASHBOARD, UserRight._DASHBOARD_CONTACT_VIEW), DASHBOARD_CAMPAIGNS_VIEW(UserRightGroup.DASHBOARD, UserRight._CAMPAIGN_VIEW), DASHBOARD_SAMPLES_VIEW(UserRightGroup.DASHBOARD, UserRight._SAMPLE_VIEW), + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW(UserRightGroup.DASHBOARD, UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW), CASE_CLINICIAN_VIEW(UserRightGroup.CASE_MANAGEMENT, UserRight._CASE_VIEW), @@ -324,6 +331,11 @@ public enum UserRight { public static final String _IMMUNIZATION_EDIT = "IMMUNIZATION_EDIT"; public static final String _IMMUNIZATION_DELETE = "IMMUNIZATION_DELETE"; public static final String _IMMUNIZATION_ARCHIVE = "IMMUNIZATION_ARCHIVE"; + public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW"; + public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE"; + public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT"; + public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE"; + public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE"; public static final String _PERSON_VIEW = "PERSON_VIEW"; public static final String _PERSON_EDIT = "PERSON_EDIT"; public static final String _PERSON_DELETE = "PERSON_DELETE"; @@ -419,6 +431,7 @@ public enum UserRight { public static final String _DASHBOARD_CONTACT_VIEW_TRANSMISSION_CHAINS = "DASHBOARD_CONTACT_VIEW_TRANSMISSION_CHAINS"; public static final String _DASHBOARD_CAMPAIGNS_VIEW = "DASHBOARD_CAMPAIGNS_VIEW"; public static final String _DASHBOARD_SAMPLES_VIEW = "DASHBOARD_SAMPLES_VIEW"; + public static final String _DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW = "DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW"; public static final String _CASE_CLINICIAN_VIEW = "CASE_CLINICIAN_VIEW"; public static final String _THERAPY_VIEW = "THERAPY_VIEW"; public static final String _PRESCRIPTION_CREATE = "PRESCRIPTION_CREATE"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRightGroup.java b/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRightGroup.java index ece02973739..7fa2dd34069 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRightGroup.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRightGroup.java @@ -35,6 +35,7 @@ public enum UserRightGroup { EVENT, SAMPLE, IMMUNIZATION, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION, TRAVEL_ENTRY, CAMPAIGN, ENVIRONMENT, diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index a2fdb57909f..70623db15e1 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -195,6 +195,8 @@ actionOkAndGoToPersonDirectory = Okay, and continue to person directory actionExecuteAutomaticDeletion = Execute automatic deletion actionDone = Done actionConfirmAction = Confirm action +actionAefiSelectPrimarySuspectVaccination = Select Suspect Vaccination +actionAefiAssignNewReportingIdNumber = "Assign New AEFI ID" activityAsCaseFlightNumber=Flight number ActivityAsCase=Activity as case ActivityAsCase.startDate=Start of activity @@ -1003,6 +1005,7 @@ dashboardAggregatedNumber=Count dashboardProportion=Proportion (%) dashboardViewAsColumnChart=View as Column Chart dashboardViewAsBarChart=View as Bar Chart +dashboardNumberOfAdverseEvents=Number of Adverse Events #SampleDashboard sampleDashboardAllSamples=All Samples sampleDashboardFinalLabResults=Final Laboratory Results @@ -1018,6 +1021,15 @@ sampleDashboardShowCaseSamples=Show Case Samples sampleDashboardShowContactSamples=Show Contact Samples sampleDashboardShowEventParticipantSamples=Show Event Participant Samples sampleDashboardShowEnvironmentSamples=Show Environment Samples +#AefiDashboard +aefiDashboardAllAefi=All AEFI +aefiDashboardSerious=Serious +aefiDashboardNonSerious=Non Serious +aefiDashboardSeriousAefi=Serious AEFI +aefiDashboardNonSeriousAefi=Non-Serious AEFI +aefiDashboardShowSeriousAefi=Show Serious AEFI +aefiDashboardShowNonSeriousAefi=Show Non-Serious AEFI + captionDefault=Default defaultRegion=Default Region defaultDistrict=Default District @@ -1697,6 +1709,7 @@ mainMenuEntries=Entries mainMenuEvents=Events mainMenuExternalMessages=Messages mainMenuImmunizations=Immunizations +mainMenuAdverseEvents=Adverse Events mainMenuReports=Reports mainMenuSamples=Samples mainMenuEnvironments=Environments @@ -1794,6 +1807,7 @@ personFindMatching=Find matching persons personSelect=Select a matching person personSearchAndSelect=Select a different person personAgeAndBirthdate=Age and birth date +personFullName=Full name personNoEventParticipantLinkedToPerson=No event participant linked to person personNoCaseLinkedToPerson=No case linked to person personNoContactLinkedToPerson=No contact linked to person @@ -2228,6 +2242,102 @@ immunizationOnlyPersonsWithOverdueImmunization=Only show persons with overdue im immunizationOverwriteImmunization=Overwrite the existing immunization with this data immunizationCreateNewImmunization=Create the new immunization anyway immunizationNoImmunizationsForPerson=There are no immunizations for this person +# Adverse Events Following Immunization +Aefi.primaryVaccineVaccinationDate=Vaccination date +Aefi.uuid=Adverse event UUID +Aefi.reportDate=Date of report +Aefi.reportingUser=Reporting user +Aefi.externalId=External ID +Aefi.responsibleRegion=Responsible region +Aefi.responsibleDistrict=Responsible district +Aefi.responsibleCommunity=Responsible community +Aefi.country=Country +Aefi.reportingIdNumber=Reporting ID Number +Aefi.pregnant=Pregnant +Aefi.trimester=Trimester +Aefi.lactating=Lactating +Aefi.onsetAgeYears=Years +Aefi.onsetAgeMonths=Months +Aefi.onsetAgeDays=Days +Aefi.ageGroup=Age group +Aefi.healthFacility=Facility +Aefi.healthFacilityDetails=Facility name & description +Aefi.reporterName=Name +Aefi.reporterInstitution=Insitution +Aefi.reporterDesignation=Designation +Aefi.reporterDepartment=Department +Aefi.reporterPhone=Phone number +Aefi.reporterEmail=Email +Aefi.todaysDate=Today's date +Aefi.startDateTime=Date and time AEFI started +Aefi.aefiDescription=Describe AEFI (Signs and symptoms) +Aefi.serious=Serious +Aefi.seriousReason=Reason for serious +Aefi.seriousReasonDetails=Reason for serious details +Aefi.outcome=Outcome +Aefi.deathDate=Date of death +Aefi.autopsyDone=Autopsy done +Aefi.pastMedicalHistory=Past medical history +Aefi.investigationNeeded=Investigation needed +Aefi.investigationPlannedDate=Date investigation planned +Aefi.receivedAtNationalLevelDate=Date received at national +Aefi.worldwideId=AEFI worldwide unique ID +Aefi.nationalLevelComment=Comment +Aefi.deletionReason=Reason for deletion +Aefi.otherDeletionReason=Reason for deletion details +Aefi.creationDate=Creation date +Aefi.changeDate=Date of last change +aefiNewAdverseEvent=New Adverse Event +aefiAefiList=Adverse Events List +aefiAefiDataView=Adverse Event +aefiVaccinationsVaccineInformation=Vaccine Information +aefiVaccinationsPrimaryVaccine=Primary +aefiVaccinationsVaccineDetails=Vaccine details +aefiVaccinationsDiluentInformation=Diluent Information +aefiVaccinationsDiluentBatchLotNumber=Batch/Lot number +aefiVaccinationsDiluentExpiryDate=Expiry date +aefiVaccinationsDiluentTimeOfReconstitution=Reconstitution time +aefiActiveAdverseEvents = Active adverse events +aefiArchivedAdverseEvents = Archived adverse events +aefiAllActiveAndArchivedAdverseEvents = All active and archived adverse events +aefiDeletedAdverseEvents = Deleted adverse events +# Adverse Events +AdverseEvents.severeLocalReaction=Severe local reaction +AdverseEvents.severeLocalReactionMoreThanThreeDays=>3 days +AdverseEvents.severeLocalReactionBeyondNearestJoint=Beyond nearest joint +AdverseEvents.seizures=Seizures +AdverseEvents.seizureType=Seizure type +AdverseEvents.abscess=Abscess +AdverseEvents.sepsis=Sepsis +AdverseEvents.encephalopathy=Encephalopathy +AdverseEvents.toxicShockSyndrome=Toxic shock syndrome +AdverseEvents.thrombocytopenia=Thrombocytopenia +AdverseEvents.anaphylaxis=Anaphylaxis +AdverseEvents.feverishFeeling=Fever >38degC +AdverseEvents.otherAdverseEventDetails=Other (specify) +# Adverse Events Index +AefiIndex.uuid=AEFI ID +AefiIndex.immunizationUuid=Immunization ID +AefiIndex.personUuid=Person ID +AefiIndex.personFirstName=First Name +AefiIndex.personLastName=Last Name +AefiIndex.reportDate=Date of report +AefiIndex.disease=Disease +AefiIndex.ageAndBirthDate=Age and birth date +AefiIndex.sex=Sex +AefiIndex.region=Region +AefiIndex.district=District +AefiIndex.serious=Serious +AefiIndex.primaryVaccine=Suspect Vaccine +AefiIndex.outcome=Outcome +AefiIndex.vaccinationDate=Date of vaccination +AefiIndex.startDateTime=Date of AEFI onset +AefiIndex.adverseEvents=Adverse events +# Advers Events Criteria +AefiCriteria.aefiType=AEFI Type +AefiCriteria.outcome=Outcome +AefiCriteria.vaccineName=Vaccine +AefiCriteria.vaccineManufacturer=Manufacturer # Statistics statisticsAddFilter=Add filter statisticsAttribute=Attribute @@ -2712,6 +2822,7 @@ View.dashboard.contacts=Contacts Dashboard View.dashboard.surveillance=Surveillance Dashboard View.dashboard.campaigns=Campaigns Dashboard View.dashboard.samples=Samples Dashboard +View.dashboard.adverseevents=Adverse Events Dashboard View.events=Event Directory View.events.archive=Event Archive View.events.data=Event Information @@ -2728,6 +2839,7 @@ View.samples.data=Sample Information View.samples.sub= View.travelEntries=Travel Entries Directory View.immunizations=Immunization Directory +View.adverseevents=Adverse Events Directory View.statistics=Statistics View.statistics.database-export=Database export View.tasks=Task Management diff --git a/sormas-api/src/main/resources/descriptions.properties b/sormas-api/src/main/resources/descriptions.properties index 74693af7bd6..66bc7df81fd 100644 --- a/sormas-api/src/main/resources/descriptions.properties +++ b/sormas-api/src/main/resources/descriptions.properties @@ -232,4 +232,8 @@ sampleDashboardRegionFilter= The region of the associated Case/Contact/Event par sampleDashboardDistrictFilter= The district of the associated Case/Contact/Event participant sampleDashboardDiseaseFilter= The disease of the associated Case/Contact/Event participant's event sampleDashboardCountsByShipmentStatus=Only samples with the purpose external lab testing are considered -sampleDashboardCountsBySpecimenCondition=Only samples received and with the purpose external lab testing are considered \ No newline at end of file +sampleDashboardCountsBySpecimenCondition=Only samples received and with the purpose external lab testing are considered + +aefiDashboardRegionFilter= The region of the associated immunization +aefiDashboardDistrictFilter= The district of the associated immunization +aefiDashboardDiseaseFilter= The disease of the associated immunization \ No newline at end of file diff --git a/sormas-api/src/main/resources/enum.properties b/sormas-api/src/main/resources/enum.properties index 35fa86fe9f5..5880c9e0f00 100644 --- a/sormas-api/src/main/resources/enum.properties +++ b/sormas-api/src/main/resources/enum.properties @@ -291,6 +291,7 @@ CustomizableEnumType.PATHOGEN = Pathogen DashboardType.CONTACTS = Contacts DashboardType.SURVEILLANCE = Surveillance DashboardType.CAMPAIGNS = Campaigns +DashboardType.ADVERSE_EVENTS = Adverse Events # DatabaseTable DatabaseTable.ACTIONS = Actions @@ -1550,6 +1551,11 @@ UserRight.IMMUNIZATION_CREATE = Create new immunizations and vaccinations UserRight.IMMUNIZATION_EDIT = Edit existing immunizations and vaccinations UserRight.IMMUNIZATION_DELETE = Delete immunizations and vaccinations from the system UserRight.IMMUNIZATION_ARCHIVE = Archive immunizations +UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW = View existing adverse events following immunization +UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE = Create new adverse event following immunization +UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT = Edit existing adverse event following immunization +UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE = Delete adverse events following immunization from the system +UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE = Archive adverse events following immunization UserRight.PERSON_EXPORT = Export persons UserRight.CONTACT_MERGE = Merge contacts UserRight.EVENTGROUP_CREATE = Create new event groups @@ -1768,6 +1774,11 @@ UserRight.Desc.IMMUNIZATION_CREATE = Able to create new immunizations and vaccin UserRight.Desc.IMMUNIZATION_EDIT = Able to edit existing immunizations and vaccinations UserRight.Desc.IMMUNIZATION_DELETE = Able to delete immunizations and vaccinations from the system UserRight.Desc.IMMUNIZATION_ARCHIVE = Able to archive immunizations +UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW = Able to view existing adverse events following immunization +UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE = Able to create new adverse event following immunization +UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT = Able to edit existing adverse events following immunization +UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE = Able to delete adverse events following immunization from the system +UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE = Able to archive adverse events following immunization UserRight.Desc.PERSON_EXPORT = Able to export persons UserRight.Desc.CONTACT_MERGE = Able to merge contacts UserRight.Desc.EVENTGROUP_CREATE = Able to create new event groups @@ -1838,6 +1849,7 @@ UserRightGroup.CONTACT = Contact Surveillance UserRightGroup.VISIT = Follow-Up UserRightGroup.SAMPLE = Sample Testing UserRightGroup.IMMUNIZATION = Immunization +UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION = Adverse Events Following Immunization UserRightGroup.TRAVEL_ENTRY = Travel Entries UserRightGroup.TASK = Tasks UserRightGroup.EVENT = Events @@ -2333,3 +2345,43 @@ SelfReportInvestigationStatus.REJECTED=Rejected # SelfReportProcessingStatus SelfReportProcessingStatus.UNPROCESSED=Unprocessed SelfReportProcessingStatus.PROCESSED=Processed + +# AefiAgeGroup +AefiAgeGroup.ZERO_TO_ONE = 0 < 1 year +AefiAgeGroup.ONE_TO_FIVE = 1- 5 years +AefiAgeGroup.FIVE_TO_EIGHTEEN = > 5 years - 18 years +AefiAgeGroup.EIGHTEEN_TO_SIXTY = > 18 years - 60 years +AefiAgeGroup.SIXY_AND_ABOVE = > 60 years + +# SeizureType +SeizureType.FEBRILE=Febrile +SeizureType.AFEBRILE=Afebrile + +# SeriousAefiReason +SeriousAefiReason.DEATH=Death +SeriousAefiReason.LIFE_THREATENING=Life threatening +SeriousAefiReason.DISABILITY=Disability +SeriousAefiReason.HOSPITALIZATION=Hospitalization +SeriousAefiReason.CONGENITAL_ANOMALY=Congenital anomaly +SeriousAefiReason.OTHER=Other + +# AefiOutcome +AefiOutcome.RECOVERING=Recovering +AefiOutcome.RECOVERED=Recovered +AefiOutcome.RECOVERED_WITH_SEQUELAE=Recovered with sequelae +AefiOutcome.NOT_RECOVERED=Not Recovered +AefiOutcome.UNKNOWN=Unknown +AefiOutcome.DIED=Died + +# AefiType +AefiType.SERIOUS=Serious +AefiType.NON_SERIOUS=Non-serious + +# AefiDateType +AefiDateType.REPORT_DATE=Date of report +AefiDateType.START_DATE=Date of onset +AefiDateType.VACCINATION_DATE=Date of vaccination + +# AefiDashboardFilterDateType +AefiDashboardFilterDateType.REPORT_DATE=Date of report +AefiDashboardFilterDateType.START_DATE=Date of onset \ No newline at end of file diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index fc1854df695..845fd0c6f1f 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -136,6 +136,7 @@ confirmationArchiveEvent = Are you sure you want to archive this event? This wil confirmationArchiveEvents = Are you sure you want to archive all %d selected events? confirmationArchiveEventParticipant = Are you sure you want to archive this event participant? This will not remove it from the system or any statistics, but only hide it from the list of event participants. confirmationArchiveImmunization = Are you sure you want to archive this immunization? This will not remove it from the system or any statistics, but only hide it from the normal immunization directory. +confirmationArchiveAdverseEvent = Are you sure you want to archive this adverse event? This will not remove it from the system or any statistics, but only hide it from the normal adverse events directory. confirmationArchiveTask = Are you sure you want to archive this task? This will not remove it from the system or any statistics, but only hide it from the normal task management. confirmationArchiveTasks = Are you sure you want to archive all %d selected tasks? confirmationArchiveTravelEntry = Are you sure you want to archive this travel entry? This will not remove it from the system or any statistics, but only hide it from the normal travel entry directory. @@ -217,6 +218,7 @@ confirmationDearchiveCommunities = Are you sure you want to de-archive all %d se confirmationArchiveFacilities = Are you sure you want to archive all %d selected facilities? confirmationDearchiveFacilities = Are you sure you want to de-archive all %d selected facilities? confirmationDearchiveImmunization = Are you sure you want to de-archive this immunization? This will make it appear in the normal immunization directory again. +confirmationDearchiveAdverseEvent = Are you sure you want to de-archive this adverse event? This will make it appear in the normal adverse events directory again. confirmationArchiveLaboratories = Are you sure you want to archive all %d selected laboratories? confirmationDearchiveLaboratories = Are you sure you want to de-archive all %d selected laboratories? confirmationDearchiveTask = Are you sure you want to de-archive this task? This will make it appear in the normal task directory again. @@ -339,6 +341,12 @@ entityUsers = Users entityVaccinations = Vaccinations entityVisits = Visits entityWeeklyReports = Weekly reports +entityOutbreaks = Outbreaks +entityCustomizableEnumValues = Customizable enum values +entityCampaignFormMeta = Campaign form meta +entityCampaignFormData = Campaign form data +entityAdverseEvent = Adverse event +entityAdverseEvents = Adverse events # Error Messages errorAccessDenied=You do not have the required rights to view this page. @@ -426,6 +434,7 @@ errorEnvironmentSampleNoDispatchRight = You do not have the necessary user right errorEnvironmentSampleNoReceivalRight = You do not have the necessary user right to change the receival status of this environment sample errorSendingExternalEmail = Email could not be sent. Please contact an admin and notify them about this problem. errorExternalEmailAttachmentCannotEncrypt=Can't send email with attachments. The person has no national health id or primary phone number to send the password to or the SMS service is not set up in the system. +errorAdverseEventNotEditable = This adverse event is not editable anymore errorExternalEmailMissingPersonEmailAddress=This person does not have an email address # headings @@ -442,6 +451,7 @@ headingArchiveEvent = Archive event headingArchiveEventParticipant = Archive event participant headingArchiveEventGroup = Archive event group headingArchiveImmunization = Archive immunization +headingArchiveAdverseEvent = Archive adverse event headingArchiveTravelEntry = Archive travel entry headingCampaignBasics = Campaign basics headingCampaignData = Campaign data @@ -873,6 +883,16 @@ headingLimitedDiseases=Disease restrictions headingExternalEmailSend=Send email headingExternalEmailDetails=Email details headingCustomizableEnumConfigurationInfo=Customizable enum configuration +headingImmunizationAdverseEvents=Adverse events +headingAefiReportingInformation=Reporting information +headingAefiPatientsIdentification=Patients identification +headingAefiPatientsAgeAtOnset=Age at onset +headingAefiVaccinations=Vaccinations +headingAefiAdverseEvents=Adverse events +headingAefiFirstDecisionLevel=First decision Level +headingAefiNationalDecisionLevel=National Decision Level +headingAefiReportersInformation=Reporter's information +headingAefiPickPrimarySuspectVaccine=Select primary suspect vaccine # Info texts infoActivityAsCaseInvestigation = Please document ALL relevant activities after infection: @@ -1032,6 +1052,7 @@ infoContactsViewRegionDistrictFilter = When you select a region and/or district infoDeveloperOptions = You can use the controls below to generate dummy cases and contacts based on the selected restraints. Please note that generating a lot of data at once might take some time.
Generated data is neither fully deterministic, nor fully random, and intended for testing and demonstration purposes only. infoDeveloperOptionsContactGeneration = When generating contacts, the generator will pick random existing cases as source. Please make sure the case database is not empty before generating contacts. infoDeveloperOptionsSeedUsage = Using the seed will create identical datasets, provided the starting conditions (configuration & database) are also identical +infoCreateNewSampleDiscardsChangesCase = Creating a new sample will discard all unsaved changes made to this case infoCreateNewSampleDiscardsChangesContact = Creating a new sample will discard all unsaved changes made to this contact infoCreateNewSampleDiscardsChangesEventParticipant = Creating a new sample will discard all unsaved changes made to this event participant infoUsageOfEditableCampaignGrids = You can edit the campaign data and dashboard definitions by clicking inside one of the cells in the grid, and you can reorder the dashboard elements by dragging and dropping the grid rows @@ -1074,6 +1095,9 @@ infoNoEnvironmentSamples = No samples have been created for this environment infoRestrictDiseasesDescription=Mark all diseases that the user is supposed to have access to infoNoCustomizableEnumTranslations = Click on the + button below to add translations to this customizable enum value. infoCustomizableEnumConfigurationInfo = Customizable enums are value sets that can be customized in order to react to the individual needs of your country or a specific epidemiological situation. The table on this screen contains all customizable enum values in the database. Each value is associated with a data type, e.g. disease variants or occupation types. Some of these data types have default values that are automatically added to the database when SORMAS is set up or new data types are added to the system.

You can add new enum values or edit existing ones, add translations for languages supported by SORMAS, select the diseases that the value should be visible for (by default, customizable enum values are visible for all diseases), and configure additional properties.

Properties are used to further control the behaviour of customizable enum values. E.g. the "has details" property that is supported by most enum values toggles whether selecting this enum value would bring up an additional text field that users can add more information to. +infoNoImmunizationAdverseEvents = No adverse events have been created for this immunization +infoAefiSelectPrimarySuspectVaccine = The list below contains all vaccinations of the immunization. Please select the suspect vaccination related to this adverse event. +infoArchivedAefiEntries = Adverse event entries are automatically archived after %d days without changes to the data. # Messages messageActionOutsideJurisdictionDeletionDenied = The action outside user's jurisdiction cannot be deleted @@ -1255,6 +1279,8 @@ messageImmunizationDearchived = Immunization has been de-archived messageImmunizationOutsideJurisdictionDeletionDenied = The immunization outside user's jurisdiction cannot be deleted messageImmunizationSaved = Immunization data saved. messageImmunizationSavedVaccinationStatusUpdated = Immunization data saved. The vaccination status of matching cases, contacts, and event participants of the immunization person has been updated to vaccinated. +messageAdverseEventArchived = Adverse event has been archived +messageAdverseEventDearchived = Adverse event has been de-archived messageImportCanceled = Import canceled!
The import has been canceled. All already processed rows have been successfully imported. You can now close this window. messageImportCanceledErrors = Import canceled!
The import has been canceled. Some of the already processed rows could not be imported due to malformed data.
Please close this window and download the error report. messageImportError = Could not import file. @@ -1520,6 +1546,7 @@ messageExternalEmailNoAttachments=No attachments messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. +messageAdverseEventSaved=Adverse event saved messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. @@ -1642,6 +1669,7 @@ promptImmunizationDateFrom = New immunizations from... promptImmunizationDateTo = ... to promptImmunizationEpiWeekFrom = New immunizations from epi week... promptImmunizationEpiWeekTo = ... to epi week +promptEmail = Email: promptExternalMessagesSearchField = UUID, name, postal code, reporter name, reporter postal code promptExternalMessagesContentSearchField = External message content promptExternalMessagesDateFrom = Message date from... @@ -1660,6 +1688,7 @@ promptSampleEpiWeekTo = ... to epi week promtSampleDataType = Sample reference date promptSearch = Search... promptTaskSearchField = Case or contact ID/name, Event ID/title +promptTelephoneNumber = Phone number: promptTaskDateType = Task reference date promptTaskDateFrom = Tasks from... promptTaskDateTo = ... to @@ -1685,7 +1714,13 @@ promptAllDistricts=All districts promptAllCommunities=All communities promptExternalIdExternalSurveillanceTool=Will adopt external reporting tool GUID promptExternalJournalForceDeletion=Do you want to force the cancellation in SORMAS? This would mark the person as deleted from the external journal in SORMAS, while there is a high probability of personal data still remaining in the external journal. -promptPersonDuplicateSearchIdExternalId=First Name, Last Name, ID, External ID, External token +promptPersonDuplicateSearchIdExternalId=First Name, Last Name, ID, External ID, External token +promptAefiDateType = Aefi reference date +promptAefiValidFrom = New adverse events valid from... +promptAefiDateFrom = New adverse events from... +promptAefiDateTo = ... to +promptAefiEpiWeekFrom = New adverse events from epi week... +promptAefiEpiWeekTo = ... to epi week #DiseaseNetworkDiagram DiseaseNetworkDiagram.Classification.HEALTHY = Healthy DiseaseNetworkDiagram.heading = Disease network diagram @@ -1798,6 +1833,11 @@ headingSampleDashboardMap=Sample Status Map infoHeadingSampleDashboardMap=Samples are shown using the GPS coordinate of the person's home address. infoHeadingEnvironmentSampleDashboardMap=Environment samples are shown using the GPS coordinates of those samples or, if not available, their associated environment. +promptAefiDashboardFilterDateType=AEFI reference date +headingAefiDashboardEpiCurve=Adverse Events Type Chart +headingAefiDashboardMap=Adverse Events Status Map +infoHeadingAefiDashboardMap=Adverse events are shown using the GPS coordinate of the facility or person's home address. + headingSpecailCaseAccess = Grant special access headingCreateSpecailCaseAccess = Create new special access headingEditSpecailCaseAccess = Edit special access diff --git a/sormas-api/src/main/resources/validations.properties b/sormas-api/src/main/resources/validations.properties index 2e875813934..ac49b31d7e3 100644 --- a/sormas-api/src/main/resources/validations.properties +++ b/sormas-api/src/main/resources/validations.properties @@ -297,6 +297,9 @@ customizableEnumValueEmptyTranslations = Please select languages and enter capti customizableEnumValueDuplicateLanguage = Please only add one translation per language. customizableEnumValueDuplicateValue = The value %s already exists for data type %s. Enum values have to be unique. attachedDocumentNotRelatedToEntity=The attached document is not related to the entity. +aefiWithoutSuspectVaccine=You have to select at least one suspect vaccine +aefiWithoutPrimarySuspectVaccine=You have to select the primary suspect vaccine +aefiWithoutAdverseEvents=You have to select at least one adverse event invalidNationalHealthId=This value does not seem to be a correct national health ID invalidSelfReportType = Invalid SelfReport type selfReportAlreadyProcessedError = The self report was processed by another user in the meantime. diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AdverseEventsMapper.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AdverseEventsMapper.java new file mode 100644 index 00000000000..24b4abb88ea --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AdverseEventsMapper.java @@ -0,0 +1,82 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import javax.ejb.LocalBean; +import javax.ejb.Stateless; +import javax.validation.constraints.NotNull; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents; +import de.symeda.sormas.backend.util.DtoHelper; + +@LocalBean +@Stateless(name = "AdverseEventsMapper") +public class AdverseEventsMapper { + + public static AdverseEventsDto toDto(AdverseEvents source) { + + if (source == null) { + return null; + } + + AdverseEventsDto target = new AdverseEventsDto(); + DtoHelper.fillDto(target, source); + + target.setSevereLocalReaction(source.getSevereLocalReaction()); + target.setSevereLocalReactionMoreThanThreeDays(source.isSevereLocalReactionMoreThanThreeDays()); + target.setSevereLocalReactionBeyondNearestJoint(source.isSevereLocalReactionBeyondNearestJoint()); + target.setSeizures(source.getSeizures()); + target.setSeizureType(source.getSeizureType()); + target.setAbscess(source.getAbscess()); + target.setSepsis(source.getSepsis()); + target.setEncephalopathy(source.getEncephalopathy()); + target.setToxicShockSyndrome(source.getToxicShockSyndrome()); + target.setThrombocytopenia(source.getThrombocytopenia()); + target.setAnaphylaxis(source.getAnaphylaxis()); + target.setFeverishFeeling(source.getFeverishFeeling()); + target.setOtherAdverseEventDetails(source.getOtherAdverseEventDetails()); + + return target; + } + + public AdverseEvents fillOrBuildEntity(@NotNull AdverseEventsDto source, AdverseEvents target, boolean checkChangeDate) { + if (source == null) { + return null; + } + + target = DtoHelper.fillOrBuildEntity(source, target, AdverseEvents::new, checkChangeDate); + + target.setSevereLocalReaction(source.getSevereLocalReaction()); + target.setSevereLocalReactionMoreThanThreeDays(source.isSevereLocalReactionMoreThanThreeDays()); + target.setSevereLocalReactionBeyondNearestJoint(source.isSevereLocalReactionBeyondNearestJoint()); + target.setSeizures(source.getSeizures()); + target.setSeizureType(source.getSeizureType()); + target.setAbscess(source.getAbscess()); + target.setSepsis(source.getSepsis()); + target.setEncephalopathy(source.getEncephalopathy()); + target.setToxicShockSyndrome(source.getToxicShockSyndrome()); + target.setThrombocytopenia(source.getThrombocytopenia()); + target.setAnaphylaxis(source.getAnaphylaxis()); + target.setFeverishFeeling(source.getFeverishFeeling()); + target.setOtherAdverseEventDetails(source.getOtherAdverseEventDetails()); + + return target; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java new file mode 100644 index 00000000000..0d91053215e --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java @@ -0,0 +1,455 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; +import javax.inject.Inject; +import javax.validation.Valid; +import javax.validation.constraints.NotNull; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiFacade; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiReferenceDto; +import de.symeda.sormas.api.common.DeletableEntityType; +import de.symeda.sormas.api.common.DeletionDetails; +import de.symeda.sormas.api.common.progress.ProcessedEntity; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.api.utils.AccessDeniedException; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.api.utils.SortProperty; +import de.symeda.sormas.api.utils.ValidationRuntimeException; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.backend.FacadeHelper; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.common.AbstractCoreFacadeEjb; +import de.symeda.sormas.backend.immunization.ImmunizationFacadeEjb; +import de.symeda.sormas.backend.immunization.ImmunizationFacadeEjb.ImmunizationFacadeEjbLocal; +import de.symeda.sormas.backend.immunization.ImmunizationService; +import de.symeda.sormas.backend.infrastructure.community.CommunityFacadeEjb; +import de.symeda.sormas.backend.infrastructure.community.CommunityService; +import de.symeda.sormas.backend.infrastructure.country.CountryFacadeEjb; +import de.symeda.sormas.backend.infrastructure.country.CountryService; +import de.symeda.sormas.backend.infrastructure.district.DistrictFacadeEjb; +import de.symeda.sormas.backend.infrastructure.district.DistrictService; +import de.symeda.sormas.backend.infrastructure.facility.FacilityFacadeEjb; +import de.symeda.sormas.backend.infrastructure.facility.FacilityService; +import de.symeda.sormas.backend.infrastructure.region.RegionFacadeEjb; +import de.symeda.sormas.backend.infrastructure.region.RegionService; +import de.symeda.sormas.backend.location.LocationFacadeEjb; +import de.symeda.sormas.backend.location.LocationFacadeEjb.LocationFacadeEjbLocal; +import de.symeda.sormas.backend.person.PersonFacadeEjb; +import de.symeda.sormas.backend.person.PersonService; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.user.UserFacadeEjb; +import de.symeda.sormas.backend.util.DtoHelper; +import de.symeda.sormas.backend.util.Pseudonymizer; +import de.symeda.sormas.backend.util.RightsAllowed; +import de.symeda.sormas.backend.vaccination.Vaccination; +import de.symeda.sormas.backend.vaccination.VaccinationFacadeEjb; +import de.symeda.sormas.backend.vaccination.VaccinationFacadeEjb.VaccinationFacadeEjbLocal; +import de.symeda.sormas.backend.vaccination.VaccinationService; + +@Stateless(name = "AefiFacade") +@RightsAllowed(UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW) +public class AefiFacadeEjb extends AbstractCoreFacadeEjb + implements AefiFacade { + + private final Logger logger = LoggerFactory.getLogger(AefiFacadeEjb.class); + + @EJB + private ImmunizationFacadeEjbLocal immunizationFacade; + @EJB + private ImmunizationService immunizationService; + @EJB + private PersonService personService; + @EJB + private LocationFacadeEjbLocal locationFacade; + @EJB + private VaccinationFacadeEjbLocal vaccinationFacade; + @EJB + private VaccinationService vaccinationService; + @EJB + private RegionService regionService; + @EJB + private DistrictService districtService; + @EJB + private CommunityService communityService; + @EJB + private FacilityService facilityService; + @EJB + private CountryService countryService; + @EJB + private AdverseEventsMapper adverseEventsMapper; + + public AefiFacadeEjb() { + } + + @Inject + public AefiFacadeEjb(AefiService service) { + super(Aefi.class, AefiDto.class, service); + } + + @Override + @RightsAllowed({ + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT }) + public AefiDto save(@Valid @NotNull AefiDto dto) { + return save(dto, true, true); + } + + @RightsAllowed({ + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT }) + public AefiDto save(@Valid @NotNull AefiDto dto, boolean checkChangeDate, boolean internal) { + Aefi existingAefi = service.getByUuid(dto.getUuid()); + + FacadeHelper.checkCreateAndEditRights( + existingAefi, + userService, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT); + + if (internal && existingAefi != null && !service.isEditAllowed(existingAefi)) { + throw new AccessDeniedException(I18nProperties.getString(Strings.errorAdverseEventNotEditable)); + } + + AefiDto existingDto = toDto(existingAefi); + + Pseudonymizer pseudonymizer = createPseudonymizer(existingAefi); + restorePseudonymizedDto(dto, existingDto, existingAefi, pseudonymizer); + + validate(dto); + + Aefi aefi = fillOrBuildEntity(dto, existingAefi, checkChangeDate); + + service.ensurePersisted(aefi); + + return toPseudonymizedDto(aefi, pseudonymizer); + } + + @Override + public long count(AefiCriteria criteria) { + return service.count(criteria); + } + + @Override + public List getIndexList(AefiCriteria criteria, Integer first, Integer max, List sortProperties) { + List resultsList = service.getIndexList(criteria, first, max, sortProperties); + Pseudonymizer pseudonymizer = createGenericPlaceholderPseudonymizer(); + pseudonymizer.pseudonymizeDtoCollection(AefiIndexDto.class, resultsList, AefiIndexDto::isInJurisdiction, null); + return resultsList; + } + + @Override + public List getEntriesList(AefiListCriteria criteria, Integer first, Integer max) { + Long immunizationId = immunizationService.getIdByUuid(criteria.getImmunization().getUuid()); + return service.getEntriesList(immunizationId, first, max); + } + + @Override + public void validate(AefiDto aefiDto) throws ValidationRuntimeException { + if (DateHelper.isDateAfter(aefiDto.getStartDateTime(), aefiDto.getReportDate())) { + String validationError = String.format( + I18nProperties.getValidationError(Validations.afterDate), + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, AefiDto.START_DATE_TIME), + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, AefiDto.REPORT_DATE)); + throw new ValidationRuntimeException(validationError); + } + + // Check whether any required field that does not have a not null constraint in the database is empty + if (aefiDto.getImmunization() == null) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.validImmunization)); + } + + if (aefiDto.getPrimarySuspectVaccine() == null) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.aefiWithoutPrimarySuspectVaccine)); + } + + AdverseEventsDto adverseEvents = aefiDto.getAdverseEvents(); + if (adverseEvents == null) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.aefiWithoutAdverseEvents)); + } else { + if (adverseEvents.getSevereLocalReaction() == null + && adverseEvents.getSeizures() == null + && adverseEvents.getAbscess() == null + && adverseEvents.getSepsis() == null + && adverseEvents.getEncephalopathy() == null + && adverseEvents.getToxicShockSyndrome() == null + && adverseEvents.getThrombocytopenia() == null + && adverseEvents.getAnaphylaxis() == null + && adverseEvents.getFeverishFeeling() == null + && StringUtils.isBlank(adverseEvents.getOtherAdverseEventDetails())) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.aefiWithoutAdverseEvents)); + } + + boolean adverseEventSelected = adverseEvents.getSevereLocalReaction() == AdverseEventState.YES + || adverseEvents.getSeizures() == AdverseEventState.YES + || adverseEvents.getAbscess() == AdverseEventState.YES + || adverseEvents.getSepsis() == AdverseEventState.YES + || adverseEvents.getEncephalopathy() == AdverseEventState.YES + || adverseEvents.getToxicShockSyndrome() == AdverseEventState.YES + || adverseEvents.getThrombocytopenia() == AdverseEventState.YES + || adverseEvents.getAnaphylaxis() == AdverseEventState.YES + || adverseEvents.getFeverishFeeling() == AdverseEventState.YES + || !StringUtils.isBlank(adverseEvents.getOtherAdverseEventDetails()); + + if (!adverseEventSelected) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.aefiWithoutAdverseEvents)); + } + } + + if (aefiDto.getReportingUser() == null && !aefiDto.isPseudonymized()) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.validReportingUser)); + } + } + + @Override + public List getArchivedUuidsSince(Date since) { + return null; + } + + @Override + public List delete(List uuids, DeletionDetails deletionDetails) { + return null; + } + + @Override + public List restore(List uuids) { + return null; + } + + @Override + protected Aefi fillOrBuildEntity(AefiDto source, Aefi target, boolean checkChangeDate) { + return fillOrBuildEntity(source, target, checkChangeDate, false); + } + + protected Aefi fillOrBuildEntity(@NotNull AefiDto source, Aefi target, boolean checkChangeDate, boolean includeVaccinations) { + + target = DtoHelper.fillOrBuildEntity(source, target, Aefi::build, checkChangeDate); + + target.setImmunization(immunizationService.getByReferenceDto(source.getImmunization())); + target.setPerson(personService.getByReferenceDto(source.getPerson())); + target.setAddress(locationFacade.fillOrBuildEntity(source.getAddress(), target.getAddress(), checkChangeDate)); + + if (includeVaccinations) { + List vaccinationEntities = new ArrayList<>(); + for (VaccinationDto vaccinationDto : source.getVaccinations()) { + Vaccination vaccination = vaccinationService.getByUuid(vaccinationDto.getUuid()); + vaccination = vaccinationFacade.fillOrBuildEntity(vaccinationDto, vaccination, checkChangeDate); + vaccinationEntities.add(vaccination); + } + target.getVaccinations().clear(); + target.getVaccinations().addAll(vaccinationEntities); + } + + target.setPrimarySuspectVaccine(vaccinationService.getByUuid(source.getPrimarySuspectVaccine().getUuid())); + target.setAdverseEvents(adverseEventsMapper.fillOrBuildEntity(source.getAdverseEvents(), target.getAdverseEvents(), checkChangeDate)); + target.setReportDate(source.getReportDate()); + target.setReportingUser(userService.getByReferenceDto(source.getReportingUser())); + target.setExternalId(source.getExternalId()); + target.setResponsibleRegion(regionService.getByReferenceDto(source.getResponsibleRegion())); + target.setResponsibleDistrict(districtService.getByReferenceDto(source.getResponsibleDistrict())); + target.setResponsibleCommunity(communityService.getByReferenceDto(source.getResponsibleCommunity())); + target.setCountry(countryService.getByReferenceDto(source.getCountry())); + target.setReportingIdNumber(source.getReportingIdNumber()); + target.setPhoneNumber(source.getPhoneNumber()); + target.setPregnant(source.getPregnant()); + target.setTrimester(source.getTrimester()); + target.setLactating(source.getLactating()); + target.setOnsetAgeYears(source.getOnsetAgeYears()); + target.setOnsetAgeMonths(source.getOnsetAgeMonths()); + target.setOnsetAgeDays(source.getOnsetAgeDays()); + target.setAgeGroup(source.getAgeGroup()); + target.setHealthFacility(facilityService.getByReferenceDto(source.getHealthFacility())); + target.setHealthFacilityDetails(source.getHealthFacilityDetails()); + target.setReporterName(source.getReporterName()); + target.setReporterInstitution(facilityService.getByReferenceDto(source.getReporterInstitution())); + target.setReporterDesignation(source.getReporterDesignation()); + target.setReporterDepartment(source.getReporterDepartment()); + target.setReporterAddress(locationFacade.fillOrBuildEntity(source.getReporterAddress(), target.getReporterAddress(), checkChangeDate)); + target.setReporterPhone(source.getReporterPhone()); + target.setReporterEmail(source.getReporterEmail()); + target.setTodaysDate(source.getTodaysDate()); + target.setStartDateTime(source.getStartDateTime()); + target.setAefiDescription(source.getAefiDescription()); + target.setSerious(source.getSerious()); + target.setSeriousReason(source.getSeriousReason()); + target.setSeriousReasonDetails(source.getSeriousReasonDetails()); + target.setOutcome(source.getOutcome()); + target.setDeathDate(source.getDeathDate()); + target.setAutopsyDone(source.getAutopsyDone()); + target.setPastMedicalHistory(source.getPastMedicalHistory()); + target.setInvestigationNeeded(source.getInvestigationNeeded()); + target.setInvestigationPlannedDate(source.getInvestigationPlannedDate()); + target.setReceivedAtNationalLevelDate(source.getReceivedAtNationalLevelDate()); + target.setWorldwideId(source.getWorldwideId()); + target.setNationalLevelComment(source.getNationalLevelComment()); + target.setArchived(source.isArchived()); + target.setDeleted(source.isDeleted()); + target.setDeletionReason(source.getDeletionReason()); + target.setOtherDeletionReason(source.getOtherDeletionReason()); + + return target; + } + + @Override + protected AefiDto toDto(Aefi entity) { + return toAefiDto(entity); + } + + public static AefiDto toAefiDto(Aefi entity) { + + if (entity == null) { + return null; + } + AefiDto dto = new AefiDto(); + DtoHelper.fillDto(dto, entity); + + dto.setImmunization(ImmunizationFacadeEjb.toReferenceDto(entity.getImmunization())); + dto.setPerson(PersonFacadeEjb.toReferenceDto(entity.getPerson())); + dto.setAddress(LocationFacadeEjb.toDto(entity.getAddress())); + + List vaccinationDtos = new ArrayList<>(); + for (Vaccination vaccination : entity.getImmunization().getVaccinations()) { + VaccinationDto vaccinationDto = VaccinationFacadeEjb.toVaccinationDto(vaccination); + vaccinationDtos.add(vaccinationDto); + } + dto.setVaccinations(vaccinationDtos); + + dto.setPrimarySuspectVaccine(VaccinationFacadeEjb.toVaccinationDto(entity.getPrimarySuspectVaccine())); + dto.setAdverseEvents(AdverseEventsMapper.toDto(entity.getAdverseEvents())); + dto.setReportDate(entity.getReportDate()); + dto.setReportingUser(UserFacadeEjb.toReferenceDto(entity.getReportingUser())); + dto.setExternalId(entity.getExternalId()); + dto.setResponsibleRegion(RegionFacadeEjb.toReferenceDto(entity.getResponsibleRegion())); + dto.setResponsibleDistrict(DistrictFacadeEjb.toReferenceDto(entity.getResponsibleDistrict())); + dto.setResponsibleCommunity(CommunityFacadeEjb.toReferenceDto(entity.getResponsibleCommunity())); + dto.setCountry(CountryFacadeEjb.toReferenceDto(entity.getCountry())); + dto.setReportingIdNumber(entity.getReportingIdNumber()); + dto.setPhoneNumber(entity.getPhoneNumber()); + dto.setPregnant(entity.getPregnant()); + dto.setTrimester(entity.getTrimester()); + dto.setLactating(entity.getLactating()); + dto.setOnsetAgeYears(entity.getOnsetAgeYears()); + dto.setOnsetAgeMonths(entity.getOnsetAgeMonths()); + dto.setOnsetAgeDays(entity.getOnsetAgeDays()); + dto.setAgeGroup(entity.getAgeGroup()); + dto.setHealthFacility(FacilityFacadeEjb.toReferenceDto(entity.getHealthFacility())); + dto.setHealthFacilityDetails(entity.getHealthFacilityDetails()); + dto.setReporterName(entity.getReporterName()); + dto.setReporterInstitution(FacilityFacadeEjb.toReferenceDto(entity.getReporterInstitution())); + dto.setReporterDesignation(entity.getReporterDesignation()); + dto.setReporterDepartment(entity.getReporterDepartment()); + dto.setReporterAddress(LocationFacadeEjb.toDto(entity.getReporterAddress())); + dto.setReporterPhone(entity.getReporterPhone()); + dto.setReporterEmail(entity.getReporterEmail()); + dto.setTodaysDate(entity.getTodaysDate()); + dto.setStartDateTime(entity.getStartDateTime()); + dto.setAefiDescription(entity.getAefiDescription()); + dto.setSerious(entity.getSerious()); + dto.setSeriousReason(entity.getSeriousReason()); + dto.setSeriousReasonDetails(entity.getSeriousReasonDetails()); + dto.setOutcome(entity.getOutcome()); + dto.setDeathDate(entity.getDeathDate()); + dto.setAutopsyDone(entity.getAutopsyDone()); + dto.setPastMedicalHistory(entity.getPastMedicalHistory()); + dto.setInvestigationNeeded(entity.getInvestigationNeeded()); + dto.setInvestigationPlannedDate(entity.getInvestigationPlannedDate()); + dto.setReceivedAtNationalLevelDate(entity.getReceivedAtNationalLevelDate()); + dto.setWorldwideId(entity.getWorldwideId()); + dto.setNationalLevelComment(entity.getNationalLevelComment()); + dto.setArchived(entity.isArchived()); + dto.setDeleted(entity.isDeleted()); + dto.setDeletionReason(entity.getDeletionReason()); + dto.setOtherDeletionReason(entity.getOtherDeletionReason()); + + return dto; + } + + @Override + protected AefiReferenceDto toRefDto(Aefi aefi) { + return toReferenceDto(aefi); + } + + public static AefiReferenceDto toReferenceDto(Aefi entity) { + + if (entity == null) { + return null; + } + + return new AefiReferenceDto(entity.getUuid(), "", ""); + } + + @Override + protected void pseudonymizeDto(Aefi source, AefiDto dto, Pseudonymizer pseudonymizer, boolean inJurisdiction) { + + if (dto != null) { + pseudonymizer.pseudonymizeDto(AefiDto.class, dto, inJurisdiction, c -> { + pseudonymizer.pseudonymizeUser(source.getReportingUser(), userService.getCurrentUser(), dto::setReportingUser, dto); + }); + } + } + + @Override + protected void restorePseudonymizedDto(AefiDto dto, AefiDto existingDto, Aefi entity, Pseudonymizer pseudonymizer) { + + if (existingDto != null) { + final boolean inJurisdiction = service.inJurisdictionOrOwned(entity); + final User currentUser = userService.getCurrentUser(); + pseudonymizer.restoreUser(entity.getReportingUser(), currentUser, dto, dto::setReportingUser); + pseudonymizer.restorePseudonymizedValues(AefiDto.class, dto, existingDto, inJurisdiction); + } + } + + @Override + protected DeletableEntityType getDeletableEntityType() { + return null; + } + + @LocalBean + @Stateless + public static class AefiFacadeEjbLocal extends AefiFacadeEjb { + + public AefiFacadeEjbLocal() { + super(); + } + + @Inject + public AefiFacadeEjbLocal(AefiService service) { + super(service); + } + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java new file mode 100644 index 00000000000..215572a9d48 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java @@ -0,0 +1,108 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import java.util.List; + +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.Path; +import javax.persistence.criteria.Predicate; + +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiJoins; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.util.PredicateJurisdictionValidator; + +public class AefiJurisdictionPredicateValidator extends PredicateJurisdictionValidator { + + private final AefiJoins joins; + + public AefiJurisdictionPredicateValidator( + CriteriaBuilder cb, + AefiJoins joins, + User user, + List jurisdictionValidators) { + super(cb, user, null, jurisdictionValidators); + this.joins = joins; + } + + public static AefiJurisdictionPredicateValidator of(AefiQueryContext qc, User user) { + return new AefiJurisdictionPredicateValidator(qc.getCriteriaBuilder(), qc.getJoins(), user, null); + } + + @Override + public Predicate isRootInJurisdiction() { + return isInJurisdictionByJurisdictionLevel(user.getJurisdictionLevel()); + } + + @Override + public Predicate isRootInJurisdictionOrOwned() { + final Path reportingUserPath = joins.getRoot().get(Immunization.REPORTING_USER); + final Predicate reportedByCurrentUser = cb.and(cb.isNotNull(reportingUserPath), cb.equal(reportingUserPath.get(User.ID), user.getId())); + return cb.or(reportedByCurrentUser, this.isRootInJurisdiction()); + } + + @Override + public Predicate isRootInJurisdictionForRestrictedAccess() { + return null; + } + + @Override + protected Predicate whenNotAllowed() { + return cb.disjunction(); + } + + @Override + protected Predicate whenNationalLevel() { + return cb.conjunction(); + } + + @Override + protected Predicate whenRegionalLevel() { + return cb.equal(joins.getRoot().get(Aefi.IMMUNIZATION).get(Immunization.RESPONSIBLE_REGION).get(Region.ID), user.getRegion().getId()); + } + + @Override + protected Predicate whenDistrictLevel() { + return cb.equal(joins.getRoot().get(Aefi.IMMUNIZATION).get(Immunization.RESPONSIBLE_DISTRICT).get(District.ID), user.getDistrict().getId()); + } + + @Override + protected Predicate whenCommunityLevel() { + return cb + .equal(joins.getRoot().get(Aefi.IMMUNIZATION).get(Immunization.RESPONSIBLE_COMMUNITY).get(Community.ID), user.getCommunity().getId()); + } + + @Override + protected Predicate whenFacilityLevel() { + return cb.equal(joins.getRoot().get(Aefi.IMMUNIZATION).get(Immunization.HEALTH_FACILITY).get(Facility.ID), user.getHealthFacility().getId()); + } + + @Override + protected Predicate whenPointOfEntryLevel() { + return cb.disjunction(); + } + + @Override + protected Predicate whenLaboratoryLevel() { + return cb.disjunction(); + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiQueryContext.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiQueryContext.java new file mode 100644 index 00000000000..fa7ec727d58 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiQueryContext.java @@ -0,0 +1,45 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.Expression; +import javax.persistence.criteria.From; + +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiJoins; +import de.symeda.sormas.backend.common.QueryContext; + +public class AefiQueryContext extends QueryContext { + + public AefiQueryContext(CriteriaBuilder cb, CriteriaQuery query, From root) { + super(cb, query, root, new AefiJoins(root)); + } + + public AefiQueryContext(CriteriaBuilder cb, CriteriaQuery query, AefiJoins joins) { + super(cb, query, joins.getRoot(), joins); + } + + public AefiQueryContext(CriteriaBuilder cb, CriteriaQuery query, From root, AefiJoins joins) { + super(cb, query, root, joins); + } + + @Override + protected Expression createExpression(String name) { + return null; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java new file mode 100644 index 00000000000..5d18a99795b --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java @@ -0,0 +1,521 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import static de.symeda.sormas.backend.common.CriteriaBuilderHelper.andEquals; + +import java.util.ArrayList; +import java.util.List; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; +import javax.persistence.Tuple; +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.Expression; +import javax.persistence.criteria.From; +import javax.persistence.criteria.Join; +import javax.persistence.criteria.Order; +import javax.persistence.criteria.Path; +import javax.persistence.criteria.Predicate; +import javax.persistence.criteria.Root; +import javax.persistence.criteria.Selection; + +import org.apache.commons.collections4.CollectionUtils; + +import de.symeda.sormas.api.EntityRelevanceStatus; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.common.DeletableEntityType; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.feature.FeatureTypeProperty; +import de.symeda.sormas.api.person.PersonIndexDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.SortProperty; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiJoins; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers.AefiIndexDtoResultTransformer; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers.AefiListEntryDtoResultTransformer; +import de.symeda.sormas.backend.common.AbstractCoreAdoService; +import de.symeda.sormas.backend.common.CriteriaBuilderHelper; +import de.symeda.sormas.backend.feature.FeatureConfigurationFacadeEjb; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.person.Person; +import de.symeda.sormas.backend.person.PersonJoins; +import de.symeda.sormas.backend.person.PersonJurisdictionPredicateValidator; +import de.symeda.sormas.backend.person.PersonQueryContext; +import de.symeda.sormas.backend.person.PersonService; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.user.UserService; +import de.symeda.sormas.backend.util.IterableHelper; +import de.symeda.sormas.backend.util.JurisdictionHelper; +import de.symeda.sormas.backend.util.ModelConstants; +import de.symeda.sormas.backend.util.QueryHelper; +import de.symeda.sormas.backend.vaccination.Vaccination; + +@Stateless +@LocalBean +public class AefiService extends AbstractCoreAdoService { + + @EJB + private PersonService personService; + @EJB + private UserService userService; + @EJB + private FeatureConfigurationFacadeEjb.FeatureConfigurationFacadeEjbLocal featureConfigurationFacade; + + public AefiService() { + super(Aefi.class, DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION); + } + + public List getEntriesList(Long immunizationId, Integer first, Integer max) { + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Tuple.class); + final Root aefi = cq.from(Aefi.class); + + AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefi); + AefiJoins joins = aefiQueryContext.getJoins(); + + Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + Join adverseEvents = joins.getAdverseEvents(); + + cq.multiselect( + aefi.get(Aefi.UUID), + aefi.get(Aefi.SERIOUS), + primarySuspectVaccine.get(Vaccination.VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.VACCINE_DOSE), + primarySuspectVaccine.get(Vaccination.VACCINATION_DATE), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT), + adverseEvents.get(AdverseEvents.SEIZURES), + adverseEvents.get(AdverseEvents.SEIZURE_TYPE), + adverseEvents.get(AdverseEvents.ABSCESS), + adverseEvents.get(AdverseEvents.SEPSIS), + adverseEvents.get(AdverseEvents.ENCEPHALOPATHY), + adverseEvents.get(AdverseEvents.TOXIC_SHOCK_SYNDROME), + adverseEvents.get(AdverseEvents.THROMBOCYTOPENIA), + adverseEvents.get(AdverseEvents.ANAPHYLAXIS), + adverseEvents.get(AdverseEvents.FEVERISH_FEELING), + adverseEvents.get(AdverseEvents.OTHER_ADVERSE_EVENT_DETAILS), + aefi.get(Aefi.CHANGE_DATE), + JurisdictionHelper.booleanSelector(cb, isInJurisdictionOrOwned(aefiQueryContext))); + + final Predicate criteriaFilter = buildCriteriaFilter(immunizationId, aefiQueryContext); + if (criteriaFilter != null) { + cq.where(criteriaFilter); + } + + cq.orderBy(cb.desc(aefi.get(Aefi.CHANGE_DATE))); + + cq.distinct(true); + + return QueryHelper.getResultList(em, cq, new AefiListEntryDtoResultTransformer(), first, max); + } + + private Predicate buildCriteriaFilter(Long immunizationId, AefiQueryContext aefiQueryContext) { + + final CriteriaBuilder cb = aefiQueryContext.getCriteriaBuilder(); + final From from = aefiQueryContext.getRoot(); + + Predicate filter = cb.equal(from.get(Aefi.IMMUNIZATION_ID), immunizationId); + + filter = CriteriaBuilderHelper.and(cb, filter, cb.isFalse(from.get(Aefi.DELETED))); + + return filter; + } + + public long count(AefiCriteria criteria) { + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Long.class); + final Root aefi = cq.from(Aefi.class); + + AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefi); + + buildWhereCondition(criteria, cb, cq, aefiQueryContext, null); + + cq.select(cb.countDistinct(aefi)); + return em.createQuery(cq).getSingleResult(); + } + + public List getIndexList(AefiCriteria criteria, Integer first, Integer max, List sortProperties) { + + List indexListIds = getIndexListIds(criteria, first, max, sortProperties); + List aefiIndexDtos = new ArrayList<>(); + + IterableHelper.executeBatched(indexListIds, ModelConstants.PARAMETER_LIMIT, batchedIds -> { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Tuple.class); + final Root aefi = cq.from(Aefi.class); + + AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefi); + AefiJoins joins = aefiQueryContext.getJoins(); + + final Join immunization = joins.getImmunization(); + final Join person = joins.getImmunizationJoins().getPerson(); + + final Join responsibleRegion = joins.getImmunizationJoins().getResponsibleRegion(); + final Join responsibleDistrict = joins.getImmunizationJoins().getResponsibleDistrict(); + + final Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + final Join adverseEvents = joins.getAdverseEvents(); + + cq.multiselect( + Stream + .concat( + Stream.of( + aefi.get(Aefi.UUID), + immunization.get(Immunization.UUID), + immunization.get(Immunization.DISEASE), + person.get(Person.UUID), + person.get(Person.FIRST_NAME), + person.get(Person.LAST_NAME), + person.get(Person.APPROXIMATE_AGE), + person.get(Person.APPROXIMATE_AGE_TYPE), + person.get(Person.BIRTHDATE_DD), + person.get(Person.BIRTHDATE_MM), + person.get(Person.BIRTHDATE_YYYY), + person.get(Person.SEX), + responsibleRegion.get(Region.NAME), + responsibleDistrict.get(District.NAME), + aefi.get(Aefi.SERIOUS), + primarySuspectVaccine.get(Vaccination.VACCINE_NAME), + aefi.get(Aefi.OUTCOME), + primarySuspectVaccine.get(Vaccination.VACCINATION_DATE), + aefi.get(Aefi.REPORT_DATE), + aefi.get(Aefi.START_DATE_TIME), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT), + adverseEvents.get(AdverseEvents.SEIZURES), + adverseEvents.get(AdverseEvents.SEIZURE_TYPE), + adverseEvents.get(AdverseEvents.ABSCESS), + adverseEvents.get(AdverseEvents.SEPSIS), + adverseEvents.get(AdverseEvents.ENCEPHALOPATHY), + adverseEvents.get(AdverseEvents.TOXIC_SHOCK_SYNDROME), + adverseEvents.get(AdverseEvents.THROMBOCYTOPENIA), + adverseEvents.get(AdverseEvents.ANAPHYLAXIS), + adverseEvents.get(AdverseEvents.FEVERISH_FEELING), + adverseEvents.get(AdverseEvents.OTHER_ADVERSE_EVENT_DETAILS), + aefi.get(Aefi.DELETION_REASON), + aefi.get(Aefi.OTHER_DELETION_REASON), + JurisdictionHelper.booleanSelector(cb, isInJurisdictionOrOwned(aefiQueryContext)), + aefi.get(Aefi.CHANGE_DATE)), + // add sort properties to select + sortBy(sortProperties, aefiQueryContext).stream()) + .collect(Collectors.toList())); + + buildWhereCondition(criteria, cb, cq, aefiQueryContext, aefi.get(Aefi.ID).in(batchedIds)); + cq.distinct(true); + + aefiIndexDtos.addAll(QueryHelper.getResultList(em, cq, new AefiIndexDtoResultTransformer(), null, null)); + }); + + return aefiIndexDtos; + } + + private List getIndexListIds(AefiCriteria criteria, Integer first, Integer max, List sortProperties) { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createTupleQuery(); + final Root aefi = cq.from(Aefi.class); + + AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefi); + + List> selections = new ArrayList<>(); + selections.add(aefi.get(Aefi.ID)); + selections.addAll(sortBy(sortProperties, aefiQueryContext)); + + cq.multiselect(selections); + + buildWhereCondition(criteria, cb, cq, aefiQueryContext, null); + cq.distinct(true); + + List aefiResultList = QueryHelper.getResultList(em, cq, first, max); + return aefiResultList.stream().map(t -> t.get(0, Long.class)).collect(Collectors.toList()); + } + + private List> sortBy(List sortProperties, AefiQueryContext aefiQueryContext) { + + List> selections = new ArrayList<>(); + CriteriaBuilder cb = aefiQueryContext.getCriteriaBuilder(); + CriteriaQuery cq = aefiQueryContext.getQuery(); + + if (CollectionUtils.isNotEmpty(sortProperties)) { + List order = new ArrayList<>(sortProperties.size()); + for (SortProperty sortProperty : sortProperties) { + Expression expression; + switch (sortProperty.propertyName) { + case AefiIndexDto.UUID: + case AefiIndexDto.DISEASE: + case AefiIndexDto.START_DATE_TIME: + expression = aefiQueryContext.getRoot().get(sortProperty.propertyName); + break; + case AefiIndexDto.PERSON_UUID: + expression = aefiQueryContext.getJoins().getImmunizationJoins().getPerson().get(Person.UUID); + break; + case AefiIndexDto.PERSON_FIRST_NAME: + expression = cb.lower(aefiQueryContext.getJoins().getImmunizationJoins().getPerson().get(Person.FIRST_NAME)); + break; + case AefiIndexDto.PERSON_LAST_NAME: + expression = cb.lower(aefiQueryContext.getJoins().getImmunizationJoins().getPerson().get(Person.LAST_NAME)); + break; + case AefiIndexDto.AGE_AND_BIRTH_DATE: + expression = aefiQueryContext.getJoins().getImmunizationJoins().getPerson().get(Person.APPROXIMATE_AGE); + break; + case AefiIndexDto.SEX: + expression = aefiQueryContext.getJoins().getImmunizationJoins().getPerson().get(Person.SEX); + break; + case AefiIndexDto.REGION: + expression = + cb.lower(aefiQueryContext.getJoins().getImmunizationJoins().getPersonJoins().getAddressJoins().getRegion().get(Region.NAME)); + break; + case AefiIndexDto.DISTRICT: + expression = cb.lower( + aefiQueryContext.getJoins().getImmunizationJoins().getPersonJoins().getAddressJoins().getDistrict().get(District.NAME)); + break; + case AefiIndexDto.PRIMARY_VACCINE_NAME: + expression = aefiQueryContext.getJoins().getPrimarySuspectVaccination().get(Vaccination.VACCINE_NAME); + break; + case AefiIndexDto.VACCINATION_DATE: + expression = aefiQueryContext.getJoins().getPrimarySuspectVaccination().get(Vaccination.VACCINATION_DATE); + break; + case AefiIndexDto.SERIOUS: + expression = aefiQueryContext.getRoot().get(Aefi.SERIOUS); + break; + case AefiIndexDto.OUTCOME: + expression = aefiQueryContext.getRoot().get(Aefi.OUTCOME); + break; + default: + throw new IllegalArgumentException(sortProperty.propertyName); + } + order.add(sortProperty.ascending ? cb.asc(expression) : cb.desc(expression)); + selections.add(expression); + } + cq.orderBy(order); + } else { + Path changeDate = aefiQueryContext.getRoot().get(Aefi.CHANGE_DATE); + cq.orderBy(cb.desc(changeDate)); + selections.add(changeDate); + } + + return selections; + } + + private void buildWhereCondition( + AefiCriteria criteria, + CriteriaBuilder cb, + CriteriaQuery cq, + AefiQueryContext aefiQueryContext, + Predicate additionalFilter) { + + Predicate filter = createUserFilter(aefiQueryContext); + if (additionalFilter != null) { + filter = CriteriaBuilderHelper.and(cb, additionalFilter, filter); + } + + if (criteria != null) { + final Predicate criteriaFilter = buildCriteriaFilter(criteria, aefiQueryContext); + filter = CriteriaBuilderHelper.and(cb, filter, criteriaFilter); + } + + if (filter != null) { + cq.where(filter); + } + } + + public Predicate buildCriteriaFilter(AefiCriteria criteria, AefiQueryContext aefiQueryContext) { + + final AefiJoins joins = aefiQueryContext.getJoins(); + final CriteriaBuilder cb = aefiQueryContext.getCriteriaBuilder(); + final From from = aefiQueryContext.getRoot(); + final Join immunization = joins.getImmunization(); + final Join person = joins.getImmunizationJoins().getPerson(); + final Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + + final Join location = joins.getImmunizationJoins().getPersonJoins().getAddress(); + + Predicate filter = null; + if (criteria.getDisease() != null) { + filter = CriteriaBuilderHelper.and(cb, null, cb.equal(immunization.get(Immunization.DISEASE), criteria.getDisease())); + } + + if (!DataHelper.isNullOrEmpty(criteria.getNameAddressPhoneEmailLike())) { + final CriteriaQuery cq = cb.createQuery(PersonIndexDto.class); + final PersonQueryContext personQueryContext = new PersonQueryContext(cb, cq, joins.getImmunizationJoins().getPersonJoins()); + + String[] textFilters = criteria.getNameAddressPhoneEmailLike().split("\\s+"); + + for (String textFilter : textFilters) { + if (DataHelper.isNullOrEmpty(textFilter)) { + continue; + } + + Predicate likeFilters = cb.or( + CriteriaBuilderHelper.ilike(cb, from.get(Aefi.UUID), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, person.get(Person.FIRST_NAME), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, person.get(Person.LAST_NAME), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.UUID), textFilter), + CriteriaBuilderHelper.ilike(cb, personQueryContext.getSubqueryExpression(PersonQueryContext.PERSON_EMAIL_SUBQUERY), textFilter), + phoneNumberPredicate(cb, personQueryContext.getSubqueryExpression(PersonQueryContext.PERSON_PHONE_SUBQUERY), textFilter), + CriteriaBuilderHelper + .ilike(cb, personQueryContext.getSubqueryExpression(PersonQueryContext.PERSON_PRIMARY_OTHER_SUBQUERY), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, location.get(Location.STREET), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, location.get(Location.CITY), textFilter), + CriteriaBuilderHelper.ilike(cb, location.get(Location.POSTAL_CODE), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.INTERNAL_TOKEN), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.EXTERNAL_ID), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.EXTERNAL_TOKEN), textFilter)); + filter = CriteriaBuilderHelper.and(cb, filter, likeFilters); + } + } + if (criteria.getAefiType() != null) { + if (criteria.getAefiType() == AefiType.SERIOUS) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Aefi.SERIOUS), YesNoUnknown.YES)); + } else { + filter = CriteriaBuilderHelper.and(cb, filter, cb.notEqual(from.get(Aefi.SERIOUS), YesNoUnknown.YES)); + } + } + if (criteria.getOutcome() != null) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Aefi.OUTCOME), criteria.getOutcome())); + } + if (criteria.getVaccineName() != null) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(primarySuspectVaccine.get(Vaccination.VACCINE_NAME), criteria.getVaccineName())); + } + if (criteria.getVaccineManufacturer() != null) { + filter = CriteriaBuilderHelper + .and(cb, filter, cb.equal(primarySuspectVaccine.get(Vaccination.VACCINE_MANUFACTURER), criteria.getVaccineManufacturer())); + } + filter = andEquals(cb, () -> joins.getImmunizationJoins().getResponsibleRegion(), filter, criteria.getRegion()); + filter = andEquals(cb, () -> joins.getImmunizationJoins().getResponsibleDistrict(), filter, criteria.getDistrict()); + filter = andEquals(cb, () -> joins.getImmunizationJoins().getResponsibleCommunity(), filter, criteria.getCommunity()); + if (criteria.getFacilityType() != null) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(immunization.get(Immunization.FACILITY_TYPE), criteria.getFacilityType())); + } + filter = andEquals(cb, () -> joins.getImmunizationJoins().getHealthFacility(), filter, criteria.getHealthFacility()); + if (criteria.getAefiDateType() != null) { + Path path = buildPathForDateFilter(criteria.getAefiDateType(), aefiQueryContext); + if (path != null) { + filter = CriteriaBuilderHelper.applyDateFilter(cb, filter, path, criteria.getFromDate(), criteria.getToDate()); + } + } + + if (criteria.getRelevanceStatus() != null) { + if (criteria.getRelevanceStatus() == EntityRelevanceStatus.ACTIVE) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.or(cb.equal(from.get(Aefi.ARCHIVED), false), cb.isNull(from.get(Aefi.ARCHIVED)))); + } else if (criteria.getRelevanceStatus() == EntityRelevanceStatus.ARCHIVED) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Aefi.ARCHIVED), true)); + } else if (criteria.getRelevanceStatus() == EntityRelevanceStatus.DELETED) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Aefi.DELETED), true)); + } + } + if (criteria.getRelevanceStatus() != EntityRelevanceStatus.DELETED) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.isFalse(from.get(Aefi.DELETED))); + } + + return filter; + } + + private Path buildPathForDateFilter(AefiDateType aefiDateType, AefiQueryContext aefiQueryContext) { + Path path = null; + String dateField = getDateFieldFromDateType(aefiDateType); + if (dateField != null) { + if (Vaccination.VACCINATION_DATE.equals(dateField)) { + final Join primarySuspectVaccination = aefiQueryContext.getJoins().getPrimarySuspectVaccination(); + path = primarySuspectVaccination.get(Vaccination.VACCINATION_DATE); + } else { + path = aefiQueryContext.getRoot().get(dateField); + } + } + return path; + } + + private String getDateFieldFromDateType(AefiDateType aefiDateType) { + switch (aefiDateType) { + case REPORT_DATE: + return Aefi.REPORT_DATE; + case START_DATE: + return Aefi.START_DATE_TIME; + case VACCINATION_DATE: + return Vaccination.VACCINATION_DATE; + } + return null; + } + + public Predicate createUserFilter(AefiQueryContext qc) { + + User currentUser = getCurrentUser(); + if (currentUser == null) { + return null; + } + final CriteriaBuilder cb = qc.getCriteriaBuilder(); + + Predicate filter = isInJurisdictionOrOwned(qc); + + filter = CriteriaBuilderHelper.and( + cb, + filter, + CriteriaBuilderHelper.limitedDiseasePredicate(cb, currentUser, qc.getRoot().get(Aefi.IMMUNIZATION).get(Immunization.DISEASE))); + + return filter; + } + + private Predicate isInJurisdictionOrOwned(AefiQueryContext qc) { + + final User currentUser = userService.getCurrentUser(); + CriteriaBuilder cb = qc.getCriteriaBuilder(); + Predicate filter; + if (!featureConfigurationFacade + .isPropertyValueTrue(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT, FeatureTypeProperty.REDUCED)) { + filter = AefiJurisdictionPredicateValidator.of(qc, currentUser).inJurisdictionOrOwned(); + } else { + filter = CriteriaBuilderHelper.or( + cb, + cb.equal(qc.getRoot().get(Immunization.REPORTING_USER), currentUser), + PersonJurisdictionPredicateValidator + .of(qc.getQuery(), cb, new PersonJoins(qc.getJoins().getPerson()), currentUser, personService.getPermittedAssociations()) + .inJurisdictionOrOwned()); + } + return filter; + } + + @Override + protected Predicate createUserFilterInternal(CriteriaBuilder cb, CriteriaQuery cq, From from) { + return createUserFilter(new AefiQueryContext(cb, cq, from)); + } + + @Override + protected AefiJoins toJoins(From adoPath) { + return new AefiJoins(adoPath); + } + + @Override + public Predicate inJurisdictionOrOwned(CriteriaBuilder cb, CriteriaQuery query, From from) { + return isInJurisdictionOrOwned(new AefiQueryContext(cb, query, from)); + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AdverseEvents.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AdverseEvents.java new file mode 100644 index 00000000000..2513cb06548 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AdverseEvents.java @@ -0,0 +1,179 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity; + +import javax.persistence.Entity; +import javax.persistence.EnumType; +import javax.persistence.Enumerated; +import javax.persistence.Table; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeizureType; +import de.symeda.sormas.backend.common.AbstractDomainObject; + +@Entity +@Table(name = "adverseevents") +public class AdverseEvents extends AbstractDomainObject { + + private static final long serialVersionUID = 5407524640930885029L; + + public static final String TABLE_NAME = "adverseevents"; + + public static final String SEVERE_LOCAL_REACTION = "severeLocalReaction"; + public static final String SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS = "severeLocalReactionMoreThanThreeDays"; + public static final String SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT = "severeLocalReactionBeyondNearestJoint"; + public static final String SEIZURES = "seizures"; + public static final String SEIZURE_TYPE = "seizureType"; + public static final String ABSCESS = "abscess"; + public static final String SEPSIS = "sepsis"; + public static final String ENCEPHALOPATHY = "encephalopathy"; + public static final String TOXIC_SHOCK_SYNDROME = "toxicShockSyndrome"; + public static final String THROMBOCYTOPENIA = "thrombocytopenia"; + public static final String ANAPHYLAXIS = "anaphylaxis"; + public static final String FEVERISH_FEELING = "feverishFeeling"; + public static final String OTHER_ADVERSE_EVENT_DETAILS = "otherAdverseEventDetails"; + + private AdverseEventState severeLocalReaction; + private boolean severeLocalReactionMoreThanThreeDays; + private boolean severeLocalReactionBeyondNearestJoint; + private AdverseEventState seizures; + private SeizureType seizureType; + private AdverseEventState abscess; + private AdverseEventState sepsis; + private AdverseEventState encephalopathy; + private AdverseEventState toxicShockSyndrome; + private AdverseEventState thrombocytopenia; + private AdverseEventState anaphylaxis; + private AdverseEventState feverishFeeling; + private String otherAdverseEventDetails; + + @Enumerated(EnumType.STRING) + public AdverseEventState getSevereLocalReaction() { + return severeLocalReaction; + } + + public void setSevereLocalReaction(AdverseEventState severeLocalReaction) { + this.severeLocalReaction = severeLocalReaction; + } + + public boolean isSevereLocalReactionMoreThanThreeDays() { + return severeLocalReactionMoreThanThreeDays; + } + + public void setSevereLocalReactionMoreThanThreeDays(boolean severeLocalReactionMoreThanThreeDays) { + this.severeLocalReactionMoreThanThreeDays = severeLocalReactionMoreThanThreeDays; + } + + public boolean isSevereLocalReactionBeyondNearestJoint() { + return severeLocalReactionBeyondNearestJoint; + } + + public void setSevereLocalReactionBeyondNearestJoint(boolean severeLocalReactionBeyondNearestJoint) { + this.severeLocalReactionBeyondNearestJoint = severeLocalReactionBeyondNearestJoint; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getSeizures() { + return seizures; + } + + public void setSeizures(AdverseEventState seizures) { + this.seizures = seizures; + } + + @Enumerated(EnumType.STRING) + public SeizureType getSeizureType() { + return seizureType; + } + + public void setSeizureType(SeizureType seizureType) { + this.seizureType = seizureType; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getAbscess() { + return abscess; + } + + public void setAbscess(AdverseEventState abscess) { + this.abscess = abscess; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getSepsis() { + return sepsis; + } + + public void setSepsis(AdverseEventState sepsis) { + this.sepsis = sepsis; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getEncephalopathy() { + return encephalopathy; + } + + public void setEncephalopathy(AdverseEventState encephalopathy) { + this.encephalopathy = encephalopathy; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getToxicShockSyndrome() { + return toxicShockSyndrome; + } + + public void setToxicShockSyndrome(AdverseEventState toxicShockSyndrome) { + this.toxicShockSyndrome = toxicShockSyndrome; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getThrombocytopenia() { + return thrombocytopenia; + } + + public void setThrombocytopenia(AdverseEventState thrombocytopenia) { + this.thrombocytopenia = thrombocytopenia; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getAnaphylaxis() { + return anaphylaxis; + } + + public void setAnaphylaxis(AdverseEventState anaphylaxis) { + this.anaphylaxis = anaphylaxis; + } + + @Enumerated(EnumType.STRING) + public AdverseEventState getFeverishFeeling() { + return feverishFeeling; + } + + public void setFeverishFeeling(AdverseEventState feverishFeeling) { + this.feverishFeeling = feverishFeeling; + } + + public String getOtherAdverseEventDetails() { + return otherAdverseEventDetails; + } + + public void setOtherAdverseEventDetails(String otherAdverseEventDetails) { + this.otherAdverseEventDetails = otherAdverseEventDetails; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java new file mode 100644 index 00000000000..1b62575f5e4 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java @@ -0,0 +1,606 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity; + +import static de.symeda.sormas.api.utils.FieldConstraints.CHARACTER_LIMIT_DEFAULT; + +import java.util.Date; +import java.util.HashSet; +import java.util.Set; + +import javax.persistence.CascadeType; +import javax.persistence.Column; +import javax.persistence.Entity; +import javax.persistence.EnumType; +import javax.persistence.Enumerated; +import javax.persistence.FetchType; +import javax.persistence.JoinColumn; +import javax.persistence.JoinTable; +import javax.persistence.ManyToMany; +import javax.persistence.ManyToOne; +import javax.persistence.OneToOne; +import javax.persistence.Table; +import javax.persistence.Temporal; +import javax.persistence.TemporalType; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiAgeGroup; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiOutcome; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeriousAefiReason; +import de.symeda.sormas.api.caze.Trimester; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.backend.common.CoreAdo; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.country.Country; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.person.Person; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.vaccination.Vaccination; + +@Entity +@Table(name = "adverseeventsfollowingimmunization") +public class Aefi extends CoreAdo { + + private static final long serialVersionUID = -7845660472641846292L; + + public static final String TABLE_NAME = "adverseeventsfollowingimmunization"; + public static final String AEFI_VACCINATIONS_TABLE_NAME = "adverseeventsfollowingimmunization_vaccinations"; + + public static final String IMMUNIZATION = "immunization"; + public static final String IMMUNIZATION_ID = "immunizationId"; + public static final String ADDRESS = "address"; + public static final String VACCINATIONS = "vaccinations"; + public static final String PRIMARY_SUSPECT_VACCINE = "primarySuspectVaccine"; + public static final String ADVERSE_EVENTS = "adverseEvents"; + public static final String PERSON = "person"; + public static final String PERSON_ID = "personId"; + public static final String REPORT_DATE = "reportDate"; + public static final String REPORTING_USER = "reportingUser"; + public static final String EXTERNAL_ID = "externalId"; + public static final String RESPONSIBLE_REGION = "responsibleRegion"; + public static final String RESPONSIBLE_DISTRICT = "responsibleDistrict"; + public static final String RESPONSIBLE_COMMUNITY = "responsibleCommunity"; + public static final String COUNTRY = "country"; + public static final String REPORTINGID_NUMBER = "reportingIdNumber"; + public static final String PREGNANT = "pregnant"; + public static final String PHONE_NUMBER = "phoneNumber"; + public static final String TRIMESTER = "trimester"; + public static final String LACTATING = "lactating"; + public static final String ONSET_AGE_YEARS = "onsetAgeYears"; + public static final String ONSET_AGE_MONTHS = "onsetAgeMonths"; + public static final String ONSET_AGE_DAYS = "onsetAgeDays"; + public static final String AGE_GROUP = "ageGroup"; + public static final String HEALTH_FACILITY = "healthFacility"; + public static final String HEALTH_FACILITY_DETAILS = "healthFacilityDetails"; + public static final String REPORTER_NAME = "reporterName"; + public static final String REPORTER_INSTITUTION = "reporterInstitution"; + public static final String REPORTER_DESIGNATION = "reporterDesignation"; + public static final String REPORTER_DEPARTMENT = "reporterDepartment"; + public static final String REPORTER_ADDRESS = "reporterAddress"; + public static final String REPORTER_PHONE = "reporterPhone"; + public static final String REPORTER_EMAIL = "reporterEmail"; + public static final String TODAYS_DATE = "todaysDate"; + public static final String START_DATE_TIME = "startDateTime"; + public static final String AEFI_DESCRIPTION = "aefiDescription"; + public static final String SERIOUS = "serious"; + public static final String SERIOUS_REASON = "seriousReason"; + public static final String SERIOUS_REASON_DETAILS = "seriousReasonDetails"; + public static final String OUTCOME = "outcome"; + public static final String DEATH_DATE = "deathDate"; + public static final String AUTOPSY_DONE = "autopsyDone"; + public static final String PAST_MEDICAL_HISTORY = "pastMedicalHistory"; + public static final String INVESTIGATION_NEEDED = "investigationNeeded"; + public static final String INVESTIGATION_PLANNED_DATE = "investigationPlannedDate"; + public static final String RECEIVED_AT_NATIONAL_LEVEL_DATE = "receivedAtNationalLevelDate"; + public static final String WORLD_WIDE_ID = "worldwideId"; + public static final String NATIONAL_LEVEL_COMMENT = "nationalLevelComment"; + + private Immunization immunization; + private Long immunizationId; + private Person person; + private Long personId; + private Location address; + private Set vaccinations = new HashSet<>(); + private Vaccination primarySuspectVaccine; + private AdverseEvents adverseEvents; + private Date reportDate; + private User reportingUser; + private String externalId; + private Region responsibleRegion; + private District responsibleDistrict; + private Community responsibleCommunity; + private Country country; + private String reportingIdNumber; + private String phoneNumber; + private YesNoUnknown pregnant; + private Trimester trimester; + private YesNoUnknown lactating; + private Integer onsetAgeYears; + private Integer onsetAgeMonths; + private Integer onsetAgeDays; + private AefiAgeGroup ageGroup; + private Facility healthFacility; + private String healthFacilityDetails; + private String reporterName; + private Facility reporterInstitution; + private String reporterDesignation; + private String reporterDepartment; + private Location reporterAddress; + private String reporterPhone; + private String reporterEmail; + private Date todaysDate; + private Date startDateTime; + private String aefiDescription; + private YesNoUnknown serious; + private SeriousAefiReason seriousReason; + private String seriousReasonDetails; + private AefiOutcome outcome; + private Date deathDate; + private YesNoUnknown autopsyDone; + private String pastMedicalHistory; + private YesNoUnknown investigationNeeded; + private Date investigationPlannedDate; + private Date receivedAtNationalLevelDate; + private String worldwideId; + private String nationalLevelComment; + + public static Aefi build() { + Aefi aefi = new Aefi(); + return aefi; + } + + @ManyToOne + @JoinColumn(nullable = false) + public Immunization getImmunization() { + return immunization; + } + + public void setImmunization(Immunization immunization) { + this.immunization = immunization; + } + + @Column(name = "immunization_id", updatable = false, insertable = false) + public Long getImmunizationId() { + return immunizationId; + } + + public void setImmunizationId(Long immunizationId) { + this.immunizationId = immunizationId; + } + + @ManyToMany(fetch = FetchType.LAZY) + @JoinTable(name = AEFI_VACCINATIONS_TABLE_NAME, + joinColumns = @JoinColumn(name = "adverseeventsfollowingimmunization_id"), + inverseJoinColumns = @JoinColumn(name = "vaccination_id")) + public Set getVaccinations() { + return vaccinations; + } + + public void setVaccinations(Set vaccinations) { + this.vaccinations = vaccinations; + } + + @OneToOne + public Vaccination getPrimarySuspectVaccine() { + return primarySuspectVaccine; + } + + public void setPrimarySuspectVaccine(Vaccination primarySuspectVaccine) { + this.primarySuspectVaccine = primarySuspectVaccine; + } + + @OneToOne(cascade = CascadeType.ALL, fetch = FetchType.EAGER) + @JoinColumn(nullable = false) + //@ManyToOne + public AdverseEvents getAdverseEvents() { + return adverseEvents; + } + + public void setAdverseEvents(AdverseEvents adverseEvents) { + this.adverseEvents = adverseEvents; + } + + @ManyToOne + public Person getPerson() { + return person; + } + + public void setPerson(Person person) { + this.person = person; + } + + @Column(name = "person_id", updatable = false, insertable = false) + public Long getPersonId() { + return personId; + } + + public void setPersonId(Long personId) { + this.personId = personId; + } + + @ManyToOne(cascade = CascadeType.ALL, fetch = FetchType.LAZY) + public Location getAddress() { + if (address == null) { + address = new Location(); + } + return address; + } + + public void setAddress(Location address) { + this.address = address; + } + + @Temporal(TemporalType.TIMESTAMP) + @Column(nullable = false) + public Date getReportDate() { + return reportDate; + } + + public void setReportDate(Date reportDate) { + this.reportDate = reportDate; + } + + @ManyToOne(fetch = FetchType.LAZY) + @JoinColumn(nullable = false) + public User getReportingUser() { + return reportingUser; + } + + public void setReportingUser(User reportingUser) { + this.reportingUser = reportingUser; + } + + @Column(length = CHARACTER_LIMIT_DEFAULT) + public String getExternalId() { + return externalId; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Region getResponsibleRegion() { + return responsibleRegion; + } + + public void setResponsibleRegion(Region responsibleRegion) { + this.responsibleRegion = responsibleRegion; + } + + @ManyToOne(fetch = FetchType.LAZY) + public District getResponsibleDistrict() { + return responsibleDistrict; + } + + public void setResponsibleDistrict(District responsibleDistrict) { + this.responsibleDistrict = responsibleDistrict; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Community getResponsibleCommunity() { + return responsibleCommunity; + } + + public void setResponsibleCommunity(Community responsibleCommunity) { + this.responsibleCommunity = responsibleCommunity; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Country getCountry() { + return country; + } + + public void setCountry(Country country) { + this.country = country; + } + + public void setExternalId(String externalId) { + this.externalId = externalId; + } + + @Column(length = CHARACTER_LIMIT_DEFAULT) + public String getReportingIdNumber() { + return reportingIdNumber; + } + + public void setReportingIdNumber(String reportingIdNumber) { + this.reportingIdNumber = reportingIdNumber; + } + + public String getPhoneNumber() { + return phoneNumber; + } + + public void setPhoneNumber(String phoneNumber) { + this.phoneNumber = phoneNumber; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getPregnant() { + return pregnant; + } + + public void setPregnant(YesNoUnknown pregnant) { + this.pregnant = pregnant; + } + + @Enumerated(EnumType.STRING) + public Trimester getTrimester() { + return trimester; + } + + public void setTrimester(Trimester trimester) { + this.trimester = trimester; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getLactating() { + return lactating; + } + + public void setLactating(YesNoUnknown lactating) { + this.lactating = lactating; + } + + public Integer getOnsetAgeYears() { + return onsetAgeYears; + } + + public void setOnsetAgeYears(Integer onsetAgeYears) { + this.onsetAgeYears = onsetAgeYears; + } + + public Integer getOnsetAgeMonths() { + return onsetAgeMonths; + } + + public void setOnsetAgeMonths(Integer onsetAgeMonths) { + this.onsetAgeMonths = onsetAgeMonths; + } + + public Integer getOnsetAgeDays() { + return onsetAgeDays; + } + + public void setOnsetAgeDays(Integer onsetAgeDays) { + this.onsetAgeDays = onsetAgeDays; + } + + @Enumerated(EnumType.STRING) + public AefiAgeGroup getAgeGroup() { + return ageGroup; + } + + public void setAgeGroup(AefiAgeGroup ageGroup) { + this.ageGroup = ageGroup; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Facility getHealthFacility() { + return healthFacility; + } + + public void setHealthFacility(Facility healthFacility) { + this.healthFacility = healthFacility; + } + + @Column(length = CHARACTER_LIMIT_DEFAULT) + public String getHealthFacilityDetails() { + return healthFacilityDetails; + } + + public void setHealthFacilityDetails(String healthFacilityDetails) { + this.healthFacilityDetails = healthFacilityDetails; + } + + public String getReporterName() { + return reporterName; + } + + public void setReporterName(String reporterName) { + this.reporterName = reporterName; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Facility getReporterInstitution() { + return reporterInstitution; + } + + public void setReporterInstitution(Facility reporterInstitution) { + this.reporterInstitution = reporterInstitution; + } + + public String getReporterDesignation() { + return reporterDesignation; + } + + public void setReporterDesignation(String reporterDesignation) { + this.reporterDesignation = reporterDesignation; + } + + public String getReporterDepartment() { + return reporterDepartment; + } + + public void setReporterDepartment(String reporterDepartment) { + this.reporterDepartment = reporterDepartment; + } + + @ManyToOne(cascade = CascadeType.ALL, fetch = FetchType.LAZY) + @JoinColumn(name = "reporteraddress_id") + public Location getReporterAddress() { + return reporterAddress; + } + + public void setReporterAddress(Location reporterAddress) { + this.reporterAddress = reporterAddress; + } + + public String getReporterPhone() { + return reporterPhone; + } + + public void setReporterPhone(String reporterPhone) { + this.reporterPhone = reporterPhone; + } + + public String getReporterEmail() { + return reporterEmail; + } + + public void setReporterEmail(String reporterEmail) { + this.reporterEmail = reporterEmail; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getTodaysDate() { + return todaysDate; + } + + public void setTodaysDate(Date todaysDate) { + this.todaysDate = todaysDate; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getStartDateTime() { + return startDateTime; + } + + public void setStartDateTime(Date startDateTime) { + this.startDateTime = startDateTime; + } + + @Column(columnDefinition = "text") + public String getAefiDescription() { + return aefiDescription; + } + + public void setAefiDescription(String aefiDescription) { + this.aefiDescription = aefiDescription; + } + + @Enumerated(EnumType.STRING) + @Column(nullable = false) + public YesNoUnknown getSerious() { + return serious; + } + + public void setSerious(YesNoUnknown serious) { + this.serious = serious; + } + + @Enumerated(EnumType.STRING) + public SeriousAefiReason getSeriousReason() { + return seriousReason; + } + + public void setSeriousReason(SeriousAefiReason seriousReason) { + this.seriousReason = seriousReason; + } + + @Column(columnDefinition = "text") + public String getSeriousReasonDetails() { + return seriousReasonDetails; + } + + public void setSeriousReasonDetails(String seriousReasonDetails) { + this.seriousReasonDetails = seriousReasonDetails; + } + + @Enumerated(EnumType.STRING) + @Column(nullable = false) + public AefiOutcome getOutcome() { + return outcome; + } + + public void setOutcome(AefiOutcome outcome) { + this.outcome = outcome; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getDeathDate() { + return deathDate; + } + + public void setDeathDate(Date deathDate) { + this.deathDate = deathDate; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAutopsyDone() { + return autopsyDone; + } + + public void setAutopsyDone(YesNoUnknown autopsyDone) { + this.autopsyDone = autopsyDone; + } + + @Column(columnDefinition = "text") + public String getPastMedicalHistory() { + return pastMedicalHistory; + } + + public void setPastMedicalHistory(String pastMedicalHistory) { + this.pastMedicalHistory = pastMedicalHistory; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getInvestigationNeeded() { + return investigationNeeded; + } + + public void setInvestigationNeeded(YesNoUnknown investigationNeeded) { + this.investigationNeeded = investigationNeeded; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getInvestigationPlannedDate() { + return investigationPlannedDate; + } + + public void setInvestigationPlannedDate(Date investigationPlannedDate) { + this.investigationPlannedDate = investigationPlannedDate; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getReceivedAtNationalLevelDate() { + return receivedAtNationalLevelDate; + } + + public void setReceivedAtNationalLevelDate(Date receivedAtNationalLevelDate) { + this.receivedAtNationalLevelDate = receivedAtNationalLevelDate; + } + + public String getWorldwideId() { + return worldwideId; + } + + public void setWorldwideId(String worldwideId) { + this.worldwideId = worldwideId; + } + + @Column(columnDefinition = "text") + public String getNationalLevelComment() { + return nationalLevelComment; + } + + public void setNationalLevelComment(String nationalLevelComment) { + this.nationalLevelComment = nationalLevelComment; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java new file mode 100644 index 00000000000..33f71570b4f --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java @@ -0,0 +1,162 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity; + +import javax.persistence.criteria.From; +import javax.persistence.criteria.Join; +import javax.persistence.criteria.JoinType; + +import de.symeda.sormas.backend.common.QueryJoins; +import de.symeda.sormas.backend.immunization.ImmunizationJoins; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.country.Country; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.person.Person; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.vaccination.Vaccination; + +public class AefiJoins extends QueryJoins { + + private Join immunization; + private Join address; + private Join primarySuspectVaccination; + private Join adverseEvents; + private Join person; + private Join reportingUser; + private Join responsibleRegion; + private Join responsibleDistrict; + private Join responsibleCommunity; + private Join country; + private Join healthFacility; + private Join reporterInstitution; + + private ImmunizationJoins immunizationJoins; + + public AefiJoins(From root) { + super(root); + } + + public Join getImmunization() { + return getOrCreate(immunization, Aefi.IMMUNIZATION, JoinType.LEFT, this::setImmunization); + } + + public void setImmunization(Join immunization) { + this.immunization = immunization; + } + + public ImmunizationJoins getImmunizationJoins() { + return getOrCreate(immunizationJoins, () -> new ImmunizationJoins(getImmunization()), this::setImmunizationJoins); + } + + public void setImmunizationJoins(ImmunizationJoins immunizationJoins) { + this.immunizationJoins = immunizationJoins; + } + + public Join getAddress() { + return getOrCreate(address, Aefi.ADDRESS, JoinType.LEFT, this::setAddress); + } + + public void setAddress(Join address) { + this.address = address; + } + + public Join getPrimarySuspectVaccination() { + return getOrCreate(primarySuspectVaccination, Aefi.PRIMARY_SUSPECT_VACCINE, JoinType.LEFT, this::setPrimarySuspectVaccination); + } + + public void setPrimarySuspectVaccination(Join primarySuspectVaccination) { + this.primarySuspectVaccination = primarySuspectVaccination; + } + + public Join getAdverseEvents() { + return getOrCreate(adverseEvents, Aefi.ADVERSE_EVENTS, JoinType.LEFT, this::setAdverseEvents); + } + + public void setAdverseEvents(Join adverseEvents) { + this.adverseEvents = adverseEvents; + } + + public Join getPerson() { + return getOrCreate(person, Aefi.PERSON, JoinType.LEFT, this::setPerson); + } + + public void setPerson(Join person) { + this.person = person; + } + + public Join getReportingUser() { + return getOrCreate(reportingUser, Aefi.REPORTING_USER, JoinType.LEFT, this::setReportingUser); + } + + public void setReportingUser(Join reportingUser) { + this.reportingUser = reportingUser; + } + + public Join getResponsibleRegion() { + return getOrCreate(responsibleRegion, Aefi.RESPONSIBLE_REGION, JoinType.LEFT, this::setResponsibleRegion); + } + + public void setResponsibleRegion(Join responsibleRegion) { + this.responsibleRegion = responsibleRegion; + } + + public Join getResponsibleDistrict() { + return getOrCreate(responsibleDistrict, Aefi.RESPONSIBLE_DISTRICT, JoinType.LEFT, this::setResponsibleDistrict); + } + + public void setResponsibleDistrict(Join responsibleDistrict) { + this.responsibleDistrict = responsibleDistrict; + } + + public Join getResponsibleCommunity() { + return getOrCreate(responsibleCommunity, Aefi.RESPONSIBLE_COMMUNITY, JoinType.LEFT, this::setResponsibleCommunity); + } + + public void setResponsibleCommunity(Join responsibleCommunity) { + this.responsibleCommunity = responsibleCommunity; + } + + public Join getCountry() { + return getOrCreate(country, Aefi.COUNTRY, JoinType.LEFT, this::setCountry); + } + + public void setCountry(Join country) { + this.country = country; + } + + public Join getHealthFacility() { + return getOrCreate(healthFacility, Aefi.HEALTH_FACILITY, JoinType.LEFT, this::setHealthFacility); + } + + public void setHealthFacility(Join healthFacility) { + this.healthFacility = healthFacility; + } + + public Join getReporterInstitution() { + return getOrCreate(reporterInstitution, Aefi.REPORTER_INSTITUTION, JoinType.LEFT, this::setReporterInstitution); + } + + public void setReporterInstitution(Join reporterInstitution) { + this.reporterInstitution = reporterInstitution; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java new file mode 100644 index 00000000000..73594a78f5f --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java @@ -0,0 +1,86 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers; + +import java.util.Date; +import java.util.List; + +import org.hibernate.transform.ResultTransformer; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiHelper; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiOutcome; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeizureType; +import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.person.ApproximateAgeType; +import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.utils.YesNoUnknown; + +public class AefiIndexDtoResultTransformer implements ResultTransformer { + + @Override + public Object transformTuple(Object[] objects, String[] strings) { + + Integer age = objects[6] != null ? (int) objects[6] : null; + ApproximateAgeType approximateAgeType = (ApproximateAgeType) objects[7]; + Integer birthdateDD = objects[8] != null ? (int) objects[8] : null; + Integer birthdateMM = objects[9] != null ? (int) objects[9] : null; + Integer birthdateYYYY = objects[10] != null ? (int) objects[10] : null; + + //@formatter:off + String adverseEvents = AefiHelper + .buildAdverseEventsString((AdverseEventState) objects[20], (boolean) objects[21], (boolean) objects[22], + (AdverseEventState) objects[23], (SeizureType) objects[24], (AdverseEventState) objects[25], + (AdverseEventState) objects[26], (AdverseEventState) objects[27], (AdverseEventState) objects[28], + (AdverseEventState) objects[29], (AdverseEventState) objects[30], (AdverseEventState) objects[31], + (String) objects[32]); + //@formatter:on + + return new AefiIndexDto( + (String) objects[0], + (String) objects[1], + (String) objects[3], + (String) objects[4], + (String) objects[5], + (Disease) objects[2], + new AgeAndBirthDateDto(age, approximateAgeType, birthdateDD, birthdateMM, birthdateYYYY), + (Sex) objects[11], + (String) objects[12], + (String) objects[13], + (YesNoUnknown) objects[14], + (Vaccine) objects[15], + (AefiOutcome) objects[16], + (Date) objects[17], + (Date) objects[18], + (Date) objects[19], + adverseEvents, + (DeletionReason) objects[33], + (String) objects[34], + (boolean) objects[35]); + } + + @Override + public List transformList(List list) { + return list; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiListEntryDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiListEntryDtoResultTransformer.java new file mode 100644 index 00000000000..02e9082cace --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiListEntryDtoResultTransformer.java @@ -0,0 +1,60 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers; + +import java.util.Date; +import java.util.List; + +import org.hibernate.transform.ResultTransformer; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiHelper; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeizureType; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.utils.YesNoUnknown; + +public class AefiListEntryDtoResultTransformer implements ResultTransformer { + + @Override + public Object transformTuple(Object[] objects, String[] strings) { + + //@formatter:off + String adverseEvents = AefiHelper + .buildAdverseEventsString((AdverseEventState) objects[5], (boolean) objects[6], (boolean) objects[7], + (AdverseEventState) objects[8], (SeizureType) objects[9], (AdverseEventState) objects[10], + (AdverseEventState) objects[11], (AdverseEventState) objects[12], (AdverseEventState) objects[13], + (AdverseEventState) objects[14], (AdverseEventState) objects[15], (AdverseEventState) objects[16], + (String) objects[17]); + //@formatter:on + + return new AefiListEntryDto( + (String) objects[0], + (YesNoUnknown) objects[1], + (Vaccine) objects[2], + (String) objects[3], + (Date) objects[4], + adverseEvents); + } + + @Override + public List transformList(List list) { + return list; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java new file mode 100644 index 00000000000..91471089e92 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java @@ -0,0 +1,76 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.dashboard.adverseeventsfollowingimmunization; + +import java.util.List; +import java.util.Map; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartData; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiDashboardFacade; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.MapAefiDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.backend.util.RightsAllowed; + +@Stateless(name = "AefiDashboardFacade") +@RightsAllowed(UserRight._DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW) +public class AefiDashboardFacadeEjb implements AefiDashboardFacade { + + @EJB + private AefiDashboardService aefiDashboardService; + + @Override + public Map getAefiCountsByType(AefiDashboardCriteria dashboardCriteria) { + return aefiDashboardService.getAefiCountsByType(dashboardCriteria); + } + + @Override + public Map> getAefiCountsByVaccine(AefiDashboardCriteria dashboardCriteria) { + return aefiDashboardService.getAefiCountsByVaccine(dashboardCriteria); + } + + @Override + public AefiChartData getAefiByVaccineDoseChartData(AefiDashboardCriteria dashboardCriteria) { + return aefiDashboardService.getAefiByVaccineDoseChartData(dashboardCriteria); + } + + @Override + public AefiChartData getAefiEventsByGenderChartData(AefiDashboardCriteria dashboardCriteria) { + return aefiDashboardService.getAefiEventsByGenderChartData(dashboardCriteria); + } + + @Override + public Long countAefiForMap(AefiDashboardCriteria criteria) { + return aefiDashboardService.countAefiForMap(criteria); + } + + @Override + public List getAefiForMap(AefiDashboardCriteria criteria) { + return aefiDashboardService.getAefiForMap(criteria); + } + + @LocalBean + @Stateless + public static class AefiDashboardFacadeEjbLocal extends AefiDashboardFacadeEjb { + + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java new file mode 100644 index 00000000000..41a50bed7a5 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java @@ -0,0 +1,515 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.dashboard.adverseeventsfollowingimmunization; + +import java.math.BigInteger; +import java.text.SimpleDateFormat; +import java.util.ArrayList; +import java.util.Date; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.function.BiFunction; +import java.util.function.Function; +import java.util.stream.Collectors; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; +import javax.persistence.EntityManager; +import javax.persistence.PersistenceContext; +import javax.persistence.Query; +import javax.persistence.Tuple; +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.From; +import javax.persistence.criteria.Join; +import javax.persistence.criteria.Path; +import javax.persistence.criteria.Predicate; +import javax.persistence.criteria.Root; + +import org.apache.commons.lang3.StringUtils; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDashboardFilterDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartData; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartSeries; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.MapAefiDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.AefiQueryContext; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.AefiService; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiJoins; +import de.symeda.sormas.backend.common.AbstractDomainObject; +import de.symeda.sormas.backend.common.CriteriaBuilderHelper; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.district.DistrictService; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.infrastructure.region.RegionService; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.person.Person; +import de.symeda.sormas.backend.util.ModelConstants; +import de.symeda.sormas.backend.util.QueryHelper; + +@Stateless +@LocalBean +public class AefiDashboardService { + + private final Logger logger = LoggerFactory.getLogger(AefiDashboardService.class); + + @PersistenceContext(unitName = ModelConstants.PERSISTENCE_UNIT_NAME) + private EntityManager em; + + public static final String SERIOUS_SERIES_COLOR = "#E04A5B"; + public static final String NON_SERIOUS_SERIES_COLOR = "#7565B8"; + public static final String MALE_SERIES_COLOR = "#544FC5"; + public static final String FEMALE_SERIES_COLOR = "#2CAFFE"; + + @EJB + private AefiService aefiService; + @EJB + private RegionService regionService; + @EJB + private DistrictService districtService; + + public Map getAefiCountsByType(AefiDashboardCriteria dashboardCriteria) { + + Map result = new HashMap<>(); + + Map dataMap = getAefiCountsByEnumProperty(Aefi.SERIOUS, YesNoUnknown.class, dashboardCriteria, null); + + result.put(AefiType.SERIOUS, dataMap.getOrDefault(YesNoUnknown.YES, 0L)); + result.put(AefiType.NON_SERIOUS, dataMap.getOrDefault(YesNoUnknown.NO, 0L) + dataMap.getOrDefault(YesNoUnknown.UNKNOWN, 0L)); + + return result; + } + + public Map> getAefiCountsByVaccine(AefiDashboardCriteria dashboardCriteria) { + + Map> countsByVaccine = new HashMap<>(); + + Disease disease = dashboardCriteria.getDisease(); + RegionReferenceDto regionReference = dashboardCriteria.getRegion(); + DistrictReferenceDto districtReference = dashboardCriteria.getDistrict(); + + String whereConditions = createAefiNativeQueryFilter(dashboardCriteria); + if (StringUtils.isBlank(whereConditions)) { + whereConditions = " primaryvaccine.vaccinename is not null"; + } else { + whereConditions = whereConditions + " AND primaryvaccine.vaccinename is not null"; + } + + //@formatter:off + String queryString = "select primaryvaccine.vaccinename as vaccine_name " + + " , count(CASE WHEN aefi.serious = 'YES' THEN 1 END) as serious" + + " , count(CASE WHEN aefi.serious <> 'YES' THEN 1 END) as non_serious" + + " from adverseeventsfollowingimmunization aefi" + + " join immunization on aefi.immunization_id = immunization.id" + + " join vaccination primaryvaccine on aefi.primarysuspectvaccine_id = primaryvaccine.id" + + " where " + whereConditions + + " group by primaryvaccine.vaccinename" + + " order by primaryvaccine.vaccinename"; + //@formatter:on + + Query dataQuery = em.createNativeQuery(queryString); + + if (disease != null) { + dataQuery.setParameter("disease", disease.name()); + } + + if (regionReference != null) { + Region region = regionService.getByReferenceDto(regionReference); + dataQuery.setParameter("responsibleregion_id", region.getId()); + } + + if (districtReference != null) { + District district = districtService.getByReferenceDto(districtReference); + dataQuery.setParameter("responsibledistrict_id", district.getId()); + } + + if (dashboardCriteria.getDateFrom() != null && dashboardCriteria.getDateTo() != null) { + Date dateFrom = DateHelper.getStartOfDay(dashboardCriteria.getDateFrom()); + Date dateTo = DateHelper.getEndOfDay(dashboardCriteria.getDateTo()); + + SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); + String dateFromStr = DateHelper.formatLocalDate(dateFrom, simpleDateFormat); + String dateToStr = DateHelper.formatLocalDate(dateTo, simpleDateFormat); + + dataQuery.setParameter("dateFrom", dateFromStr); + dataQuery.setParameter("dateTo", dateToStr); + } + + @SuppressWarnings("unchecked") + List resultList = dataQuery.getResultList(); + + Map aefiTypeDataMap; + for (Object[] result : resultList) { + aefiTypeDataMap = new HashMap<>(); + aefiTypeDataMap.put(AefiType.SERIOUS, ((BigInteger) result[1]).longValue()); + aefiTypeDataMap.put(AefiType.NON_SERIOUS, ((BigInteger) result[2]).longValue()); + + countsByVaccine.put(Vaccine.valueOf((String) result[0]), aefiTypeDataMap); + } + + return countsByVaccine; + } + + public AefiChartData getAefiByVaccineDoseChartData(AefiDashboardCriteria dashboardCriteria) { + + AefiChartData chartData = new AefiChartData(); + + AefiChartSeries seriousSeries = new AefiChartSeries(AefiType.SERIOUS); + seriousSeries.setColor(SERIOUS_SERIES_COLOR); + AefiChartSeries nonSeriousSeries = new AefiChartSeries(AefiType.NON_SERIOUS); + nonSeriousSeries.setColor(NON_SERIOUS_SERIES_COLOR); + + chartData.addSeries(seriousSeries); + chartData.addSeries(nonSeriousSeries); + + Disease disease = dashboardCriteria.getDisease(); + RegionReferenceDto regionReference = dashboardCriteria.getRegion(); + DistrictReferenceDto districtReference = dashboardCriteria.getDistrict(); + + String whereConditions = createAefiNativeQueryFilter(dashboardCriteria); + if (StringUtils.isBlank(whereConditions)) { + whereConditions = " primaryvaccine.vaccinedose is not null"; + } else { + whereConditions = whereConditions + " AND primaryvaccine.vaccinedose is not null"; + } + + //@formatter:off + String queryString = "select primaryvaccine.vaccinedose as vaccine_dose " + + " , count(CASE WHEN aefi.serious = 'YES' THEN 1 END) as serious" + + " , count(CASE WHEN aefi.serious <> 'YES' THEN 1 END) as non_serious" + + " from adverseeventsfollowingimmunization aefi" + + " join immunization on aefi.immunization_id = immunization.id" + + " join vaccination primaryvaccine on aefi.primarysuspectvaccine_id = primaryvaccine.id" + + " where " + whereConditions + + " group by primaryvaccine.vaccinedose" + + " order by primaryvaccine.vaccinedose"; + //@formatter:on + + Query dataQuery = em.createNativeQuery(queryString); + + if (disease != null) { + dataQuery.setParameter("disease", disease.name()); + } + + if (regionReference != null) { + Region region = regionService.getByReferenceDto(regionReference); + dataQuery.setParameter("responsibleregion_id", region.getId()); + } + + if (districtReference != null) { + District district = districtService.getByReferenceDto(districtReference); + dataQuery.setParameter("responsibledistrict_id", district.getId()); + } + + if (dashboardCriteria.getDateFrom() != null && dashboardCriteria.getDateTo() != null) { + Date dateFrom = DateHelper.getStartOfDay(dashboardCriteria.getDateFrom()); + Date dateTo = DateHelper.getEndOfDay(dashboardCriteria.getDateTo()); + + SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); + String dateFromStr = DateHelper.formatLocalDate(dateFrom, simpleDateFormat); + String dateToStr = DateHelper.formatLocalDate(dateTo, simpleDateFormat); + + dataQuery.setParameter("dateFrom", dateFromStr); + dataQuery.setParameter("dateTo", dateToStr); + } + + @SuppressWarnings("unchecked") + List resultList = dataQuery.getResultList(); + + for (Object[] result : resultList) { + chartData.addXAxisCategory(result[0]); + seriousSeries.addData(result[1].toString()); + nonSeriousSeries.addData(result[2].toString()); + } + + return chartData; + } + + public AefiChartData getAefiEventsByGenderChartData(AefiDashboardCriteria dashboardCriteria) { + + AefiChartData chartData = new AefiChartData(); + + AefiChartSeries maleSeries = new AefiChartSeries(Sex.MALE); + maleSeries.setColor(MALE_SERIES_COLOR); + AefiChartSeries femaleSeries = new AefiChartSeries(Sex.FEMALE); + femaleSeries.setColor(FEMALE_SERIES_COLOR); + + chartData.addSeries(maleSeries); + chartData.addSeries(femaleSeries); + + Disease disease = dashboardCriteria.getDisease(); + RegionReferenceDto regionReference = dashboardCriteria.getRegion(); + DistrictReferenceDto districtReference = dashboardCriteria.getDistrict(); + + List adverseEventsList = new ArrayList<>(); + adverseEventsList.add(AdverseEvents.SEVERE_LOCAL_REACTION); + adverseEventsList.add(AdverseEvents.SEIZURES); + adverseEventsList.add(AdverseEvents.ABSCESS); + adverseEventsList.add(AdverseEvents.SEPSIS); + adverseEventsList.add(AdverseEvents.ENCEPHALOPATHY); + adverseEventsList.add(AdverseEvents.TOXIC_SHOCK_SYNDROME); + adverseEventsList.add(AdverseEvents.THROMBOCYTOPENIA); + adverseEventsList.add(AdverseEvents.ANAPHYLAXIS); + adverseEventsList.add(AdverseEvents.FEVERISH_FEELING); + + String queryString; + String queryFilter = createAefiNativeQueryFilter(dashboardCriteria); + String whereConditions; + String adverseEventCondition; + String lowerCaseAdverseEvent; + + for (String adverseEvent : adverseEventsList) { + + whereConditions = queryFilter; + lowerCaseAdverseEvent = adverseEvent.toLowerCase(); + adverseEventCondition = " adverseevents." + lowerCaseAdverseEvent + " = 'YES'"; + if (StringUtils.isBlank(whereConditions)) { + whereConditions = adverseEventCondition; + } else { + whereConditions = whereConditions + " AND " + adverseEventCondition; + } + + //@formatter:off + queryString = "select adverseevents." + lowerCaseAdverseEvent + " as " + lowerCaseAdverseEvent + + " , count(CASE WHEN person.sex = 'MALE' THEN 1 END) as total_male" + + " , count(CASE WHEN person.sex = 'FEMALE' THEN 1 END) as total_female" + + " from adverseeventsfollowingimmunization aefi" + + " join immunization on aefi.immunization_id = immunization.id" + + " join person on immunization.person_id = person.id" + + " join adverseevents on aefi.adverseevents_id = adverseevents.id" + + " where " + whereConditions + + " group by adverseevents." + lowerCaseAdverseEvent; + //@formatter:on + + Query dataQuery = em.createNativeQuery(queryString); + + if (disease != null) { + dataQuery.setParameter("disease", disease.name()); + } + + if (regionReference != null) { + Region region = regionService.getByReferenceDto(regionReference); + dataQuery.setParameter("responsibleregion_id", region.getId()); + } + + if (districtReference != null) { + District district = districtService.getByReferenceDto(districtReference); + dataQuery.setParameter("responsibledistrict_id", district.getId()); + } + + if (dashboardCriteria.getDateFrom() != null && dashboardCriteria.getDateTo() != null) { + Date dateFrom = DateHelper.getStartOfDay(dashboardCriteria.getDateFrom()); + Date dateTo = DateHelper.getEndOfDay(dashboardCriteria.getDateTo()); + + SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); + String dateFromStr = DateHelper.formatLocalDate(dateFrom, simpleDateFormat); + String dateToStr = DateHelper.formatLocalDate(dateTo, simpleDateFormat); + + dataQuery.setParameter("dateFrom", dateFromStr); + dataQuery.setParameter("dateTo", dateToStr); + } + + chartData.addXAxisCategory(adverseEvent); + + @SuppressWarnings("unchecked") + List resultList = dataQuery.getResultList(); + + if (!resultList.isEmpty()) { + Object[] firstResult = resultList.get(0); + maleSeries.addData(firstResult[1].toString()); + femaleSeries.addData(firstResult[2].toString()); + } else { + maleSeries.addData("0"); + femaleSeries.addData("0"); + } + } + + return chartData; + } + + private > Map getAefiCountsByEnumProperty( + String property, + Class propertyType, + AefiDashboardCriteria dashboardCriteria, + BiFunction, Predicate> additionalFilters) { + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createTupleQuery(); + final Root aefiRoot = cq.from(Aefi.class); + final Path groupingProperty = aefiRoot.get(property); + + cq.multiselect(groupingProperty, cb.count(aefiRoot)); + + final Predicate criteriaFilter = createAefiFilter(new AefiQueryContext(cb, cq, aefiRoot), dashboardCriteria); + cq.where(CriteriaBuilderHelper.and(cb, criteriaFilter, additionalFilters != null ? additionalFilters.apply(cb, aefiRoot) : null)); + + cq.groupBy(groupingProperty); + + return QueryHelper.getResultList(em, cq, null, null, Function.identity()) + .stream() + .collect(Collectors.toMap(t -> propertyType.cast(t.get(0)), t -> (Long) t.get(1))); + } + + public Long countAefiForMap(AefiDashboardCriteria criteria) { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Long.class); + final Root aefiRoot = cq.from(Aefi.class); + + AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefiRoot); + AefiJoins joins = aefiQueryContext.getJoins(); + + cq.select(cb.count(aefiRoot)); + + final Predicate criteriaFilter = createAefiFilter(aefiQueryContext, criteria); + + cq.where(CriteriaBuilderHelper.and(cb, criteriaFilter)); + + return QueryHelper.getSingleResult(em, cq); + } + + public List getAefiForMap(AefiDashboardCriteria criteria) { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(MapAefiDto.class); + final Root aefiRoot = cq.from(Aefi.class); + + AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefiRoot); + AefiJoins joins = aefiQueryContext.getJoins(); + + Join aefiFacility = joins.getHealthFacility(); + Join immunizationFacility = joins.getImmunizationJoins().getHealthFacility(); + Join immunizationPersonAddress = joins.getImmunizationJoins().getPersonJoins().getAddress(); + + cq.multiselect( + aefiFacility.get(Facility.LATITUDE), + aefiFacility.get(Facility.LONGITUDE), + immunizationFacility.get(Facility.LATITUDE), + immunizationFacility.get(Facility.LONGITUDE), + immunizationPersonAddress.get(Facility.LATITUDE), + immunizationPersonAddress.get(Facility.LONGITUDE), + aefiRoot.get(Aefi.SERIOUS)); + + final Predicate criteriaFilter = createAefiFilter(aefiQueryContext, criteria); + + cq.where(CriteriaBuilderHelper.and(cb, criteriaFilter)); + + return QueryHelper.getResultList(em, cq, null, null); + } + + private Predicate createAefiFilter(AefiQueryContext queryContext, AefiDashboardCriteria criteria) { + CriteriaBuilder cb = queryContext.getCriteriaBuilder(); + CriteriaQuery cq = queryContext.getQuery(); + From aefiRoot = queryContext.getRoot(); + AefiJoins joins = queryContext.getJoins(); + + Predicate filter = aefiService.createUserFilter(queryContext); + + filter = CriteriaBuilderHelper.and( + cb, + filter, + aefiService.buildCriteriaFilter( + new AefiCriteria().disease(criteria.getDisease()) + .region(criteria.getRegion()) + .district(criteria.getDistrict()) + .aefiType(criteria.getAefiType()), + queryContext)); + + if (criteria.getDateFrom() != null && criteria.getDateTo() != null) { + final Predicate dateFilter; + Date dateFrom = DateHelper.getStartOfDay(criteria.getDateFrom()); + Date dateTo = DateHelper.getEndOfDay(criteria.getDateTo()); + + AefiDashboardFilterDateType aefiDashboardFilterDateType = criteria.getAefiDashboardFilterDateType() != null + ? criteria.getAefiDashboardFilterDateType() + : AefiDashboardFilterDateType.REPORT_DATE; + + switch (aefiDashboardFilterDateType) { + case REPORT_DATE: + dateFilter = cb.between(aefiRoot.get(Aefi.REPORT_DATE), dateFrom, dateTo); + break; + case START_DATE: + dateFilter = cb.between(aefiRoot.get(Aefi.START_DATE_TIME), dateFrom, dateTo); + break; + default: + throw new RuntimeException("Unhandled date type [" + aefiDashboardFilterDateType + "]"); + } + + filter = CriteriaBuilderHelper.and(cb, filter, dateFilter); + } + + return CriteriaBuilderHelper.and( + cb, + filter, + // Exclude deleted adverse events. Archived adverse events should stay included + cb.isFalse(aefiRoot.get(Aefi.DELETED))); + } + + private String createAefiNativeQueryFilter(AefiDashboardCriteria criteria) { + + List whereConditions = new ArrayList<>(); + + if (criteria.getDisease() != null) { + whereConditions.add("immunization.disease = :disease"); + } + + if (criteria.getRegion() != null) { + whereConditions.add("immunization.responsibleregion_id = :responsibleregion_id"); + } + + if (criteria.getDistrict() != null) { + whereConditions.add("immunization.responsibledistrict_id = :responsibledistrict_id"); + } + + if (criteria.getDateFrom() != null && criteria.getDateTo() != null) { + AefiDashboardFilterDateType aefiDashboardFilterDateType = criteria.getAefiDashboardFilterDateType() != null + ? criteria.getAefiDashboardFilterDateType() + : AefiDashboardFilterDateType.REPORT_DATE; + + switch (aefiDashboardFilterDateType) { + case REPORT_DATE: + whereConditions.add("(aefi.reportdate >= cast(:dateFrom as timestamp) and aefi.reportdate <= cast(:dateTo as timestamp))"); + break; + case START_DATE: + whereConditions.add("(aefi.startdatetime >= cast(:dateFrom as timestamp) and aefi.startdatetime <= cast(:dateTo as timestamp))"); + break; + default: + throw new RuntimeException("Unhandled date type [" + aefiDashboardFilterDateType + "]"); + } + } + + return String.join(" AND ", whereConditions); + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java index 237cdad9507..a52daa2c309 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java @@ -30,16 +30,19 @@ import javax.ejb.TransactionAttribute; import javax.ejb.TransactionAttributeType; import javax.persistence.Tuple; +import javax.persistence.TypedQuery; import javax.persistence.criteria.CriteriaBuilder; import javax.persistence.criteria.CriteriaQuery; import javax.persistence.criteria.CriteriaUpdate; import javax.persistence.criteria.From; import javax.persistence.criteria.Join; import javax.persistence.criteria.JoinType; +import javax.persistence.criteria.ParameterExpression; import javax.persistence.criteria.Path; import javax.persistence.criteria.Predicate; import javax.persistence.criteria.Root; import javax.persistence.criteria.Subquery; +import javax.validation.constraints.NotNull; import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.EditPermissionType; @@ -108,6 +111,24 @@ public ImmunizationService() { super(Immunization.class, DeletableEntityType.IMMUNIZATION); } + public Long getIdByUuid(@NotNull String uuid) { + + if (uuid == null) { + return null; + } + + CriteriaBuilder cb = em.getCriteriaBuilder(); + ParameterExpression uuidParam = cb.parameter(String.class, AbstractDomainObject.UUID); + CriteriaQuery cq = cb.createQuery(Long.class); + Root from = cq.from(Immunization.class); + cq.select(from.get(AbstractDomainObject.ID)); + cq.where(cb.equal(from.get(AbstractDomainObject.UUID), uuidParam)); + + TypedQuery q = em.createQuery(cq).setParameter(uuidParam, uuid); + + return q.getResultList().stream().findFirst().orElse(null); + } + @Override public void deletePermanent(Immunization immunization) { diff --git a/sormas-backend/src/main/resources/META-INF/persistence.xml b/sormas-backend/src/main/resources/META-INF/persistence.xml index b792ff39147..e0dd26ad9b2 100644 --- a/sormas-backend/src/main/resources/META-INF/persistence.xml +++ b/sormas-backend/src/main/resources/META-INF/persistence.xml @@ -86,6 +86,9 @@ de.symeda.sormas.backend.externalmessage.labmessage.TestReport de.symeda.sormas.backend.deletionconfiguration.DeletionConfiguration de.symeda.sormas.backend.externalmessage.labmessage.SampleReport + de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi + de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents + de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess de.symeda.sormas.backend.selfreport.SelfReport diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java index 12b32cf2f3c..7e2f4581fc1 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java @@ -18,6 +18,7 @@ package de.symeda.sormas.ui; import de.symeda.sormas.ui.action.ActionController; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.AefiController; import de.symeda.sormas.ui.campaign.CampaignController; import de.symeda.sormas.ui.caze.CaseController; import de.symeda.sormas.ui.caze.surveillancereport.SurveillanceReportController; @@ -89,6 +90,7 @@ public class ControllerProvider extends BaseControllerProvider { private final DocGenerationController docGenerationController; private final TravelEntryController travelEntryController; private final ImmunizationController immunizationController; + private final AefiController aefiController; private final VaccinationController vaccinationController; private final ArchivingController archivingController; private final DeleteRestoreController deleteRestoreController; @@ -132,6 +134,7 @@ public ControllerProvider() { docGenerationController = new DocGenerationController(); travelEntryController = new TravelEntryController(); immunizationController = new ImmunizationController(); + aefiController = new AefiController(); vaccinationController = new VaccinationController(); archivingController = new ArchivingController(); deleteRestoreController = new DeleteRestoreController(); @@ -264,6 +267,10 @@ public static ImmunizationController getImmunizationController() { return get().immunizationController; } + public static AefiController getAefiController() { + return get().aefiController; + } + public static VaccinationController getVaccinationController() { return get().vaccinationController; } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java index 890097e8b3b..00ecea96b08 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java @@ -52,6 +52,7 @@ import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.DataHelper; import de.symeda.sormas.api.utils.criteria.BaseCriteria; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.AefiView; import de.symeda.sormas.ui.campaign.AbstractCampaignView; import de.symeda.sormas.ui.campaign.campaigndata.CampaignDataView; import de.symeda.sormas.ui.campaign.campaigns.CampaignsView; @@ -69,6 +70,7 @@ import de.symeda.sormas.ui.configuration.outbreak.OutbreaksView; import de.symeda.sormas.ui.contact.ContactsView; import de.symeda.sormas.ui.dashboard.AbstractDashboardView; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.AefiDashboardView; import de.symeda.sormas.ui.dashboard.campaigns.CampaignDashboardView; import de.symeda.sormas.ui.dashboard.contacts.ContactsDashboardView; import de.symeda.sormas.ui.dashboard.sample.SampleDashboardView; @@ -231,6 +233,12 @@ public View getView(String viewName) { AbstractDashboardView.ROOT_VIEW_NAME, I18nProperties.getCaption(Captions.mainMenuDashboard), VaadinIcons.DASHBOARD); + } else if (aefiDashboardPermitted()) { + menu.addView( + AefiDashboardView.class, + AbstractDashboardView.ROOT_VIEW_NAME, + I18nProperties.getCaption(Captions.mainMenuDashboard), + VaadinIcons.DASHBOARD); } if (permitted(FeatureType.TASK_MANAGEMENT, UserRight.TASK_VIEW)) { @@ -299,6 +307,13 @@ public View getView(String viewName) { VaadinIcons.HEALTH_CARD); } + if (permitted(FeatureType.IMMUNIZATION_MANAGEMENT, UserRight.IMMUNIZATION_VIEW) + && !FacadeProvider.getFeatureConfigurationFacade().isPropertyValueTrue(FeatureType.IMMUNIZATION_MANAGEMENT, FeatureTypeProperty.REDUCED) + && permitted(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT, UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + ControllerProvider.getAefiController().registerViews(navigator); + menu.addView(AefiView.class, AefiView.VIEW_NAME, I18nProperties.getCaption(Captions.mainMenuAdverseEvents), VaadinIcons.BELL_SLASH); + } + if (permitted(FeatureType.TRAVEL_ENTRIES, UserRight.TRAVEL_ENTRY_MANAGEMENT_ACCESS) && FacadeProvider.getConfigFacade().isConfiguredCountry(CountryHelper.COUNTRY_CODE_GERMANY)) { ControllerProvider.getTravelEntryController().registerViews(navigator); @@ -439,6 +454,12 @@ private static boolean sampleDashboardPermitted() { return permitted(EnumSet.of(FeatureType.DASHBOARD_SAMPLES, FeatureType.SAMPLES_LAB), UserRight.DASHBOARD_SAMPLES_VIEW); } + private static boolean aefiDashboardPermitted() { + return permitted( + EnumSet.of(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT), + UserRight.DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW); + } + private static Set initKnownViews() { final Set views = new HashSet<>( Arrays.asList( @@ -468,7 +489,8 @@ private static Set initKnownViews() { CountriesView.VIEW_NAME, ExternalMessagesView.VIEW_NAME, TravelEntriesView.VIEW_NAME, - ImmunizationsView.VIEW_NAME)); + ImmunizationsView.VIEW_NAME, + AefiView.VIEW_NAME)); if (surveillanceDashboardPermitted()) { views.add(SurveillanceDashboardView.VIEW_NAME); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java new file mode 100644 index 00000000000..3e082e0657a --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java @@ -0,0 +1,115 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import com.vaadin.navigator.ViewChangeListener.ViewChangeEvent; + +import de.symeda.sormas.api.EditPermissionFacade; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiReferenceDto; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.SubMenu; +import de.symeda.sormas.ui.immunization.ImmunizationDataView; +import de.symeda.sormas.ui.immunization.ImmunizationPersonView; +import de.symeda.sormas.ui.utils.AbstractEditAllowedDetailView; + +public abstract class AbstractAefiView extends AbstractEditAllowedDetailView { + + public static final String ROOT_VIEW_NAME = AefiView.VIEW_NAME; + + protected AbstractAefiView(String viewName) { + super(viewName); + } + + @Override + protected AefiReferenceDto getReferenceByUuid(String uuid) { + + final AefiReferenceDto reference; + + boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(uuid); + + if (isCreateAction) { + reference = new AefiReferenceDto(); + reference.setUuid(DataHelper.createUuid()); + } else { + if (FacadeProvider.getAefiFacade().exists(uuid)) { + reference = FacadeProvider.getAefiFacade().getReferenceByUuid(uuid); + } else { + reference = null; + } + } + return reference; + } + + @Override + protected String getRootViewName() { + return ROOT_VIEW_NAME; + } + + @Override + protected void initView(String params) { + + } + + @Override + protected EditPermissionFacade getEditPermissionFacade() { + return FacadeProvider.getAefiFacade(); + } + + @Override + public void enter(ViewChangeEvent event) { + + super.enter(event); + initOrRedirect(event); + } + + @Override + public void refreshMenu(SubMenu menu, String params) { + + if (!findReferenceByParams(params)) { + return; + } + + boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(params); + + String immunizationUuid = ""; + if (isCreateAction) { + immunizationUuid = ControllerProvider.getAefiController().getCreateActionImmunizationUuid(params); + } else { + AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(params); + immunizationUuid = aefiDto.getImmunization().getUuid(); + } + + menu.removeAllViews(); + menu.addView(AefiView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiList)); + menu.addView(ImmunizationDataView.VIEW_NAME, I18nProperties.getCaption(ImmunizationDto.I18N_PREFIX), immunizationUuid); + menu.addView( + ImmunizationPersonView.VIEW_NAME, + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, ImmunizationDto.PERSON), + immunizationUuid); + menu.addView(AefiDataView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiDataView), params); + + setMainHeaderComponent(ControllerProvider.getAefiController().getAefiViewTitleLayout(params, immunizationUuid, isCreateAction)); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java new file mode 100644 index 00000000000..a3200c25c27 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java @@ -0,0 +1,195 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import java.util.function.Consumer; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.navigator.Navigator; +import com.vaadin.server.Sizeable; +import com.vaadin.ui.Notification; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.deletionconfiguration.DeletionInfoDto; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.SormasUI; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiPrimarySuspectVaccinationSelectionField; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.AefiDataForm; +import de.symeda.sormas.ui.utils.ArchiveHandlers; +import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; +import de.symeda.sormas.ui.utils.VaadinUiUtil; +import de.symeda.sormas.ui.utils.components.automaticdeletion.DeletionLabel; +import de.symeda.sormas.ui.utils.components.page.title.TitleLayout; +import de.symeda.sormas.ui.utils.components.page.title.TitleLayoutHelper; + +public class AefiController { + + public void registerViews(Navigator navigator) { + navigator.addView(AefiView.VIEW_NAME, AefiView.class); + //navigator.addView(ImmunizationDataView.VIEW_NAME, ImmunizationDataView.class); + //navigator.addView(ImmunizationPersonView.VIEW_NAME, ImmunizationPersonView.class); + navigator.addView(AefiDataView.VIEW_NAME, AefiDataView.class); + } + + public void selectPrimarySuspectVaccination(AefiDto aefiDto, Consumer commitCallback) { + + AefiPrimarySuspectVaccinationSelectionField selectionField = + new AefiPrimarySuspectVaccinationSelectionField(aefiDto.getVaccinations(), aefiDto.getPrimarySuspectVaccine()); + selectionField.setWidth(1024, Sizeable.Unit.PIXELS); + + final CommitDiscardWrapperComponent component = + new CommitDiscardWrapperComponent<>(selectionField); + component.addCommitListener(() -> { + VaccinationDto selectedVaccination = selectionField.getValue(); + if (selectedVaccination != null) { + aefiDto.setPrimarySuspectVaccine(selectedVaccination); + + if (commitCallback != null) { + commitCallback.accept(selectedVaccination); + } + } + }); + + selectionField.setSelectionChangeCallback((commitAllowed) -> component.getCommitButton().setEnabled(commitAllowed)); + VaadinUiUtil.showModalPopupWindow(component, I18nProperties.getString(Strings.headingAefiPickPrimarySuspectVaccine)); + } + + public void navigateToAefi(String uuid) { + navigateToView(AefiDataView.VIEW_NAME, uuid); + } + + public void navigateToView(String viewName, String uuid) { + final String navigationState = viewName + "/" + uuid; + SormasUI.get().getNavigator().navigateTo(navigationState); + } + + public CommitDiscardWrapperComponent getAefiDataEditComponent( + boolean isCreateAction, + AefiDto aefiDto, + Consumer actionCallback) { + + AefiDataForm aefiDataForm = new AefiDataForm(isCreateAction, aefiDto.isPseudonymized(), aefiDto.isInJurisdiction(), actionCallback); + aefiDataForm.setValue(aefiDto); + + CommitDiscardWrapperComponent editComponent = + new CommitDiscardWrapperComponent<>(aefiDataForm, true, aefiDataForm.getFieldGroup()); + + if (!isCreateAction) { + DeletionInfoDto automaticDeletionInfoDto = FacadeProvider.getAefiFacade().getAutomaticDeletionInfo(aefiDto.getUuid()); + DeletionInfoDto manuallyDeletionInfoDto = FacadeProvider.getAefiFacade().getManuallyDeletionInfo(aefiDto.getUuid()); + + editComponent.getButtonsPanel() + .addComponentAsFirst(new DeletionLabel(automaticDeletionInfoDto, manuallyDeletionInfoDto, aefiDto.isDeleted(), AefiDto.I18N_PREFIX)); + + if (aefiDto.isDeleted()) { + editComponent.getWrappedComponent().getField(AefiDto.DELETION_REASON).setVisible(true); + if (editComponent.getWrappedComponent().getField(AefiDto.DELETION_REASON).getValue() == DeletionReason.OTHER_REASON) { + editComponent.getWrappedComponent().getField(AefiDto.OTHER_DELETION_REASON).setVisible(true); + } + } + } + + editComponent.addCommitListener(() -> { + if (!aefiDataForm.getFieldGroup().isModified()) { + AefiDto aefiDataFormValue = aefiDataForm.getValue(); + + AefiDto savedAefiDto = FacadeProvider.getAefiFacade().save(aefiDataFormValue); + Notification.show(I18nProperties.getString(Strings.messageAdverseEventSaved), Notification.Type.WARNING_MESSAGE); + + if (isCreateAction) { + navigateToAefi(savedAefiDto.getUuid()); + } else { + SormasUI.refreshView(); + } + } + }); + + if (!isCreateAction) { + // Initialize 'Delete' button + if (UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE)) { + editComponent.addDeleteWithReasonOrRestoreListener( + AefiView.VIEW_NAME, + null, + I18nProperties.getString(Strings.entityAdverseEvent), + aefiDto.getUuid(), + FacadeProvider.getAefiFacade()); + } + + // Initialize 'Archive' button + if (UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE)) { + ControllerProvider.getArchiveController() + .addArchivingButton(aefiDto, ArchiveHandlers.forAefi(), editComponent, () -> navigateToAefi(aefiDto.getUuid())); + } + + editComponent.restrictEditableComponentsOnEditView( + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT, + null, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE, + FacadeProvider.getAefiFacade().getEditPermissionType(aefiDto.getUuid()), + aefiDto.isInJurisdiction()); + } + + return editComponent; + } + + public TitleLayout getAefiViewTitleLayout(String aefiUuid, String immunizationUuid, boolean isCreateAction) { + + if (!isCreateAction) { + AefiDto aefiDto = findAefi(aefiUuid); + immunizationUuid = aefiDto.getImmunization().getUuid(); + } + + TitleLayout titleLayout = new TitleLayout(); + + String shortUuid = DataHelper.getShortUuid(immunizationUuid); + ImmunizationDto immunizationDto = FacadeProvider.getImmunizationFacade().getByUuid(immunizationUuid); + PersonDto person = FacadeProvider.getPersonFacade().getByUuid(immunizationDto.getPerson().getUuid()); + StringBuilder mainRowText = TitleLayoutHelper.buildPersonString(person); + mainRowText.append(mainRowText.length() > 0 ? " (" + shortUuid + ")" : shortUuid); + titleLayout.addMainRow(mainRowText.toString()); + + return titleLayout; + } + + private AefiDto findAefi(String uuid) { + return FacadeProvider.getAefiFacade().getByUuid(uuid); + } + + public boolean isCreateAction(String params) { + return StringUtils.startsWith(params, "immunization") && StringUtils.endsWith(params, "create"); + } + + public String getCreateActionImmunizationUuid(String params) { + return StringUtils.contains(params, "/") + ? StringUtils.substringBetween(params, "/", "/") + : StringUtils.substringBetween(params, "immunization", "create"); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java new file mode 100644 index 00000000000..adc07afc106 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java @@ -0,0 +1,128 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import java.util.ArrayList; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.EditPermissionType; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.feature.FeatureTypeProperty; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.AefiDataForm; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiImmunizationInfo; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiPersonInfo; +import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DetailSubComponentWrapper; +import de.symeda.sormas.ui.utils.LayoutWithSidePanel; + +public class AefiDataView extends AbstractAefiView { + + public static final String VIEW_NAME = ROOT_VIEW_NAME + "/data"; + + public static final String ADVERSE_EVENT_LOC = "adverseEventLoc"; + public static final String PERSON_LOC = "personLoc"; + public static final String IMMUNIZATION_LOC = "immunizationLoc"; + public static final String INVESTIGATIONS_LOC = "investigationsLoc"; + + private CommitDiscardWrapperComponent editComponent; + + public AefiDataView() { + super(VIEW_NAME); + } + + @Override + protected String getRootViewName() { + return super.getRootViewName(); + } + + @Override + protected void initView(String params) { + setHeightUndefined(); + + AefiDto aefi; + ImmunizationDto immunization; + + boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(params); + if (isCreateAction) { + aefi = AefiDto.build(getReference()); + + String immunizationUuid = ControllerProvider.getAefiController().getCreateActionImmunizationUuid(params); + immunization = FacadeProvider.getImmunizationFacade().getByUuid(immunizationUuid); + + aefi.setImmunization(immunization.toReference()); + aefi.setVaccinations(new ArrayList<>(immunization.getVaccinations())); + aefi.setReportingUser(UserProvider.getCurrent().getUserReference()); + } else { + aefi = FacadeProvider.getAefiFacade().getByUuid(getReference().getUuid()); + immunization = FacadeProvider.getImmunizationFacade().getByUuid(aefi.getImmunization().getUuid()); + } + + editComponent = ControllerProvider.getAefiController().getAefiDataEditComponent(isCreateAction, aefi, this::showUnsavedChangesPopup); + + DetailSubComponentWrapper container = new DetailSubComponentWrapper(() -> editComponent); + container.setWidth(100, Unit.PERCENTAGE); + container.setMargin(true); + setSubComponent(container); + container.setEnabled(true); + + LayoutWithSidePanel layout = new LayoutWithSidePanel(editComponent, PERSON_LOC, IMMUNIZATION_LOC, INVESTIGATIONS_LOC); + + container.addComponent(layout); + + UserProvider currentUser = UserProvider.getCurrent(); + if (currentUser.hasAllUserRights(UserRight.PERSON_VIEW)) { + PersonDto personDto = FacadeProvider.getPersonFacade().getByUuid(immunization.getPerson().getUuid()); + Disease disease = immunization.getDisease(); + + AefiPersonInfo aefiPersonInfo = new AefiPersonInfo(personDto, disease); + CssStyles.style(aefiPersonInfo, CssStyles.VIEW_SECTION); + + layout.addSidePanelComponent(aefiPersonInfo, PERSON_LOC); + } + + if (FacadeProvider.getFeatureConfigurationFacade().isFeatureEnabled(FeatureType.IMMUNIZATION_MANAGEMENT) + && currentUser != null + && currentUser.hasUserRight(UserRight.IMMUNIZATION_VIEW)) { + if (!FacadeProvider.getFeatureConfigurationFacade() + .isPropertyValueTrue(FeatureType.IMMUNIZATION_MANAGEMENT, FeatureTypeProperty.REDUCED)) { + + AefiImmunizationInfo aefiImmunizationInfo = new AefiImmunizationInfo(immunization, (r) -> { + }); + CssStyles.style(aefiImmunizationInfo, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + + layout.addSidePanelComponent(aefiImmunizationInfo, IMMUNIZATION_LOC); + } + } + + if (!isCreateAction) { + final String uuid = aefi.getUuid(); + final EditPermissionType aefiEditAllowed = FacadeProvider.getAefiFacade().getEditPermissionType(uuid); + final boolean deleted = FacadeProvider.getAefiFacade().isDeleted(uuid); + layout.disableIfNecessary(deleted, aefiEditAllowed); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java new file mode 100644 index 00000000000..daa30665c50 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java @@ -0,0 +1,208 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import java.util.Objects; + +import com.vaadin.icons.VaadinIcons; +import com.vaadin.navigator.ViewChangeListener; +import com.vaadin.shared.ui.ContentMode; +import com.vaadin.ui.Alignment; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; +import com.vaadin.v7.ui.ComboBox; + +import de.symeda.sormas.api.EntityRelevanceStatus; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.common.DeletableEntityType; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.feature.FeatureTypeProperty; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.ViewModelProviders; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory.AefiDataLayout; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory.AefiFilterFormLayout; +import de.symeda.sormas.ui.utils.AbstractView; +import de.symeda.sormas.ui.utils.ComboBoxHelper; +import de.symeda.sormas.ui.utils.CssStyles; + +public class AefiView extends AbstractView { + + public static final String VIEW_NAME = "adverseevents"; + + private final AefiCriteria criteria; + + private AefiFilterFormLayout filterFormLayout; + private final AefiDataLayout dataLayout; + + // Filters + private Label relevanceStatusInfoLabel; + private ComboBox relevanceStatusFilter; + + public AefiView() { + super(VIEW_NAME); + + CssStyles.style(getViewTitleLabel(), CssStyles.PAGE_TITLE); + + criteria = ViewModelProviders.of(AefiView.class).get(AefiCriteria.class); + if (criteria.getRelevanceStatus() == null) { + criteria.setRelevanceStatus(EntityRelevanceStatus.ACTIVE); + } + dataLayout = new AefiDataLayout(criteria); + + final VerticalLayout mainLayout = new VerticalLayout(); + mainLayout.addComponent(createFilterBar()); + + final VerticalLayout gridLayout = new VerticalLayout(); + gridLayout.setMargin(false); + gridLayout.setSpacing(false); + gridLayout.setSizeFull(); + CssStyles.style(gridLayout, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + + gridLayout.addComponent(createStatusFilterBar()); + gridLayout.addComponent(dataLayout); + gridLayout.setExpandRatio(dataLayout, 1); + + mainLayout.addComponent(gridLayout); + + mainLayout.setMargin(true); + mainLayout.setSpacing(false); + mainLayout.setSizeFull(); + mainLayout.setExpandRatio(gridLayout, 1); + mainLayout.setStyleName("crud-main-layout"); + + addComponent(mainLayout); + } + + private void updateFilterComponents() { + // TODO replace with Vaadin 8 databinding + applyingCriteria = true; + + if (relevanceStatusFilter != null) { + relevanceStatusFilter.setValue(criteria.getRelevanceStatus()); + } + + filterFormLayout.setValue(criteria); + + applyingCriteria = false; + } + + private AefiFilterFormLayout createFilterBar() { + filterFormLayout = new AefiFilterFormLayout(); + + filterFormLayout.addResetHandler(clickEvent -> { + ViewModelProviders.of(AefiView.class).remove(AefiCriteria.class); + navigateTo(null, true); + }); + + filterFormLayout.addApplyHandler(clickEvent -> { + dataLayout.refreshGrid(); + }); + + return filterFormLayout; + } + + public HorizontalLayout createStatusFilterBar() { + HorizontalLayout statusFilterLayout = new HorizontalLayout(); + statusFilterLayout.setSpacing(true); + statusFilterLayout.setMargin(false); + statusFilterLayout.setWidth(100, Unit.PERCENTAGE); + statusFilterLayout.addStyleName(CssStyles.VSPACE_3); + + HorizontalLayout actionButtonsLayout = new HorizontalLayout(); + actionButtonsLayout.setSpacing(true); + + // Show active/archived/all dropdown + if (Objects.nonNull(UserProvider.getCurrent()) + && UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + + if (FacadeProvider.getFeatureConfigurationFacade() + .isFeatureEnabled(FeatureType.AUTOMATIC_ARCHIVING, DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION)) { + + int daysAfterAefiEntryGetsArchived = FacadeProvider.getFeatureConfigurationFacade() + .getProperty( + FeatureType.AUTOMATIC_ARCHIVING, + DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION, + FeatureTypeProperty.THRESHOLD_IN_DAYS, + Integer.class); + if (daysAfterAefiEntryGetsArchived > 0) { + relevanceStatusInfoLabel = new Label( + VaadinIcons.INFO_CIRCLE.getHtml() + " " + + String.format(I18nProperties.getString(Strings.infoArchivedAefiEntries), daysAfterAefiEntryGetsArchived), + ContentMode.HTML); + relevanceStatusInfoLabel.setVisible(false); + relevanceStatusInfoLabel.addStyleName(CssStyles.LABEL_VERTICAL_ALIGN_SUPER); + actionButtonsLayout.addComponent(relevanceStatusInfoLabel); + actionButtonsLayout.setComponentAlignment(relevanceStatusInfoLabel, Alignment.MIDDLE_RIGHT); + } + } + relevanceStatusFilter = ComboBoxHelper.createComboBoxV7(); + relevanceStatusFilter.setId("relevanceStatus"); + relevanceStatusFilter.setWidth(260, Unit.PIXELS); + relevanceStatusFilter.setNullSelectionAllowed(false); + relevanceStatusFilter.setTextInputAllowed(false); + relevanceStatusFilter.addItems((Object[]) EntityRelevanceStatus.values()); + relevanceStatusFilter.setItemCaption(EntityRelevanceStatus.ACTIVE, I18nProperties.getCaption(Captions.aefiActiveAdverseEvents)); + relevanceStatusFilter.setItemCaption(EntityRelevanceStatus.ARCHIVED, I18nProperties.getCaption(Captions.aefiArchivedAdverseEvents)); + relevanceStatusFilter + .setItemCaption(EntityRelevanceStatus.ACTIVE_AND_ARCHIVED, I18nProperties.getCaption(Captions.aefiAllActiveAndArchivedAdverseEvents)); + relevanceStatusFilter.setCaption(null); + relevanceStatusFilter.addStyleName(CssStyles.VSPACE_NONE); + + if (UserProvider.getCurrent().hasUserRight(UserRight.IMMUNIZATION_DELETE)) { + relevanceStatusFilter.setItemCaption(EntityRelevanceStatus.DELETED, I18nProperties.getCaption(Captions.aefiDeletedAdverseEvents)); + } else { + relevanceStatusFilter.removeItem(EntityRelevanceStatus.DELETED); + } + + relevanceStatusFilter.addValueChangeListener(e -> { + if (relevanceStatusInfoLabel != null) { + relevanceStatusInfoLabel.setVisible(EntityRelevanceStatus.ARCHIVED.equals(e.getProperty().getValue())); + } + criteria.setRelevanceStatus((EntityRelevanceStatus) e.getProperty().getValue()); + navigateTo(criteria); + }); + actionButtonsLayout.addComponent(relevanceStatusFilter); + } + + if (actionButtonsLayout.getComponentCount() > 0) { + statusFilterLayout.addComponent(actionButtonsLayout); + statusFilterLayout.setComponentAlignment(actionButtonsLayout, Alignment.TOP_RIGHT); + statusFilterLayout.setExpandRatio(actionButtonsLayout, 1); + } + + return statusFilterLayout; + } + + @Override + public void enter(ViewChangeListener.ViewChangeEvent event) { + + String params = event.getParameters().trim(); + if (params.startsWith("?")) { + params = params.substring(1); + criteria.fromUrlParams(params); + } + updateFilterComponents(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java new file mode 100644 index 00000000000..1c88db1072b --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java @@ -0,0 +1,89 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefilink; + +import java.util.List; +import java.util.function.Consumer; + +import org.apache.commons.collections.CollectionUtils; + +import com.vaadin.ui.Button; +import com.vaadin.ui.Label; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.utils.PaginationList; + +@SuppressWarnings("serial") +public class AefiList extends PaginationList { + + private static final int MAX_DISPLAYED_ENTRIES = 5; + + private AefiListCriteria aefiListCriteria; + private Consumer actionCallback; + private final boolean isEditable; + private Label noAdverseEventsLabel; + + public AefiList(AefiListCriteria aefiListCriteria, Consumer actionCallback, boolean isEditable) { + + super(MAX_DISPLAYED_ENTRIES); + + this.aefiListCriteria = aefiListCriteria; + this.actionCallback = actionCallback; + this.isEditable = isEditable; + + noAdverseEventsLabel = new Label(I18nProperties.getString(Strings.infoNoImmunizationAdverseEvents)); + } + + @Override + public void reload() { + + List aefiListEntries = FacadeProvider.getAefiFacade().getEntriesList(aefiListCriteria, 0, maxDisplayedEntries * 20); + + setEntries(aefiListEntries); + if (CollectionUtils.isNotEmpty(aefiListEntries)) { + showPage(1); + } else { + listLayout.removeAllComponents(); + updatePaginationLayout(); + listLayout.addComponent(noAdverseEventsLabel); + } + } + + @Override + protected void drawDisplayedEntries() { + + List displayedEntries = getDisplayedEntries(); + for (AefiListEntryDto aefiListEntry : displayedEntries) { + AefiListEntry listEntry = new AefiListEntry(aefiListEntry); + + String aefiUuid = aefiListEntry.getUuid(); + listEntry.addEditButton( + "edit-aefi-" + aefiUuid, + (Button.ClickListener) event -> ControllerProvider.getAefiController().navigateToAefi(aefiUuid)); + + listEntry.setEnabled(isEditable); + listLayout.addComponent(listEntry); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java new file mode 100644 index 00000000000..400715182cb --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java @@ -0,0 +1,56 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefilink; + +import java.util.function.Consumer; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListCriteria; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponent; + +@SuppressWarnings("serial") +public class AefiListComponent extends SideComponent { + + public AefiListComponent(AefiListCriteria aefiListCriteria, Consumer actionCallback, boolean isEditAllowed, int totalVaccinations) { + super(I18nProperties.getString(Strings.headingImmunizationAdverseEvents), actionCallback); + + setMargin(false); + setWidth(100, Unit.PERCENTAGE); + + if (isEditAllowed) { + addCreateButton( + I18nProperties.getCaption(Captions.aefiNewAdverseEvent), + () -> ControllerProvider.getAefiController() + .navigateToAefi("immunization/" + aefiListCriteria.getImmunization().getUuid() + "/create"), + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE); + + if (totalVaccinations == 0) { + createButton.setEnabled(false); + } + } + + AefiList aefiList = new AefiList(aefiListCriteria, actionCallback, isEditAllowed); + addComponent(aefiList); + aefiList.reload(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListEntry.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListEntry.java new file mode 100644 index 00000000000..a47a70bfe60 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListEntry.java @@ -0,0 +1,79 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefilink; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.ui.Label; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponentField; + +@SuppressWarnings("serial") +public class AefiListEntry extends SideComponentField { + + public static final String SEPARATOR = ": "; + + private final AefiListEntryDto aefiListEntryDto; + + public AefiListEntry(AefiListEntryDto aefiListEntryDto) { + + this.aefiListEntryDto = aefiListEntryDto; + + Label labelAefiType = new Label(AefiType.toString(aefiListEntryDto.getSerious())); + CssStyles.style(labelAefiType, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + if (aefiListEntryDto.getSerious() == YesNoUnknown.YES) { + CssStyles.style(labelAefiType, CssStyles.LABEL_CRITICAL); + } + addComponentToField(labelAefiType); + + Label labelVaccineName = new Label(aefiListEntryDto.getPrimaryVaccineName().toString()); + CssStyles.style(labelVaccineName, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelVaccineName); + + if (!StringUtils.isBlank(aefiListEntryDto.getPrimaryVaccineDose())) { + Label labelVaccineDose = new Label( + I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_DOSE) + + SEPARATOR + + aefiListEntryDto.getPrimaryVaccineDose()); + CssStyles.style(labelVaccineDose, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelVaccineDose); + } + + Label labelVaccineDate = new Label( + I18nProperties.getPrefixCaption(AefiListEntryDto.I18N_PREFIX, AefiListEntryDto.PRIMARY_VACCINE_VACCINATION_DATE) + + SEPARATOR + + DateFormatHelper.formatLocalDate(aefiListEntryDto.getPrimaryVaccineVaccinationDate())); + addComponentToField(labelVaccineDate); + + Label labelAdverseEvents = new Label(StringUtils.abbreviate(aefiListEntryDto.getAdverseEvents(), 56)); + CssStyles.style(labelAdverseEvents, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelAdverseEvents); + } + + public AefiListEntryDto getAefiListEntryDto() { + return aefiListEntryDto; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java new file mode 100644 index 00000000000..fa96538ee1e --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java @@ -0,0 +1,42 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; + +public class AefiDataLayout extends VerticalLayout { + + private final AefiGrid grid; + + public AefiDataLayout(AefiCriteria criteria) { + grid = new AefiGrid(criteria); + addComponent(grid); + + setMargin(false); + setSpacing(false); + setSizeFull(); + setExpandRatio(grid, 1); + } + + public void refreshGrid() { + grid.reload(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java new file mode 100644 index 00000000000..3f83e93210e --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java @@ -0,0 +1,420 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import static de.symeda.sormas.ui.utils.LayoutUtil.divCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.filterLocs; +import static de.symeda.sormas.ui.utils.LayoutUtil.loc; +import static de.symeda.sormas.ui.utils.LayoutUtil.locCss; + +import java.util.Date; +import java.util.stream.Stream; + +import org.apache.commons.lang3.ArrayUtils; + +import com.vaadin.ui.CustomLayout; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.v7.data.Property; +import com.vaadin.v7.ui.ComboBox; +import com.vaadin.v7.ui.Field; +import com.vaadin.v7.ui.TextField; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDateType; +import de.symeda.sormas.api.i18n.Descriptions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityType; +import de.symeda.sormas.api.infrastructure.facility.FacilityTypeGroup; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.user.JurisdictionLevel; +import de.symeda.sormas.api.user.UserDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.DateFilterOption; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.api.utils.EpiWeek; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.utils.AbstractFilterForm; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.EpiWeekAndDateFilterComponent; +import de.symeda.sormas.ui.utils.FieldConfiguration; +import de.symeda.sormas.ui.utils.FieldHelper; + +public class AefiFilterForm extends AbstractFilterForm { + + private static final String ACTION_BUTTONS_ID = "actionButtons"; + private static final String MORE_FILTERS_ID = "moreFilters"; + private static final String WEEK_AND_DATE_FILTER = "weekAndDateFilter"; + + private static final String MORE_FILTERS_HTML = loc(WEEK_AND_DATE_FILTER); + + public AefiFilterForm() { + super( + AefiCriteria.class, + AefiCriteria.I18N_PREFIX, + JurisdictionFieldConfig.of(AefiCriteria.REGION, AefiCriteria.DISTRICT, AefiCriteria.COMMUNITY)); + } + + @Override + protected String createHtmlLayout() { + return divCss( + CssStyles.VIEW_SECTION, + filterLocs(ArrayUtils.addAll(getMainFilterLocators(), ACTION_BUTTONS_ID)) + locCss(CssStyles.VSPACE_TOP_NONE, MORE_FILTERS_ID)); + + } + + @Override + protected String[] getMainFilterLocators() { + return new String[] { + AefiCriteria.DISEASE, + AefiCriteria.NAME_ADDRESS_PHONE_EMAIL_LIKE, + AefiCriteria.AEFI_TYPE, + AefiCriteria.VACCINE_NAME, + AefiCriteria.REGION, + AefiCriteria.DISTRICT, + AefiCriteria.COMMUNITY, + AefiCriteria.OUTCOME }; + } + + @Override + protected String createMoreFiltersHtmlLayout() { + return MORE_FILTERS_HTML; + } + + @Override + protected void addFields() { + addField(FieldConfiguration.pixelSized(AefiCriteria.DISEASE, 140)); + + final TextField searchField = addField( + FieldConfiguration.withCaptionAndPixelSized( + AefiCriteria.NAME_ADDRESS_PHONE_EMAIL_LIKE, + I18nProperties.getString(Strings.promptPersonsSearchField), + 200)); + searchField.setNullRepresentation(""); + + addFields(FieldConfiguration.pixelSized(AefiCriteria.AEFI_TYPE, 140), FieldConfiguration.pixelSized(AefiCriteria.VACCINE_NAME, 140)); + + if (currentUserDto().getRegion() == null) { + ComboBox regionFilter = addField(getContent(), FieldConfiguration.pixelSized(AefiCriteria.REGION, 140)); + regionFilter.addItems(FacadeProvider.getRegionFacade().getAllActiveByServerCountry()); + } + + ComboBox districtFilter = addField(getContent(), FieldConfiguration.pixelSized(AefiCriteria.DISTRICT, 140)); + districtFilter.setDescription(I18nProperties.getDescription(Descriptions.descDistrictFilter)); + if (currentUserDto().getDistrict() != null) { + districtFilter.setVisible(false); + } + + addField(getContent(), FieldConfiguration.pixelSized(AefiCriteria.COMMUNITY, 140)); + + addField(FieldConfiguration.pixelSized(AefiCriteria.OUTCOME, 140)); + } + + @Override + public void addMoreFilters(CustomLayout moreFiltersContainer) { + moreFiltersContainer.addComponent(buildWeekAndDateFilter(), WEEK_AND_DATE_FILTER); + } + + @Override + protected void applyDependenciesOnFieldChange(String propertyId, Property.ValueChangeEvent event) { + + super.applyDependenciesOnFieldChange(propertyId, event); + + final AefiCriteria criteria = getValue(); + + final ComboBox facilityTypeGroupField = getField(AefiCriteria.FACILITY_TYPE_GROUP); + final ComboBox facilityTypeField = getField(AefiCriteria.FACILITY_TYPE); + final ComboBox facilityField = getField(AefiCriteria.HEALTH_FACILITY); + + final UserDto user = currentUserDto(); + final DistrictReferenceDto currentDistrict = + user.getDistrict() != null ? user.getDistrict() : (DistrictReferenceDto) districtFilter.getValue(); + + switch (propertyId) { + case AefiCriteria.REGION: { + final RegionReferenceDto region = user.getRegion() != null ? user.getRegion() : (RegionReferenceDto) event.getProperty().getValue(); + + if (!DataHelper.equal(region, criteria.getRegion())) { + if (region != null) { + enableFields(districtFilter); + FieldHelper.updateItems(districtFilter, FacadeProvider.getDistrictFacade().getAllActiveByRegion(region.getUuid())); + } else { + clearAndDisableFields(districtFilter); + } + clearAndDisableFields(communityFilter, facilityField, facilityTypeField, facilityTypeGroupField); + } + + break; + } + case AefiCriteria.DISTRICT: { + final DistrictReferenceDto newDistrict = (DistrictReferenceDto) event.getProperty().getValue(); + + if (!DataHelper.equal(newDistrict, criteria.getDistrict())) { + if (newDistrict != null) { + enableFields(communityFilter, facilityTypeGroupField); + + clearAndDisableFields(facilityField); + if (facilityTypeGroupField != null) { + if (facilityTypeGroupField.getValue() != null && facilityTypeField.getValue() != null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade() + .getActiveFacilitiesByDistrictAndType(newDistrict, (FacilityType) facilityTypeField.getValue(), true, false)); + enableFields(facilityField); + } else { + FieldHelper.updateEnumData(facilityTypeGroupField, FacilityTypeGroup.getAccomodationGroups()); + } + } + + FieldHelper.updateItems(communityFilter, FacadeProvider.getCommunityFacade().getAllActiveByDistrict(newDistrict.getUuid())); + } else { + clearAndDisableFields(communityFilter, facilityField, facilityTypeField, facilityTypeGroupField); + } + } + + break; + } + case AefiCriteria.COMMUNITY: { + CommunityReferenceDto community = (CommunityReferenceDto) event.getProperty().getValue(); + if (!DataHelper.equal(community, criteria.getCommunity())) { + if (facilityField != null) { + facilityField.setValue(null); + } + + final FacilityType facilityType = facilityTypeField != null ? (FacilityType) facilityTypeField.getValue() : null; + + if (facilityType == null && facilityField != null) { + facilityField.removeAllItems(); + } else if (facilityField != null) { + if (community == null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByDistrictAndType(currentDistrict, facilityType, true, false)); + } else { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByCommunityAndType(community, facilityType, true, false)); + } + } + } + break; + } + case AefiCriteria.FACILITY_TYPE_GROUP: { + FacilityTypeGroup typeGroup = (FacilityTypeGroup) event.getProperty().getValue(); + if (!DataHelper.equal(typeGroup, criteria.getFacilityTypeGroup())) { + if (typeGroup != null) { + enableFields(AefiCriteria.FACILITY_TYPE); + FieldHelper.updateEnumData(facilityTypeField, FacilityType.getAccommodationTypes(typeGroup)); + facilityField.setValue(null); + } else { + clearAndDisableFields(facilityTypeField, facilityField); + } + } + + break; + } + case AefiCriteria.FACILITY_TYPE: { + FacilityType facilityType = (FacilityType) event.getProperty().getValue(); + if (!DataHelper.equal(facilityType, criteria.getFacilityType())) { + if (facilityType == null) { + clearAndDisableFields(facilityField); + } else { + enableFields(facilityField); + facilityField.setValue(null); + + CommunityReferenceDto community = (CommunityReferenceDto) communityFilter.getValue(); + if (community != null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByCommunityAndType(community, facilityType, true, false)); + } else if (currentDistrict != null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByDistrictAndType(currentDistrict, facilityType, true, false)); + } + } + } + break; + } + } + } + + @Override + protected void applyDependenciesOnNewValue(AefiCriteria criteria) { + + final UserDto user = currentUserDto(); + + UserProvider currentUserProvider = UserProvider.getCurrent(); + final JurisdictionLevel userJurisdictionLevel = currentUserProvider != null ? UserProvider.getCurrent().getJurisdictionLevel() : null; + + final ComboBox facilityTypeGroupField = getField(AefiCriteria.FACILITY_TYPE_GROUP); + final ComboBox facilityTypeField = getField(AefiCriteria.FACILITY_TYPE); + final ComboBox facilityField = getField(AefiCriteria.HEALTH_FACILITY); + + // Disable all fields + clearAndDisableFields(districtFilter, communityFilter, facilityTypeGroupField, facilityTypeField, facilityField); + + // Get initial field values according to user and criteria + final RegionReferenceDto region = user.getRegion() == null ? criteria.getRegion() : user.getRegion(); + final DistrictReferenceDto district = user.getDistrict() == null ? criteria.getDistrict() : user.getDistrict(); + final CommunityReferenceDto community = user.getCommunity() == null ? criteria.getCommunity() : user.getCommunity(); + final FacilityTypeGroup facilityTypeGroup = criteria.getFacilityTypeGroup(); + final FacilityType facilityType = criteria.getFacilityType(); + + // district + if (region != null) { + enableFields(districtFilter); + districtFilter.addItems(FacadeProvider.getDistrictFacade().getAllActiveByRegion(region.getUuid())); + // community + if (district != null) { + districtFilter.setValue(district); + communityFilter.addItems(FacadeProvider.getCommunityFacade().getAllActiveByDistrict(district.getUuid())); + enableFields(communityFilter); + if (community != null) { + communityFilter.setValue(community); + } + } else { + clearAndDisableFields(communityFilter); + } + } else { + clearAndDisableFields(districtFilter, communityFilter); + } + + // facility + if (userJurisdictionLevel == JurisdictionLevel.HEALTH_FACILITY) { + facilityField.setValue(user.getHealthFacility()); + disableFields(facilityTypeGroupField, facilityTypeField, facilityField); + } else if (facilityTypeGroupField != null && district != null) { + enableFields(facilityTypeGroupField); + FieldHelper.updateEnumData(facilityTypeGroupField, FacilityTypeGroup.getAccomodationGroups()); + if (facilityTypeGroup != null) { + facilityTypeGroupField.setValue(facilityTypeGroup); + enableFields(facilityTypeField); + FieldHelper.updateEnumData(facilityTypeField, FacilityType.getAccommodationTypes(facilityTypeGroup)); + if (facilityType != null) { + facilityTypeField.setValue(facilityType); + enableFields(facilityField); + if (community != null) { + facilityField + .addItems(FacadeProvider.getFacilityFacade().getActiveFacilitiesByCommunityAndType(community, facilityType, true, false)); + } else { + facilityField + .addItems(FacadeProvider.getFacilityFacade().getActiveFacilitiesByDistrictAndType(district, facilityType, true, false)); + } + } else { + disableFields(facilityField); + } + } else { + disableFields(facilityTypeField); + } + } + + // Disable fields according to user & jurisdiction + if (userJurisdictionLevel == JurisdictionLevel.DISTRICT) { + clearAndDisableFields(districtFilter); + } else if (userJurisdictionLevel == JurisdictionLevel.COMMUNITY) { + clearAndDisableFields(districtFilter, communityFilter); + } else if (userJurisdictionLevel == JurisdictionLevel.HEALTH_FACILITY) { + clearAndDisableFields(districtFilter, communityFilter, facilityTypeGroupField, facilityTypeField, facilityField); + } + + // Date/Epi week filter + HorizontalLayout dateFilterLayout = (HorizontalLayout) getMoreFiltersContainer().getComponent(WEEK_AND_DATE_FILTER); + @SuppressWarnings("unchecked") + EpiWeekAndDateFilterComponent weekAndDateFilter = + (EpiWeekAndDateFilterComponent) dateFilterLayout.getComponent(0); + + AefiDateType aefiDateType = criteria.getAefiDateType(); + weekAndDateFilter.getDateTypeSelector().setValue(aefiDateType); + weekAndDateFilter.getDateFilterOptionFilter().setValue(criteria.getDateFilterOption()); + Date dateFrom = criteria.getFromDate(); + Date dateTo = criteria.getToDate(); + + if (DateFilterOption.EPI_WEEK.equals(criteria.getDateFilterOption())) { + weekAndDateFilter.getWeekFromFilter().setValue(dateFrom == null ? null : DateHelper.getEpiWeek(dateFrom)); + weekAndDateFilter.getWeekToFilter().setValue(dateTo == null ? null : DateHelper.getEpiWeek(dateTo)); + } else { + weekAndDateFilter.getDateFromFilter().setValue(dateFrom); + weekAndDateFilter.getDateToFilter().setValue(dateTo); + } + } + + @Override + protected Stream streamFieldsForEmptyCheck(CustomLayout layout) { + HorizontalLayout dateFilterLayout = (HorizontalLayout) getMoreFiltersContainer().getComponent(WEEK_AND_DATE_FILTER); + EpiWeekAndDateFilterComponent weekAndDateFilter = + (EpiWeekAndDateFilterComponent) dateFilterLayout.getComponent(0); + + return super.streamFieldsForEmptyCheck(layout).filter(f -> f != weekAndDateFilter.getDateFilterOptionFilter()); + } + + private HorizontalLayout buildWeekAndDateFilter() { + + EpiWeekAndDateFilterComponent weekAndDateFilter = new EpiWeekAndDateFilterComponent<>( + false, + false, + null, + AefiDateType.values(), + I18nProperties.getString(Strings.promptAefiDateType), + null, + this); + weekAndDateFilter.getWeekFromFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiEpiWeekFrom)); + weekAndDateFilter.getWeekToFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiEpiWeekTo)); + weekAndDateFilter.getDateFromFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiDateFrom)); + weekAndDateFilter.getDateToFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiDateTo)); + + addApplyHandler(e -> onApplyClick(weekAndDateFilter)); + + HorizontalLayout dateFilterRowLayout = new HorizontalLayout(); + dateFilterRowLayout.setSpacing(true); + dateFilterRowLayout.setSizeUndefined(); + + dateFilterRowLayout.addComponent(weekAndDateFilter); + + return dateFilterRowLayout; + } + + private void onApplyClick(EpiWeekAndDateFilterComponent weekAndDateFilter) { + AefiCriteria criteria = getValue(); + + DateFilterOption dateFilterOption = (DateFilterOption) weekAndDateFilter.getDateFilterOptionFilter().getValue(); + Date fromDate, toDate; + if (dateFilterOption == DateFilterOption.DATE) { + Date dateFrom = weekAndDateFilter.getDateFromFilter().getValue(); + fromDate = dateFrom != null ? DateHelper.getStartOfDay(dateFrom) : null; + Date dateTo = weekAndDateFilter.getDateToFilter().getValue(); + toDate = dateFrom != null ? DateHelper.getEndOfDay(dateTo) : null; + } else { + fromDate = DateHelper.getEpiWeekStart((EpiWeek) weekAndDateFilter.getWeekFromFilter().getValue()); + toDate = DateHelper.getEpiWeekEnd((EpiWeek) weekAndDateFilter.getWeekToFilter().getValue()); + } + if ((fromDate != null && toDate != null) || (fromDate == null && toDate == null)) { + criteria.setDateFilterOption(dateFilterOption); + AefiDateType AefiDateType = (AefiDateType) weekAndDateFilter.getDateTypeSelector().getValue(); + criteria.setAefiDateType(AefiDateType); + criteria.setFromDate(fromDate); + criteria.setToDate(toDate); + } else { + weekAndDateFilter.setNotificationsForMissingFilters(); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterFormLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterFormLayout.java new file mode 100644 index 00000000000..801b80ae08c --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterFormLayout.java @@ -0,0 +1,53 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import com.vaadin.ui.Button; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; + +public class AefiFilterFormLayout extends VerticalLayout { + + private static final float PERCENTAGE_WIDTH = 100; + + private final AefiFilterForm filterForm; + + public AefiFilterFormLayout() { + setSpacing(false); + setMargin(false); + setWidth(PERCENTAGE_WIDTH, Unit.PERCENTAGE); + + filterForm = new AefiFilterForm(); + addComponent(filterForm); + } + + public AefiCriteria getValue() { + return filterForm.getValue(); + } + + public void setValue(AefiCriteria criteria) { + filterForm.setValue(criteria); + } + + public void addResetHandler(Button.ClickListener listener) { + filterForm.addResetHandler(listener); + } + + public void addApplyHandler(Button.ClickListener listener) { + filterForm.addApplyHandler(listener); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java new file mode 100644 index 00000000000..8a27926187e --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java @@ -0,0 +1,132 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import java.util.Date; + +import com.vaadin.ui.renderers.DateRenderer; +import com.vaadin.ui.renderers.TextRenderer; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; +import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.person.PersonHelper; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.UiUtil; +import de.symeda.sormas.ui.immunization.ImmunizationPersonView; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.FieldAccessColumnStyleGenerator; +import de.symeda.sormas.ui.utils.FilteredGrid; +import de.symeda.sormas.ui.utils.ShowDetailsListener; +import de.symeda.sormas.ui.utils.UuidRenderer; + +public class AefiGrid extends FilteredGrid { + + public AefiGrid(AefiCriteria criteria) { + super(AefiIndexDto.class); + setSizeFull(); + setLazyDataProvider(); + setCriteria(criteria); + + Column deleteColumn = addColumn(entry -> { + if (entry.getDeletionReason() != null) { + return entry.getDeletionReason() + (entry.getOtherDeletionReason() != null ? ": " + entry.getOtherDeletionReason() : ""); + } else { + return "-"; + } + }); + deleteColumn.setId(DELETE_REASON_COLUMN); + deleteColumn.setSortable(false); + deleteColumn.setCaption(I18nProperties.getCaption(Captions.deletionReason)); + + initColumns(); + + addItemClickListener(new ShowDetailsListener<>(AefiIndexDto.UUID, e -> ControllerProvider.getAefiController().navigateToAefi(e.getUuid()))); + addItemClickListener(new ShowDetailsListener<>(AefiIndexDto.IMMUNIZATION_UUID, e -> { + ControllerProvider.getImmunizationController().navigateToImmunization(e.getImmunizationUuid()); + })); + addItemClickListener(new ShowDetailsListener<>(AefiIndexDto.PERSON_UUID, e -> { + ControllerProvider.getImmunizationController().navigateToView(ImmunizationPersonView.VIEW_NAME, e.getImmunizationUuid()); + })); + } + + public void reload() { + getDataProvider().refreshAll(); + } + + private void initColumns() { + setColumns( + AefiIndexDto.UUID, + AefiIndexDto.IMMUNIZATION_UUID, + AefiIndexDto.PERSON_UUID, + AefiIndexDto.PERSON_FIRST_NAME, + AefiIndexDto.PERSON_LAST_NAME, + AefiIndexDto.DISEASE, + AefiIndexDto.AGE_AND_BIRTH_DATE, + AefiIndexDto.SEX, + AefiIndexDto.REGION, + AefiIndexDto.DISTRICT, + AefiIndexDto.SERIOUS, + AefiIndexDto.PRIMARY_VACCINE_NAME, + AefiIndexDto.OUTCOME, + AefiIndexDto.VACCINATION_DATE, + AefiIndexDto.REPORT_DATE, + AefiIndexDto.START_DATE_TIME, + AefiIndexDto.ADVERSE_EVENTS, + DELETE_REASON_COLUMN); + + ((Column) getColumn(AefiIndexDto.UUID)).setRenderer(new UuidRenderer()); + ((Column) getColumn(AefiIndexDto.IMMUNIZATION_UUID)).setRenderer(new UuidRenderer()); + ((Column) getColumn(AefiIndexDto.PERSON_UUID)).setRenderer(new UuidRenderer()); + + ((Column) getColumn(AefiIndexDto.AGE_AND_BIRTH_DATE)).setRenderer( + value -> value == null + ? "" + : PersonHelper.getAgeAndBirthdateString( + value.getAge(), + value.getAgeType(), + value.getDateOfBirthDD(), + value.getDateOfBirthMM(), + value.getDateOfBirthYYYY()), + new TextRenderer()); + + ((Column) getColumn(AefiIndexDto.VACCINATION_DATE)).setRenderer(new DateRenderer(DateFormatHelper.getDateFormat())); + ((Column) getColumn(AefiIndexDto.REPORT_DATE)).setRenderer(new DateRenderer(DateFormatHelper.getDateFormat())); + ((Column) getColumn(AefiIndexDto.START_DATE_TIME)).setRenderer(new DateRenderer(DateFormatHelper.getDateFormat())); + + for (Column column : getColumns()) { + column.setCaption(I18nProperties.findPrefixCaptionWithDefault(column.getId(), column.getCaption(), AefiIndexDto.I18N_PREFIX)); + column.setStyleGenerator(FieldAccessColumnStyleGenerator.getDefault(getBeanType(), column.getId())); + } + + if (UiUtil.enabled(FeatureType.HIDE_JURISDICTION_FIELDS)) { + getColumn(AefiIndexDto.REGION).setHidden(true); + getColumn(AefiIndexDto.DISTRICT).setHidden(true); + } + } + + private void setLazyDataProvider() { + + setLazyDataProvider(FacadeProvider.getAefiFacade()::getIndexList, FacadeProvider.getAefiFacade()::count); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiPrimarySuspectVaccinationSelectionField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiPrimarySuspectVaccinationSelectionField.java new file mode 100644 index 00000000000..6f86f5792bc --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiPrimarySuspectVaccinationSelectionField.java @@ -0,0 +1,146 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines; + +import java.util.List; +import java.util.function.Consumer; + +import com.vaadin.shared.ui.grid.HeightMode; +import com.vaadin.ui.Component; +import com.vaadin.ui.CustomField; +import com.vaadin.ui.Grid; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.caze.VaccineManufacturer; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.VaadinUiUtil; + +public class AefiPrimarySuspectVaccinationSelectionField extends CustomField { + + private VerticalLayout mainLayout; + private Grid vaccinationGrid; + private final String infoAefiSelectPrimarySuspectVaccine; + private Consumer selectionChangeCallback; + private List vaccinationDtoList; + private VaccinationDto primarySuspectVaccine; + + public AefiPrimarySuspectVaccinationSelectionField(List vaccinationDtoList, VaccinationDto primarySuspectVaccine) { + this.vaccinationDtoList = vaccinationDtoList; + this.primarySuspectVaccine = primarySuspectVaccine; + this.infoAefiSelectPrimarySuspectVaccine = I18nProperties.getString(Strings.infoAefiSelectPrimarySuspectVaccine); + + initializeGrid(); + } + + private void addInfoComponent() { + mainLayout.addComponent(VaadinUiUtil.createInfoComponent(infoAefiSelectPrimarySuspectVaccine)); + } + + public void initializeGrid() { + + vaccinationGrid = new Grid<>(); + vaccinationGrid.setSizeFull(); + vaccinationGrid.setSelectionMode(Grid.SelectionMode.SINGLE); + vaccinationGrid.setHeightByRows(5); + vaccinationGrid.setHeightMode(HeightMode.ROW); + + vaccinationGrid.setItems(vaccinationDtoList); + + if (primarySuspectVaccine != null) { + vaccinationGrid.select(primarySuspectVaccine); + } + + vaccinationGrid.addColumn(vaccinationDto -> DateFormatHelper.formatDate(vaccinationDto.getVaccinationDate())) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINATION_DATE)); + + vaccinationGrid + .addColumn( + vaccinationDto -> Vaccine.OTHER.equals(vaccinationDto.getVaccineName()) + ? vaccinationDto.getOtherVaccineName() + : vaccinationDto.getVaccineName()) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_NAME)); + + vaccinationGrid + .addColumn( + vaccinationDto -> VaccineManufacturer.OTHER.equals(vaccinationDto.getVaccineManufacturer()) + ? vaccinationDto.getOtherVaccineManufacturer() + : vaccinationDto.getVaccineManufacturer()) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_MANUFACTURER)); + + vaccinationGrid.addColumn(VaccinationDto::getVaccineType) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_TYPE)); + + vaccinationGrid.addColumn(VaccinationDto::getVaccineDose) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_DOSE)); + + vaccinationGrid.addSelectionListener(e -> { + + if (selectionChangeCallback != null) { + selectionChangeCallback.accept(!e.getAllSelectedItems().isEmpty()); + } + }); + } + + @Override + protected Component initContent() { + + mainLayout = new VerticalLayout(); + mainLayout.setSpacing(true); + mainLayout.setMargin(false); + mainLayout.setSizeUndefined(); + mainLayout.setWidth(100, Unit.PERCENTAGE); + CssStyles.style(mainLayout, CssStyles.VSPACE_2); + + addInfoComponent(); + + mainLayout.addComponent(vaccinationGrid); + + return mainLayout; + } + + @Override + protected void doSetValue(VaccinationDto vaccinationDto) { + if (vaccinationDto != null) { + vaccinationGrid.select(vaccinationDto); + } + } + + @Override + public VaccinationDto getValue() { + if (vaccinationGrid != null) { + VaccinationDto value = vaccinationGrid.getSelectedItems().stream().findFirst().orElse(null); + return value; + } + + return null; + } + + public void setSelectionChangeCallback(Consumer callback) { + this.selectionChangeCallback = callback; + } + + public Grid getVaccinationGrid() { + return vaccinationGrid; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java new file mode 100644 index 00000000000..03742120e3a --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java @@ -0,0 +1,200 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines; + +import java.util.function.Consumer; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.icons.VaadinIcons; +import com.vaadin.shared.ui.ContentMode; +import com.vaadin.ui.Label; +import com.vaadin.v7.data.util.BeanItemContainer; +import com.vaadin.v7.ui.Table; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.utils.DateFormatHelper; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.caze.AbstractTableField; + +public class AefiVaccinationsField extends AbstractTableField { + + private AefiDto aefiDto; + private VaccinationDto primarySuspectVaccination; + private final UiFieldAccessCheckers fieldAccessCheckers; + + public AefiVaccinationsField(UiFieldAccessCheckers fieldAccessCheckers) { + super(fieldAccessCheckers); + this.fieldAccessCheckers = fieldAccessCheckers; + + updateAddButtonCaption(); + } + + @Override + public Class getEntryType() { + return VaccinationDto.class; + } + + @Override + protected void editEntry(VaccinationDto entry, boolean create, Consumer commitCallback) { + + if (create) { + ControllerProvider.getAefiController().selectPrimarySuspectVaccination(aefiDto, this::selectPrimarySuspectVaccination); + } + } + + public void updateAddButtonCaption() { + getAddButton().setCaption(I18nProperties.getCaption(Captions.actionAefiSelectPrimarySuspectVaccination)); + } + + /* + * @Override + * protected VaccinationDto createEntry() { + * UserDto user = UserProvider.getCurrent().getUser(); + * return VaccinationDto.build(user.toReference()); + * } + */ + + /* + * @Override + * protected Table createTable() { + * Table table = super.createTable(); + * table.addGeneratedColumn(VaccinationDto.UUID, (Table.ColumnGenerator) (source, itemId, columnId) -> { + * Label textField = new Label(DataHelper.getShortUuid(((EntityDto) itemId).getUuid())); + * return textField; + * }); + * table.addGeneratedColumn(VaccinationDto.VACCINATION_DATE, (Table.ColumnGenerator) (source, itemId, columnId) -> { + * Label textField = new Label(DateFormatHelper.formatDate(((VaccinationDto) itemId).getVaccinationDate())); + * return textField; + * }); + * return table; + * } + */ + + @Override + protected void updateColumns() { + Table table = getTable(); + + table.addGeneratedColumn(Captions.aefiVaccinationsPrimaryVaccine, (Table.ColumnGenerator) (source, itemId, columnId) -> { + VaccinationDto vaccinationDto = (VaccinationDto) itemId; + return new Label( + primarySuspectVaccination != null && StringUtils.equals(vaccinationDto.getUuid(), primarySuspectVaccination.getUuid()) + ? VaadinIcons.CHECK_CIRCLE.getHtml() + : "", + ContentMode.HTML); + }); + + table.addGeneratedColumn(Captions.aefiVaccinationsVaccineDetails, (Table.ColumnGenerator) (source, item, columnId) -> { + VaccinationDto vaccinationDto = (VaccinationDto) item; + + StringBuilder vaccineDetailsBuilder = new StringBuilder(); + vaccineDetailsBuilder.append(vaccinationDto.getVaccineManufacturer()); + vaccineDetailsBuilder.append(", ") + .append( + Vaccine.OTHER.equals(vaccinationDto.getVaccineName()) ? vaccinationDto.getOtherVaccineName() : vaccinationDto.getVaccineName()); + + if (vaccinationDto.getVaccinationDate() != null) { + vaccineDetailsBuilder.append(", ").append(DateFormatHelper.formatDate(vaccinationDto.getVaccinationDate())); + } + + if (!StringUtils.isBlank(vaccinationDto.getVaccineDose())) { + vaccineDetailsBuilder.append(", Dose ").append(vaccinationDto.getVaccineDose()); + } + + return vaccineDetailsBuilder.toString(); + }); + + table.addGeneratedColumn(Captions.aefiVaccinationsDiluentBatchLotNumber, (Table.ColumnGenerator) (source, item, columnId) -> { + VaccinationDto vaccinationDto = (VaccinationDto) item; + return "-"; + }); + + table.addGeneratedColumn(Captions.aefiVaccinationsDiluentExpiryDate, (Table.ColumnGenerator) (source, item, columnId) -> { + VaccinationDto vaccinationDto = (VaccinationDto) item; + return "-"; + }); + + table.addGeneratedColumn(Captions.aefiVaccinationsDiluentTimeOfReconstitution, (Table.ColumnGenerator) (source, item, columnId) -> { + VaccinationDto vaccinationDto = (VaccinationDto) item; + return "-"; + }); + + table.setVisibleColumns( + Captions.aefiVaccinationsPrimaryVaccine, + Captions.aefiVaccinationsVaccineDetails, + Captions.aefiVaccinationsDiluentBatchLotNumber, + Captions.aefiVaccinationsDiluentExpiryDate, + Captions.aefiVaccinationsDiluentTimeOfReconstitution); + + for (Object columnId : table.getVisibleColumns()) { + if (columnId.equals(ACTION_COLUMN_ID)) { + table.setColumnHeader(columnId, " "); + } else if (columnId.equals(Captions.aefiVaccinationsPrimaryVaccine)) { + table.setColumnHeader(columnId, I18nProperties.getCaption(Captions.aefiVaccinationsPrimaryVaccine)); + } else if (columnId.equals(Captions.aefiVaccinationsVaccineDetails)) { + table.setColumnHeader(columnId, I18nProperties.getCaption(Captions.aefiVaccinationsVaccineDetails)); + } else if (columnId.equals(Captions.aefiVaccinationsDiluentBatchLotNumber)) { + table.setColumnHeader(columnId, I18nProperties.getCaption(Captions.aefiVaccinationsDiluentBatchLotNumber)); + } else if (columnId.equals(Captions.aefiVaccinationsDiluentExpiryDate)) { + table.setColumnHeader(columnId, I18nProperties.getCaption(Captions.aefiVaccinationsDiluentExpiryDate)); + } else if (columnId.equals(Captions.aefiVaccinationsDiluentTimeOfReconstitution)) { + table.setColumnHeader(columnId, I18nProperties.getCaption(Captions.aefiVaccinationsDiluentTimeOfReconstitution)); + } else { + table.setColumnHeader(columnId, I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, (String) columnId)); + } + } + } + + public void refreshTable() { + Table table = getTable(); + + BeanItemContainer container = getContainer(); + if (container == null) { + return; + } + + container.removeAllItems(); + container.addAll(aefiDto.getVaccinations()); + table.refreshRowCache(); + } + + public void setAefiDto(AefiDto aefiDto) { + this.aefiDto = aefiDto; + } + + public void selectPrimarySuspectVaccination(VaccinationDto vaccinationDto) { + primarySuspectVaccination = vaccinationDto; + refreshTable(); + } + + @Override + protected boolean isEmpty(VaccinationDto entry) { + return false; + } + + @Override + protected boolean isModified(VaccinationDto oldEntry, VaccinationDto newEntry) { + return false; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java new file mode 100644 index 00000000000..659ae920839 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java @@ -0,0 +1,198 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines; + +import java.util.ArrayList; +import java.util.Collection; +import java.util.List; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.vaadin.ui.Alignment; +import com.vaadin.ui.Button; +import com.vaadin.ui.Component; +import com.vaadin.ui.Grid; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; +import com.vaadin.ui.themes.ValoTheme; +import com.vaadin.v7.data.Property; +import com.vaadin.v7.data.util.converter.Converter; +import com.vaadin.v7.ui.CustomField; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.caze.VaccineManufacturer; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.utils.ButtonHelper; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateFormatHelper; + +@SuppressWarnings({ + "serial", + "rawtypes" }) +public class AefiVaccinationsField_2 extends CustomField { + + private final Logger logger = LoggerFactory.getLogger(getClass()); + + private VerticalLayout mainLayout; + private Label captionLabel; + private Button addButton; + private Grid vaccinationGrid; + private List value = new ArrayList<>(); + private AefiDto aefiDto; + private VaccinationDto primarySuspectVaccination; + protected UiFieldAccessCheckers fieldAccessCheckers; + + public AefiVaccinationsField_2(UiFieldAccessCheckers fieldAccessCheckers) { + this.fieldAccessCheckers = fieldAccessCheckers; + + getContent(); + //setValue(value); + } + + public void initializeGrid() { + + vaccinationGrid = new Grid<>(); + vaccinationGrid.setSizeFull(); + vaccinationGrid.setSelectionMode(Grid.SelectionMode.SINGLE); + /* vaccinationGrid.setHeightMode(HeightMode.ROW); */ + + vaccinationGrid.addColumn(vaccinationDto -> DateFormatHelper.formatDate(vaccinationDto.getVaccinationDate())) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINATION_DATE)); + + vaccinationGrid + .addColumn( + vaccinationDto -> Vaccine.OTHER.equals(vaccinationDto.getVaccineName()) + ? vaccinationDto.getOtherVaccineName() + : vaccinationDto.getVaccineName()) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_NAME)); + + vaccinationGrid + .addColumn( + vaccinationDto -> VaccineManufacturer.OTHER.equals(vaccinationDto.getVaccineManufacturer()) + ? vaccinationDto.getOtherVaccineManufacturer() + : vaccinationDto.getVaccineManufacturer()) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_MANUFACTURER)); + + vaccinationGrid.addColumn(VaccinationDto::getVaccineType) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_TYPE)); + + vaccinationGrid.addColumn(VaccinationDto::getVaccineDose) + .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_DOSE)); + + vaccinationGrid.setStyleGenerator(vaccinationDto -> { + if (primarySuspectVaccination != null) { + return vaccinationDto.getUuid().equals(primarySuspectVaccination.getUuid()) ? CssStyles.GRID_ROW_SELECTED : null; + } + return null; + }); + } + + @Override + protected Component initContent() { + mainLayout = new VerticalLayout(); + mainLayout.setSpacing(false); + mainLayout.setMargin(false); + + HorizontalLayout headerLayout = new HorizontalLayout(); + { + headerLayout.setWidth(100, Unit.PERCENTAGE); + + captionLabel = new Label(getCaption()); + captionLabel.setSizeUndefined(); + headerLayout.addComponent(captionLabel); + headerLayout.setComponentAlignment(captionLabel, Alignment.BOTTOM_LEFT); + headerLayout.setExpandRatio(captionLabel, 0); + + addButton = ButtonHelper.createButton(Captions.actionAefiSelectPrimarySuspectVaccination, (event) -> { + ControllerProvider.getAefiController().selectPrimarySuspectVaccination(aefiDto, this::selectPrimarySuspectVaccination); + }, ValoTheme.BUTTON_LINK); + headerLayout.addComponent(addButton); + headerLayout.setComponentAlignment(addButton, Alignment.BOTTOM_RIGHT); + headerLayout.setExpandRatio(addButton, 1); + } + mainLayout.addComponent(headerLayout); + + initializeGrid(); + mainLayout.addComponent(vaccinationGrid); + + return mainLayout; + } + + @Override + public Class getType() { + return Collection.class; + } + + @Override + public void setPropertyDataSource(Property newDataSource) { + super.setPropertyDataSource(newDataSource); + } + + /* + * @Override + * protected void setValue(Collection newFieldValue, boolean repaintIsNotNeeded, boolean ignoreReadOnly) + * throws ReadOnlyException, Converter.ConversionException, Validator.InvalidValueException { + * super.setValue(newFieldValue, repaintIsNotNeeded, ignoreReadOnly); + * value = new ArrayList<>(newFieldValue); + * vaccinationGrid.setItems(newFieldValue); + * fireValueChange(repaintIsNotNeeded); + * } + */ + + @Override + public void setValue(Collection newFieldValue) throws ReadOnlyException, Converter.ConversionException { + value = new ArrayList<>(newFieldValue); + vaccinationGrid.setItems(newFieldValue); + + super.setValue(newFieldValue); + } + + /* + * @Override + * protected void doSetValue(Collection collection) { + * value = new ArrayList<>(collection); + * vaccinationGrid.setItems(collection); + * } + */ + + @Override + public Collection getValue() { + return value; + } + + public void setAefiDto(AefiDto aefiDto) { + this.aefiDto = aefiDto; + } + + public void setPrimarySuspectVaccination(VaccinationDto primarySuspectVaccination) { + this.primarySuspectVaccination = primarySuspectVaccination; + } + + public void selectPrimarySuspectVaccination(VaccinationDto vaccinationDto) { + primarySuspectVaccination = vaccinationDto; + vaccinationGrid.select(primarySuspectVaccination); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java new file mode 100644 index 00000000000..4eade7d908a --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java @@ -0,0 +1,149 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form; + +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.ABSCESS; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.ANAPHYLAXIS; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.ENCEPHALOPATHY; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.FEVERISH_FEELING; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.I18N_PREFIX; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.OTHER_ADVERSE_EVENT_DETAILS; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.SEIZURES; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.SEIZURE_TYPE; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.SEPSIS; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.SEVERE_LOCAL_REACTION; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.THROMBOCYTOPENIA; +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto.TOXIC_SHOCK_SYNDROME; +import static de.symeda.sormas.ui.utils.CssStyles.H3; +import static de.symeda.sormas.ui.utils.LayoutUtil.divCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidColumn; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRow; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowLocs; +import static de.symeda.sormas.ui.utils.LayoutUtil.loc; +import static de.symeda.sormas.ui.utils.LayoutUtil.locs; + +import java.util.Arrays; + +import com.vaadin.ui.CustomLayout; +import com.vaadin.ui.Label; +import com.vaadin.v7.ui.CheckBox; +import com.vaadin.v7.ui.Field; +import com.vaadin.v7.ui.TextArea; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto; +import de.symeda.sormas.api.i18n.Descriptions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; +import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.FieldHelper; + +public class AdverseEventsForm extends AbstractEditForm { + + private static final long serialVersionUID = 5081846814610543073L; + + private static final String ADVERSE_EVENTS_HEADINGS_LOC = "adverseEventsHeadingLoc"; + private static final String EMPTY_LABEL_LOC = "emptyLabelLoc"; + + //@formatter:off + private static final String HTML_LAYOUT = + loc(ADVERSE_EVENTS_HEADINGS_LOC) + + fluidRow( + fluidColumn(6, 0, locs( + SEVERE_LOCAL_REACTION, SEIZURES, + ABSCESS, SEPSIS, ENCEPHALOPATHY, + TOXIC_SHOCK_SYNDROME, THROMBOCYTOPENIA, + ANAPHYLAXIS, FEVERISH_FEELING)), + fluidColumn(5, 0, + divCss(CssStyles.VSPACE_TOP_4, + fluidRowLocs(4, SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS, + 6, SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT, + 2, EMPTY_LABEL_LOC)) + + divCss(CssStyles.VSPACE_TOP_3, + fluidRowLocs(10, SEIZURE_TYPE))) + ) + + loc(OTHER_ADVERSE_EVENT_DETAILS); + //@formatter:on + + public AdverseEventsForm(FieldVisibilityCheckers fieldVisibilityCheckers, UiFieldAccessCheckers fieldAccessCheckers) { + super(AdverseEventsDto.class, I18N_PREFIX, true, fieldVisibilityCheckers, fieldAccessCheckers); + } + + @Override + protected String createHtmlLayout() { + return HTML_LAYOUT; + } + + @Override + protected void addFields() { + Label adverseEventsHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiAdverseEvents)); + adverseEventsHeadingLabel.addStyleName(H3); + getContent().addComponent(adverseEventsHeadingLabel, ADVERSE_EVENTS_HEADINGS_LOC); + + addFields( + SEVERE_LOCAL_REACTION, + SEIZURES, + ABSCESS, + SEPSIS, + ENCEPHALOPATHY, + TOXIC_SHOCK_SYNDROME, + THROMBOCYTOPENIA, + ANAPHYLAXIS, + FEVERISH_FEELING); + + addField(SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS, CheckBox.class); + addField(SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT, CheckBox.class); + + Label emptyLabel = new Label(""); + emptyLabel.addStyleName(H3); + getContent().addComponent(emptyLabel, EMPTY_LABEL_LOC); + + addField(SEIZURE_TYPE); + + TextArea otherAdverseEvents = addField(OTHER_ADVERSE_EVENT_DETAILS, TextArea.class); + otherAdverseEvents.setRows(6); + otherAdverseEvents.setDescription( + I18nProperties.getPrefixDescription(AdverseEventsDto.I18N_PREFIX, OTHER_ADVERSE_EVENT_DETAILS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + initializeVisibilitiesAndAllowedVisibilities(); + initializeAccessAndAllowedAccesses(); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + Arrays.asList(SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS, SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT), + SEVERE_LOCAL_REACTION, + Arrays.asList(AdverseEventState.YES), + true); + + FieldHelper.setVisibleWhen(getFieldGroup(), SEIZURE_TYPE, SEIZURES, Arrays.asList(AdverseEventState.YES), true); + } + + @Override + protected F addFieldToLayout(CustomLayout layout, String propertyId, F field) { + field.addValueChangeListener(e -> fireValueChange(false)); + + return super.addFieldToLayout(layout, propertyId, field); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java new file mode 100644 index 00000000000..3bbd3e2e4c0 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java @@ -0,0 +1,303 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form; + +import static de.symeda.sormas.ui.utils.CssStyles.FORCE_CAPTION; +import static de.symeda.sormas.ui.utils.CssStyles.H3; +import static de.symeda.sormas.ui.utils.CssStyles.H4; +import static de.symeda.sormas.ui.utils.LayoutUtil.divCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRow; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowLocs; +import static de.symeda.sormas.ui.utils.LayoutUtil.loc; + +import java.util.Arrays; +import java.util.function.Consumer; + +import com.vaadin.ui.Button; +import com.vaadin.ui.Label; +import com.vaadin.ui.themes.ValoTheme; +import com.vaadin.v7.data.util.converter.Converter; +import com.vaadin.v7.ui.ComboBox; +import com.vaadin.v7.ui.DateField; +import com.vaadin.v7.ui.PasswordField; +import com.vaadin.v7.ui.TextArea; +import com.vaadin.v7.ui.TextField; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiOutcome; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeriousAefiReason; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.Descriptions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField; +import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.ButtonHelper; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldHelper; +import de.symeda.sormas.ui.utils.NullableOptionGroup; +import de.symeda.sormas.ui.utils.UserField; + +@SuppressWarnings("deprecation") +public class AefiDataForm extends AbstractEditForm { + + private static final String ASSIGN_NEW_AEFI_ID_LOC = "assignNewAefiIdLoc"; + private static final String REPORTING_INFORMATION_HEADING_LOC = "reportingInformationHeadingLoc"; + private static final String PATIENTS_IDENTIFICATION_HEADING_LOC = "patientsIdentificationHeadingLoc"; + private static final String PATIENTS_IDENTIFICATION_AGE_AT_ONSET = "patientsIdentificationAgeAtOnsetHeadingLoc"; + private static final String VACCINATIONS_HEADING_LOC = "vaccinationsHeadingLoc"; + private static final String ADVERSE_EVENTS_HEADING_LOC = "adverseEventsHeadingLoc"; + private static final String FIRST_DECISION_LEVEL_HEADING_LOC = "firstDecisionLevelHeadingLoc"; + private static final String NATIONAL_DECISION_LEVEL_HEADING_LOC = "nationalDecisionLevelHeadingLoc"; + private static final String REPORTERS_INFORMATION_HEADING_LOC = "reportersInformationHeadingLoc"; + + //@formatter:off + private static final String HTML_LAYOUT = + divCss(CssStyles.VIEW_SECTION_MARGIN_TOP_4_MARGIN_X_4, + loc(REPORTING_INFORMATION_HEADING_LOC) + + fluidRowLocs(4, AefiDto.UUID, 3, AefiDto.REPORT_DATE, 3, AefiDto.REPORTING_USER, 2, "") + + fluidRowLocs(4, AefiDto.REPORTING_ID_NUMBER, 3, ASSIGN_NEW_AEFI_ID_LOC) + ) + + divCss(CssStyles.VIEW_SECTION_MARGIN_X_4 + " " + CssStyles.VSPACE_TOP_3, + loc(PATIENTS_IDENTIFICATION_HEADING_LOC) + + fluidRowLocs(AefiDto.PREGNANT, AefiDto.TRIMESTER, AefiDto.LACTATING) + + loc(PATIENTS_IDENTIFICATION_AGE_AT_ONSET) + + fluidRow( + fluidRowLocs(AefiDto.ONSET_AGE_YEARS, AefiDto.ONSET_AGE_MONTHS, AefiDto.ONSET_AGE_DAYS), + fluidRowLocs(6, AefiDto.AGE_GROUP) + ) + ) + + divCss(CssStyles.VIEW_SECTION_MARGIN_X_4 + " " + CssStyles.VSPACE_TOP_3, + loc(VACCINATIONS_HEADING_LOC) + + fluidRowLocs(AefiDto.VACCINATIONS) + ) + + divCss(CssStyles.VIEW_SECTION_MARGIN_X_4 + " " + CssStyles.VSPACE_TOP_3, + loc(ADVERSE_EVENTS_HEADING_LOC) + + fluidRowLocs(AefiDto.ADVERSE_EVENTS) + + fluidRowLocs(6, AefiDto.START_DATE_TIME) + + fluidRowLocs(AefiDto.AEFI_DESCRIPTION) + + fluidRowLocs(4, AefiDto.SERIOUS, 4, AefiDto.SERIOUS_REASON, 4, AefiDto.SERIOUS_REASON_DETAILS) + + fluidRowLocs(AefiDto.OUTCOME) + + fluidRowLocs(4, AefiDto.DEATH_DATE, 4, AefiDto.AUTOPSY_DONE) + + fluidRowLocs(AefiDto.PAST_MEDICAL_HISTORY) + ) + + divCss(CssStyles.VIEW_SECTION_MARGIN_X_4 + " " + CssStyles.VSPACE_TOP_3, + loc(FIRST_DECISION_LEVEL_HEADING_LOC) + + fluidRowLocs(4, AefiDto.INVESTIGATION_NEEDED, 4, AefiDto.INVESTIGATION_PLANNED_DATE) + ) + + divCss(CssStyles.VIEW_SECTION_MARGIN_X_4 + " " + CssStyles.VSPACE_2 + " " + CssStyles.VSPACE_TOP_3, + loc(NATIONAL_DECISION_LEVEL_HEADING_LOC) + + fluidRowLocs(4, AefiDto.RECEIVED_AT_NATIONAL_LEVEL_DATE, 4, AefiDto.WORLD_WIDE_ID) + + fluidRowLocs(AefiDto.NATIONAL_LEVEL_COMMENT) + ); + //@formatter:on + + private boolean isCreateAction; + private final Consumer actionCallback; + private AefiVaccinationsField vaccinationsField; + + public AefiDataForm(boolean isCreateAction, boolean isPseudonymized, boolean inJurisdiction, Consumer actionCallback) { + super( + AefiDto.class, + AefiDto.I18N_PREFIX, + false, + FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), + UiFieldAccessCheckers.forDataAccessLevel(UserProvider.getCurrent().getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + + this.isCreateAction = isCreateAction; + this.actionCallback = actionCallback; + + if (isCreateAction) { + hideValidationUntilNextCommit(); + } + + addFields(); + } + + @Override + protected String createHtmlLayout() { + return HTML_LAYOUT; + } + + @Override + protected void addFields() { + + Label reportingInformationHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiReportingInformation)); + reportingInformationHeadingLabel.addStyleName(H3); + getContent().addComponent(reportingInformationHeadingLabel, REPORTING_INFORMATION_HEADING_LOC); + + if (isCreateAction) { + addField(AefiDto.UUID, PasswordField.class); + } else { + addField(AefiDto.UUID); + } + addField(AefiDto.REPORT_DATE, DateField.class); + addField(ImmunizationDto.REPORTING_USER, UserField.class); + + TextField reportIdField = addField(AefiDto.REPORTING_ID_NUMBER, TextField.class); + /* + * reportIdField.setInvalidCommitted(true); + * reportIdField.setMaxLength(24); + * style(reportIdField, ERROR_COLOR_PRIMARY); + */ + + // Button to automatically assign a new reporting ID + Button assignNewReportingIdNumberButton = ButtonHelper.createButton(Captions.actionAefiAssignNewReportingIdNumber, e -> { + }, ValoTheme.BUTTON_DANGER, FORCE_CAPTION); + + getContent().addComponent(assignNewReportingIdNumberButton, ASSIGN_NEW_AEFI_ID_LOC); + assignNewReportingIdNumberButton.setVisible(false); + + Label patientsIdentificationHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiPatientsIdentification)); + patientsIdentificationHeadingLabel.addStyleName(H3); + getContent().addComponent(patientsIdentificationHeadingLabel, PATIENTS_IDENTIFICATION_HEADING_LOC); + + addField(AefiDto.PREGNANT, NullableOptionGroup.class); + addField(AefiDto.TRIMESTER, NullableOptionGroup.class); + addField(AefiDto.LACTATING, NullableOptionGroup.class); + + Label patientsAgeAtOnsetHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiPatientsAgeAtOnset)); + patientsAgeAtOnsetHeadingLabel.addStyleName(H4); + getContent().addComponent(patientsAgeAtOnsetHeadingLabel, PATIENTS_IDENTIFICATION_AGE_AT_ONSET); + + TextField onsetAgeYearsField = addField(AefiDto.ONSET_AGE_YEARS, TextField.class); + onsetAgeYearsField + .setConversionError(I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, onsetAgeYearsField.getCaption())); + + TextField onsetAgeMonthsField = addField(AefiDto.ONSET_AGE_MONTHS, TextField.class); + onsetAgeMonthsField + .setConversionError(I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, onsetAgeMonthsField.getCaption())); + + TextField onsetAgeDaysField = addField(AefiDto.ONSET_AGE_DAYS, TextField.class); + onsetAgeDaysField + .setConversionError(I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, onsetAgeDaysField.getCaption())); + + addField(AefiDto.AGE_GROUP, ComboBox.class); + + Label vaccinationsHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiVaccinations)); + vaccinationsHeadingLabel.addStyleName(H3); + getContent().addComponent(vaccinationsHeadingLabel, VACCINATIONS_HEADING_LOC); + + vaccinationsField = addField(AefiDto.VACCINATIONS, AefiVaccinationsField.class); + + addField(AefiDto.ADVERSE_EVENTS, AdverseEventsForm.class).setCaption(null); + + final DateTimeField startDateField = addField(AefiDto.START_DATE_TIME, DateTimeField.class); + startDateField.setInvalidCommitted(false); + + TextArea aefiDescriptionField = addField(AefiDto.AEFI_DESCRIPTION, TextArea.class); + aefiDescriptionField.setRows(6); + aefiDescriptionField.setDescription( + I18nProperties.getPrefixDescription(AefiDto.I18N_PREFIX, AefiDto.AEFI_DESCRIPTION, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + addField(AefiDto.SERIOUS, NullableOptionGroup.class); + addField(AefiDto.SERIOUS_REASON, ComboBox.class); + addField(AefiDto.SERIOUS_REASON_DETAILS, TextField.class); + addField(AefiDto.OUTCOME, NullableOptionGroup.class); + addField(AefiDto.DEATH_DATE, DateField.class); + addField(AefiDto.AUTOPSY_DONE, NullableOptionGroup.class); + + TextArea pastMedicalHistoryField = addField(AefiDto.PAST_MEDICAL_HISTORY, TextArea.class); + pastMedicalHistoryField.setRows(6); + pastMedicalHistoryField.setDescription( + I18nProperties.getPrefixDescription(AefiDto.I18N_PREFIX, AefiDto.PAST_MEDICAL_HISTORY, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + Label firstDecisionLevelHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiFirstDecisionLevel)); + firstDecisionLevelHeadingLabel.addStyleName(H3); + getContent().addComponent(firstDecisionLevelHeadingLabel, FIRST_DECISION_LEVEL_HEADING_LOC); + + addField(AefiDto.INVESTIGATION_NEEDED, NullableOptionGroup.class); + addField(AefiDto.INVESTIGATION_PLANNED_DATE, DateField.class); + + Label nationalDecisionLevelHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiNationalDecisionLevel)); + nationalDecisionLevelHeadingLabel.addStyleName(H3); + getContent().addComponent(nationalDecisionLevelHeadingLabel, NATIONAL_DECISION_LEVEL_HEADING_LOC); + + addField(AefiDto.RECEIVED_AT_NATIONAL_LEVEL_DATE, DateField.class); + addField(AefiDto.WORLD_WIDE_ID, TextField.class); + + TextArea nationalLevelCommentField = addField(AefiDto.NATIONAL_LEVEL_COMMENT, TextArea.class); + nationalLevelCommentField.setRows(6); + nationalLevelCommentField.setDescription( + I18nProperties.getPrefixDescription(AefiDto.I18N_PREFIX, AefiDto.NATIONAL_LEVEL_COMMENT, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + //set visibility, read only and required status + FieldHelper.setVisibleWhen(getFieldGroup(), AefiDto.TRIMESTER, AefiDto.PREGNANT, Arrays.asList(YesNoUnknown.YES), true); + FieldHelper.setVisibleWhen(getFieldGroup(), AefiDto.SERIOUS_REASON, AefiDto.SERIOUS, Arrays.asList(YesNoUnknown.YES), true); + FieldHelper + .setVisibleWhen(getFieldGroup(), AefiDto.SERIOUS_REASON_DETAILS, AefiDto.SERIOUS_REASON, Arrays.asList(SeriousAefiReason.OTHER), true); + FieldHelper.setVisibleWhen( + getFieldGroup(), + Arrays.asList(AefiDto.DEATH_DATE, AefiDto.AUTOPSY_DONE), + AefiDto.OUTCOME, + Arrays.asList(AefiOutcome.DIED), + true); + FieldHelper + .setVisibleWhen(getFieldGroup(), AefiDto.INVESTIGATION_PLANNED_DATE, AefiDto.INVESTIGATION_NEEDED, Arrays.asList(YesNoUnknown.YES), true); + + setReadOnly(true, AefiDto.UUID, AefiDto.REPORTING_USER); + + setRequired(true, AefiDto.REPORT_DATE, AefiDto.REPORTING_ID_NUMBER, AefiDto.SERIOUS, AefiDto.OUTCOME); + FieldHelper.setRequiredWhen(getFieldGroup(), AefiDto.SERIOUS_REASON, Arrays.asList(AefiDto.SERIOUS_REASON), Arrays.asList(YesNoUnknown.YES)); + } + + @Override + public void attach() { + super.attach(); + + vaccinationsField.setAefiDto(getValue()); + if (getValue().getPrimarySuspectVaccine() != null) { + vaccinationsField.selectPrimarySuspectVaccination(getValue().getPrimarySuspectVaccine()); + } + } + + /* + * @Override + * public AefiDto getValue() { + * return super.getValue(); + * } + */ + + @Override + public void setValue(AefiDto newFieldValue) throws ReadOnlyException, Converter.ConversionException { + super.setValue(newFieldValue); + + getValue(); + + // HACK: Binding to the fields will call field listeners that may clear/modify the values of other fields. + // this hopefully resets everything to its correct value + discard(); + } + + @Override + public void discard() throws SourceException { + super.discard(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiImmunizationInfo.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiImmunizationInfo.java new file mode 100644 index 00000000000..94341c9e0bd --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiImmunizationInfo.java @@ -0,0 +1,109 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information; + +import java.util.function.Consumer; + +import com.vaadin.ui.Alignment; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.immunization.ImmunizationListEntryDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponent; + +public class AefiImmunizationInfo extends SideComponent { + + public static final String SEPARATOR = ": "; + + private ImmunizationDto immunization; + private final VerticalLayout mainLayout; + + public AefiImmunizationInfo(ImmunizationDto immunization, Consumer actionCallback) { + super(I18nProperties.getString(Strings.entityImmunization), actionCallback); + + this.immunization = immunization; + mainLayout = new VerticalLayout(); + mainLayout.setWidth(100, Unit.PERCENTAGE); + mainLayout.setMargin(false); + mainLayout.setSpacing(false); + + buildMainLayout(); + addComponent(mainLayout); + } + + public void buildMainLayout() { + + HorizontalLayout uuidReportLayout = new HorizontalLayout(); + uuidReportLayout.setMargin(false); + uuidReportLayout.setSpacing(true); + + Label immunizationUuidLabel = new Label(DataHelper.getShortUuid(immunization.getUuid())); + immunizationUuidLabel.setDescription(immunization.getUuid()); + CssStyles.style(immunizationUuidLabel, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + uuidReportLayout.addComponent(immunizationUuidLabel); + + Label diseaseLabel = new Label(DataHelper.toStringNullable(immunization.getDisease())); + CssStyles.style(diseaseLabel, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + uuidReportLayout.addComponent(diseaseLabel); + + uuidReportLayout.setWidthFull(); + uuidReportLayout.setComponentAlignment(immunizationUuidLabel, Alignment.MIDDLE_LEFT); + uuidReportLayout.setComponentAlignment(diseaseLabel, Alignment.MIDDLE_RIGHT); + mainLayout.addComponent(uuidReportLayout); + + HorizontalLayout meansOfImmunizationLayout = new HorizontalLayout(); + Label meansOfImmunizationLabel = new Label( + I18nProperties.getPrefixCaption(ImmunizationListEntryDto.I18N_PREFIX, ImmunizationListEntryDto.MEANS_OF_IMMUNIZATION) + + SEPARATOR + + DataHelper.toStringNullable(immunization.getMeansOfImmunization())); + meansOfImmunizationLayout.addComponent(meansOfImmunizationLabel); + mainLayout.addComponent(meansOfImmunizationLayout); + + HorizontalLayout immunizationStatusLayout = new HorizontalLayout(); + Label immunizationStatusLabel = new Label( + I18nProperties.getPrefixCaption(ImmunizationListEntryDto.I18N_PREFIX, ImmunizationListEntryDto.IMMUNIZATION_STATUS) + + SEPARATOR + + DataHelper.toStringNullable(immunization.getImmunizationStatus())); + immunizationStatusLayout.addComponent(immunizationStatusLabel); + mainLayout.addComponent(immunizationStatusLayout); + + HorizontalLayout managementStatusLayout = new HorizontalLayout(); + Label managementStatusLabel = new Label( + I18nProperties.getPrefixCaption(ImmunizationListEntryDto.I18N_PREFIX, ImmunizationListEntryDto.IMMUNIZATION_MANAGEMENT_STATUS) + + SEPARATOR + + DataHelper.toStringNullable(immunization.getImmunizationManagementStatus())); + managementStatusLayout.addComponent(managementStatusLabel); + mainLayout.addComponent(managementStatusLayout); + + HorizontalLayout immunizationPeriodLayout = new HorizontalLayout(); + Label reportDateLabel = new Label( + I18nProperties.getPrefixCaption(ImmunizationListEntryDto.I18N_PREFIX, ImmunizationListEntryDto.IMMUNIZATION_PERIOD) + + SEPARATOR + + DateFormatHelper.buildPeriodString(immunization.getStartDate(), immunization.getEndDate())); + immunizationPeriodLayout.addComponent(reportDateLabel); + mainLayout.addComponent(immunizationPeriodLayout); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java new file mode 100644 index 00000000000..3f378ccf0b0 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java @@ -0,0 +1,142 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information; + +import com.vaadin.server.Sizeable; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.person.ApproximateAgeType; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.ui.AbstractInfoLayout; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.utils.CssStyles; + +@SuppressWarnings("serial") +public class AefiPersonInfo extends AbstractInfoLayout { + + public static final String PERSON_FULL_NAME = "personFullName"; + public static final String DISEASE = "disease"; + + private PersonDto personDto; + private Disease disease; + + public AefiPersonInfo(PersonDto personDto, Disease disease) { + super( + PersonDto.class, + UiFieldAccessCheckers.forDataAccessLevel( + UserProvider.getCurrent().getPseudonymizableDataAccessLevel(personDto.isInJurisdiction()), + personDto.isPseudonymized())); + + this.personDto = personDto; + this.disease = disease; + + setSpacing(true); + setMargin(false); + setWidth(100, Unit.PERCENTAGE); + updatePersonInfo(); + } + + private void updatePersonInfo() { + + this.removeAllComponents(); + + VerticalLayout mainLayout = new VerticalLayout(); + mainLayout.setWidth(100, Sizeable.Unit.PERCENTAGE); + CssStyles.style(mainLayout, CssStyles.PADDING_NONE); + + HorizontalLayout componentHeader = new HorizontalLayout(); + componentHeader.setMargin(false); + componentHeader.setSpacing(false); + componentHeader.setWidth(100, Sizeable.Unit.PERCENTAGE); + + Label headingLabel = new Label(I18nProperties.getString(Strings.headingPersonInformation)); + headingLabel.addStyleName(CssStyles.H3); + componentHeader.addComponent(headingLabel); + componentHeader.setExpandRatio(headingLabel, 1); + + HorizontalLayout infoColumnsLayout = new HorizontalLayout(); + infoColumnsLayout.setMargin(false); + infoColumnsLayout.setSpacing(false); + infoColumnsLayout.setWidth(100, Sizeable.Unit.PERCENTAGE); + + VerticalLayout leftColumnLayout = new VerticalLayout(); + leftColumnLayout.setMargin(false); + leftColumnLayout.setSpacing(true); + boolean hasUserRightPersonView = UserProvider.getCurrent().hasUserRight(UserRight.PERSON_VIEW); + { + final Label personIdLabel = addDescLabel( + leftColumnLayout, + PersonDto.UUID, + DataHelper.getShortUuid(personDto.getUuid()), + I18nProperties.getPrefixCaption(PersonDto.I18N_PREFIX, PersonDto.UUID)); + personIdLabel.setId("personIdLabel"); + personIdLabel.setDescription(personDto.getUuid()); + + if (hasUserRightPersonView) { + addDescLabel(leftColumnLayout, PersonDto.LAST_NAME, personDto.buildCaption(), I18nProperties.getCaption(PERSON_FULL_NAME)); + + HorizontalLayout ageSexLayout = new HorizontalLayout(); + ageSexLayout.setMargin(false); + ageSexLayout.setSpacing(true); + addCustomDescLabel( + ageSexLayout, + PersonDto.class, + PersonDto.APPROXIMATE_AGE, + ApproximateAgeType.ApproximateAgeHelper.formatApproximateAge(personDto.getApproximateAge(), personDto.getApproximateAgeType()), + I18nProperties.getPrefixCaption(PersonDto.I18N_PREFIX, PersonDto.APPROXIMATE_AGE)); + addCustomDescLabel( + ageSexLayout, + PersonDto.class, + PersonDto.SEX, + personDto.getSex(), + I18nProperties.getPrefixCaption(PersonDto.I18N_PREFIX, PersonDto.SEX)); + leftColumnLayout.addComponent(ageSexLayout); + } + } + infoColumnsLayout.addComponent(leftColumnLayout); + + VerticalLayout rightColumnLayout = new VerticalLayout(); + rightColumnLayout.setMargin(false); + rightColumnLayout.setSpacing(true); + { + addDescLabel(rightColumnLayout, DISEASE, disease, I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, ImmunizationDto.DISEASE)); + + addDescLabel( + rightColumnLayout, + PersonDto.PHONE, + personDto.getPhone(), + I18nProperties.getPrefixCaption(PersonDto.I18N_PREFIX, PersonDto.PHONE)); + } + infoColumnsLayout.addComponent(rightColumnLayout); + + mainLayout.addComponent(componentHeader); + mainLayout.addComponent(infoColumnsLayout); + + this.addComponent(mainLayout); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/AbstractDashboardView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/AbstractDashboardView.java index 2dc0fdf58b1..aec6e9ecf83 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/AbstractDashboardView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/AbstractDashboardView.java @@ -29,6 +29,7 @@ import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.ui.SormasUI; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.AefiDashboardView; import de.symeda.sormas.ui.dashboard.campaigns.CampaignDashboardView; import de.symeda.sormas.ui.dashboard.contacts.ContactsDashboardView; import de.symeda.sormas.ui.dashboard.sample.SampleDashboardView; @@ -66,6 +67,11 @@ protected AbstractDashboardView(String viewName) { dashboardSwitcher.setItemCaption(DashboardType.SAMPLES, I18nProperties.getEnumCaption(DashboardType.SAMPLES)); } + if (permitted(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT, UserRight.DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + dashboardSwitcher.addItem(DashboardType.ADVERSE_EVENTS); + dashboardSwitcher.setItemCaption(DashboardType.ADVERSE_EVENTS, I18nProperties.getEnumCaption(DashboardType.ADVERSE_EVENTS)); + } + if (permitted(FeatureType.CAMPAIGNS, UserRight.DASHBOARD_CAMPAIGNS_VIEW)) { dashboardSwitcher.addItem(DashboardType.CAMPAIGNS); dashboardSwitcher.setItemCaption(DashboardType.CAMPAIGNS, I18nProperties.getEnumCaption(DashboardType.CAMPAIGNS)); @@ -96,6 +102,8 @@ protected void navigateToDashboardView(Property.ValueChangeEvent e) { SormasUI.get().getNavigator().navigateTo(ContactsDashboardView.VIEW_NAME); } else if (DashboardType.SAMPLES.equals(e.getProperty().getValue())) { SormasUI.get().getNavigator().navigateTo(SampleDashboardView.VIEW_NAME); + } else if (DashboardType.ADVERSE_EVENTS.equals(e.getProperty().getValue())) { + SormasUI.get().getNavigator().navigateTo(AefiDashboardView.VIEW_NAME); } else { SormasUI.get().getNavigator().navigateTo(CampaignDashboardView.VIEW_NAME); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardController.java index d65f29e12a0..d8a6414fbd7 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardController.java @@ -23,6 +23,7 @@ import de.symeda.sormas.api.feature.FeatureType; import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.AefiDashboardView; import de.symeda.sormas.ui.dashboard.campaigns.CampaignDashboardView; import de.symeda.sormas.ui.dashboard.contacts.ContactsDashboardView; import de.symeda.sormas.ui.dashboard.sample.SampleDashboardView; @@ -48,5 +49,9 @@ public void registerViews(Navigator navigator) { if (permitted(FeatureType.SAMPLES_LAB, UserRight.DASHBOARD_SAMPLES_VIEW)) { navigator.addView(SampleDashboardView.VIEW_NAME, SampleDashboardView.class); } + + if (permitted(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT, UserRight.DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + navigator.addView(AefiDashboardView.VIEW_NAME, AefiDashboardView.class); + } } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardType.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardType.java index 245918b091d..d1325b7ccb7 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardType.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/DashboardType.java @@ -24,7 +24,8 @@ public enum DashboardType { SURVEILLANCE, CONTACTS, CAMPAIGNS, - SAMPLES; + SAMPLES, + ADVERSE_EVENTS; @Override public String toString() { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java new file mode 100644 index 00000000000..fb6a77afc5a --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java @@ -0,0 +1,81 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization; + +import java.util.Map; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDashboardFilterDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartData; +import de.symeda.sormas.ui.dashboard.AbstractDashboardDataProvider; + +public class AefiDashboardDataProvider extends AbstractDashboardDataProvider { + + private AefiDashboardFilterDateType dateType = AefiDashboardFilterDateType.REPORT_DATE; + + private Map aefiCountsByType; + private Map> aefiCountsByVaccine; + private AefiChartData aefiByVaccineDoseChartData; + private AefiChartData aefiEventsByGenderChartData; + + @Override + public void refreshData() { + aefiCountsByType = FacadeProvider.getAefiDashboardFacade().getAefiCountsByType(buildDashboardCriteriaWithDates()); + aefiCountsByVaccine = FacadeProvider.getAefiDashboardFacade().getAefiCountsByVaccine(buildDashboardCriteriaWithDates()); + aefiByVaccineDoseChartData = FacadeProvider.getAefiDashboardFacade().getAefiByVaccineDoseChartData(buildDashboardCriteriaWithDates()); + aefiEventsByGenderChartData = FacadeProvider.getAefiDashboardFacade().getAefiEventsByGenderChartData(buildDashboardCriteriaWithDates()); + } + + @Override + protected AefiDashboardCriteria newCriteria() { + return new AefiDashboardCriteria(); + } + + @Override + protected AefiDashboardCriteria buildDashboardCriteria() { + return super.buildDashboardCriteria().aefiDashboardDateType(dateType); + } + + public AefiDashboardFilterDateType getDateType() { + return dateType; + } + + public void setDateType(AefiDashboardFilterDateType dateType) { + this.dateType = dateType; + } + + public Map getAefiCountsByType() { + return aefiCountsByType; + } + + public Map> getAefiCountsByVaccine() { + return aefiCountsByVaccine; + } + + public AefiChartData getAefiByVaccineDoseChartData() { + return aefiByVaccineDoseChartData; + } + + public AefiChartData getAefiEventsByGenderChartData() { + return aefiEventsByGenderChartData; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java new file mode 100644 index 00000000000..0c4df88ccf6 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java @@ -0,0 +1,112 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization; + +import java.util.List; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import com.vaadin.v7.ui.ComboBox; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDashboardFilterDateType; +import de.symeda.sormas.api.i18n.Descriptions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.ui.dashboard.components.DashboardFilterLayout; +import de.symeda.sormas.ui.utils.ComboBoxHelper; +import de.symeda.sormas.ui.utils.components.datetypeselector.DateTypeSelectorComponent; + +public class AefiDashboardFilterLayout extends DashboardFilterLayout { + + public static final String DATE_TYPE_FILTER = "dateTypeFilter"; + public static final String DISEASE_FILTER = "diseaseFilter"; + + private final static String[] FILTERS = new String[] { + DATE_TYPE_FILTER, + REGION_FILTER, + DISTRICT_FILTER, + DISEASE_FILTER }; + + public AefiDashboardFilterLayout(AefiDashboardView dashboardView, AefiDashboardDataProvider dashboardDataProvider) { + super(dashboardView, dashboardDataProvider, FILTERS); + } + + @Override + public void populateLayout() { + super.populateLayout(); + + createDateTypeSelector(); + createRegionFilter(I18nProperties.getDescription(Descriptions.aefiDashboardRegionFilter)); + createDistrictFilter(I18nProperties.getDescription(Descriptions.aefiDashboardDistrictFilter)); + createDiseaseFilter(); + } + + private void createDateTypeSelector() { + @SuppressWarnings("unchecked") + DateTypeSelectorComponent dateTypeSelectorComponent = new DateTypeSelectorComponent.Builder<>(AefiDashboardFilterDateType.class) + .dateTypePrompt(I18nProperties.getString(Strings.promptAefiDashboardFilterDateType)) + .defaultDateType(dashboardDataProvider.getDateType()) + .build(); + + dateTypeSelectorComponent.addValueChangeListener(e -> { + dashboardDataProvider.setDateType((AefiDashboardFilterDateType) e.getProperty().getValue()); + }); + + addCustomComponent(dateTypeSelectorComponent, DATE_TYPE_FILTER); + } + + private void createDiseaseFilter() { + ComboBox diseaseFilter = ComboBoxHelper.createComboBoxV7(); + diseaseFilter.setWidth(200, Unit.PIXELS); + diseaseFilter.setInputPrompt(I18nProperties.getString(Strings.promptDisease)); + diseaseFilter.setDescription(I18nProperties.getDescription(Descriptions.aefiDashboardDiseaseFilter)); + List availableDisease = FacadeProvider.getDiseaseConfigurationFacade().getAllDiseasesWithFollowUp(true, true, true); + + diseaseFilter + .addItems(Stream.concat(availableDisease.stream(), Stream.of(AefiDashboardCustomDiseaseFilter.values())).collect(Collectors.toList())); + diseaseFilter.setValue(dashboardDataProvider.getDisease()); + + diseaseFilter.addValueChangeListener(e -> { + Object filterValue = diseaseFilter.getValue(); + if (filterValue instanceof Disease) { + dashboardDataProvider.setDisease((Disease) filterValue); + } else if (filterValue == AefiDashboardCustomDiseaseFilter.NO_DISEASE) { + dashboardDataProvider.setDisease(null); + } else if (filterValue == null) { + dashboardDataProvider.setDisease(null); + } else { + throw new RuntimeException("Disease filter [" + filterValue + "] not handled!"); + } + }); + + addCustomComponent(diseaseFilter, DISEASE_FILTER); + } + + enum AefiDashboardCustomDiseaseFilter { + + NO_DISEASE; + + @Override + public String toString() { + return I18nProperties.getEnumCaptionShort(this); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java new file mode 100644 index 00000000000..15bc987c333 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java @@ -0,0 +1,230 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization; + +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidColumn; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidColumnLoc; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.locCss; + +import com.vaadin.ui.Component; +import com.vaadin.ui.CustomLayout; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.ui.dashboard.AbstractDashboardView; +import de.symeda.sormas.ui.dashboard.DashboardCssStyles; +import de.symeda.sormas.ui.dashboard.DashboardType; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiByVaccineDoseChart; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiCountTilesComponent; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiDashboardMapComponent; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiReactionsByGenderChart; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiTypeStatisticsGroupComponent; +import de.symeda.sormas.ui.dashboard.components.DashboardHeadingComponent; +import de.symeda.sormas.ui.utils.CssStyles; + +public class AefiDashboardView extends AbstractDashboardView { + + public static final String VIEW_NAME = ROOT_VIEW_NAME + "/adverseevents"; + + private static final int EPI_CURVE_AND_MAP_HEIGHT = 555; + + private static final String HEADING_LOC = "headingLoc"; + private static final String AEFI_TYPE_LOC = "aefiTypeLoc"; + private static final String VACCINES_LOC = "vaccinesLoc"; + private static final String VACCINE_DOSE_LOC = "vaccineDoseLoc"; + private static final String REACTIONS_LOC = "reactionsLoc"; + + private final AefiDashboardDataProvider dataProvider; + + private final CustomLayout aefiCountsLayout; + private final HorizontalLayout epiCurveAndMapLayout; + private final VerticalLayout epiCurveLayout; + private final VerticalLayout mapLayout; + + private final DashboardHeadingComponent heading; + private final AefiCountTilesComponent aefiCountsByType; + private final AefiTypeStatisticsGroupComponent aefiCountsByVaccine; + private final AefiByVaccineDoseChart vaccineDoseChart; + private final AefiReactionsByGenderChart reactionsByGenderChart; + private final AefiEpiCurveComponent epiCurveComponent; + private final AefiDashboardMapComponent mapComponent; + + public AefiDashboardView() { + super(VIEW_NAME); + + CssStyles.style(getViewTitleLabel(), CssStyles.PAGE_TITLE); + + dashboardSwitcher.setValue(DashboardType.ADVERSE_EVENTS); + dashboardSwitcher.addValueChangeListener(this::navigateToDashboardView); + + dashboardLayout.setHeightUndefined(); + + dataProvider = new AefiDashboardDataProvider(); + AefiDashboardFilterLayout filterLayout = new AefiDashboardFilterLayout(this, dataProvider); + + dashboardLayout.addComponent(filterLayout); + dashboardLayout.setExpandRatio(filterLayout, 0); + + aefiCountsLayout = new CustomLayout(); + //@formatter:off + aefiCountsLayout.setTemplateContents( + fluidRowCss( + CssStyles.PADDING_X_20 + " " + CssStyles.VSPACE_TOP_2, + fluidColumn(6, 0, locCss("", HEADING_LOC)), + fluidColumn(4, 0, locCss("", AEFI_TYPE_LOC)) + ) + + fluidRowCss( + CssStyles.VSPACE_TOP_2, + fluidColumnLoc(12, 0, 12, 0, VACCINES_LOC) + ) + + fluidRowCss( + CssStyles.PADDING_X_20 + " " + CssStyles.VSPACE_TOP_2, + fluidColumn(6, 0, locCss("", VACCINE_DOSE_LOC)), + fluidColumn(6, 0, locCss("", REACTIONS_LOC)) + ) + ); + //@formatter:on + dashboardLayout.addComponent(aefiCountsLayout); + + heading = new DashboardHeadingComponent(Captions.aefiDashboardAllAefi, null); + heading.setMargin(false); + aefiCountsLayout.addComponent(heading, HEADING_LOC); + + aefiCountsByType = new AefiCountTilesComponent<>(AefiType.class, "", this::getBackgroundStyleForAefiCountByType, null); + aefiCountsByType.setTitleStyleNames(CssStyles.H3, CssStyles.VSPACE_TOP_5); + aefiCountsByType.setGroupLabelStyle(CssStyles.LABEL_LARGE + " " + CssStyles.LABEL_WHITE_SPACE_NORMAL); + aefiCountsLayout.addComponent(aefiCountsByType, AEFI_TYPE_LOC); + + aefiCountsByVaccine = new AefiTypeStatisticsGroupComponent(); + aefiCountsByVaccine.addStyleNames(CssStyles.PADDING_X_20); + aefiCountsLayout.addComponent(aefiCountsByVaccine, VACCINES_LOC); + + vaccineDoseChart = new AefiByVaccineDoseChart(); + aefiCountsLayout.addComponent(vaccineDoseChart, VACCINE_DOSE_LOC); + + reactionsByGenderChart = new AefiReactionsByGenderChart(); + aefiCountsLayout.addComponent(reactionsByGenderChart, REACTIONS_LOC); + + epiCurveComponent = new AefiEpiCurveComponent(dataProvider); + epiCurveLayout = createEpiCurveLayout(); + + mapComponent = new AefiDashboardMapComponent(dataProvider); + mapLayout = createMapLayout(mapComponent); + + epiCurveAndMapLayout = createEpiCurveAndMapLayout(epiCurveLayout, mapLayout); + epiCurveAndMapLayout.addStyleName(CssStyles.VSPACE_TOP_1); + dashboardLayout.addComponent(epiCurveAndMapLayout); + dashboardLayout.setExpandRatio(epiCurveAndMapLayout, 1); + } + + @Override + public void refreshDashboard() { + dataProvider.refreshData(); + + heading.updateTotalLabel(String.valueOf(dataProvider.getAefiCountsByType().values().stream().mapToLong(Long::longValue).sum())); + + aefiCountsByType.update(dataProvider.getAefiCountsByType()); + aefiCountsByVaccine.update(dataProvider.getAefiCountsByVaccine()); + vaccineDoseChart.update(dataProvider.getAefiByVaccineDoseChartData()); + reactionsByGenderChart.update(dataProvider.getAefiEventsByGenderChartData()); + epiCurveComponent.clearAndFillEpiCurveChart(); + mapComponent.refreshMap(); + } + + private String getBackgroundStyleForAefiCountByType(AefiType aefiType) { + return aefiType == AefiType.SERIOUS ? "background-shipment-status-not-shipped" : "background-internal-lab-samples"; + } + + protected HorizontalLayout createEpiCurveAndMapLayout(VerticalLayout epiCurveLayout, VerticalLayout mapLayout) { + HorizontalLayout layout = new HorizontalLayout(epiCurveLayout, mapLayout); + layout.addStyleName(DashboardCssStyles.CURVE_AND_MAP_LAYOUT); + layout.setWidth(100, Unit.PERCENTAGE); + layout.setMargin(false); + layout.setSpacing(false); + + return layout; + } + + protected VerticalLayout createEpiCurveLayout() { + if (epiCurveComponent == null) { + throw new UnsupportedOperationException("EpiCurveComponent needs to be initialized before calling createEpiCurveLayout"); + } + + VerticalLayout layout = new VerticalLayout(); + layout.setMargin(false); + layout.setSpacing(false); + layout.setHeight(EPI_CURVE_AND_MAP_HEIGHT, Unit.PIXELS); + + epiCurveComponent.setSizeFull(); + + layout.addComponent(epiCurveComponent); + layout.setExpandRatio(epiCurveComponent, 1); + + epiCurveComponent.setExpandListener(expanded -> { + setExpanded(expanded, layout, mapLayout, 1); + }); + + return layout; + } + + private VerticalLayout createMapLayout(AefiDashboardMapComponent mapComponent) { + VerticalLayout layout = new VerticalLayout(); + layout.setMargin(false); + layout.setSpacing(false); + layout.setHeight(EPI_CURVE_AND_MAP_HEIGHT, Unit.PIXELS); + + mapComponent.setSizeFull(); + + layout.addComponent(mapComponent); + layout.setExpandRatio(mapComponent, 1); + + mapComponent.setExpandListener(expanded -> { + setExpanded(expanded, layout, epiCurveLayout, 0); + }); + + return layout; + } + + private void setExpanded(Boolean expanded, Component componentToExpand, Component componentToRemove, int removedComponentIndex) { + if (expanded) { + dashboardLayout.removeComponent(heading); + dashboardLayout.removeComponent(aefiCountsLayout); + epiCurveAndMapLayout.removeComponent(componentToRemove); + setHeight(100, Unit.PERCENTAGE); + + epiCurveAndMapLayout.setHeightFull(); + setHeightFull(); + componentToExpand.setSizeFull(); + dashboardLayout.setHeightFull(); + } else { + dashboardLayout.addComponent(heading, 1); + dashboardLayout.addComponent(aefiCountsLayout, 2); + epiCurveAndMapLayout.addComponent(componentToRemove, removedComponentIndex); + mapComponent.refreshMap(); + + componentToExpand.setHeight(EPI_CURVE_AND_MAP_HEIGHT, Unit.PIXELS); + setHeightUndefined(); + epiCurveAndMapLayout.setHeightUndefined(); + dashboardLayout.setHeightUndefined(); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiEpiCurveComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiEpiCurveComponent.java new file mode 100644 index 00000000000..0cc3a0ef910 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiEpiCurveComponent.java @@ -0,0 +1,97 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization; + +import java.util.Arrays; +import java.util.Date; +import java.util.List; +import java.util.Map; + +import com.vaadin.ui.AbstractComponent; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; +import de.symeda.sormas.api.dashboard.EpiCurveGrouping; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.ui.dashboard.diagram.AbstractEpiCurveBuilder; +import de.symeda.sormas.ui.dashboard.diagram.AbstractEpiCurveComponent; +import de.symeda.sormas.ui.dashboard.diagram.EpiCurveSeriesElement; + +public class AefiEpiCurveComponent extends AbstractEpiCurveComponent { + + private static final long serialVersionUID = 6767165953640006853L; + + public AefiEpiCurveComponent(AefiDashboardDataProvider dashboardDataProvider) { + super(dashboardDataProvider); + epiCurveLabel.setValue(I18nProperties.getString(Strings.headingAefiDashboardEpiCurve)); + } + + @Override + protected AbstractComponent createEpiCurveModeSelector() { + return null; + } + + @Override + public void clearAndFillEpiCurveChart() { + epiCurveChart.setHcjs(new AefiEpiCurveBuilder(epiCurveGrouping).buildFrom(buildListOfFilteredDates(), dashboardDataProvider)); + } + + private static class AefiEpiCurveBuilder extends AbstractEpiCurveBuilder { + + public AefiEpiCurveBuilder(EpiCurveGrouping epiCurveGrouping) { + super(Captions.dashboardNumberOfAdverseEvents, epiCurveGrouping); + } + + @Override + protected List getEpiCurveElements(List datesGroupedBy, AefiDashboardDataProvider dashboardDataProvider) { + int[] seriousNumbers = new int[datesGroupedBy.size()]; + int[] nonSeriousNumbers = new int[datesGroupedBy.size()]; + + for (int i = 0; i < datesGroupedBy.size(); i++) { + Date date = datesGroupedBy.get(i); + + AefiDashboardCriteria criteria = + dashboardDataProvider.buildDashboardCriteria().aefiDashboardDateType(dashboardDataProvider.getDateType()); + if (epiCurveGrouping == EpiCurveGrouping.DAY) { + criteria.dateBetween(DateHelper.getStartOfDay(date), DateHelper.getEndOfDay(date)); + } else if (epiCurveGrouping == EpiCurveGrouping.WEEK) { + criteria.dateBetween(DateHelper.getStartOfWeek(date), DateHelper.getEndOfWeek(date)); + } else { + criteria.dateBetween(DateHelper.getStartOfMonth(date), DateHelper.getEndOfMonth(date)); + } + + Map aefiCounts = FacadeProvider.getAefiDashboardFacade().getAefiCountsByType(criteria); + + Long seriousCount = aefiCounts.get(AefiType.SERIOUS); + Long nonSeriousCount = aefiCounts.get(AefiType.NON_SERIOUS); + + seriousNumbers[i] = seriousCount != null ? seriousCount.intValue() : 0; + nonSeriousNumbers[i] = nonSeriousCount != null ? nonSeriousCount.intValue() : 0; + } + + return Arrays.asList( + new EpiCurveSeriesElement(AefiType.SERIOUS, "#c80000", seriousNumbers), + new EpiCurveSeriesElement(AefiType.NON_SERIOUS, "#32CD32", nonSeriousNumbers)); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiByVaccineDoseChart.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiByVaccineDoseChart.java new file mode 100644 index 00000000000..964db0c07cd --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiByVaccineDoseChart.java @@ -0,0 +1,158 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components; + +import java.util.List; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartData; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartSeries; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.ui.highcharts.HighChart; + +public class AefiByVaccineDoseChart extends VerticalLayout { + + protected final HighChart chart; + + public AefiByVaccineDoseChart() { + + setMargin(false); + setSpacing(false); + setSizeFull(); + + chart = new HighChart(); + chart.setSizeFull(); + + addComponent(chart); + setExpandRatio(chart, 1); + } + + public void update(AefiChartData chartData) { + + StringBuilder hcjs = new StringBuilder(); + + //@formatter:off + hcjs.append( + "var options = {" + + " chart: {" + + " type: 'bar'," + + " borderRadius: '8px'" + + " }," + + " title: {" + + " text: 'Adverse events by vaccine dose'," + + " align: 'left'," + + " style: {" + + " fontSize: '15px'," + + " fontWeight: 'bold'" + + " }" + + " }," + + " subtitle: {" + + " text: ''," + + " align: 'left'" + + " }," + ); + //@formatter:on + + List xAxisCategories = chartData.getxAxisCategories(); + + StringBuilder categoryBuilder = new StringBuilder(); + if (!xAxisCategories.isEmpty()) { + categoryBuilder.append("["); + } + for (Object s : xAxisCategories) { + if (xAxisCategories.indexOf(s) == xAxisCategories.size() - 1) { + categoryBuilder.append("'" + I18nProperties.getCaption(String.valueOf(s)) + "']"); + } else { + categoryBuilder.append("'" + I18nProperties.getCaption(String.valueOf(s)) + "', "); + } + } + String categories = !StringUtils.isBlank(categoryBuilder.toString()) ? categoryBuilder.toString() : "[]"; + + //@formatter:off + hcjs.append("xAxis: { " + + " categories: " + categories + "," + + " title: {" + + " text: 'Vaccine dose'," + + " style: {" + + " fontWeight: 'bold'" + + " }" + + " }," + + "},"); + //@formatter:on + + //@formatter:off + hcjs.append("yAxis: {" + + " min: 0," + + " title: {" + + " text: 'No. of AEFI Reports'," + + " align: 'high'," + + " style: {" + + " fontWeight: 'bold'" + + " }" + + " }," + + " labels: {" + + " overflow: 'justify'" + + " }," + + " gridLineWidth: 0" + + " }," + + " tooltip: {" + + " valueSuffix: ''" + + " }," + + " plotOptions: {" + + " bar: {" + + " dataLabels: {" + + " enabled: true" + + " }," + + " groupPadding: 0.1" + + " }" + + " }," + + " credits: {" + + " enabled: false" + + " },"); + //@formatter:on + + hcjs.append("series: ["); + List chartSeries = chartData.getSeries(); + for (AefiChartSeries series : chartSeries) { + hcjs.append("{") + .append("name: '") + .append(series.getName()) + .append("',") + .append("color: '") + .append(series.getColor()) + .append("',") + .append("data: [") + .append(String.join(",", series.getSeriesData())) + .append("]") + .append("}"); + + if (chartSeries.indexOf(series) < chartSeries.size() - 1) { + hcjs.append(","); + } + } + hcjs.append("]"); + + hcjs.append("};"); + + chart.setHcjs(hcjs.toString()); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java new file mode 100644 index 00000000000..a3308e23824 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java @@ -0,0 +1,204 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components; + +import java.math.BigDecimal; +import java.math.MathContext; +import java.math.RoundingMode; +import java.util.Map; +import java.util.function.Function; + +import javax.annotation.Nullable; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.icons.VaadinIcons; +import com.vaadin.shared.ui.ContentMode; +import com.vaadin.ui.Alignment; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.ui.utils.CssStyles; + +public class AefiCountTilesComponent> extends VerticalLayout { + + private static final long serialVersionUID = 3342746874277667267L; + private final Class groupType; + private final Function tileBackgroundFactory; + private boolean withPercentage; + private String groupLabelStyle; + + private final Label title; + private Label infoIcon; + private final HorizontalLayout countsLayout; + private String[] titleStyleNames = new String[] { + CssStyles.H3 }; + + public AefiCountTilesComponent( + Class groupType, + String titleCaption, + Function tileBackgroundFactory, + @Nullable String descriptionTag) { + this.groupType = groupType; + this.tileBackgroundFactory = tileBackgroundFactory; + + setMargin(false); + setSpacing(false); + setWidthFull(); + + HorizontalLayout titleLayout = new HorizontalLayout(); + titleLayout.setMargin(false); + titleLayout.setSpacing(false); + addComponent(titleLayout); + + title = new Label(I18nProperties.getCaption(titleCaption)); + title.addStyleNames(titleStyleNames); + + titleLayout.addComponent(title); + + if (StringUtils.isNotBlank(descriptionTag)) { + infoIcon = new Label(VaadinIcons.INFO_CIRCLE.getHtml(), ContentMode.HTML); + infoIcon.setDescription(I18nProperties.getDescription(descriptionTag)); + infoIcon.addStyleName(CssStyles.HSPACE_LEFT_4); + infoIcon.addStyleNames(titleStyleNames); + + titleLayout.addComponent(infoIcon); + } + + countsLayout = new HorizontalLayout(); + countsLayout.setWidthFull(); + addComponent(countsLayout); + } + + public void update(Map counts) { + countsLayout.removeAllComponents(); + + Long total = null; + if (withPercentage) { + total = counts.values().stream().reduce(0L, Long::sum); + } + + addTiles(counts, total); + } + + private void addTiles(Map counts, @Nullable Long total) { + for (T group : groupType.getEnumConstants()) { + addTileForGroup(counts, group, total, String.valueOf(group)); + } + + if (counts.containsKey(null)) { + addTileForGroup(counts, null, total, I18nProperties.getCaption(Captions.notSpecified)); + } + } + + private void addTileForGroup(Map counts, T group, Long total, String groupCaption) { + Long count = counts.getOrDefault(group, 0L); + + BigDecimal percentage = null; + if (total != null) { + percentage = total == 0 + ? BigDecimal.ZERO + : BigDecimal.valueOf(count).divide(BigDecimal.valueOf(total), MathContext.DECIMAL32).multiply(new BigDecimal(100)); + percentage = percentage.setScale(0, RoundingMode.HALF_UP); + } + + TileComponent tile = new TileComponent<>(groupCaption, count, percentage, groupLabelStyle, tileBackgroundFactory.apply(group)); + tile.setWidthFull(); + + countsLayout.addComponent(tile); + countsLayout.setExpandRatio(tile, 1); + } + + public void setTitleStyleNames(String... styleNames) { + title.removeStyleNames(titleStyleNames); + title.addStyleNames(styleNames); + + if (infoIcon != null) { + infoIcon.removeStyleNames(titleStyleNames); + infoIcon.addStyleNames(styleNames); + } + + this.titleStyleNames = styleNames; + } + + public void setWithPercentage(boolean withPercentage) { + this.withPercentage = withPercentage; + } + + public void setGroupLabelStyle(String groupLabelStyle) { + this.groupLabelStyle = groupLabelStyle; + } + + private static class TileComponent extends VerticalLayout { + + private static final long serialVersionUID = 5055236377479070515L; + + TileComponent(String group, Long count, @Nullable BigDecimal percentage, String groupLabelStyle, String backgroundStyle) { + setSpacing(false); + setMargin(false); + + VerticalLayout numbersLayout = new VerticalLayout(); + numbersLayout.setWidthFull(); + numbersLayout.setMargin(false); + numbersLayout.setSpacing(false); + numbersLayout.addStyleNames(backgroundStyle, CssStyles.ROUNDED_BORDER_TOP); + addComponent(numbersLayout); + + Label countLabel = new Label(count.toString()); + countLabel + .addStyleNames(CssStyles.LABEL_WHITE, CssStyles.LABEL_BOLD, CssStyles.LABEL_LARGE, CssStyles.ALIGN_CENTER, CssStyles.VSPACE_TOP_4); + + numbersLayout.addComponent(countLabel); + numbersLayout.setComponentAlignment(countLabel, Alignment.TOP_CENTER); + + if (percentage != null) { + Label percentageLabel = new Label(percentage + "%"); + percentageLabel.addStyleNames(CssStyles.LABEL_WHITE, CssStyles.ALIGN_CENTER, CssStyles.VSPACE_4); + + numbersLayout.addComponent(percentageLabel); + numbersLayout.setComponentAlignment(percentageLabel, Alignment.MIDDLE_CENTER); + } + + Label groupLabel = new Label(group); + groupLabel.addStyleNames( + CssStyles.LABEL_WHITE, + CssStyles.LABEL_BOLD, + CssStyles.VSPACE_TOP_4, + CssStyles.VSPACE_4, + CssStyles.HSPACE_LEFT_4, + CssStyles.HSPACE_RIGHT_4); + + if (groupLabelStyle != null) { + groupLabel.addStyleName(groupLabelStyle); + } + + VerticalLayout groupLabelLayout = new VerticalLayout(); + groupLabelLayout.setMargin(false); + groupLabelLayout.setSpacing(false); + groupLabelLayout.addStyleNames(backgroundStyle, CssStyles.BACKGROUND_DARKER, CssStyles.ROUNDED_BORDER_BOTTOM); + groupLabelLayout.addComponent(groupLabel); + groupLabelLayout.setComponentAlignment(groupLabel, Alignment.TOP_CENTER); + + addComponent(groupLabelLayout); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiDashboardMapComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiDashboardMapComponent.java new file mode 100644 index 00000000000..be94e75e9a4 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiDashboardMapComponent.java @@ -0,0 +1,161 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.Date; +import java.util.List; +import java.util.stream.Collectors; + +import com.vaadin.ui.Component; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.VerticalLayout; +import com.vaadin.v7.ui.CheckBox; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.MapAefiDto; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.AefiDashboardDataProvider; +import de.symeda.sormas.ui.dashboard.map.BaseDashboardMapComponent; +import de.symeda.sormas.ui.map.LeafletMarker; +import de.symeda.sormas.ui.map.MarkerIcon; +import de.symeda.sormas.ui.utils.CssStyles; + +public class AefiDashboardMapComponent extends BaseDashboardMapComponent { + + public AefiDashboardMapComponent(AefiDashboardDataProvider dashboardDataProvider) { + super(Strings.headingAefiDashboardMap, dashboardDataProvider, Strings.infoHeadingAefiDashboardMap); + } + + @Override + protected void addComponents() { + super.addComponents(); + } + + @Override + protected Long getMarkerCount(Date fromDate, Date toDate, int maxCount) { + return FacadeProvider.getAefiDashboardFacade().countAefiForMap(dashboardDataProvider.buildDashboardCriteriaWithDates()); + } + + @Override + protected void loadMapData(Date fromDate, Date toDate) { + String markerGroup = "adverseevents"; + map.removeGroup(markerGroup); + + AefiDashboardCriteria criteria = dashboardDataProvider.buildDashboardCriteriaWithDates(); + List aefiMapData = FacadeProvider.getAefiDashboardFacade().getAefiForMap(criteria); + + //temporary fix: remove data without coordinates + //ideally do this in the backend code + List filteredAefiMapData = new ArrayList<>(); + for (MapAefiDto mapAefiDto : aefiMapData) { + if (mapAefiDto.getLatitude() != null && mapAefiDto.getLongitude() != null) { + filteredAefiMapData.add(mapAefiDto); + } + } + + List markers = new ArrayList<>(aefiMapData.size()); + + markers.addAll(filteredAefiMapData.stream().map(mapAefiDto -> { + LeafletMarker marker = new LeafletMarker(); + switch (mapAefiDto.getAefiType()) { + case SERIOUS: + marker.setIcon(MarkerIcon.SAMPLE_CASE); + break; + case NON_SERIOUS: + marker.setIcon(MarkerIcon.SAMPLE_CONTACT); + break; + default: + marker.setIcon(MarkerIcon.SAMPLE_EVENT_PARTICIPANT); + } + marker.setLatLon(mapAefiDto.getLatitude(), mapAefiDto.getLongitude()); + + return marker; + }).collect(Collectors.toList())); + + map.addMarkerGroup(markerGroup, markers); + } + + @Override + protected void addLayerOptions(VerticalLayout layersLayout) { + + CheckBox showSeriousAefiCheckBox = new CheckBox(); + showSeriousAefiCheckBox.setId(Captions.aefiDashboardShowSeriousAefi); + showSeriousAefiCheckBox.setCaption(I18nProperties.getCaption(Captions.aefiDashboardShowSeriousAefi)); + showSeriousAefiCheckBox.setValue(shouldShowSeriousAefi()); + showSeriousAefiCheckBox.addValueChangeListener(e -> { + dashboardDataProvider.buildDashboardCriteriaWithDates().aefiType(AefiType.SERIOUS); + + refreshMap(true); + }); + + layersLayout.addComponent(showSeriousAefiCheckBox); + + CheckBox showNonSeriousAefiCheckBox = new CheckBox(); + showNonSeriousAefiCheckBox.setId(Captions.aefiDashboardShowNonSeriousAefi); + showNonSeriousAefiCheckBox.setCaption(I18nProperties.getCaption(Captions.aefiDashboardShowNonSeriousAefi)); + showNonSeriousAefiCheckBox.setValue(shouldShowNonSeriousAefi()); + showNonSeriousAefiCheckBox.addValueChangeListener(e -> { + dashboardDataProvider.buildDashboardCriteriaWithDates().aefiType(AefiType.NON_SERIOUS); + + refreshMap(true); + }); + + layersLayout.addComponent(showNonSeriousAefiCheckBox); + } + + @Override + protected List getLegendComponents() { + + HorizontalLayout samplesLegendLayout = new HorizontalLayout(); + samplesLegendLayout.setSpacing(false); + samplesLegendLayout.setMargin(false); + + //if (shouldShowSeriousAefi()) { + HorizontalLayout seriousLegendEntry = + buildMarkerLegendEntry(MarkerIcon.SAMPLE_CASE, I18nProperties.getCaption(Captions.aefiDashboardSeriousAefi)); + CssStyles.style(seriousLegendEntry, CssStyles.HSPACE_RIGHT_3); + samplesLegendLayout.addComponent(seriousLegendEntry); + //} + + //if (shouldShowNonSeriousAefi()) { + HorizontalLayout nonSeriousLegendEntry = + buildMarkerLegendEntry(MarkerIcon.SAMPLE_CONTACT, I18nProperties.getCaption(Captions.aefiDashboardNonSeriousAefi)); + CssStyles.style(nonSeriousLegendEntry, CssStyles.HSPACE_RIGHT_3); + samplesLegendLayout.addComponent(nonSeriousLegendEntry); + //} + + return Collections.singletonList(samplesLegendLayout); + } + + private boolean shouldShowSeriousAefi() { + return dashboardDataProvider.buildDashboardCriteriaWithDates().getAefiType() == AefiType.SERIOUS; + } + + private boolean shouldShowNonSeriousAefi() { + return dashboardDataProvider.buildDashboardCriteriaWithDates().getAefiType() == AefiType.NON_SERIOUS; + } + + @Override + protected void onMarkerClicked(String groupId, int markerIndex) { + + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java new file mode 100644 index 00000000000..464872634f0 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java @@ -0,0 +1,166 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components; + +import java.util.List; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartData; +import de.symeda.sormas.api.dashboard.adverseeventsfollowingimmunization.AefiChartSeries; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.ui.highcharts.HighChart; + +public class AefiReactionsByGenderChart extends VerticalLayout { + + protected final HighChart chart; + + public AefiReactionsByGenderChart() { + + setMargin(false); + setSpacing(false); + setSizeFull(); + + chart = new HighChart(); + chart.setSizeFull(); + + addComponent(chart); + setExpandRatio(chart, 1); + } + + public void update(AefiChartData chartData) { + + StringBuilder hcjs = new StringBuilder(); + + //@formatter:off + hcjs.append( + "var options = {" + + " chart: {" + + " type: 'bar'," + + " borderRadius: '8px'" + + " }," + + " title: {" + + " text: 'Proportion of AEFI reactions (events) by gender'," + + " align: 'left'," + + " style: {" + + " fontSize: '15px'," + + " fontWeight: 'bold'" + + " }" + + " }," + + " subtitle: {" + + " text: ''," + + " align: 'left'" + + " }," + ); + //@formatter:on + + List xAxisCategories = chartData.getxAxisCategories(); + + StringBuilder categoryBuilder = new StringBuilder(); + if (!xAxisCategories.isEmpty()) { + categoryBuilder.append("["); + } + for (Object s : xAxisCategories) { + if (xAxisCategories.indexOf(s) == xAxisCategories.size() - 1) { + categoryBuilder.append("'" + I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, String.valueOf(s)) + "']"); + } else { + categoryBuilder.append("'" + I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, String.valueOf(s)) + "', "); + } + } + String categories = !StringUtils.isBlank(categoryBuilder.toString()) ? categoryBuilder.toString() : "[]"; + + //@formatter:off + hcjs.append("xAxis: [{" + + " categories: " + categories + "," + + " title: {" + + " text: 'AEFI Reactions (events)'," + + " style: {" + + " fontWeight: 'bold'" + + " }" + + " }," + + " reversed: true," + + " labels: {" + + " step: 1" + + " }" + + " }, { " + + " title: {" + + " text: 'AEFI Reactions (events)'," + + " style: {" + + " fontWeight: 'bold'" + + " }" + + " }," + + " opposite: true," + + " reversed: false," + + " categories: " + categories + "," + + " linkedTo: 0," + + " labels: {" + + " step: 1" + + " }" + + " }],"); + //@formatter:on + + //@formatter:off + hcjs.append("yAxis: {" + + " title: {" + + " text: 'Proportion of AEFI reactions'," + + " align: 'high'," + + " style: {" + + " fontWeight: 'bold'" + + " }" + + " }," + + " }," + + " plotOptions: {" + + " series: {" + + " stacking: 'normal'" + + " }" + + " }," + + " credits: {" + + " enabled: false" + + " },"); + //@formatter:on + + hcjs.append("series: ["); + List chartSeries = chartData.getSeries(); + for (AefiChartSeries series : chartSeries) { + hcjs.append("{") + .append("name: '") + .append(series.getName()) + .append("',") + .append("color: '") + .append(series.getColor()) + .append("',") + .append("data: [") + .append(String.join(",", series.getSeriesData())) + .append("]") + .append("}"); + + if (chartSeries.indexOf(series) < chartSeries.size() - 1) { + hcjs.append(","); + } + } + hcjs.append("]"); + + hcjs.append("};"); + + chart.setHcjs(hcjs.toString()); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsComponent.java new file mode 100644 index 00000000000..697d17fe4d3 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsComponent.java @@ -0,0 +1,100 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components; + +import java.util.Map; + +import com.vaadin.ui.Label; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.ui.dashboard.components.DashboardHeadingComponent; +import de.symeda.sormas.ui.dashboard.statistics.CountElementStyle; +import de.symeda.sormas.ui.dashboard.statistics.DashboardStatisticsCountElement; +import de.symeda.sormas.ui.dashboard.surveillance.components.statistics.DiseaseSectionStatisticsComponent; +import de.symeda.sormas.ui.utils.CssStyles; + +public class AefiTypeStatisticsComponent extends DiseaseSectionStatisticsComponent { + + private final DashboardStatisticsCountElement seriousCount; + private final DashboardStatisticsCountElement nonSeriousCount; + private boolean withPercentage; + + public AefiTypeStatisticsComponent(String titleCaption, String description, String subtitleCaption, boolean showInfoIcon) { + super(titleCaption, description, ""); + + setWidthUndefined(); + + if (subtitleCaption != null) { + Label subTitleLabel = new Label(I18nProperties.getCaption(subtitleCaption)); + CssStyles.style(subTitleLabel, CssStyles.H3, CssStyles.VSPACE_TOP_5); + addComponent(subTitleLabel); + } + + // Count layout + seriousCount = new DashboardStatisticsCountElement(I18nProperties.getCaption(Captions.aefiDashboardSerious), CountElementStyle.CRITICAL); + nonSeriousCount = + new DashboardStatisticsCountElement(I18nProperties.getCaption(Captions.aefiDashboardNonSerious), CountElementStyle.POSITIVE); + + buildCountLayout(seriousCount, nonSeriousCount); + } + + public void update(Map aefiTypeData) { + if (aefiTypeData != null) { + Long totalCount = null; + Long seriousTotal = aefiTypeData.getOrDefault(AefiType.SERIOUS, 0L); + Long nonSeriousTotal = aefiTypeData.getOrDefault(AefiType.NON_SERIOUS, 0L); + + //updateTotalLabel(((Long) aefiTypeData.values().stream().mapToLong(Long::longValue).sum()).toString()); + if (withPercentage) { + totalCount = aefiTypeData.values().stream().reduce(0L, Long::sum); + seriousCount.updateCountLabel(seriousTotal + " (" + calculatePercentage(totalCount, seriousTotal) + " %)"); + nonSeriousCount.updateCountLabel(nonSeriousTotal + " (" + calculatePercentage(totalCount, nonSeriousTotal) + " %)"); + } else { + seriousCount.updateCountLabel(seriousTotal.toString()); + nonSeriousCount.updateCountLabel(nonSeriousTotal.toString()); + } + } + } + + public int calculatePercentage(Long totalCount, Long labResultCount) { + return totalCount == 0 ? 0 : (int) ((labResultCount * 100.0f) / totalCount); + } + + public void setWithPercentage(boolean withPercentage) { + this.withPercentage = withPercentage; + } + + public void setTitleStyleNamesOnTitleLabel(String... styleNames) { + DashboardHeadingComponent dashboardHeadingComponent = this.getHeading(); + Label titleLabel = dashboardHeadingComponent.getTitleLabel(); + + titleLabel.removeStyleNames(dashboardHeadingComponent.getTitleStyleNames()); + titleLabel.addStyleNames(styleNames); + } + + public void setTitleStyleNamesOnTotalLabel(String... styleNames) { + DashboardHeadingComponent dashboardHeadingComponent = this.getHeading(); + Label totalLabel = dashboardHeadingComponent.getTotalLabel(); + + totalLabel.removeStyleNames(dashboardHeadingComponent.getTotalLabelStyleNames()); + totalLabel.addStyleNames(styleNames); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java new file mode 100644 index 00000000000..45e459ac760 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java @@ -0,0 +1,64 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components; + +import java.util.HashMap; +import java.util.Map; + +import com.vaadin.ui.CssLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.ui.utils.CssStyles; + +public class AefiTypeStatisticsGroupComponent extends CssLayout { + + Map componentMap = new HashMap<>(); + + public AefiTypeStatisticsGroupComponent() { + + setWidthFull(); + } + + public void update(Map> countsByVaccineData) { + + //temporary fix: re-use stored components and hide/unhide if not in new update + if (countsByVaccineData.isEmpty()) { + removeAllComponents(); + componentMap.clear(); + } + + AefiTypeStatisticsComponent statisticsComponent; + + for (Map.Entry> entry : countsByVaccineData.entrySet()) { + if (componentMap.containsKey(entry.getKey())) { + componentMap.get(entry.getKey()).update(entry.getValue()); + } else { + statisticsComponent = new AefiTypeStatisticsComponent("", null, entry.getKey().toString(), false); + //statisticsComponent.getHeading().getTitleLabel().setValue(entry.getKey().toString()); + statisticsComponent.hideHeading(); + statisticsComponent.update(entry.getValue()); + statisticsComponent.addStyleNames(CssStyles.VIEW_SECTION, CssStyles.PADDING_X_8, CssStyles.HSPACE_RIGHT_3); + componentMap.put(entry.getKey(), statisticsComponent); + + addComponent(statisticsComponent); + } + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationDataView.java index bc59daabd97..d2cba6a96d0 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationDataView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationDataView.java @@ -4,20 +4,25 @@ import de.symeda.sormas.api.EditPermissionType; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListCriteria; import de.symeda.sormas.api.feature.FeatureType; import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.immunization.MeansOfImmunization; import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefilink.AefiListComponent; import de.symeda.sormas.ui.immunization.components.form.ImmunizationDataForm; import de.symeda.sormas.ui.sormastosormas.SormasToSormasListComponent; import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DetailSubComponentWrapper; import de.symeda.sormas.ui.utils.LayoutWithSidePanel; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponentLayout; public class ImmunizationDataView extends AbstractImmunizationView { public static final String VIEW_NAME = ROOT_VIEW_NAME + "/data"; public static final String IMMUNIZATION_LOC = "immunization"; + public static final String ADVERSE_EVENTS_LOC = "adverseEvents"; public static final String SORMAS_TO_SORMAS_LOC = "sormsToSormas"; private CommitDiscardWrapperComponent editComponent; @@ -46,11 +51,24 @@ protected void initView(String params) { setSubComponent(container); container.setEnabled(true); - LayoutWithSidePanel layout = new LayoutWithSidePanel(editComponent, SORMAS_TO_SORMAS_LOC); + LayoutWithSidePanel layout = new LayoutWithSidePanel(editComponent, SORMAS_TO_SORMAS_LOC, ADVERSE_EVENTS_LOC); container.addComponent(layout); ImmunizationDto immunization = FacadeProvider.getImmunizationFacade().getImmunizationByUuid(getReference().getUuid()); + + if (FacadeProvider.getFeatureConfigurationFacade().isFeatureEnabled(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT) + && (immunization.getMeansOfImmunization() == MeansOfImmunization.VACCINATION + || immunization.getMeansOfImmunization() == MeansOfImmunization.VACCINATION_RECOVERY)) { + AefiListCriteria aefiListCriteria = new AefiListCriteria.Builder(getReference()).build(); + + AefiListComponent aefiListComponent = + new AefiListComponent(aefiListCriteria, this::showUnsavedChangesPopup, isEditAllowed(), immunization.getVaccinations().size()); + CssStyles.style(aefiListComponent, CssStyles.VIEW_SECTION); + + layout.addSidePanelComponent(new SideComponentLayout(aefiListComponent), ADVERSE_EVENTS_LOC); + } + boolean sormasToSormasEnabled = FacadeProvider.getSormasToSormasFacade() .isAnyFeatureConfigured( FeatureType.SORMAS_TO_SORMAS_SHARE_CASES, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java index 48f3db9e746..5bfa1ac1249 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java @@ -30,6 +30,8 @@ import de.symeda.sormas.api.CoreFacade; import de.symeda.sormas.api.EntityDto; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiFacade; import de.symeda.sormas.api.campaign.CampaignDto; import de.symeda.sormas.api.campaign.CampaignFacade; import de.symeda.sormas.api.caze.CaseDataDto; @@ -93,6 +95,10 @@ public static CoreEntityArchiveHandler forI return new CoreEntityArchiveHandler<>(FacadeProvider.getImmunizationFacade(), ArchiveMessages.IMMUNIZATION); } + public static CoreEntityArchiveHandler forAefi() { + return new CoreEntityArchiveHandler<>(FacadeProvider.getAefiFacade(), ArchiveMessages.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION); + } + public static CoreEntityArchiveHandler forTravelEntry() { return new CoreEntityArchiveHandler<>(FacadeProvider.getTravelEntryFacade(), ArchiveMessages.TRAVEL_ENTRY); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java index 8b0aef3c5e9..079f167ae09 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java @@ -106,6 +106,23 @@ public enum ArchiveMessages { null, null), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION(Strings.headingArchiveImmunization, + Strings.confirmationArchiveImmunization, + null, + Strings.messageImmunizationArchived, + Strings.headingDearchiveImmunization, + Strings.confirmationDearchiveImmunization, + null, + Strings.messageImmunizationDearchived, + null, + null, + null, + null, + null, + null, + null, + null), + TRAVEL_ENTRY(Strings.headingArchiveTravelEntry, Strings.confirmationArchiveTravelEntry, null, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java index 130e7bfad5f..1c82e66f59f 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java @@ -80,6 +80,7 @@ private CssStyles() { public static final String INDENT_LEFT_1 = "indent-left-1"; public static final String INDENT_LEFT_2 = "indent-left-2"; public static final String INDENT_LEFT_3 = "indent-left-3"; + public static final String PADDING_NONE = "padding-none"; public static final String VAADIN_LABEL = "v-label"; public static final String VAADIN_CAPTION = "v-caption"; @@ -238,6 +239,7 @@ private CssStyles() { public static final String GRID_ROW_STATUS_PROGRESS = "status-progress"; public static final String GRID_ROW_TITLE = "row-title"; + public static final String GRID_ROW_SELECTED = "v-grid-row-selected"; public static final String LABEL_CONFIGURATION_SEVERITY_INDICATOR = "severity-indicator"; public static final String BADGE = "badge"; @@ -309,6 +311,21 @@ private CssStyles() { @Deprecated public static final String CALLOUT = "callout"; + public static final String VIEW_SECTION = "view-section"; + public static final String PAGE_TITLE = "page-title"; + public static final String ROUNDED_BORDER = "rounded-border"; + public static final String ROUNDED_BORDER_TOP = "rounded-border-top"; + public static final String ROUNDED_BORDER_BOTTOM = "rounded-border-bottom"; + public static final String ROUNDED_BORDER_SM = "rounded-border-sm"; + public static final String ROUNDED_BORDER_TOP_SM = "rounded-border-sm-top"; + public static final String ROUNDED_BORDER_BOTTOM_SM = "rounded-border-sm-bottom"; + public static final String MARGIN_X_4 = "margin-x-4"; + public static final String MARGIN_TOP_4 = "margin-top-4"; + public static final String PADDING_X_8 = "padding-x-8"; + public static final String PADDING_X_20 = "padding-x-20"; + public static final String VIEW_SECTION_MARGIN_X_4 = VIEW_SECTION + " " + MARGIN_X_4; + public static final String VIEW_SECTION_MARGIN_TOP_4_MARGIN_X_4 = VIEW_SECTION + " " + MARGIN_X_4 + " " + MARGIN_TOP_4; + public static String buildVaadinStyle(String primaryStyle, String... styles) { StringBuilder styleBuilder = new StringBuilder(); styleBuilder.append(primaryStyle); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java index ad062f3b4b8..6e0290496cd 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java @@ -19,6 +19,9 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.ReferenceDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeizureType; +import de.symeda.sormas.api.caze.Trimester; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.symptoms.SymptomState; import de.symeda.sormas.api.utils.FieldConstraints; @@ -26,6 +29,9 @@ import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.ActivityAsCase.ActivityAsCaseField; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField_2; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.AdverseEventsForm; import de.symeda.sormas.ui.clinicalcourse.HealthConditionsForm; import de.symeda.sormas.ui.exposure.ExposuresField; import de.symeda.sormas.ui.hospitalization.PreviousHospitalizationsField; @@ -72,7 +78,11 @@ public SormasFieldGroupFieldFactory( public T createField(Class type, Class fieldType) { if (type.isEnum()) { if (fieldType.isAssignableFrom(Field.class) // no specific fieldType defined? - && (SymptomState.class.isAssignableFrom(type) || YesNoUnknown.class.isAssignableFrom(type))) { + && (SymptomState.class.isAssignableFrom(type) + || YesNoUnknown.class.isAssignableFrom(type) + || AdverseEventState.class.isAssignableFrom(type) + || SeizureType.class.isAssignableFrom(type) + || Trimester.class.isAssignableFrom(type))) { NullableOptionGroup field = new NullableOptionGroup(); field.setImmediate(true); populateWithEnumData(field, (Class) type); @@ -177,7 +187,11 @@ public T createField(Class type, Class fieldType) { return (T) new CustomizableEnumPropertiesComponent(); } else if (UserField.class.isAssignableFrom(fieldType)) { return (T) new UserField(); - } else if (CheckBoxTree.class.isAssignableFrom(fieldType)) { + } else if (AefiVaccinationsField.class.isAssignableFrom(fieldType)) { + return (T) new AefiVaccinationsField(fieldAccessCheckers); + } else if (AdverseEventsForm.class.isAssignableFrom(fieldType)) { + return (T) new AdverseEventsForm(fieldVisibilityCheckers, fieldAccessCheckers); + }else if (CheckBoxTree.class.isAssignableFrom(fieldType)){ return (T) new CheckBoxTree<>(); } return super.createField(type, fieldType); diff --git a/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss b/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss index 6e6c108f8eb..a84710ea9e3 100644 --- a/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss +++ b/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss @@ -250,6 +250,27 @@ margin-left: 45px; } + .margin-x-4{ + margin-left: 4px; + margin-right: 4px; + } + + .margin-top-4{ + margin-top: 4px; + } + + .padding-none{ + padding: 0 !important; + } + + .padding-x-8{ + padding: 0 8px !important; + } + + .padding-x-20{ + padding: 0 20px !important; + } + .vertical-rule { border-right: 1px solid #CDD8EC; } @@ -343,6 +364,52 @@ padding: 6px; } + .view-section{ + padding: 10px 10px; + border-radius: 4px; + background-color: #ffffff; + border-width: 0; + border-style: solid; + border-color: #ffffff; + + /*box-shadow: 0px 2px 1px -1px rgba(0, 0, 0, .2), 0px 1px 1px 0px rgba(0, 0, 0, .14), 0px 1px 3px 0px rgba(0, 0, 0, .12);*/ + + -webkit-box-shadow: 0 0 2px rgba(0,0,0,0.12), 0 2px 4px rgba(0,0,0,0.14);; + box-shadow: 0 0 2px rgba(0,0,0,0.12), 0 2px 4px rgba(0,0,0,0.14);; + } + + .page-title{ + font-size: 24px !important; + } + + .rounded-border{ + border-radius: 8px; + } + + .rounded-border-top{ + border-top-left-radius: 8px; + border-top-right-radius: 8px; + } + + .rounded-border-bottom{ + border-bottom-left-radius: 8px; + border-bottom-right-radius: 8px; + } + + .rounded-border-sm{ + border-radius: 4px; + } + + .rounded-border-sm-top{ + border-top-left-radius: 4px; + border-top-right-radius: 4px; + } + + .rounded-border-sm-bottom{ + border-bottom-left-radius: 4px; + border-bottom-right-radius: 4px; + } + .spacing-small { .v-spacing { width: 4px; From 0b1cd5850698b08cd49ce5919e71daa57404b69f Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Sun, 24 Mar 2024 12:58:34 +0100 Subject: [PATCH 05/56] #12634 - AEFI module - AEFI Investigation - #13053 - Create a new AEFI investigation entity and IndexDTO - #13056 - Add an AEFI Investigations directory for the web app - #13054 - Add a create/edit form for AEFI investigations in the web app - #13055 - Add an AEFI investigations list to AEFI report forms --- .../de/symeda/sormas/api/FacadeProvider.java | 5 + .../AefiCausality.java | 29 + .../AefiClassification.java | 30 + .../AefiClassificationSubType.java | 39 + .../AefiCriteria.java | 35 +- .../AefiDto.java | 91 +- .../AefiExportDto.java | 568 +++++ .../AefiFacade.java | 6 +- .../AefiHelper.java | 46 +- .../AefiImmunizationPeriod.java | 30 + .../AefiIndexDto.java | 16 +- .../AefiInvestigationCriteria.java | 240 +++ .../AefiInvestigationDateType.java | 30 + .../AefiInvestigationDto.java | 1684 +++++++++++++++ .../AefiInvestigationFacade.java | 29 + .../AefiInvestigationIndexDto.java | 372 ++++ .../AefiInvestigationListCriteria.java | 44 + .../AefiInvestigationListEntryDto.java | 156 ++ .../AefiInvestigationReferenceDto.java | 42 + .../AefiInvestigationStage.java | 30 + .../AefiInvestigationStatus.java | 33 + .../AefiReferenceDto.java | 7 +- .../AefiVaccinationPeriod.java | 30 + .../BirthTerm.java | 30 + .../DeliveryProcedure.java | 31 + .../PatientStatusAtAefiInvestigation.java | 32 + .../PlaceOfVaccination.java | 30 + .../SeriousAefiInfoSource.java | 31 + .../SyringeType.java | 31 + .../VaccinationActivity.java | 30 + .../VaccinationSite.java | 32 + .../VaccineCarrier.java | 30 + .../de/symeda/sormas/api/caze/Vaccine.java | 12 + .../sormas/api/caze/VaccineManufacturer.java | 12 + .../AefiDashboardFacade.java | 9 +- .../de/symeda/sormas/api/i18n/Captions.java | 261 +++ .../de/symeda/sormas/api/i18n/Strings.java | 26 +- .../symeda/sormas/api/i18n/Validations.java | 5 +- .../de/symeda/sormas/api/user/UserRight.java | 2 + .../src/main/resources/captions.properties | 270 ++- sormas-api/src/main/resources/enum.properties | 102 +- .../src/main/resources/strings.properties | 33 +- .../src/main/resources/validations.properties | 5 +- .../AefiFacadeEjb.java | 42 +- .../AefiInvestigationFacadeEjb.java | 641 ++++++ ...igationJurisdictionPredicateValidator.java | 116 ++ .../AefiInvestigationQueryContext.java | 45 + .../AefiInvestigationService.java | 548 +++++ .../AefiJurisdictionPredicateValidator.java | 2 +- .../AefiService.java | 166 +- .../entity/Aefi.java | 99 +- .../entity/AefiInvestigation.java | 1832 +++++++++++++++++ .../entity/AefiInvestigationJoins.java | 138 ++ .../entity/AefiJoins.java | 5 +- .../AefiIndexDtoResultTransformer.java | 24 +- ...nvestigationIndexDtoResultTransformer.java | 84 + ...tigationListEntryDtoResultTransformer.java | 53 + .../AefiDashboardFacadeEjb.java | 13 + .../AefiDashboardService.java | 177 +- .../main/resources/META-INF/persistence.xml | 4 +- .../symeda/sormas/ui/ControllerProvider.java | 7 + .../AbstractAefiDataView.java | 113 + .../AbstractAefiInvestigationDataView.java | 122 ++ .../AbstractAefiView.java | 87 +- .../AefiController.java | 38 +- .../AefiDataView.java | 21 +- .../AefiInvestigationController.java | 229 +++ .../AefiInvestigationDataForm.java | 950 +++++++++ .../AefiInvestigationDataView.java | 132 ++ .../AefiInvestigationView.java | 244 +++ .../AefiView.java | 69 +- .../AefiInvestigationList.java | 88 + .../AefiInvestigationListComponent.java | 57 + .../AefiInvestigationListEntry.java | 76 + .../aefilink/AefiListComponent.java | 5 +- .../components/directory/AefiDataLayout.java | 7 +- .../components/directory/AefiFilterForm.java | 12 +- .../components/directory/AefiGrid.java | 26 +- .../AefiInvestigationDataLayout.java | 43 + .../AefiInvestigationFilterForm.java | 418 ++++ .../AefiInvestigationFilterFormLayout.java | 53 + .../directory/AefiInvestigationGrid.java | 151 ++ .../vaccines/AefiVaccinationsField.java | 52 +- .../vaccines/AefiVaccinationsField_2.java | 198 -- .../components/form/AdverseEventsForm.java | 9 +- .../components/form/AefiDataForm.java | 45 +- .../components/form/FormSectionAccordion.java | 55 + .../form/FormSectionAccordionPanel.java | 77 + .../components/information/AefiInfo.java | 89 + .../AefiDashboardDataProvider.java | 30 +- .../AefiDashboardFilterLayout.java | 5 +- .../AefiDashboardView.java | 103 +- .../components/AefiCountTilesComponent.java | 5 +- .../AefiReactionsByGenderChart.java | 15 +- .../AefiTypeStatisticsGroupComponent.java | 11 +- .../sormas/ui/utils/AefiDownloadUtil.java | 67 + .../sormas/ui/utils/ArchiveHandlers.java | 8 + .../sormas/ui/utils/ArchiveMessages.java | 28 +- .../de/symeda/sormas/ui/utils/CssStyles.java | 2 + .../sormas/ui/utils/ExportEntityName.java | 3 + .../utils/SormasFieldGroupFieldFactory.java | 9 +- 101 files changed, 11747 insertions(+), 607 deletions(-) create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCausality.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassification.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassificationSubType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiExportDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiImmunizationPeriod.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationCriteria.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDateType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationFacade.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationIndexDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListCriteria.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListEntryDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationReferenceDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStage.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStatus.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiVaccinationPeriod.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/BirthTerm.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/DeliveryProcedure.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PatientStatusAtAefiInvestigation.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PlaceOfVaccination.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiInfoSource.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SyringeType.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationActivity.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationSite.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccineCarrier.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationFacadeEjb.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationJurisdictionPredicateValidator.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationQueryContext.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationService.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigationJoins.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationIndexDtoResultTransformer.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationListEntryDtoResultTransformer.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiDataView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiInvestigationDataView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationController.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListComponent.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListEntry.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationDataLayout.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterForm.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterFormLayout.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationGrid.java delete mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordion.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordionPanel.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiInfo.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AefiDownloadUtil.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java b/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java index f7570d1c269..bf85edda351 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/FacadeProvider.java @@ -20,6 +20,7 @@ import de.symeda.sormas.api.action.ActionFacade; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiFacade; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationFacade; import de.symeda.sormas.api.audit.AuditLoggerFacade; import de.symeda.sormas.api.bagexport.BAGExportFacade; import de.symeda.sormas.api.campaign.CampaignFacade; @@ -154,6 +155,10 @@ public static AefiFacade getAefiFacade() { return get().lookupEjbRemote(AefiFacade.class); } + public static AefiInvestigationFacade getAefiInvestigationFacade() { + return get().lookupEjbRemote(AefiInvestigationFacade.class); + } + public static VaccinationFacade getVaccinationFacade() { return get().lookupEjbRemote(VaccinationFacade.class); } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCausality.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCausality.java new file mode 100644 index 00000000000..c29b81af2a7 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCausality.java @@ -0,0 +1,29 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiCausality { + + CONFIRMED, + INCONCLUSIVE; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassification.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassification.java new file mode 100644 index 00000000000..8f8771c4c18 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassification.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiClassification { + + RELATED_TO_VACCINE_OR_VACCINATION, + COINCIDENTAL_ADVERSE_EVENT, + UNDETERMINED; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassificationSubType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassificationSubType.java new file mode 100644 index 00000000000..e94d6a7010b --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiClassificationSubType.java @@ -0,0 +1,39 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import static de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification.RELATED_TO_VACCINE_OR_VACCINATION; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiClassificationSubType { + + VACCINE_PRODUCT_RELATED(RELATED_TO_VACCINE_OR_VACCINATION), + VACCINE_QUALITY_DEFECT_RELATED(RELATED_TO_VACCINE_OR_VACCINATION), + IMMUNIZATION_ERROR_RELATED(RELATED_TO_VACCINE_OR_VACCINATION), + IMMUNIZATION_ANXIETY_RELATED(RELATED_TO_VACCINE_OR_VACCINATION); + + private AefiClassification aefiClassification; + + AefiClassificationSubType(AefiClassification aefiClassification) { + this.aefiClassification = aefiClassification; + } + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java index 742df524387..4edcde120d6 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiCriteria.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -39,14 +36,14 @@ public class AefiCriteria extends BaseCriteria implements Serializable { public static final String I18N_PREFIX = "AefiCriteria"; public static final String DISEASE = "disease"; - public static final String NAME_ADDRESS_PHONE_EMAIL_LIKE = "nameAddressPhoneEmailLike"; + public static final String PERSON_LIKE = "personLike"; public static final String AEFI_TYPE = "aefiType"; - public static final String OUTCOME = "outcome"; public static final String VACCINE_NAME = "vaccineName"; public static final String VACCINE_MANUFACTURER = "vaccineManufacturer"; public static final String REGION = "region"; public static final String DISTRICT = "district"; public static final String COMMUNITY = "community"; + public static final String OUTCOME = "outcome"; public static final String FACILITY_TYPE_GROUP = "facilityTypeGroup"; public static final String FACILITY_TYPE = "facilityType"; public static final String HEALTH_FACILITY = "healthFacility"; @@ -57,14 +54,14 @@ public class AefiCriteria extends BaseCriteria implements Serializable { public static final String RELEVANCE_STATUS = "relevanceStatus"; private Disease disease; - private String nameAddressPhoneEmailLike; + private String personLike; private AefiType aefiType; - private AefiOutcome outcome; private Vaccine vaccineName; private VaccineManufacturer vaccineManufacturer; private RegionReferenceDto region; private DistrictReferenceDto district; private CommunityReferenceDto community; + private AefiOutcome outcome; private FacilityTypeGroup facilityTypeGroup; private FacilityType facilityType; private FacilityReferenceDto healthFacility; @@ -82,12 +79,12 @@ public void setDisease(Disease disease) { this.disease = disease; } - public String getNameAddressPhoneEmailLike() { - return nameAddressPhoneEmailLike; + public String getPersonLike() { + return personLike; } - public void setNameAddressPhoneEmailLike(String nameAddressPhoneEmailLike) { - this.nameAddressPhoneEmailLike = nameAddressPhoneEmailLike; + public void setPersonLike(String personLike) { + this.personLike = personLike; } public AefiType getAefiType() { @@ -98,14 +95,6 @@ public void setAefiType(AefiType aefiType) { this.aefiType = aefiType; } - public AefiOutcome getOutcome() { - return outcome; - } - - public void setOutcome(AefiOutcome outcome) { - this.outcome = outcome; - } - public Vaccine getVaccineName() { return vaccineName; } @@ -146,6 +135,14 @@ public void setCommunity(CommunityReferenceDto community) { this.community = community; } + public AefiOutcome getOutcome() { + return outcome; + } + + public void setOutcome(AefiOutcome outcome) { + this.outcome = outcome; + } + public FacilityTypeGroup getFacilityTypeGroup() { return facilityTypeGroup; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java index 7d32221109d..e98c7ca0d9e 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiDto.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -81,13 +78,13 @@ public class AefiDto extends PseudonymizableDto { public static final String AGE_GROUP = "ageGroup"; public static final String HEALTH_FACILITY = "healthFacility"; public static final String HEALTH_FACILITY_DETAILS = "healthFacilityDetails"; - public static final String REPORTER_NAME = "reporterName"; - public static final String REPORTER_INSTITUTION = "reporterInstitution"; - public static final String REPORTER_DESIGNATION = "reporterDesignation"; - public static final String REPORTER_DEPARTMENT = "reporterDepartment"; - public static final String REPORTER_ADDRESS = "reporterAddress"; - public static final String REPORTER_PHONE = "reporterPhone"; - public static final String REPORTER_EMAIL = "reporterEmail"; + public static final String REPORTING_OFFICER_NAME = "reportingOfficerName"; + public static final String REPORTING_OFFICER_FACILITY = "reportingOfficerFacility"; + public static final String REPORTING_OFFICER_DESIGNATION = "reportingOfficerDesignation"; + public static final String REPORTING_OFFICER_DEPARTMENT = "reportingOfficerDepartment"; + public static final String REPORTING_OFFICER_ADDRESS = "reportingOfficerAddress"; + public static final String REPORTING_OFFICER_PHONE_NUMBER = "reportingOfficerPhoneNumber"; + public static final String REPORTING_OFFICER_EMAIL = "reportingOfficerEmail"; public static final String TODAYS_DATE = "todaysDate"; public static final String START_DATE_TIME = "startDateTime"; public static final String AEFI_DESCRIPTION = "aefiDescription"; @@ -110,7 +107,7 @@ public class AefiDto extends PseudonymizableDto { private ImmunizationReferenceDto immunization; private PersonReferenceDto person; private LocationDto address; - @NotEmpty(message = Validations.aefiWithoutSuspectVaccine) + @NotEmpty(message = Validations.aefiWithoutSuspectVaccines) private List vaccinations = new ArrayList<>(); @NotNull(message = Validations.aefiWithoutPrimarySuspectVaccine) private VaccinationDto primarySuspectVaccine; @@ -139,17 +136,17 @@ public class AefiDto extends PseudonymizableDto { @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String healthFacilityDetails; @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) - private String reporterName; - private FacilityReferenceDto reporterInstitution; + private String reportingOfficerName; + private FacilityReferenceDto reportingOfficerFacility; @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) - private String reporterDesignation; + private String reportingOfficerDesignation; @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) - private String reporterDepartment; - private LocationDto reporterAddress; + private String reportingOfficerDepartment; + private LocationDto reportingOfficerAddress; @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) - private String reporterPhone; + private String reportingOfficerPhoneNumber; @Size(max = FieldConstraints.CHARACTER_LIMIT_SMALL, message = Validations.textTooLong) - private String reporterEmail; + private String reportingOfficerEmail; private Date todaysDate; private Date startDateTime; @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) @@ -209,7 +206,7 @@ public static AefiDto build(AefiReferenceDto aefiReferenceDto) { } public AefiReferenceDto toReference() { - return new AefiReferenceDto(getUuid(), getPerson().getCaption(), getExternalId()); + return new AefiReferenceDto(getUuid(), getExternalId()); } public ImmunizationReferenceDto getImmunization() { @@ -404,60 +401,60 @@ public void setHealthFacilityDetails(String healthFacilityDetails) { this.healthFacilityDetails = healthFacilityDetails; } - public String getReporterName() { - return reporterName; + public String getReportingOfficerName() { + return reportingOfficerName; } - public void setReporterName(String reporterName) { - this.reporterName = reporterName; + public void setReportingOfficerName(String reportingOfficerName) { + this.reportingOfficerName = reportingOfficerName; } - public FacilityReferenceDto getReporterInstitution() { - return reporterInstitution; + public FacilityReferenceDto getReportingOfficerFacility() { + return reportingOfficerFacility; } - public void setReporterInstitution(FacilityReferenceDto reporterInstitution) { - this.reporterInstitution = reporterInstitution; + public void setReportingOfficerFacility(FacilityReferenceDto reportingOfficerFacility) { + this.reportingOfficerFacility = reportingOfficerFacility; } - public String getReporterDesignation() { - return reporterDesignation; + public String getReportingOfficerDesignation() { + return reportingOfficerDesignation; } - public void setReporterDesignation(String reporterDesignation) { - this.reporterDesignation = reporterDesignation; + public void setReportingOfficerDesignation(String reportingOfficerDesignation) { + this.reportingOfficerDesignation = reportingOfficerDesignation; } - public String getReporterDepartment() { - return reporterDepartment; + public String getReportingOfficerDepartment() { + return reportingOfficerDepartment; } - public void setReporterDepartment(String reporterDepartment) { - this.reporterDepartment = reporterDepartment; + public void setReportingOfficerDepartment(String reportingOfficerDepartment) { + this.reportingOfficerDepartment = reportingOfficerDepartment; } - public LocationDto getReporterAddress() { - return reporterAddress; + public LocationDto getReportingOfficerAddress() { + return reportingOfficerAddress; } - public void setReporterAddress(LocationDto reporterAddress) { - this.reporterAddress = reporterAddress; + public void setReportingOfficerAddress(LocationDto reportingOfficerAddress) { + this.reportingOfficerAddress = reportingOfficerAddress; } - public String getReporterPhone() { - return reporterPhone; + public String getReportingOfficerPhoneNumber() { + return reportingOfficerPhoneNumber; } - public void setReporterPhone(String reporterPhone) { - this.reporterPhone = reporterPhone; + public void setReportingOfficerPhoneNumber(String reportingOfficerPhoneNumber) { + this.reportingOfficerPhoneNumber = reportingOfficerPhoneNumber; } - public String getReporterEmail() { - return reporterEmail; + public String getReportingOfficerEmail() { + return reportingOfficerEmail; } - public void setReporterEmail(String reporterEmail) { - this.reporterEmail = reporterEmail; + public void setReportingOfficerEmail(String reportingOfficerEmail) { + this.reportingOfficerEmail = reportingOfficerEmail; } public Date getTodaysDate() { diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiExportDto.java new file mode 100644 index 00000000000..2c41a8827ac --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiExportDto.java @@ -0,0 +1,568 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import org.apache.commons.lang3.StringUtils; + +import de.symeda.sormas.api.caze.BirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.caze.VaccineManufacturer; +import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.utils.Order; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableIndexDto; + +public class AefiExportDto extends PseudonymizableIndexDto implements Serializable { + + private static final long serialVersionUID = 4568112808032880384L; + + public static final String I18N_PREFIX = "AefiExport"; + + public static final String RECEIVED_AT_NATIONAL_LEVEL_DATE = "receivedAtNationalLevelDate"; + public static final String VACCINATION_FACILITY_NAME = "vaccinationFacilityName"; + public static final String VACCINATION_FACILITY_REGION = "vaccinationFacilityRegion"; + public static final String VACCINATION_FACILITY_DISTRICT = "vaccinationFacilityDistrict"; + public static final String VACCINATION_FACILITY_COMMUNITY = "vaccinationFacilityCommunity"; + public static final String REPORTING_OFFICER_ADDRESS_COUNTRY_NAME = "reportingOfficerAddressCountryName"; + public static final String PATIENT_ADDRESS_REGION = "patientAddressRegion"; + public static final String PATIENT_ADDRESS_DISTRICT = "patientAddressDistrict"; + public static final String PATIENT_ADDRESS_COMMUNITY = "patientAddressCommunity"; + public static final String PATIENT_ADDRESS_DETAILS = "patientAddressDetails"; + public static final String REPORTING_ID_NUMBER = "reportingIdNumber"; + public static final String WORLDWIDE_ID = "worldWideId"; + public static final String FIRST_NAME = "firstName"; + public static final String LAST_NAME = "lastName"; + public static final String BIRTH_DATE = "birthDate"; + public static final String ONSET_AGE_YEARS = "onsetAgeYears"; + public static final String ONSET_AGE_MONTHS = "onsetAgeMonths"; + public static final String ONSET_AGE_DAYS = "onsetAgeDays"; + public static final String ONSET_AGE_GROUP = "onsetAgeGroup"; + public static final String SEX = "sex"; + public static final String AEFI_DESCRIPTION = "aefiDescription"; + public static final String PRIMARY_SUSPECT_VACCINE_NAME = "primarySuspectVaccineName"; + public static final String PRIMARY_SUSPECT_VACCINE_OTHER_NAME = "primarySuspectVaccineOtherName"; + public static final String PRIMARY_SUSPECT_VACCINE_BRAND = "primarySuspectVaccineBrand"; + public static final String PRIMARY_SUSPECT_VACCINE_MANUFACTURER = "primarySuspectVaccineManufacturer"; + public static final String PRIMARY_SUSPECT_VACCINE_BATCH_NUMBER = "primarySuspectVaccineBatchNumber"; + public static final String PRIMARY_SUSPECT_VACCINE_DOSE = "primarySuspectVaccineDose"; + public static final String PRIMARY_SUSPECT_VACCINE_DILUENT_BATCH_NUMBER = "primarySuspectVaccineDiluentBatchNumber"; + public static final String PRIMARY_SUSPECT_VACCINE_VACCINATION_DATE = "primarySuspectVaccineVaccinationDate"; + public static final String START_DATE_TIME = "startDateTime"; + public static final String SEVERE_LOCAL_REACTION = "severeLocalReaction"; + public static final String SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS = "severeLocalReactionMoreThanThreeDays"; + public static final String SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT = "severeLocalReactionBeyondNearestJoint"; + public static final String SEIZURES = "seizures"; + public static final String SEIZURE_TYPE = "seizureType"; + public static final String ABSCESS = "abscess"; + public static final String SEPSIS = "sepsis"; + public static final String ENCEPHALOPATHY = "encephalopathy"; + public static final String TOXIC_SHOCK_SYNDROME = "toxicShockSyndrome"; + public static final String THROMBOCYTOPENIA = "thrombocytopenia"; + public static final String ANAPHYLAXIS = "anaphylaxis"; + public static final String FEVERISH_FEELING = "feverishFeeling"; + public static final String OTHER_ADVERSE_EVENT_DETAILS = "otherAdverseEventDetails"; + public static final String OUTCOME = "outcome"; + public static final String SERIOUS = "serious"; + public static final String REPORTING_OFFICER_NAME = "reportingOfficerName"; + public static final String REPORTING_OFFICER_FACILITY_NAME = "reportingOfficerFacilityName"; + public static final String REPORTING_OFFICER_FACILITY_REGION = "reportingOfficerFacilityRegion"; + public static final String REPORTING_OFFICER_FACILITY_DISTRICT = "reportingOfficerFacilityDistrict"; + public static final String REPORTING_OFFICER_FACILITY_COMMUNITY = "reportingOfficerFacilityCommunity"; + public static final String REPORTING_OFFICER_DESIGNATION = "reportingOfficerDesignation"; + public static final String REPORTING_OFFICER_DEPARTMENT = "reportingOfficerDepartment"; + public static final String REPORTING_OFFICER_EMAIL = "reportingOfficerEmail"; + public static final String REPORTING_OFFICER_PHONE_NUMBER = "reportingOfficerPhoneNumber"; + public static final String REPORT_DATE = "reportDate"; + public static final String NATIONAL_LEVEL_COMMENT = "nationalLevelComment"; + + private Date receivedAtNationalLevelDate; + private String vaccinationFacilityName; + private String vaccinationFacilityRegion; + private String vaccinationFacilityDistrict; + private String vaccinationFacilityCommunity; + private String reportingOfficerAddressCountryName; + private String patientAddressRegion; + private String patientAddressDistrict; + private String patientAddressCommunity; + private String patientAddressDetails; + private String reportingIdNumber; + private String worldWideId; + private String firstName; + private String lastName; + private BirthDateDto birthDate; + private Integer onsetAgeYears; + private Integer onsetAgeMonths; + private Integer onsetAgeDays; + private AefiAgeGroup onsetAgeGroup; + private Sex sex; + private String aefiDescription; + private Vaccine primarySuspectVaccineName; + private String primarySuspectVaccineOtherName; + private String primarySuspectVaccineBrandName; + private VaccineManufacturer primarySuspectVaccineManufacturer; + private String primarySuspectVaccineBatchNumber; + private String primarySuspectVaccineDose; + private String primarySuspectVaccineDiluentBatchNumber; + private Date primarySuspectVaccineVaccinationDate; + private Date startDateTime; + private AdverseEventState severeLocalReaction; + private boolean severeLocalReactionMoreThanThreeDays; + private boolean severeLocalReactionBeyondNearestJoint; + private AdverseEventState seizures; + private SeizureType seizureType; + private AdverseEventState abscess; + private AdverseEventState sepsis; + private AdverseEventState encephalopathy; + private AdverseEventState toxicShockSyndrome; + private AdverseEventState thrombocytopenia; + private AdverseEventState anaphylaxis; + private AdverseEventState feverishFeeling; + private String otherAdverseEventDetails; + private AefiOutcome outcome; + private YesNoUnknown serious; + private String reportingOfficerName; + private String reportingOfficerFacilityName; + private String reportingOfficerFacilityRegion; + private String reportingOfficerFacilityDistrict; + private String reportingOfficerFacilityCommunity; + private String reportingOfficerDesignation; + private String reportingOfficerDepartment; + private String reportingOfficerEmail; + private String reportingOfficerPhoneNumber; + private Date reportDate; + private String nationalLevelComment; + private Boolean isInJurisdiction; + + public AefiExportDto( + String uuid, + Date receivedAtNationalLevelDate, + String vaccinationFacilityName, + String vaccinationFacilityRegion, + String vaccinationFacilityDistrict, + String vaccinationFacilityCommunity, + String reportingOfficerAddressCountryName, + String patientAddressRegion, + String patientAddressDistrict, + String patientAddressCommunity, + String street, + String houseNumber, + String postalCode, + String city, + String reportingIdNumber, + String worldWideId, + String firstName, + String lastName, + Integer birthdateDD, + Integer birthdateMM, + Integer birthdateYYYY, + Integer onsetAgeYears, + Integer onsetAgeMonths, + Integer onsetAgeDays, + AefiAgeGroup onsetAgeGroup, + Sex sex, + String aefiDescription, + Vaccine primarySuspectVaccineName, + String primarySuspectVaccineOtherName, + VaccineManufacturer primarySuspectVaccineManufacturer, + String primarySuspectVaccineBatchNumber, + String primarySuspectVaccineDose, + Date primarySuspectVaccineVaccinationDate, + Date startDateTime, + AdverseEventState severeLocalReaction, + boolean severeLocalReactionMoreThanThreeDays, + boolean severeLocalReactionBeyondNearestJoint, + AdverseEventState seizures, + SeizureType seizureType, + AdverseEventState abscess, + AdverseEventState sepsis, + AdverseEventState encephalopathy, + AdverseEventState toxicShockSyndrome, + AdverseEventState thrombocytopenia, + AdverseEventState anaphylaxis, + AdverseEventState feverishFeeling, + String otherAdverseEventDetails, + AefiOutcome outcome, + YesNoUnknown serious, + String reportingOfficerFirstName, + String reportingOfficerLastName, + String reportingOfficerFacilityName, + String reportingOfficerFacilityRegion, + String reportingOfficerFacilityDistrict, + String reportingOfficerFacilityCommunity, + String reportingOfficerEmail, + String reportingOfficerPhoneNumber, + Date reportDate, + String nationalLevelComment, + boolean isInJurisdiction) { + super(uuid); + this.receivedAtNationalLevelDate = receivedAtNationalLevelDate; + this.vaccinationFacilityName = vaccinationFacilityName; + this.vaccinationFacilityRegion = vaccinationFacilityRegion; + this.vaccinationFacilityDistrict = vaccinationFacilityDistrict; + this.vaccinationFacilityCommunity = vaccinationFacilityCommunity; + this.reportingOfficerAddressCountryName = reportingOfficerAddressCountryName; + this.patientAddressRegion = patientAddressRegion; + this.patientAddressDistrict = patientAddressDistrict; + this.patientAddressCommunity = patientAddressCommunity; + + StringBuilder patientAddressBuilder = new StringBuilder(); + if (!StringUtils.isBlank(houseNumber)) { + patientAddressBuilder.append(houseNumber); + } + if (!StringUtils.isBlank(street)) { + patientAddressBuilder.append(", ").append(street); + } + if (!StringUtils.isBlank(postalCode)) { + patientAddressBuilder.append(", ").append(postalCode); + } + if (!StringUtils.isBlank(city)) { + patientAddressBuilder.append(", ").append(city); + } + + this.patientAddressDetails = patientAddressBuilder.toString(); + + this.reportingIdNumber = reportingIdNumber; + this.worldWideId = worldWideId; + this.firstName = firstName; + this.lastName = lastName; + this.birthDate = new BirthDateDto(birthdateDD, birthdateMM, birthdateYYYY); + this.onsetAgeYears = onsetAgeYears; + this.onsetAgeMonths = onsetAgeMonths; + this.onsetAgeDays = onsetAgeDays; + this.onsetAgeGroup = onsetAgeGroup; + this.sex = sex; + this.aefiDescription = aefiDescription; + this.primarySuspectVaccineName = primarySuspectVaccineName; + this.primarySuspectVaccineOtherName = primarySuspectVaccineOtherName; + this.primarySuspectVaccineManufacturer = primarySuspectVaccineManufacturer; + this.primarySuspectVaccineBatchNumber = primarySuspectVaccineBatchNumber; + this.primarySuspectVaccineDose = primarySuspectVaccineDose; + this.primarySuspectVaccineVaccinationDate = primarySuspectVaccineVaccinationDate; + this.startDateTime = startDateTime; + this.severeLocalReaction = severeLocalReaction; + this.severeLocalReactionMoreThanThreeDays = severeLocalReactionMoreThanThreeDays; + this.severeLocalReactionBeyondNearestJoint = severeLocalReactionBeyondNearestJoint; + this.seizures = seizures; + this.seizureType = seizureType; + this.abscess = abscess; + this.sepsis = sepsis; + this.encephalopathy = encephalopathy; + this.toxicShockSyndrome = toxicShockSyndrome; + this.thrombocytopenia = thrombocytopenia; + this.anaphylaxis = anaphylaxis; + this.feverishFeeling = feverishFeeling; + this.otherAdverseEventDetails = otherAdverseEventDetails; + this.outcome = outcome; + this.serious = serious; + this.reportingOfficerName = reportingOfficerFirstName + " " + reportingOfficerLastName; + this.reportingOfficerFacilityName = reportingOfficerFacilityName; + this.reportingOfficerFacilityRegion = reportingOfficerFacilityRegion; + this.reportingOfficerFacilityDistrict = reportingOfficerFacilityDistrict; + this.reportingOfficerFacilityCommunity = reportingOfficerFacilityCommunity; + this.reportingOfficerEmail = reportingOfficerEmail; + this.reportingOfficerPhoneNumber = reportingOfficerPhoneNumber; + this.reportDate = reportDate; + this.nationalLevelComment = nationalLevelComment; + this.isInJurisdiction = isInJurisdiction; + } + + @Order(0) + public Date getReceivedAtNationalLevelDate() { + return receivedAtNationalLevelDate; + } + + @Order(1) + public String getVaccinationFacilityName() { + return vaccinationFacilityName; + } + + @Order(2) + public String getVaccinationFacilityRegion() { + return vaccinationFacilityRegion; + } + + @Order(3) + public String getVaccinationFacilityDistrict() { + return vaccinationFacilityDistrict; + } + + @Order(4) + public String getVaccinationFacilityCommunity() { + return vaccinationFacilityCommunity; + } + + @Order(5) + public String getReportingOfficerAddressCountryName() { + return reportingOfficerAddressCountryName; + } + + @Order(6) + public String getPatientAddressRegion() { + return patientAddressRegion; + } + + @Order(7) + public String getPatientAddressDistrict() { + return patientAddressDistrict; + } + + @Order(8) + public String getPatientAddressCommunity() { + return patientAddressCommunity; + } + + @Order(9) + public String getPatientAddressDetails() { + return patientAddressDetails; + } + + @Order(10) + public String getReportingIdNumber() { + return reportingIdNumber; + } + + @Order(11) + public String getWorldWideId() { + return worldWideId; + } + + @Order(12) + public String getFirstName() { + return firstName; + } + + @Order(13) + public String getLastName() { + return lastName; + } + + @Order(14) + public BirthDateDto getBirthDate() { + return birthDate; + } + + @Order(15) + public Integer getOnsetAgeYears() { + return onsetAgeYears; + } + + @Order(16) + public Integer getOnsetAgeMonths() { + return onsetAgeMonths; + } + + @Order(17) + public Integer getOnsetAgeDays() { + return onsetAgeDays; + } + + @Order(18) + public AefiAgeGroup getOnsetAgeGroup() { + return onsetAgeGroup; + } + + @Order(19) + public Sex getSex() { + return sex; + } + + @Order(20) + public String getAefiDescription() { + return aefiDescription; + } + + @Order(21) + public Vaccine getPrimarySuspectVaccineName() { + return primarySuspectVaccineName; + } + + @Order(22) + public String getPrimarySuspectVaccineOtherName() { + return primarySuspectVaccineOtherName; + } + + @Order(23) + public String getPrimarySuspectVaccineBrandName() { + return primarySuspectVaccineBrandName; + } + + @Order(24) + public VaccineManufacturer getPrimarySuspectVaccineManufacturer() { + return primarySuspectVaccineManufacturer; + } + + @Order(25) + public String getPrimarySuspectVaccineBatchNumber() { + return primarySuspectVaccineBatchNumber; + } + + @Order(26) + public String getPrimarySuspectVaccineDose() { + return primarySuspectVaccineDose; + } + + @Order(27) + public String getPrimarySuspectVaccineDiluentBatchNumber() { + return primarySuspectVaccineDiluentBatchNumber; + } + + @Order(28) + public Date getPrimarySuspectVaccineVaccinationDate() { + return primarySuspectVaccineVaccinationDate; + } + + @Order(29) + public Date getStartDateTime() { + return startDateTime; + } + + @Order(30) + public AdverseEventState getSevereLocalReaction() { + return severeLocalReaction; + } + + @Order(31) + public boolean isSevereLocalReactionMoreThanThreeDays() { + return severeLocalReactionMoreThanThreeDays; + } + + @Order(32) + public boolean isSevereLocalReactionBeyondNearestJoint() { + return severeLocalReactionBeyondNearestJoint; + } + + @Order(33) + public AdverseEventState getSeizures() { + return seizures; + } + + @Order(34) + public SeizureType getSeizureType() { + return seizureType; + } + + @Order(35) + public AdverseEventState getAbscess() { + return abscess; + } + + @Order(36) + public AdverseEventState getSepsis() { + return sepsis; + } + + @Order(37) + public AdverseEventState getEncephalopathy() { + return encephalopathy; + } + + @Order(38) + public AdverseEventState getToxicShockSyndrome() { + return toxicShockSyndrome; + } + + @Order(39) + public AdverseEventState getThrombocytopenia() { + return thrombocytopenia; + } + + @Order(40) + public AdverseEventState getAnaphylaxis() { + return anaphylaxis; + } + + @Order(41) + public AdverseEventState getFeverishFeeling() { + return feverishFeeling; + } + + @Order(42) + public String getOtherAdverseEventDetails() { + return otherAdverseEventDetails; + } + + @Order(43) + public AefiOutcome getOutcome() { + return outcome; + } + + @Order(44) + public YesNoUnknown getSerious() { + return serious; + } + + @Order(45) + public String getReportingOfficerName() { + return reportingOfficerName; + } + + @Order(46) + public String getReportingOfficerFacilityName() { + return reportingOfficerFacilityName; + } + + @Order(47) + public String getReportingOfficerFacilityRegion() { + return reportingOfficerFacilityRegion; + } + + @Order(48) + public String getReportingOfficerFacilityDistrict() { + return reportingOfficerFacilityDistrict; + } + + @Order(49) + public String getReportingOfficerFacilityCommunity() { + return reportingOfficerFacilityCommunity; + } + + @Order(50) + public String getReportingOfficerDesignation() { + return reportingOfficerDesignation; + } + + @Order(51) + public String getReportingOfficerDepartment() { + return reportingOfficerDepartment; + } + + @Order(52) + public String getReportingOfficerEmail() { + return reportingOfficerEmail; + } + + @Order(53) + public String getReportingOfficerPhoneNumber() { + return reportingOfficerPhoneNumber; + } + + @Order(54) + public Date getReportDate() { + return reportDate; + } + + @Order(55) + public String getNationalLevelComment() { + return nationalLevelComment; + } + + public Boolean getInJurisdiction() { + return isInJurisdiction; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java index 4267d27a9b5..a7ed73ac4bb 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiFacade.java @@ -1,23 +1,21 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package de.symeda.sormas.api.adverseeventsfollowingimmunization; +import java.util.Collection; import java.util.List; import javax.ejb.Remote; @@ -28,4 +26,6 @@ public interface AefiFacade extends CoreFacade { List getEntriesList(AefiListCriteria criteria, Integer first, Integer max); + + List getExportList(AefiCriteria criteria, Collection selectedRows, int first, int max); } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java index 815c46365dc..2d4d2b80395 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiHelper.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -29,6 +26,49 @@ private AefiHelper() { } + public static String buildAdverseEventsString(AdverseEventsDto adverseEventsDto) { + + List adverseEventsList = new ArrayList<>(); + + if (adverseEventsDto.getSevereLocalReaction() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.SEVERE_LOCAL_REACTION)); + } + + if (adverseEventsDto.getSeizures() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.SEIZURES)); + } + + if (adverseEventsDto.getAbscess() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.ABSCESS)); + } + + if (adverseEventsDto.getSepsis() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.SEPSIS)); + } + + if (adverseEventsDto.getEncephalopathy() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.ENCEPHALOPATHY)); + } + + if (adverseEventsDto.getToxicShockSyndrome() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.TOXIC_SHOCK_SYNDROME)); + } + + if (adverseEventsDto.getThrombocytopenia() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.THROMBOCYTOPENIA)); + } + + if (adverseEventsDto.getAnaphylaxis() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.ANAPHYLAXIS)); + } + + if (adverseEventsDto.getFeverishFeeling() == AdverseEventState.YES) { + adverseEventsList.add(I18nProperties.getPrefixCaption(AdverseEventsDto.I18N_PREFIX, AdverseEventsDto.FEVERISH_FEELING)); + } + + return String.join(", ", adverseEventsList); + } + public static String buildAdverseEventsString( AdverseEventState severeLocalReaction, boolean severeLocalReactionMoreThanThreeDays, diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiImmunizationPeriod.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiImmunizationPeriod.java new file mode 100644 index 00000000000..e8e943fe41d --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiImmunizationPeriod.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiImmunizationPeriod { + + WITHIN_FIRST_VACCINATIONS, + WITHIN_LAST_VACCINATIONS, + UNKNOWN; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java index e389093fc00..ad982aeff95 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiIndexDto.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -33,6 +30,8 @@ public class AefiIndexDto extends PseudonymizableIndexDto implements Serializable, Cloneable { + private static final long serialVersionUID = 2873180122463056859L; + public static final String I18N_PREFIX = "AefiIndex"; public static final String UUID = "uuid"; @@ -69,6 +68,7 @@ public class AefiIndexDto extends PseudonymizableIndexDto implements Serializabl private String district; private YesNoUnknown serious; private Vaccine primaryVaccine; + private String primaryVaccineDetails; private AefiOutcome outcome; private Date vaccinationDate; private Date startDateTime; @@ -90,6 +90,7 @@ public AefiIndexDto( String district, YesNoUnknown serious, Vaccine primaryVaccine, + String primaryVaccineDetails, AefiOutcome outcome, Date vaccinationDate, Date reportDate, @@ -111,6 +112,7 @@ public AefiIndexDto( this.district = district; this.serious = serious; this.primaryVaccine = primaryVaccine; + this.primaryVaccineDetails = primaryVaccineDetails; this.outcome = outcome; this.vaccinationDate = vaccinationDate; this.reportDate = reportDate; @@ -217,6 +219,14 @@ public void setPrimaryVaccine(Vaccine primaryVaccine) { this.primaryVaccine = primaryVaccine; } + public String getPrimaryVaccineDetails() { + return primaryVaccineDetails; + } + + public void setPrimaryVaccineDetails(String primaryVaccineDetails) { + this.primaryVaccineDetails = primaryVaccineDetails; + } + public AefiOutcome getOutcome() { return outcome; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationCriteria.java new file mode 100644 index 00000000000..f54efee9f85 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationCriteria.java @@ -0,0 +1,240 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.EntityRelevanceStatus; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.caze.VaccineManufacturer; +import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityType; +import de.symeda.sormas.api.infrastructure.facility.FacilityTypeGroup; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.utils.DateFilterOption; +import de.symeda.sormas.api.utils.criteria.BaseCriteria; + +public class AefiInvestigationCriteria extends BaseCriteria implements Serializable { + + private static final long serialVersionUID = 6520144798481981253L; + + public static final String I18N_PREFIX = "AefiInvestigationCriteria"; + public static final String AEFI_REPORT_LIKE = "aefiReportLike"; + public static final String INVESTIGATION_CASE_ID = "investigationCaseId"; + public static final String DISEASE = "disease"; + public static final String PERSON_LIKE = "personLike"; + public static final String AEFI_TYPE = "aefiType"; + public static final String VACCINE_NAME = "vaccineName"; + public static final String VACCINE_MANUFACTURER = "vaccineManufacturer"; + public static final String REGION = "region"; + public static final String DISTRICT = "district"; + public static final String COMMUNITY = "community"; + public static final String STATUS_ON_DATE_OF_INVESTIGATION = "statusAtAefiInvestigation"; + public static final String AEFI_CLASSIFICATION = "aefiClassification"; + public static final String FACILITY_TYPE_GROUP = "facilityTypeGroup"; + public static final String FACILITY_TYPE = "facilityType"; + public static final String HEALTH_FACILITY = "healthFacility"; + public static final String DATE_FILTER_OPTION = "dateFilterOption"; + public static final String AEFI_DATE_TYPE = "aefiDateType"; + public static final String FROM_DATE = "fromDate"; + public static final String TO_DATE = "toDate"; + public static final String RELEVANCE_STATUS = "relevanceStatus"; + + private String aefiReportLike; + private String investigationCaseId; + private Disease disease; + private String personLike; + private AefiType aefiType; + private Vaccine vaccineName; + private VaccineManufacturer vaccineManufacturer; + private RegionReferenceDto region; + private DistrictReferenceDto district; + private CommunityReferenceDto community; + private PatientStatusAtAefiInvestigation statusAtAefiInvestigation; + private AefiClassification aefiClassification; + private FacilityTypeGroup facilityTypeGroup; + private FacilityType facilityType; + private FacilityReferenceDto healthFacility; + private DateFilterOption dateFilterOption = DateFilterOption.DATE; + private AefiInvestigationDateType aefiInvestigationDateType; + private Date fromDate; + private Date toDate; + private EntityRelevanceStatus relevanceStatus; + + public String getAefiReportLike() { + return aefiReportLike; + } + + public void setAefiReportLike(String aefiReportLike) { + this.aefiReportLike = aefiReportLike; + } + + public String getInvestigationCaseId() { + return investigationCaseId; + } + + public void setInvestigationCaseId(String investigationCaseId) { + this.investigationCaseId = investigationCaseId; + } + + public Disease getDisease() { + return disease; + } + + public void setDisease(Disease disease) { + this.disease = disease; + } + + public String getPersonLike() { + return personLike; + } + + public void setPersonLike(String personLike) { + this.personLike = personLike; + } + + public AefiType getAefiType() { + return aefiType; + } + + public void setAefiType(AefiType aefiType) { + this.aefiType = aefiType; + } + + public Vaccine getVaccineName() { + return vaccineName; + } + + public void setVaccineName(Vaccine vaccineName) { + this.vaccineName = vaccineName; + } + + public VaccineManufacturer getVaccineManufacturer() { + return vaccineManufacturer; + } + + public void setVaccineManufacturer(VaccineManufacturer vaccineManufacturer) { + this.vaccineManufacturer = vaccineManufacturer; + } + + public RegionReferenceDto getRegion() { + return region; + } + + public void setRegion(RegionReferenceDto region) { + this.region = region; + } + + public DistrictReferenceDto getDistrict() { + return district; + } + + public void setDistrict(DistrictReferenceDto district) { + this.district = district; + } + + public CommunityReferenceDto getCommunity() { + return community; + } + + public void setCommunity(CommunityReferenceDto community) { + this.community = community; + } + + public PatientStatusAtAefiInvestigation getStatusAtAefiInvestigation() { + return statusAtAefiInvestigation; + } + + public void setStatusAtAefiInvestigation(PatientStatusAtAefiInvestigation statusAtAefiInvestigation) { + this.statusAtAefiInvestigation = statusAtAefiInvestigation; + } + + public AefiClassification getAefiClassification() { + return aefiClassification; + } + + public void setAefiClassification(AefiClassification aefiClassification) { + this.aefiClassification = aefiClassification; + } + + public FacilityTypeGroup getFacilityTypeGroup() { + return facilityTypeGroup; + } + + public void setFacilityTypeGroup(FacilityTypeGroup facilityTypeGroup) { + this.facilityTypeGroup = facilityTypeGroup; + } + + public FacilityType getFacilityType() { + return facilityType; + } + + public void setFacilityType(FacilityType facilityType) { + this.facilityType = facilityType; + } + + public FacilityReferenceDto getHealthFacility() { + return healthFacility; + } + + public void setHealthFacility(FacilityReferenceDto healthFacility) { + this.healthFacility = healthFacility; + } + + public DateFilterOption getDateFilterOption() { + return dateFilterOption; + } + + public void setDateFilterOption(DateFilterOption dateFilterOption) { + this.dateFilterOption = dateFilterOption; + } + + public AefiInvestigationDateType getAefiInvestigationDateType() { + return aefiInvestigationDateType; + } + + public void setAefiInvestigationDateType(AefiInvestigationDateType aefiInvestigationDateType) { + this.aefiInvestigationDateType = aefiInvestigationDateType; + } + + public Date getFromDate() { + return fromDate; + } + + public void setFromDate(Date fromDate) { + this.fromDate = fromDate; + } + + public Date getToDate() { + return toDate; + } + + public void setToDate(Date toDate) { + this.toDate = toDate; + } + + public EntityRelevanceStatus getRelevanceStatus() { + return relevanceStatus; + } + + public void setRelevanceStatus(EntityRelevanceStatus relevanceStatus) { + this.relevanceStatus = relevanceStatus; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDateType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDateType.java new file mode 100644 index 00000000000..7d1fb23e605 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDateType.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiInvestigationDateType { + + REPORT_DATE, + INVESTIGATION_DATE, + VACCINATION_DATE; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDto.java new file mode 100644 index 00000000000..29e8ddd4cb4 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationDto.java @@ -0,0 +1,1684 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; +import java.util.Set; + +import javax.validation.constraints.NotEmpty; +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; + +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; +import de.symeda.sormas.api.infrastructure.country.CountryReferenceDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityReferenceDto; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.location.LocationDto; +import de.symeda.sormas.api.user.UserReferenceDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.DependingOnFeatureType; +import de.symeda.sormas.api.utils.FieldConstraints; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableDto; +import de.symeda.sormas.api.vaccination.VaccinationDto; + +@DependingOnFeatureType(featureType = { + FeatureType.IMMUNIZATION_MANAGEMENT, + FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT }) +public class AefiInvestigationDto extends PseudonymizableDto { + + private static final long serialVersionUID = 7811960334585576774L; + + public static final long APPROXIMATE_JSON_SIZE_IN_BYTES = 25455; + + public static final String I18N_PREFIX = "AefiInvestigation"; + + public static final String AEFI_REPORT = "aefiReport"; + public static final String ADDRESS = "address"; + public static final String VACCINATIONS = "vaccinations"; + public static final String PRIMARY_SUSPECT_VACCINE = "primarySuspectVaccine"; + public static final String REPORT_DATE = "reportDate"; + public static final String REPORTING_USER = "reportingUser"; + public static final String EXTERNAL_ID = "externalId"; + public static final String RESPONSIBLE_REGION = "responsibleRegion"; + public static final String RESPONSIBLE_DISTRICT = "responsibleDistrict"; + public static final String RESPONSIBLE_COMMUNITY = "responsibleCommunity"; + public static final String COUNTRY = "country"; + public static final String INVESTIGATION_CASE_ID = "investigationCaseId"; + public static final String PLACE_OF_VACCINATION = "placeOfVaccination"; + public static final String PLACE_OF_VACCINATION_DETAILS = "placeOfVaccinationDetails"; + public static final String VACCINATION_ACTIVITY = "vaccinationActivity"; + public static final String VACCINATION_ACTIVITY_DETAILS = "vaccinationActivityDetails"; + public static final String VACCINATION_FACILITY = "vaccinationFacility"; + public static final String VACCINATION_FACILITY_DETAILS = "vaccinationFacilityDetails"; + public static final String REPORTING_OFFICER_NAME = "reportingOfficerName"; + public static final String REPORTING_OFFICER_FACILITY = "reportingOfficerFacility"; + public static final String REPORTING_OFFICER_FACILITY_DETAILS = "reportingOfficerFacilityDetails"; + public static final String REPORTING_OFFICER_DESIGNATION = "reportingOfficerDesignation"; + public static final String REPORTING_OFFICER_DEPARTMENT = "reportingOfficerDepartment"; + public static final String REPORTING_OFFICER_ADDRESS = "reportingOfficerAddress"; + public static final String REPORTING_OFFICER_LANDLINE_PHONE_NUMBER = "reportingOfficerLandlinePhoneNumber"; + public static final String REPORTING_OFFICER_MOBILE_PHONE_NUMBER = "reportingOfficerMobilePhoneNumber"; + public static final String REPORTING_OFFICER_EMAIL = "reportingOfficerEmail"; + public static final String INVESTIGATION_DATE = "investigationDate"; + public static final String FORM_COMPLETION_DATE = "formCompletionDate"; + public static final String INVESTIGATION_STAGE = "investigationStage"; + public static final String TYPE_OF_SITE = "typeOfSite"; + public static final String TYPE_OF_SITE_DETAILS = "typeOfSiteDetails"; + public static final String KEY_SYMPTOM_DATE_TIME = "keySymptomDateTime"; + public static final String HOSPITALIZATION_DATE = "hospitalizationDate"; + public static final String REPORTED_TO_HEALTH_AUTHORITY_DATE = "reportedToHealthAuthorityDate"; + public static final String STATUS_ON_DATE_OF_INVESTIGATION = "statusOnDateOfInvestigation"; + public static final String DEATH_DATE_TIME = "deathDateTime"; + public static final String AUTOPSY_DONE = "autopsyDone"; + public static final String AUTOPSY_DATE = "autopsyDate"; + public static final String AUTOPSY_PLANNED_DATE_TIME = "autopsyPlannedDateTime"; + public static final String PAST_HISTORY_OF_SIMILAR_EVENT = "pastHistoryOfSimilarEvent"; + public static final String PAST_HISTORY_OF_SIMILAR_EVENT_DETAILS = "pastHistoryOfSimilarEventDetails"; + public static final String ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS = "adverseEventAfterPreviousVaccinations"; + public static final String ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS_DETAILS = "adverseEventAfterPreviousVaccinationsDetails"; + public static final String HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD = "historyOfAllergyToVaccineDrugOrFood"; + public static final String HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD_DETAILS = "historyOfAllergyToVaccineDrugOrFoodDetails"; + public static final String PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER = "preExistingIllnessThirtyDaysOrCongenitalDisorder"; + public static final String PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER_DETAILS = + "preExistingIllnessThirtyDaysOrCongenitalDisorderDetails"; + public static final String HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE = "historyOfHospitalizationInLastThirtyDaysWithCause"; + public static final String HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE_DETAILS = + "historyOfHospitalizationInLastThirtyDaysWithCauseDetails"; + public static final String CURRENTLY_ON_CONCOMITANT_MEDICATION = "currentlyOnConcomitantMedication"; + public static final String CURRENTLY_ON_CONCOMITANT_MEDICATION_DETAILS = "currentlyOnConcomitantMedicationDetails"; + public static final String FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY = "familyHistoryOfDiseaseOrAllergy"; + public static final String FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY_DETAILS = "familyHistoryOfDiseaseOrAllergyDetails"; + public static final String NUMBER_OF_WEEKS_PREGNANT = "numberOfWeeksPregnant"; + public static final String BIRTH_TERM = "birthTerm"; + public static final String BIRTH_WEIGHT = "birthWeight"; + public static final String DELIVERY_PROCEDURE = "deliveryProcedure"; + public static final String DELIVERY_PROCEDURE_DETAILS = "deliveryProcedureDetails"; + public static final String SERIOUS_AEFI_INFO_SOURCE = "seriousAefiInfoSource"; + public static final String SERIOUS_AEFI_INFO_SOURCE_DETAILS = "seriousAefiInfoSourceDetails"; + public static final String SERIOUS_AEFI_VERBAL_AUTOPSY_INFO_SOURCE_DETAILS = "seriousAefiVerbalAutopsyInfoSourceDetails"; + public static final String FIRST_CAREGIVERS_NAME = "firstCaregiversName"; + public static final String OTHER_CAREGIVERS_NAMES = "otherCaregiversNames"; + public static final String OTHER_SOURCES_WHO_PROVIDED_INFO = "otherSourcesWhoProvidedInfo"; + public static final String SIGNS_AND_SYMPTOMS_FROM_TIME_OF_VACCINATION = "signsAndSymptomsFromTimeOfVaccination"; + public static final String CLINICAL_DETAILS_OFFICER_NAME = "clinicalDetailsOfficerName"; + public static final String CLINICAL_DETAILS_OFFICER_PHONE_NUMBER = "clinicalDetailsOfficerPhoneNumber"; + public static final String CLINICAL_DETAILS_OFFICER_EMAIL = "clinicalDetailsOfficerEmail"; + public static final String CLINICAL_DETAILS_OFFICER_DESIGNATION = "clinicalDetailsOfficerDesignation"; + public static final String CLINICAL_DETAILS_DATE_TIME = "clinicalDetailsDateTime"; + public static final String PATIENT_RECEIVED_MEDICAL_CARE = "patientReceivedMedicalCare"; + public static final String PATIENT_RECEIVED_MEDICAL_CARE_DETAILS = "patientReceivedMedicalCareDetails"; + public static final String PROVISIONAL_OR_FINAL_DIAGNOSIS = "provisionalOrFinalDiagnosis"; + public static final String PATIENT_IMMUNIZED_PERIOD = "patientImmunizedPeriod"; + public static final String PATIENT_IMMUNIZED_PERIOD_DETAILS = "patientImmunizedPeriodDetails"; + public static final String VACCINE_GIVEN_PERIOD = "vaccineGivenPeriod"; + public static final String VACCINE_GIVEN_PERIOD_DETAILS = "vaccineGivenPeriodDetails"; + public static final String ERROR_PRESCRIBING_VACCINE = "errorPrescribingVaccine"; + public static final String ERROR_PRESCRIBING_VACCINE_DETAILS = "errorPrescribingVaccineDetails"; + public static final String VACCINE_COULD_HAVE_BEEN_UNSTERILE = "vaccineCouldHaveBeenUnSterile"; + public static final String VACCINE_COULD_HAVE_BEEN_UNSTERILE_DETAILS = "vaccineCouldHaveBeenUnSterileDetails"; + public static final String VACCINE_PHYSICAL_CONDITION_ABNORMAL = "vaccinePhysicalConditionAbnormal"; + public static final String VACCINE_PHYSICAL_CONDITION_ABNORMAL_DETAILS = "vaccinePhysicalConditionAbnormalDetails"; + public static final String ERROR_IN_VACCINE_RECONSTITUTION = "errorInVaccineReconstitution"; + public static final String ERROR_IN_VACCINE_RECONSTITUTION_DETAILS = "errorInVaccineReconstitutionDetails"; + public static final String ERROR_IN_VACCINE_HANDLING = "errorInVaccineHandling"; + public static final String ERROR_IN_VACCINE_HANDLING_DETAILS = "errorInVaccineHandlingDetails"; + public static final String VACCINE_ADMINISTERED_INCORRECTLY = "vaccineAdministeredIncorrectly"; + public static final String VACCINE_ADMINISTERED_INCORRECTLY_DETAILS = "vaccineAdministeredIncorrectlyDetails"; + public static final String NUMBER_IMMUNIZED_FROM_CONCERNED_VACCINE_VIAL = "numberImmunizedFromConcernedVaccineVial"; + public static final String NUMBER_IMMUNIZED_WITH_CONCERNED_VACCINE_IN_SAME_SESSION = "numberImmunizedWithConcernedVaccineInSameSession"; + public static final String NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_OTHER_LOCATIONS = + "numberImmunizedConcernedVaccineSameBatchNumberOtherLocations"; + public static final String NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_LOCATION_DETAILS = + "numberImmunizedConcernedVaccineSameBatchNumberLocationDetails"; + public static final String VACCINE_HAS_QUALITY_DEFECT = "vaccineHasQualityDefect"; + public static final String VACCINE_HAS_QUALITY_DEFECT_DETAILS = "vaccineHasQualityDefectDetails"; + public static final String EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION = "eventIsAStressResponseRelatedToImmunization"; + public static final String EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION_DETAILS = "eventIsAStressResponseRelatedToImmunizationDetails"; + public static final String CASE_IS_PART_OF_A_CLUSTER = "caseIsPartOfACluster"; + public static final String CASE_IS_PART_OF_A_CLUSTER_DETAILS = "caseIsPartOfAClusterDetails"; + public static final String NUMBER_OF_CASES_DETECTED_IN_CLUSTER = "numberOfCasesDetectedInCluster"; + public static final String ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL = "allCasesInClusterReceivedVaccineFromSameVial"; + public static final String ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL_DETAILS = "allCasesInClusterReceivedVaccineFromSameVialDetails"; + public static final String NUMBER_OF_VIALS_USED_IN_CLUSTER = "numberOfVialsUsedInCluster"; + public static final String NUMBER_OF_VIALS_USED_IN_CLUSTER_DETAILS = "numberOfVialsUsedInClusterDetails"; + public static final String AD_SYRINGES_USED_FOR_IMMUNIZATION = "adSyringesUsedForImmunization"; + public static final String TYPE_OF_SYRINGES_USED = "typeOfSyringesUsed"; + public static final String TYPE_OF_SYRINGES_USED_DETAILS = "typeOfSyringesUsedDetails"; + public static final String SYRINGES_USED_ADDITIONAL_DETAILS = "syringesUsedAdditionalDetails"; + public static final String SAME_RECONSTITUTION_SYRINGE_USED_FOR_MULTIPLE_VIALS_OF_SAME_VACCINE = + "sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine"; + public static final String SAME_RECONSTITUTION_SYRINGE_USED_FOR_RECONSTITUTING_DIFFERENT_VACCINES = + "sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines"; + public static final String SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINE_VIAL = "sameReconstitutionSyringeForEachVaccineVial"; + public static final String SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINATION = "sameReconstitutionSyringeForEachVaccination"; + public static final String VACCINES_AND_DILUENTS_USED_RECOMMENDED_BY_MANUFACTURER = "vaccinesAndDiluentsUsedRecommendedByManufacturer"; + public static final String RECONSTITUTION_ADDITIONAL_DETAILS = "reconstitutionAdditionalDetails"; + public static final String CORRECT_DOSE_OR_ROUTE = "correctDoseOrRoute"; + public static final String TIME_OF_RECONSTITUTION_MENTIONED_ON_THE_VIAL = "timeOfReconstitutionMentionedOnTheVial"; + public static final String NON_TOUCH_TECHNIQUE_FOLLOWED = "nonTouchTechniqueFollowed"; + public static final String CONTRAINDICATION_SCREENED_PRIOR_TO_VACCINATION = "contraIndicationScreenedPriorToVaccination"; + public static final String NUMBER_OF_AEFI_REPORTED_FROM_VACCINE_DISTRIBUTION_CENTER_LAST_THIRTY_DAYS = + "numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays"; + public static final String TRAINING_RECEIVED_BY_VACCINATOR = "trainingReceivedByVaccinator"; + public static final String LAST_TRAINING_RECEIVED_BY_VACCINATOR_DATE = "lastTrainingReceivedByVaccinatorDate"; + public static final String INJECTION_TECHNIQUE_ADDITIONAL_DETAILS = "injectionTechniqueAdditionalDetails"; + public static final String VACCINE_STORAGE_REFRIGERATOR_TEMPERATURE_MONITORED = "vaccineStorageRefrigeratorTemperatureMonitored"; + public static final String ANY_STORAGE_TEMPERATURE_DEVIATION_OUTSIDE_TWO_TO_EIGHT_DEGREES = + "anyStorageTemperatureDeviationOutsideTwoToEightDegrees"; + public static final String STORAGE_TEMPERATURE_MONITORING_ADDITIONAL_DETAILS = "storageTemperatureMonitoringAdditionalDetails"; + public static final String CORRECT_PROCEDURE_FOR_STORAGE_FOLLOWED = "correctProcedureForStorageFollowed"; + public static final String ANY_OTHER_ITEM_IN_REFRIGERATOR = "anyOtherItemInRefrigerator"; + public static final String PARTIALLY_USED_RECONSTITUTED_VACCINES_IN_REFRIGERATOR = "partiallyUsedReconstitutedVaccinesInRefrigerator"; + public static final String UNUSABLE_VACCINES_IN_REFRIGERATOR = "unusableVaccinesInRefrigerator"; + public static final String UNUSABLE_DILUENTS_IN_STORE = "unusableDiluentsInStore"; + public static final String VACCINE_STORAGE_POINT_ADDITIONAL_DETAILS = "vaccineStoragePointAdditionalDetails"; + public static final String VACCINE_CARRIER_TYPE = "vaccineCarrierType"; + public static final String VACCINE_CARRIER_TYPE_DETAILS = "vaccineCarrierTypeDetails"; + public static final String VACCINE_CARRIER_SENT_TO_SITE_ON_SAME_DATE_AS_VACCINATION = "vaccineCarrierSentToSiteOnSameDateAsVaccination"; + public static final String VACCINE_CARRIER_RETURNED_FROM_SITE_ON_SAME_DATE_AS_VACCINATION = + "vaccineCarrierReturnedFromSiteOnSameDateAsVaccination"; + public static final String CONDITIONED_ICE_PACK_USED = "conditionedIcepackUsed"; + public static final String VACCINE_TRANSPORTATION_ADDITIONAL_DETAILS = "vaccineTransportationAdditionalDetails"; + public static final String SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY = "similarEventsReportedSamePeriodAndLocality"; + public static final String SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY_DETAILS = "similarEventsReportedSamePeriodAndLocalityDetails"; + public static final String NUMBER_OF_SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY = "numberOfSimilarEventsReportedSamePeriodAndLocality"; + public static final String NUMBER_OF_THOSE_AFFECTED_VACCINATED = "numberOfThoseAffectedVaccinated"; + public static final String NUMBER_OF_THOSE_AFFECTED_NOT_VACCINATED = "numberOfThoseAffectedNotVaccinated"; + public static final String NUMBER_OF_THOSE_AFFECTED_VACCINATED_UNKNOWN = "numberOfThoseAffectedVaccinatedUnknown"; + public static final String COMMUNITY_INVESTIGATION_ADDITIONAL_DETAILS = "communityInvestigationAdditionalDetails"; + public static final String OTHER_INVESTIGATION_FINDINGS = "otherInvestigationFindings"; + public static final String INVESTIGATION_STATUS = "investigationStatus"; + public static final String INVESTIGATION_STATUS_DETAILS = "investigationStatusDetails"; + public static final String AEFI_CLASSIFICATION = "aefiClassification"; + public static final String AEFI_CLASSIFICATION_SUB_TYPE = "aefiClassificationSubType"; + public static final String AEFI_CLASSIFICATION_DETAILS = "aefiClassificationDetails"; + public static final String CAUSALITY = "causality"; + public static final String CAUSALITY_DETAILS = "causalityDetails"; + public static final String INVESTIGATION_COMPLETION_DATE = "investigationCompletionDate"; + public static final String DELETION_REASON = "deletionReason"; + public static final String OTHER_DELETION_REASON = "otherDeletionReason"; + + @NotNull(message = Validations.validAefiReport) + private AefiReferenceDto aefiReport; + private LocationDto address; + @NotEmpty(message = Validations.aefiInvestigationWithoutSuspectVaccines) + private List vaccinations = new ArrayList<>(); + private VaccinationDto primarySuspectVaccine; + private Date reportDate; + private UserReferenceDto reportingUser; + private String externalId; + private RegionReferenceDto responsibleRegion; + private DistrictReferenceDto responsibleDistrict; + private CommunityReferenceDto responsibleCommunity; + private CountryReferenceDto country; + private String investigationCaseId; + private PlaceOfVaccination placeOfVaccination; + private String placeOfVaccinationDetails; + private VaccinationActivity vaccinationActivity; + private String vaccinationActivityDetails; + private FacilityReferenceDto vaccinationFacility; + private String vaccinationFacilityDetails; + private String reportingOfficerName; + private FacilityReferenceDto reportingOfficerFacility; + private String reportingOfficerFacilityDetails; + private String reportingOfficerDesignation; + private String reportingOfficerDepartment; + private LocationDto reportingOfficerAddress; + private String reportingOfficerLandlinePhoneNumber; + private String reportingOfficerMobilePhoneNumber; + private String reportingOfficerEmail; + private Date investigationDate; + private Date formCompletionDate; + private AefiInvestigationStage investigationStage; + private VaccinationSite typeOfSite; + private String typeOfSiteDetails; + private Date keySymptomDateTime; + private Date hospitalizationDate; + private Date reportedToHealthAuthorityDate; + private PatientStatusAtAefiInvestigation statusOnDateOfInvestigation; + private Date deathDateTime; + private YesNoUnknown autopsyDone; + private Date autopsyDate; + private Date autopsyPlannedDateTime; + private YesNoUnknown pastHistoryOfSimilarEvent; + private String pastHistoryOfSimilarEventDetails; + private YesNoUnknown adverseEventAfterPreviousVaccinations; + private String adverseEventAfterPreviousVaccinationsDetails; + private YesNoUnknown historyOfAllergyToVaccineDrugOrFood; + private String historyOfAllergyToVaccineDrugOrFoodDetails; + private YesNoUnknown preExistingIllnessThirtyDaysOrCongenitalDisorder; + private String preExistingIllnessThirtyDaysOrCongenitalDisorderDetails; + private YesNoUnknown historyOfHospitalizationInLastThirtyDaysWithCause; + private String historyOfHospitalizationInLastThirtyDaysWithCauseDetails; + private YesNoUnknown currentlyOnConcomitantMedication; + private String currentlyOnConcomitantMedicationDetails; + private YesNoUnknown familyHistoryOfDiseaseOrAllergy; + private String familyHistoryOfDiseaseOrAllergyDetails; + private Integer numberOfWeeksPregnant; + private BirthTerm birthTerm; + private Float birthWeight; + private DeliveryProcedure deliveryProcedure; + private String deliveryProcedureDetails; + private Set seriousAefiInfoSource; + private String seriousAefiInfoSourceDetails; + private String seriousAefiVerbalAutopsyInfoSourceDetails; + private String firstCaregiversName; + private String otherCaregiversNames; + private String otherSourcesWhoProvidedInfo; + private String signsAndSymptomsFromTimeOfVaccination; + private String clinicalDetailsOfficerName; + private String clinicalDetailsOfficerPhoneNumber; + private String clinicalDetailsOfficerEmail; + private String clinicalDetailsOfficerDesignation; + private Date clinicalDetailsDateTime; + private YesNoUnknown patientReceivedMedicalCare; + private String patientReceivedMedicalCareDetails; + private String provisionalOrFinalDiagnosis; + private AefiImmunizationPeriod patientImmunizedPeriod; + private String patientImmunizedPeriodDetails; + private AefiVaccinationPeriod vaccineGivenPeriod; + private String vaccineGivenPeriodDetails; + private YesNoUnknown errorPrescribingVaccine; + private String errorPrescribingVaccineDetails; + private YesNoUnknown vaccineCouldHaveBeenUnSterile; + private String vaccineCouldHaveBeenUnSterileDetails; + private YesNoUnknown vaccinePhysicalConditionAbnormal; + private String vaccinePhysicalConditionAbnormalDetails; + private YesNoUnknown errorInVaccineReconstitution; + private String errorInVaccineReconstitutionDetails; + private YesNoUnknown errorInVaccineHandling; + private String errorInVaccineHandlingDetails; + private YesNoUnknown vaccineAdministeredIncorrectly; + private String vaccineAdministeredIncorrectlyDetails; + private Integer numberImmunizedFromConcernedVaccineVial; + private Integer numberImmunizedWithConcernedVaccineInSameSession; + private Integer numberImmunizedConcernedVaccineSameBatchNumberOtherLocations; + private String numberImmunizedConcernedVaccineSameBatchNumberLocationDetails; + private YesNoUnknown vaccineHasQualityDefect; + private String vaccineHasQualityDefectDetails; + private YesNoUnknown eventIsAStressResponseRelatedToImmunization; + private String eventIsAStressResponseRelatedToImmunizationDetails; + private YesNoUnknown caseIsPartOfACluster; + private String caseIsPartOfAClusterDetails; + private Integer numberOfCasesDetectedInCluster; + private YesNoUnknown allCasesInClusterReceivedVaccineFromSameVial; + private String allCasesInClusterReceivedVaccineFromSameVialDetails; + private Integer numberOfVialsUsedInCluster; + private String numberOfVialsUsedInClusterDetails; + private YesNoUnknown adSyringesUsedForImmunization; + private SyringeType typeOfSyringesUsed; + private String typeOfSyringesUsedDetails; + private String syringesUsedAdditionalDetails; + private YesNoUnknown sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine; + private YesNoUnknown sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines; + private YesNoUnknown sameReconstitutionSyringeForEachVaccineVial; + private YesNoUnknown sameReconstitutionSyringeForEachVaccination; + private YesNoUnknown vaccinesAndDiluentsUsedRecommendedByManufacturer; + private String reconstitutionAdditionalDetails; + private YesNoUnknown correctDoseOrRoute; + private YesNoUnknown timeOfReconstitutionMentionedOnTheVial; + private YesNoUnknown nonTouchTechniqueFollowed; + private YesNoUnknown contraIndicationScreenedPriorToVaccination; + private Integer numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays; + private YesNoUnknown trainingReceivedByVaccinator; + private Date lastTrainingReceivedByVaccinatorDate; + private String injectionTechniqueAdditionalDetails; + private YesNoUnknown vaccineStorageRefrigeratorTemperatureMonitored; + private YesNoUnknown anyStorageTemperatureDeviationOutsideTwoToEightDegrees; + private String storageTemperatureMonitoringAdditionalDetails; + private YesNoUnknown correctProcedureForStorageFollowed; + private YesNoUnknown anyOtherItemInRefrigerator; + private YesNoUnknown partiallyUsedReconstitutedVaccinesInRefrigerator; + private YesNoUnknown unusableVaccinesInRefrigerator; + private YesNoUnknown unusableDiluentsInStore; + private String vaccineStoragePointAdditionalDetails; + private VaccineCarrier vaccineCarrierType; + private String vaccineCarrierTypeDetails; + private YesNoUnknown vaccineCarrierSentToSiteOnSameDateAsVaccination; + private YesNoUnknown vaccineCarrierReturnedFromSiteOnSameDateAsVaccination; + private YesNoUnknown conditionedIcepackUsed; + private String vaccineTransportationAdditionalDetails; + private YesNoUnknown similarEventsReportedSamePeriodAndLocality; + private String similarEventsReportedSamePeriodAndLocalityDetails; + private Integer numberOfSimilarEventsReportedSamePeriodAndLocality; + private Integer numberOfThoseAffectedVaccinated; + private Integer numberOfThoseAffectedNotVaccinated; + private Integer numberOfThoseAffectedVaccinatedUnknown; + private String communityInvestigationAdditionalDetails; + private String otherInvestigationFindings; + private AefiInvestigationStatus investigationStatus; + private String investigationStatusDetails; + private AefiClassification aefiClassification; + private AefiClassificationSubType aefiClassificationSubType; + private String aefiClassificationDetails; + private AefiCausality causality; + private String causalityDetails; + private Date investigationCompletionDate; + private boolean archived; + private boolean deleted; + private DeletionReason deletionReason; + @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) + private String otherDeletionReason; + + public static AefiInvestigationDto build(UserReferenceDto user) { + + final AefiInvestigationDto aefiInvestigationDto = new AefiInvestigationDto(); + aefiInvestigationDto.setUuid(DataHelper.createUuid()); + aefiInvestigationDto.setReportingUser(user); + aefiInvestigationDto.setReportDate(new Date()); + + return aefiInvestigationDto; + } + + public static AefiInvestigationDto build(AefiReferenceDto aefiReferenceDto) { + + final AefiInvestigationDto aefiInvestigationDto = new AefiInvestigationDto(); + aefiInvestigationDto.setUuid(DataHelper.createUuid()); + aefiInvestigationDto.setAefiReport(aefiReferenceDto); + aefiInvestigationDto.setReportDate(new Date()); + + return aefiInvestigationDto; + } + + public static AefiInvestigationDto build(AefiInvestigationReferenceDto aefiInvestigationReferenceDto) { + + final AefiInvestigationDto aefiInvestigationDto = new AefiInvestigationDto(); + aefiInvestigationDto.setUuid(aefiInvestigationReferenceDto.getUuid()); + aefiInvestigationDto.setReportDate(new Date()); + + return aefiInvestigationDto; + } + + public AefiInvestigationReferenceDto toReference() { + return new AefiInvestigationReferenceDto(getUuid(), getExternalId()); + } + + public AefiReferenceDto getAefiReport() { + return aefiReport; + } + + public void setAefiReport(AefiReferenceDto aefiReport) { + this.aefiReport = aefiReport; + } + + public LocationDto getAddress() { + return address; + } + + public void setAddress(LocationDto address) { + this.address = address; + } + + public List getVaccinations() { + return vaccinations; + } + + public void setVaccinations(List vaccinations) { + this.vaccinations = vaccinations; + } + + public VaccinationDto getPrimarySuspectVaccine() { + return primarySuspectVaccine; + } + + public void setPrimarySuspectVaccine(VaccinationDto primarySuspectVaccine) { + this.primarySuspectVaccine = primarySuspectVaccine; + } + + public Date getReportDate() { + return reportDate; + } + + public void setReportDate(Date reportDate) { + this.reportDate = reportDate; + } + + public UserReferenceDto getReportingUser() { + return reportingUser; + } + + public void setReportingUser(UserReferenceDto reportingUser) { + this.reportingUser = reportingUser; + } + + public String getExternalId() { + return externalId; + } + + public void setExternalId(String externalId) { + this.externalId = externalId; + } + + public RegionReferenceDto getResponsibleRegion() { + return responsibleRegion; + } + + public void setResponsibleRegion(RegionReferenceDto responsibleRegion) { + this.responsibleRegion = responsibleRegion; + } + + public DistrictReferenceDto getResponsibleDistrict() { + return responsibleDistrict; + } + + public void setResponsibleDistrict(DistrictReferenceDto responsibleDistrict) { + this.responsibleDistrict = responsibleDistrict; + } + + public CommunityReferenceDto getResponsibleCommunity() { + return responsibleCommunity; + } + + public void setResponsibleCommunity(CommunityReferenceDto responsibleCommunity) { + this.responsibleCommunity = responsibleCommunity; + } + + public CountryReferenceDto getCountry() { + return country; + } + + public void setCountry(CountryReferenceDto country) { + this.country = country; + } + + public String getInvestigationCaseId() { + return investigationCaseId; + } + + public void setInvestigationCaseId(String investigationCaseId) { + this.investigationCaseId = investigationCaseId; + } + + public PlaceOfVaccination getPlaceOfVaccination() { + return placeOfVaccination; + } + + public void setPlaceOfVaccination(PlaceOfVaccination placeOfVaccination) { + this.placeOfVaccination = placeOfVaccination; + } + + public String getPlaceOfVaccinationDetails() { + return placeOfVaccinationDetails; + } + + public void setPlaceOfVaccinationDetails(String placeOfVaccinationDetails) { + this.placeOfVaccinationDetails = placeOfVaccinationDetails; + } + + public VaccinationActivity getVaccinationActivity() { + return vaccinationActivity; + } + + public void setVaccinationActivity(VaccinationActivity vaccinationActivity) { + this.vaccinationActivity = vaccinationActivity; + } + + public String getVaccinationActivityDetails() { + return vaccinationActivityDetails; + } + + public void setVaccinationActivityDetails(String vaccinationActivityDetails) { + this.vaccinationActivityDetails = vaccinationActivityDetails; + } + + public FacilityReferenceDto getVaccinationFacility() { + return vaccinationFacility; + } + + public void setVaccinationFacility(FacilityReferenceDto vaccinationFacility) { + this.vaccinationFacility = vaccinationFacility; + } + + public String getVaccinationFacilityDetails() { + return vaccinationFacilityDetails; + } + + public void setVaccinationFacilityDetails(String vaccinationFacilityDetails) { + this.vaccinationFacilityDetails = vaccinationFacilityDetails; + } + + public String getReportingOfficerName() { + return reportingOfficerName; + } + + public void setReportingOfficerName(String reportingOfficerName) { + this.reportingOfficerName = reportingOfficerName; + } + + public FacilityReferenceDto getReportingOfficerFacility() { + return reportingOfficerFacility; + } + + public void setReportingOfficerFacility(FacilityReferenceDto reportingOfficerFacility) { + this.reportingOfficerFacility = reportingOfficerFacility; + } + + public String getReportingOfficerFacilityDetails() { + return reportingOfficerFacilityDetails; + } + + public void setReportingOfficerFacilityDetails(String reportingOfficerFacilityDetails) { + this.reportingOfficerFacilityDetails = reportingOfficerFacilityDetails; + } + + public String getReportingOfficerDesignation() { + return reportingOfficerDesignation; + } + + public void setReportingOfficerDesignation(String reportingOfficerDesignation) { + this.reportingOfficerDesignation = reportingOfficerDesignation; + } + + public String getReportingOfficerDepartment() { + return reportingOfficerDepartment; + } + + public void setReportingOfficerDepartment(String reportingOfficerDepartment) { + this.reportingOfficerDepartment = reportingOfficerDepartment; + } + + public LocationDto getReportingOfficerAddress() { + return reportingOfficerAddress; + } + + public void setReportingOfficerAddress(LocationDto reportingOfficerAddress) { + this.reportingOfficerAddress = reportingOfficerAddress; + } + + public String getReportingOfficerLandlinePhoneNumber() { + return reportingOfficerLandlinePhoneNumber; + } + + public void setReportingOfficerLandlinePhoneNumber(String reportingOfficerLandlinePhoneNumber) { + this.reportingOfficerLandlinePhoneNumber = reportingOfficerLandlinePhoneNumber; + } + + public String getReportingOfficerMobilePhoneNumber() { + return reportingOfficerMobilePhoneNumber; + } + + public void setReportingOfficerMobilePhoneNumber(String reportingOfficerMobilePhoneNumber) { + this.reportingOfficerMobilePhoneNumber = reportingOfficerMobilePhoneNumber; + } + + public String getReportingOfficerEmail() { + return reportingOfficerEmail; + } + + public void setReportingOfficerEmail(String reportingOfficerEmail) { + this.reportingOfficerEmail = reportingOfficerEmail; + } + + public Date getInvestigationDate() { + return investigationDate; + } + + public void setInvestigationDate(Date investigationDate) { + this.investigationDate = investigationDate; + } + + public Date getFormCompletionDate() { + return formCompletionDate; + } + + public void setFormCompletionDate(Date formCompletionDate) { + this.formCompletionDate = formCompletionDate; + } + + public AefiInvestigationStage getInvestigationStage() { + return investigationStage; + } + + public void setInvestigationStage(AefiInvestigationStage investigationStage) { + this.investigationStage = investigationStage; + } + + public VaccinationSite getTypeOfSite() { + return typeOfSite; + } + + public void setTypeOfSite(VaccinationSite typeOfSite) { + this.typeOfSite = typeOfSite; + } + + public String getTypeOfSiteDetails() { + return typeOfSiteDetails; + } + + public void setTypeOfSiteDetails(String typeOfSiteDetails) { + this.typeOfSiteDetails = typeOfSiteDetails; + } + + public Date getKeySymptomDateTime() { + return keySymptomDateTime; + } + + public void setKeySymptomDateTime(Date keySymptomDateTime) { + this.keySymptomDateTime = keySymptomDateTime; + } + + public Date getHospitalizationDate() { + return hospitalizationDate; + } + + public void setHospitalizationDate(Date hospitalizationDate) { + this.hospitalizationDate = hospitalizationDate; + } + + public Date getReportedToHealthAuthorityDate() { + return reportedToHealthAuthorityDate; + } + + public void setReportedToHealthAuthorityDate(Date reportedToHealthAuthorityDate) { + this.reportedToHealthAuthorityDate = reportedToHealthAuthorityDate; + } + + public PatientStatusAtAefiInvestigation getStatusOnDateOfInvestigation() { + return statusOnDateOfInvestigation; + } + + public void setStatusOnDateOfInvestigation(PatientStatusAtAefiInvestigation statusOnDateOfInvestigation) { + this.statusOnDateOfInvestigation = statusOnDateOfInvestigation; + } + + public Date getDeathDateTime() { + return deathDateTime; + } + + public void setDeathDateTime(Date deathDateTime) { + this.deathDateTime = deathDateTime; + } + + public YesNoUnknown getAutopsyDone() { + return autopsyDone; + } + + public void setAutopsyDone(YesNoUnknown autopsyDone) { + this.autopsyDone = autopsyDone; + } + + public Date getAutopsyDate() { + return autopsyDate; + } + + public void setAutopsyDate(Date autopsyDate) { + this.autopsyDate = autopsyDate; + } + + public Date getAutopsyPlannedDateTime() { + return autopsyPlannedDateTime; + } + + public void setAutopsyPlannedDateTime(Date autopsyPlannedDateTime) { + this.autopsyPlannedDateTime = autopsyPlannedDateTime; + } + + public YesNoUnknown getPastHistoryOfSimilarEvent() { + return pastHistoryOfSimilarEvent; + } + + public void setPastHistoryOfSimilarEvent(YesNoUnknown pastHistoryOfSimilarEvent) { + this.pastHistoryOfSimilarEvent = pastHistoryOfSimilarEvent; + } + + public String getPastHistoryOfSimilarEventDetails() { + return pastHistoryOfSimilarEventDetails; + } + + public void setPastHistoryOfSimilarEventDetails(String pastHistoryOfSimilarEventDetails) { + this.pastHistoryOfSimilarEventDetails = pastHistoryOfSimilarEventDetails; + } + + public YesNoUnknown getAdverseEventAfterPreviousVaccinations() { + return adverseEventAfterPreviousVaccinations; + } + + public void setAdverseEventAfterPreviousVaccinations(YesNoUnknown adverseEventAfterPreviousVaccinations) { + this.adverseEventAfterPreviousVaccinations = adverseEventAfterPreviousVaccinations; + } + + public String getAdverseEventAfterPreviousVaccinationsDetails() { + return adverseEventAfterPreviousVaccinationsDetails; + } + + public void setAdverseEventAfterPreviousVaccinationsDetails(String adverseEventAfterPreviousVaccinationsDetails) { + this.adverseEventAfterPreviousVaccinationsDetails = adverseEventAfterPreviousVaccinationsDetails; + } + + public YesNoUnknown getHistoryOfAllergyToVaccineDrugOrFood() { + return historyOfAllergyToVaccineDrugOrFood; + } + + public void setHistoryOfAllergyToVaccineDrugOrFood(YesNoUnknown historyOfAllergyToVaccineDrugOrFood) { + this.historyOfAllergyToVaccineDrugOrFood = historyOfAllergyToVaccineDrugOrFood; + } + + public String getHistoryOfAllergyToVaccineDrugOrFoodDetails() { + return historyOfAllergyToVaccineDrugOrFoodDetails; + } + + public void setHistoryOfAllergyToVaccineDrugOrFoodDetails(String historyOfAllergyToVaccineDrugOrFoodDetails) { + this.historyOfAllergyToVaccineDrugOrFoodDetails = historyOfAllergyToVaccineDrugOrFoodDetails; + } + + public YesNoUnknown getPreExistingIllnessThirtyDaysOrCongenitalDisorder() { + return preExistingIllnessThirtyDaysOrCongenitalDisorder; + } + + public void setPreExistingIllnessThirtyDaysOrCongenitalDisorder(YesNoUnknown preExistingIllnessThirtyDaysOrCongenitalDisorder) { + this.preExistingIllnessThirtyDaysOrCongenitalDisorder = preExistingIllnessThirtyDaysOrCongenitalDisorder; + } + + public String getPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails() { + return preExistingIllnessThirtyDaysOrCongenitalDisorderDetails; + } + + public void setPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails(String preExistingIllnessThirtyDaysOrCongenitalDisorderDetails) { + this.preExistingIllnessThirtyDaysOrCongenitalDisorderDetails = preExistingIllnessThirtyDaysOrCongenitalDisorderDetails; + } + + public YesNoUnknown getHistoryOfHospitalizationInLastThirtyDaysWithCause() { + return historyOfHospitalizationInLastThirtyDaysWithCause; + } + + public void setHistoryOfHospitalizationInLastThirtyDaysWithCause(YesNoUnknown historyOfHospitalizationInLastThirtyDaysWithCause) { + this.historyOfHospitalizationInLastThirtyDaysWithCause = historyOfHospitalizationInLastThirtyDaysWithCause; + } + + public String getHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails() { + return historyOfHospitalizationInLastThirtyDaysWithCauseDetails; + } + + public void setHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails(String historyOfHospitalizationInLastThirtyDaysWithCauseDetails) { + this.historyOfHospitalizationInLastThirtyDaysWithCauseDetails = historyOfHospitalizationInLastThirtyDaysWithCauseDetails; + } + + public YesNoUnknown getCurrentlyOnConcomitantMedication() { + return currentlyOnConcomitantMedication; + } + + public void setCurrentlyOnConcomitantMedication(YesNoUnknown currentlyOnConcomitantMedication) { + this.currentlyOnConcomitantMedication = currentlyOnConcomitantMedication; + } + + public String getCurrentlyOnConcomitantMedicationDetails() { + return currentlyOnConcomitantMedicationDetails; + } + + public void setCurrentlyOnConcomitantMedicationDetails(String currentlyOnConcomitantMedicationDetails) { + this.currentlyOnConcomitantMedicationDetails = currentlyOnConcomitantMedicationDetails; + } + + public YesNoUnknown getFamilyHistoryOfDiseaseOrAllergy() { + return familyHistoryOfDiseaseOrAllergy; + } + + public void setFamilyHistoryOfDiseaseOrAllergy(YesNoUnknown familyHistoryOfDiseaseOrAllergy) { + this.familyHistoryOfDiseaseOrAllergy = familyHistoryOfDiseaseOrAllergy; + } + + public String getFamilyHistoryOfDiseaseOrAllergyDetails() { + return familyHistoryOfDiseaseOrAllergyDetails; + } + + public void setFamilyHistoryOfDiseaseOrAllergyDetails(String familyHistoryOfDiseaseOrAllergyDetails) { + this.familyHistoryOfDiseaseOrAllergyDetails = familyHistoryOfDiseaseOrAllergyDetails; + } + + public Integer getNumberOfWeeksPregnant() { + return numberOfWeeksPregnant; + } + + public void setNumberOfWeeksPregnant(Integer numberOfWeeksPregnant) { + this.numberOfWeeksPregnant = numberOfWeeksPregnant; + } + + public BirthTerm getBirthTerm() { + return birthTerm; + } + + public void setBirthTerm(BirthTerm birthTerm) { + this.birthTerm = birthTerm; + } + + public Float getBirthWeight() { + return birthWeight; + } + + public void setBirthWeight(Float birthWeight) { + this.birthWeight = birthWeight; + } + + public DeliveryProcedure getDeliveryProcedure() { + return deliveryProcedure; + } + + public void setDeliveryProcedure(DeliveryProcedure deliveryProcedure) { + this.deliveryProcedure = deliveryProcedure; + } + + public String getDeliveryProcedureDetails() { + return deliveryProcedureDetails; + } + + public void setDeliveryProcedureDetails(String deliveryProcedureDetails) { + this.deliveryProcedureDetails = deliveryProcedureDetails; + } + + public Set getSeriousAefiInfoSource() { + return seriousAefiInfoSource; + } + + public void setSeriousAefiInfoSource(Set seriousAefiInfoSource) { + this.seriousAefiInfoSource = seriousAefiInfoSource; + } + + public String getSeriousAefiInfoSourceDetails() { + return seriousAefiInfoSourceDetails; + } + + public void setSeriousAefiInfoSourceDetails(String seriousAefiInfoSourceDetails) { + this.seriousAefiInfoSourceDetails = seriousAefiInfoSourceDetails; + } + + public String getSeriousAefiVerbalAutopsyInfoSourceDetails() { + return seriousAefiVerbalAutopsyInfoSourceDetails; + } + + public void setSeriousAefiVerbalAutopsyInfoSourceDetails(String seriousAefiVerbalAutopsyInfoSourceDetails) { + this.seriousAefiVerbalAutopsyInfoSourceDetails = seriousAefiVerbalAutopsyInfoSourceDetails; + } + + public String getFirstCaregiversName() { + return firstCaregiversName; + } + + public void setFirstCaregiversName(String firstCaregiversName) { + this.firstCaregiversName = firstCaregiversName; + } + + public String getOtherCaregiversNames() { + return otherCaregiversNames; + } + + public void setOtherCaregiversNames(String otherCaregiversNames) { + this.otherCaregiversNames = otherCaregiversNames; + } + + public String getOtherSourcesWhoProvidedInfo() { + return otherSourcesWhoProvidedInfo; + } + + public void setOtherSourcesWhoProvidedInfo(String otherSourcesWhoProvidedInfo) { + this.otherSourcesWhoProvidedInfo = otherSourcesWhoProvidedInfo; + } + + public String getSignsAndSymptomsFromTimeOfVaccination() { + return signsAndSymptomsFromTimeOfVaccination; + } + + public void setSignsAndSymptomsFromTimeOfVaccination(String signsAndSymptomsFromTimeOfVaccination) { + this.signsAndSymptomsFromTimeOfVaccination = signsAndSymptomsFromTimeOfVaccination; + } + + public String getClinicalDetailsOfficerName() { + return clinicalDetailsOfficerName; + } + + public void setClinicalDetailsOfficerName(String clinicalDetailsOfficerName) { + this.clinicalDetailsOfficerName = clinicalDetailsOfficerName; + } + + public String getClinicalDetailsOfficerPhoneNumber() { + return clinicalDetailsOfficerPhoneNumber; + } + + public void setClinicalDetailsOfficerPhoneNumber(String clinicalDetailsOfficerPhoneNumber) { + this.clinicalDetailsOfficerPhoneNumber = clinicalDetailsOfficerPhoneNumber; + } + + public String getClinicalDetailsOfficerEmail() { + return clinicalDetailsOfficerEmail; + } + + public void setClinicalDetailsOfficerEmail(String clinicalDetailsOfficerEmail) { + this.clinicalDetailsOfficerEmail = clinicalDetailsOfficerEmail; + } + + public String getClinicalDetailsOfficerDesignation() { + return clinicalDetailsOfficerDesignation; + } + + public void setClinicalDetailsOfficerDesignation(String clinicalDetailsOfficerDesignation) { + this.clinicalDetailsOfficerDesignation = clinicalDetailsOfficerDesignation; + } + + public Date getClinicalDetailsDateTime() { + return clinicalDetailsDateTime; + } + + public void setClinicalDetailsDateTime(Date clinicalDetailsDateTime) { + this.clinicalDetailsDateTime = clinicalDetailsDateTime; + } + + public YesNoUnknown getPatientReceivedMedicalCare() { + return patientReceivedMedicalCare; + } + + public void setPatientReceivedMedicalCare(YesNoUnknown patientReceivedMedicalCare) { + this.patientReceivedMedicalCare = patientReceivedMedicalCare; + } + + public String getPatientReceivedMedicalCareDetails() { + return patientReceivedMedicalCareDetails; + } + + public void setPatientReceivedMedicalCareDetails(String patientReceivedMedicalCareDetails) { + this.patientReceivedMedicalCareDetails = patientReceivedMedicalCareDetails; + } + + public String getProvisionalOrFinalDiagnosis() { + return provisionalOrFinalDiagnosis; + } + + public void setProvisionalOrFinalDiagnosis(String provisionalOrFinalDiagnosis) { + this.provisionalOrFinalDiagnosis = provisionalOrFinalDiagnosis; + } + + public AefiImmunizationPeriod getPatientImmunizedPeriod() { + return patientImmunizedPeriod; + } + + public void setPatientImmunizedPeriod(AefiImmunizationPeriod patientImmunizedPeriod) { + this.patientImmunizedPeriod = patientImmunizedPeriod; + } + + public String getPatientImmunizedPeriodDetails() { + return patientImmunizedPeriodDetails; + } + + public void setPatientImmunizedPeriodDetails(String patientImmunizedPeriodDetails) { + this.patientImmunizedPeriodDetails = patientImmunizedPeriodDetails; + } + + public AefiVaccinationPeriod getVaccineGivenPeriod() { + return vaccineGivenPeriod; + } + + public void setVaccineGivenPeriod(AefiVaccinationPeriod vaccineGivenPeriod) { + this.vaccineGivenPeriod = vaccineGivenPeriod; + } + + public String getVaccineGivenPeriodDetails() { + return vaccineGivenPeriodDetails; + } + + public void setVaccineGivenPeriodDetails(String vaccineGivenPeriodDetails) { + this.vaccineGivenPeriodDetails = vaccineGivenPeriodDetails; + } + + public YesNoUnknown getErrorPrescribingVaccine() { + return errorPrescribingVaccine; + } + + public void setErrorPrescribingVaccine(YesNoUnknown errorPrescribingVaccine) { + this.errorPrescribingVaccine = errorPrescribingVaccine; + } + + public String getErrorPrescribingVaccineDetails() { + return errorPrescribingVaccineDetails; + } + + public void setErrorPrescribingVaccineDetails(String errorPrescribingVaccineDetails) { + this.errorPrescribingVaccineDetails = errorPrescribingVaccineDetails; + } + + public YesNoUnknown getVaccineCouldHaveBeenUnSterile() { + return vaccineCouldHaveBeenUnSterile; + } + + public void setVaccineCouldHaveBeenUnSterile(YesNoUnknown vaccineCouldHaveBeenUnSterile) { + this.vaccineCouldHaveBeenUnSterile = vaccineCouldHaveBeenUnSterile; + } + + public String getVaccineCouldHaveBeenUnSterileDetails() { + return vaccineCouldHaveBeenUnSterileDetails; + } + + public void setVaccineCouldHaveBeenUnSterileDetails(String vaccineCouldHaveBeenUnSterileDetails) { + this.vaccineCouldHaveBeenUnSterileDetails = vaccineCouldHaveBeenUnSterileDetails; + } + + public YesNoUnknown getVaccinePhysicalConditionAbnormal() { + return vaccinePhysicalConditionAbnormal; + } + + public void setVaccinePhysicalConditionAbnormal(YesNoUnknown vaccinePhysicalConditionAbnormal) { + this.vaccinePhysicalConditionAbnormal = vaccinePhysicalConditionAbnormal; + } + + public String getVaccinePhysicalConditionAbnormalDetails() { + return vaccinePhysicalConditionAbnormalDetails; + } + + public void setVaccinePhysicalConditionAbnormalDetails(String vaccinePhysicalConditionAbnormalDetails) { + this.vaccinePhysicalConditionAbnormalDetails = vaccinePhysicalConditionAbnormalDetails; + } + + public YesNoUnknown getErrorInVaccineReconstitution() { + return errorInVaccineReconstitution; + } + + public void setErrorInVaccineReconstitution(YesNoUnknown errorInVaccineReconstitution) { + this.errorInVaccineReconstitution = errorInVaccineReconstitution; + } + + public String getErrorInVaccineReconstitutionDetails() { + return errorInVaccineReconstitutionDetails; + } + + public void setErrorInVaccineReconstitutionDetails(String errorInVaccineReconstitutionDetails) { + this.errorInVaccineReconstitutionDetails = errorInVaccineReconstitutionDetails; + } + + public YesNoUnknown getErrorInVaccineHandling() { + return errorInVaccineHandling; + } + + public void setErrorInVaccineHandling(YesNoUnknown errorInVaccineHandling) { + this.errorInVaccineHandling = errorInVaccineHandling; + } + + public String getErrorInVaccineHandlingDetails() { + return errorInVaccineHandlingDetails; + } + + public void setErrorInVaccineHandlingDetails(String errorInVaccineHandlingDetails) { + this.errorInVaccineHandlingDetails = errorInVaccineHandlingDetails; + } + + public YesNoUnknown getVaccineAdministeredIncorrectly() { + return vaccineAdministeredIncorrectly; + } + + public void setVaccineAdministeredIncorrectly(YesNoUnknown vaccineAdministeredIncorrectly) { + this.vaccineAdministeredIncorrectly = vaccineAdministeredIncorrectly; + } + + public String getVaccineAdministeredIncorrectlyDetails() { + return vaccineAdministeredIncorrectlyDetails; + } + + public void setVaccineAdministeredIncorrectlyDetails(String vaccineAdministeredIncorrectlyDetails) { + this.vaccineAdministeredIncorrectlyDetails = vaccineAdministeredIncorrectlyDetails; + } + + public Integer getNumberImmunizedFromConcernedVaccineVial() { + return numberImmunizedFromConcernedVaccineVial; + } + + public void setNumberImmunizedFromConcernedVaccineVial(Integer numberImmunizedFromConcernedVaccineVial) { + this.numberImmunizedFromConcernedVaccineVial = numberImmunizedFromConcernedVaccineVial; + } + + public Integer getNumberImmunizedWithConcernedVaccineInSameSession() { + return numberImmunizedWithConcernedVaccineInSameSession; + } + + public void setNumberImmunizedWithConcernedVaccineInSameSession(Integer numberImmunizedWithConcernedVaccineInSameSession) { + this.numberImmunizedWithConcernedVaccineInSameSession = numberImmunizedWithConcernedVaccineInSameSession; + } + + public Integer getNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations() { + return numberImmunizedConcernedVaccineSameBatchNumberOtherLocations; + } + + public void setNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations( + Integer numberImmunizedConcernedVaccineSameBatchNumberOtherLocations) { + this.numberImmunizedConcernedVaccineSameBatchNumberOtherLocations = numberImmunizedConcernedVaccineSameBatchNumberOtherLocations; + } + + public String getNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails() { + return numberImmunizedConcernedVaccineSameBatchNumberLocationDetails; + } + + public void setNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails( + String numberImmunizedConcernedVaccineSameBatchNumberLocationDetails) { + this.numberImmunizedConcernedVaccineSameBatchNumberLocationDetails = numberImmunizedConcernedVaccineSameBatchNumberLocationDetails; + } + + public YesNoUnknown getVaccineHasQualityDefect() { + return vaccineHasQualityDefect; + } + + public void setVaccineHasQualityDefect(YesNoUnknown vaccineHasQualityDefect) { + this.vaccineHasQualityDefect = vaccineHasQualityDefect; + } + + public String getVaccineHasQualityDefectDetails() { + return vaccineHasQualityDefectDetails; + } + + public void setVaccineHasQualityDefectDetails(String vaccineHasQualityDefectDetails) { + this.vaccineHasQualityDefectDetails = vaccineHasQualityDefectDetails; + } + + public YesNoUnknown getEventIsAStressResponseRelatedToImmunization() { + return eventIsAStressResponseRelatedToImmunization; + } + + public void setEventIsAStressResponseRelatedToImmunization(YesNoUnknown eventIsAStressResponseRelatedToImmunization) { + this.eventIsAStressResponseRelatedToImmunization = eventIsAStressResponseRelatedToImmunization; + } + + public String getEventIsAStressResponseRelatedToImmunizationDetails() { + return eventIsAStressResponseRelatedToImmunizationDetails; + } + + public void setEventIsAStressResponseRelatedToImmunizationDetails(String eventIsAStressResponseRelatedToImmunizationDetails) { + this.eventIsAStressResponseRelatedToImmunizationDetails = eventIsAStressResponseRelatedToImmunizationDetails; + } + + public YesNoUnknown getCaseIsPartOfACluster() { + return caseIsPartOfACluster; + } + + public void setCaseIsPartOfACluster(YesNoUnknown caseIsPartOfACluster) { + this.caseIsPartOfACluster = caseIsPartOfACluster; + } + + public String getCaseIsPartOfAClusterDetails() { + return caseIsPartOfAClusterDetails; + } + + public void setCaseIsPartOfAClusterDetails(String caseIsPartOfAClusterDetails) { + this.caseIsPartOfAClusterDetails = caseIsPartOfAClusterDetails; + } + + public Integer getNumberOfCasesDetectedInCluster() { + return numberOfCasesDetectedInCluster; + } + + public void setNumberOfCasesDetectedInCluster(Integer numberOfCasesDetectedInCluster) { + this.numberOfCasesDetectedInCluster = numberOfCasesDetectedInCluster; + } + + public YesNoUnknown getAllCasesInClusterReceivedVaccineFromSameVial() { + return allCasesInClusterReceivedVaccineFromSameVial; + } + + public void setAllCasesInClusterReceivedVaccineFromSameVial(YesNoUnknown allCasesInClusterReceivedVaccineFromSameVial) { + this.allCasesInClusterReceivedVaccineFromSameVial = allCasesInClusterReceivedVaccineFromSameVial; + } + + public String getAllCasesInClusterReceivedVaccineFromSameVialDetails() { + return allCasesInClusterReceivedVaccineFromSameVialDetails; + } + + public void setAllCasesInClusterReceivedVaccineFromSameVialDetails(String allCasesInClusterReceivedVaccineFromSameVialDetails) { + this.allCasesInClusterReceivedVaccineFromSameVialDetails = allCasesInClusterReceivedVaccineFromSameVialDetails; + } + + public Integer getNumberOfVialsUsedInCluster() { + return numberOfVialsUsedInCluster; + } + + public void setNumberOfVialsUsedInCluster(Integer numberOfVialsUsedInCluster) { + this.numberOfVialsUsedInCluster = numberOfVialsUsedInCluster; + } + + public String getNumberOfVialsUsedInClusterDetails() { + return numberOfVialsUsedInClusterDetails; + } + + public void setNumberOfVialsUsedInClusterDetails(String numberOfVialsUsedInClusterDetails) { + this.numberOfVialsUsedInClusterDetails = numberOfVialsUsedInClusterDetails; + } + + public YesNoUnknown getAdSyringesUsedForImmunization() { + return adSyringesUsedForImmunization; + } + + public void setAdSyringesUsedForImmunization(YesNoUnknown adSyringesUsedForImmunization) { + this.adSyringesUsedForImmunization = adSyringesUsedForImmunization; + } + + public SyringeType getTypeOfSyringesUsed() { + return typeOfSyringesUsed; + } + + public void setTypeOfSyringesUsed(SyringeType typeOfSyringesUsed) { + this.typeOfSyringesUsed = typeOfSyringesUsed; + } + + public String getTypeOfSyringesUsedDetails() { + return typeOfSyringesUsedDetails; + } + + public void setTypeOfSyringesUsedDetails(String typeOfSyringesUsedDetails) { + this.typeOfSyringesUsedDetails = typeOfSyringesUsedDetails; + } + + public String getSyringesUsedAdditionalDetails() { + return syringesUsedAdditionalDetails; + } + + public void setSyringesUsedAdditionalDetails(String syringesUsedAdditionalDetails) { + this.syringesUsedAdditionalDetails = syringesUsedAdditionalDetails; + } + + public YesNoUnknown getSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine() { + return sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine; + } + + public void setSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine( + YesNoUnknown sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine) { + this.sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine = sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine; + } + + public YesNoUnknown getSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines() { + return sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines; + } + + public void setSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines( + YesNoUnknown sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines) { + this.sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines = sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines; + } + + public YesNoUnknown getSameReconstitutionSyringeForEachVaccineVial() { + return sameReconstitutionSyringeForEachVaccineVial; + } + + public void setSameReconstitutionSyringeForEachVaccineVial(YesNoUnknown sameReconstitutionSyringeForEachVaccineVial) { + this.sameReconstitutionSyringeForEachVaccineVial = sameReconstitutionSyringeForEachVaccineVial; + } + + public YesNoUnknown getSameReconstitutionSyringeForEachVaccination() { + return sameReconstitutionSyringeForEachVaccination; + } + + public void setSameReconstitutionSyringeForEachVaccination(YesNoUnknown sameReconstitutionSyringeForEachVaccination) { + this.sameReconstitutionSyringeForEachVaccination = sameReconstitutionSyringeForEachVaccination; + } + + public YesNoUnknown getVaccinesAndDiluentsUsedRecommendedByManufacturer() { + return vaccinesAndDiluentsUsedRecommendedByManufacturer; + } + + public void setVaccinesAndDiluentsUsedRecommendedByManufacturer(YesNoUnknown vaccinesAndDiluentsUsedRecommendedByManufacturer) { + this.vaccinesAndDiluentsUsedRecommendedByManufacturer = vaccinesAndDiluentsUsedRecommendedByManufacturer; + } + + public String getReconstitutionAdditionalDetails() { + return reconstitutionAdditionalDetails; + } + + public void setReconstitutionAdditionalDetails(String reconstitutionAdditionalDetails) { + this.reconstitutionAdditionalDetails = reconstitutionAdditionalDetails; + } + + public YesNoUnknown getCorrectDoseOrRoute() { + return correctDoseOrRoute; + } + + public void setCorrectDoseOrRoute(YesNoUnknown correctDoseOrRoute) { + this.correctDoseOrRoute = correctDoseOrRoute; + } + + public YesNoUnknown getTimeOfReconstitutionMentionedOnTheVial() { + return timeOfReconstitutionMentionedOnTheVial; + } + + public void setTimeOfReconstitutionMentionedOnTheVial(YesNoUnknown timeOfReconstitutionMentionedOnTheVial) { + this.timeOfReconstitutionMentionedOnTheVial = timeOfReconstitutionMentionedOnTheVial; + } + + public YesNoUnknown getNonTouchTechniqueFollowed() { + return nonTouchTechniqueFollowed; + } + + public void setNonTouchTechniqueFollowed(YesNoUnknown nonTouchTechniqueFollowed) { + this.nonTouchTechniqueFollowed = nonTouchTechniqueFollowed; + } + + public YesNoUnknown getContraIndicationScreenedPriorToVaccination() { + return contraIndicationScreenedPriorToVaccination; + } + + public void setContraIndicationScreenedPriorToVaccination(YesNoUnknown contraIndicationScreenedPriorToVaccination) { + this.contraIndicationScreenedPriorToVaccination = contraIndicationScreenedPriorToVaccination; + } + + public Integer getNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays() { + return numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays; + } + + public void setNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays( + Integer numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays) { + this.numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays = numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays; + } + + public YesNoUnknown getTrainingReceivedByVaccinator() { + return trainingReceivedByVaccinator; + } + + public void setTrainingReceivedByVaccinator(YesNoUnknown trainingReceivedByVaccinator) { + this.trainingReceivedByVaccinator = trainingReceivedByVaccinator; + } + + public Date getLastTrainingReceivedByVaccinatorDate() { + return lastTrainingReceivedByVaccinatorDate; + } + + public void setLastTrainingReceivedByVaccinatorDate(Date lastTrainingReceivedByVaccinatorDate) { + this.lastTrainingReceivedByVaccinatorDate = lastTrainingReceivedByVaccinatorDate; + } + + public String getInjectionTechniqueAdditionalDetails() { + return injectionTechniqueAdditionalDetails; + } + + public void setInjectionTechniqueAdditionalDetails(String injectionTechniqueAdditionalDetails) { + this.injectionTechniqueAdditionalDetails = injectionTechniqueAdditionalDetails; + } + + public YesNoUnknown getVaccineStorageRefrigeratorTemperatureMonitored() { + return vaccineStorageRefrigeratorTemperatureMonitored; + } + + public void setVaccineStorageRefrigeratorTemperatureMonitored(YesNoUnknown vaccineStorageRefrigeratorTemperatureMonitored) { + this.vaccineStorageRefrigeratorTemperatureMonitored = vaccineStorageRefrigeratorTemperatureMonitored; + } + + public YesNoUnknown getAnyStorageTemperatureDeviationOutsideTwoToEightDegrees() { + return anyStorageTemperatureDeviationOutsideTwoToEightDegrees; + } + + public void setAnyStorageTemperatureDeviationOutsideTwoToEightDegrees(YesNoUnknown anyStorageTemperatureDeviationOutsideTwoToEightDegrees) { + this.anyStorageTemperatureDeviationOutsideTwoToEightDegrees = anyStorageTemperatureDeviationOutsideTwoToEightDegrees; + } + + public String getStorageTemperatureMonitoringAdditionalDetails() { + return storageTemperatureMonitoringAdditionalDetails; + } + + public void setStorageTemperatureMonitoringAdditionalDetails(String storageTemperatureMonitoringAdditionalDetails) { + this.storageTemperatureMonitoringAdditionalDetails = storageTemperatureMonitoringAdditionalDetails; + } + + public YesNoUnknown getCorrectProcedureForStorageFollowed() { + return correctProcedureForStorageFollowed; + } + + public void setCorrectProcedureForStorageFollowed(YesNoUnknown correctProcedureForStorageFollowed) { + this.correctProcedureForStorageFollowed = correctProcedureForStorageFollowed; + } + + public YesNoUnknown getAnyOtherItemInRefrigerator() { + return anyOtherItemInRefrigerator; + } + + public void setAnyOtherItemInRefrigerator(YesNoUnknown anyOtherItemInRefrigerator) { + this.anyOtherItemInRefrigerator = anyOtherItemInRefrigerator; + } + + public YesNoUnknown getPartiallyUsedReconstitutedVaccinesInRefrigerator() { + return partiallyUsedReconstitutedVaccinesInRefrigerator; + } + + public void setPartiallyUsedReconstitutedVaccinesInRefrigerator(YesNoUnknown partiallyUsedReconstitutedVaccinesInRefrigerator) { + this.partiallyUsedReconstitutedVaccinesInRefrigerator = partiallyUsedReconstitutedVaccinesInRefrigerator; + } + + public YesNoUnknown getUnusableVaccinesInRefrigerator() { + return unusableVaccinesInRefrigerator; + } + + public void setUnusableVaccinesInRefrigerator(YesNoUnknown unusableVaccinesInRefrigerator) { + this.unusableVaccinesInRefrigerator = unusableVaccinesInRefrigerator; + } + + public YesNoUnknown getUnusableDiluentsInStore() { + return unusableDiluentsInStore; + } + + public void setUnusableDiluentsInStore(YesNoUnknown unusableDiluentsInStore) { + this.unusableDiluentsInStore = unusableDiluentsInStore; + } + + public String getVaccineStoragePointAdditionalDetails() { + return vaccineStoragePointAdditionalDetails; + } + + public void setVaccineStoragePointAdditionalDetails(String vaccineStoragePointAdditionalDetails) { + this.vaccineStoragePointAdditionalDetails = vaccineStoragePointAdditionalDetails; + } + + public VaccineCarrier getVaccineCarrierType() { + return vaccineCarrierType; + } + + public void setVaccineCarrierType(VaccineCarrier vaccineCarrierType) { + this.vaccineCarrierType = vaccineCarrierType; + } + + public String getVaccineCarrierTypeDetails() { + return vaccineCarrierTypeDetails; + } + + public void setVaccineCarrierTypeDetails(String vaccineCarrierTypeDetails) { + this.vaccineCarrierTypeDetails = vaccineCarrierTypeDetails; + } + + public YesNoUnknown getVaccineCarrierSentToSiteOnSameDateAsVaccination() { + return vaccineCarrierSentToSiteOnSameDateAsVaccination; + } + + public void setVaccineCarrierSentToSiteOnSameDateAsVaccination(YesNoUnknown vaccineCarrierSentToSiteOnSameDateAsVaccination) { + this.vaccineCarrierSentToSiteOnSameDateAsVaccination = vaccineCarrierSentToSiteOnSameDateAsVaccination; + } + + public YesNoUnknown getVaccineCarrierReturnedFromSiteOnSameDateAsVaccination() { + return vaccineCarrierReturnedFromSiteOnSameDateAsVaccination; + } + + public void setVaccineCarrierReturnedFromSiteOnSameDateAsVaccination(YesNoUnknown vaccineCarrierReturnedFromSiteOnSameDateAsVaccination) { + this.vaccineCarrierReturnedFromSiteOnSameDateAsVaccination = vaccineCarrierReturnedFromSiteOnSameDateAsVaccination; + } + + public YesNoUnknown getConditionedIcepackUsed() { + return conditionedIcepackUsed; + } + + public void setConditionedIcepackUsed(YesNoUnknown conditionedIcepackUsed) { + this.conditionedIcepackUsed = conditionedIcepackUsed; + } + + public String getVaccineTransportationAdditionalDetails() { + return vaccineTransportationAdditionalDetails; + } + + public void setVaccineTransportationAdditionalDetails(String vaccineTransportationAdditionalDetails) { + this.vaccineTransportationAdditionalDetails = vaccineTransportationAdditionalDetails; + } + + public YesNoUnknown getSimilarEventsReportedSamePeriodAndLocality() { + return similarEventsReportedSamePeriodAndLocality; + } + + public void setSimilarEventsReportedSamePeriodAndLocality(YesNoUnknown similarEventsReportedSamePeriodAndLocality) { + this.similarEventsReportedSamePeriodAndLocality = similarEventsReportedSamePeriodAndLocality; + } + + public String getSimilarEventsReportedSamePeriodAndLocalityDetails() { + return similarEventsReportedSamePeriodAndLocalityDetails; + } + + public void setSimilarEventsReportedSamePeriodAndLocalityDetails(String similarEventsReportedSamePeriodAndLocalityDetails) { + this.similarEventsReportedSamePeriodAndLocalityDetails = similarEventsReportedSamePeriodAndLocalityDetails; + } + + public Integer getNumberOfSimilarEventsReportedSamePeriodAndLocality() { + return numberOfSimilarEventsReportedSamePeriodAndLocality; + } + + public void setNumberOfSimilarEventsReportedSamePeriodAndLocality(Integer numberOfSimilarEventsReportedSamePeriodAndLocality) { + this.numberOfSimilarEventsReportedSamePeriodAndLocality = numberOfSimilarEventsReportedSamePeriodAndLocality; + } + + public Integer getNumberOfThoseAffectedVaccinated() { + return numberOfThoseAffectedVaccinated; + } + + public void setNumberOfThoseAffectedVaccinated(Integer numberOfThoseAffectedVaccinated) { + this.numberOfThoseAffectedVaccinated = numberOfThoseAffectedVaccinated; + } + + public Integer getNumberOfThoseAffectedNotVaccinated() { + return numberOfThoseAffectedNotVaccinated; + } + + public void setNumberOfThoseAffectedNotVaccinated(Integer numberOfThoseAffectedNotVaccinated) { + this.numberOfThoseAffectedNotVaccinated = numberOfThoseAffectedNotVaccinated; + } + + public Integer getNumberOfThoseAffectedVaccinatedUnknown() { + return numberOfThoseAffectedVaccinatedUnknown; + } + + public void setNumberOfThoseAffectedVaccinatedUnknown(Integer numberOfThoseAffectedVaccinatedUnknown) { + this.numberOfThoseAffectedVaccinatedUnknown = numberOfThoseAffectedVaccinatedUnknown; + } + + public String getCommunityInvestigationAdditionalDetails() { + return communityInvestigationAdditionalDetails; + } + + public void setCommunityInvestigationAdditionalDetails(String communityInvestigationAdditionalDetails) { + this.communityInvestigationAdditionalDetails = communityInvestigationAdditionalDetails; + } + + public String getOtherInvestigationFindings() { + return otherInvestigationFindings; + } + + public void setOtherInvestigationFindings(String otherInvestigationFindings) { + this.otherInvestigationFindings = otherInvestigationFindings; + } + + public AefiInvestigationStatus getInvestigationStatus() { + return investigationStatus; + } + + public void setInvestigationStatus(AefiInvestigationStatus investigationStatus) { + this.investigationStatus = investigationStatus; + } + + public String getInvestigationStatusDetails() { + return investigationStatusDetails; + } + + public void setInvestigationStatusDetails(String investigationStatusDetails) { + this.investigationStatusDetails = investigationStatusDetails; + } + + public AefiClassification getAefiClassification() { + return aefiClassification; + } + + public void setAefiClassification(AefiClassification aefiClassification) { + this.aefiClassification = aefiClassification; + } + + public AefiClassificationSubType getAefiClassificationSubType() { + return aefiClassificationSubType; + } + + public void setAefiClassificationSubType(AefiClassificationSubType aefiClassificationSubType) { + this.aefiClassificationSubType = aefiClassificationSubType; + } + + public String getAefiClassificationDetails() { + return aefiClassificationDetails; + } + + public void setAefiClassificationDetails(String aefiClassificationDetails) { + this.aefiClassificationDetails = aefiClassificationDetails; + } + + public AefiCausality getCausality() { + return causality; + } + + public void setCausality(AefiCausality causality) { + this.causality = causality; + } + + public String getCausalityDetails() { + return causalityDetails; + } + + public void setCausalityDetails(String causalityDetails) { + this.causalityDetails = causalityDetails; + } + + public Date getInvestigationCompletionDate() { + return investigationCompletionDate; + } + + public void setInvestigationCompletionDate(Date investigationCompletionDate) { + this.investigationCompletionDate = investigationCompletionDate; + } + + public boolean isArchived() { + return archived; + } + + public void setArchived(boolean archived) { + this.archived = archived; + } + + public boolean isDeleted() { + return deleted; + } + + public void setDeleted(boolean deleted) { + this.deleted = deleted; + } + + public DeletionReason getDeletionReason() { + return deletionReason; + } + + public void setDeletionReason(DeletionReason deletionReason) { + this.deletionReason = deletionReason; + } + + public String getOtherDeletionReason() { + return otherDeletionReason; + } + + public void setOtherDeletionReason(String otherDeletionReason) { + this.otherDeletionReason = otherDeletionReason; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationFacade.java new file mode 100644 index 00000000000..1fff4591f78 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationFacade.java @@ -0,0 +1,29 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.util.List; + +import javax.ejb.Remote; + +import de.symeda.sormas.api.CoreFacade; + +@Remote +public interface AefiInvestigationFacade + extends CoreFacade { + + List getEntriesList(AefiInvestigationListCriteria criteria, Integer first, Integer max); +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationIndexDto.java new file mode 100644 index 00000000000..635e82c75da --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationIndexDto.java @@ -0,0 +1,372 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.utils.PersonalData; +import de.symeda.sormas.api.utils.SensitiveData; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableIndexDto; + +public class AefiInvestigationIndexDto extends PseudonymizableIndexDto implements Serializable, Cloneable { + + private static final long serialVersionUID = 5659752736289073666L; + + public static final String I18N_PREFIX = "AefiInvestigationIndex"; + + public static final String UUID = "uuid"; + public static final String AEFI_REPORT_UUID = "aefiReportUuid"; + public static final String INVESTIGATION_CASE_ID = "investigationCaseId"; + public static final String DISEASE = "disease"; + public static final String PERSON_FIRST_NAME = "personFirstName"; + public static final String PERSON_LAST_NAME = "personLastName"; + public static final String AGE_AND_BIRTH_DATE = "ageAndBirthDate"; + public static final String SEX = "sex"; + public static final String REGION = "region"; + public static final String DISTRICT = "district"; + public static final String PLACE_OF_VACCINATION = "placeOfVaccination"; + public static final String VACCINATION_ACTIVITY = "vaccinationActivity"; + public static final String ADVERSE_EVENT_REPORT_DATE = "adverseEventReportDate"; + public static final String REPORT_DATE = "reportDate"; + public static final String INVESTIGATION_DATE = "investigationDate"; + public static final String INVESTIGATION_STAGE = "investigationStage"; + public static final String TYPE_OF_SITE = "typeOfSite"; + public static final String KEY_SYMPTOM_DATE_TIME = "keySymptomDateTime"; + public static final String HOSPITALIZATION_DATE = "hospitalizationDate"; + public static final String REPORTED_TO_HEALTH_AUTHORITY_DATE = "reportedToHealthAuthorityDate"; + public static final String STATUS_ON_DATE_OF_INVESTIGATION = "statusOnDateOfInvestigation"; + public static final String PRIMARY_VACCINE_NAME = "primaryVaccine"; + public static final String PRIMARY_VACCINE_DETAILS = "primaryVaccineDetails"; + public static final String INVESTIGATION_STATUS = "investigationStatus"; + public static final String AEFI_CLASSIFICATION = "aefiClassification"; + public static final String DELETION_REASON = "deletionReason"; + + private String aefiReportUuid; + private String investigationCaseId; + private Disease disease; + @PersonalData + @SensitiveData + private String personFirstName; + @PersonalData + @SensitiveData + private String personLastName; + private AgeAndBirthDateDto ageAndBirthDate; + private Sex sex; + private String region; + private String district; + private PlaceOfVaccination placeOfVaccination; + private VaccinationActivity vaccinationActivity; + private Date aefiReportDate; + private Date reportDate; + private Date investigationDate; + private AefiInvestigationStage investigationStage; + private VaccinationSite typeOfSite; + private Date keySymptomDateTime; + private Date hospitalizationDate; + private Date reportedToHealthAuthorityDate; + private PatientStatusAtAefiInvestigation statusOnDateOfInvestigation; + private Vaccine primaryVaccine; + private String primaryVaccineDetails; + private AefiInvestigationStatus investigationStatus; + private AefiClassification aefiClassification; + private DeletionReason deletionReason; + private String otherDeletionReason; + private boolean isInJurisdiction; + + public AefiInvestigationIndexDto( + String uuid, + String aefiReportUuid, + String investigationCaseId, + Disease disease, + String personFirstName, + String personLastName, + AgeAndBirthDateDto ageAndBirthDate, + Sex sex, + String region, + String district, + PlaceOfVaccination placeOfVaccination, + VaccinationActivity vaccinationActivity, + Date aefiReportDate, + Date reportDate, + Date investigationDate, + AefiInvestigationStage investigationStage, + VaccinationSite typeOfSite, + Date keySymptomDateTime, + Date hospitalizationDate, + Date reportedToHealthAuthorityDate, + PatientStatusAtAefiInvestigation statusOnDateOfInvestigation, + Vaccine primaryVaccine, + String primaryVaccineDetails, + AefiInvestigationStatus investigationStatus, + AefiClassification aefiClassification, + DeletionReason deletionReason, + String otherDeletionReason, + boolean isInJurisdiction) { + + super(uuid); + this.aefiReportUuid = aefiReportUuid; + this.investigationCaseId = investigationCaseId; + this.disease = disease; + this.personFirstName = personFirstName; + this.personLastName = personLastName; + this.ageAndBirthDate = ageAndBirthDate; + this.sex = sex; + this.region = region; + this.district = district; + this.placeOfVaccination = placeOfVaccination; + this.vaccinationActivity = vaccinationActivity; + this.aefiReportDate = aefiReportDate; + this.reportDate = reportDate; + this.investigationDate = investigationDate; + this.investigationStage = investigationStage; + this.typeOfSite = typeOfSite; + this.keySymptomDateTime = keySymptomDateTime; + this.hospitalizationDate = hospitalizationDate; + this.reportedToHealthAuthorityDate = reportedToHealthAuthorityDate; + this.statusOnDateOfInvestigation = statusOnDateOfInvestigation; + this.primaryVaccine = primaryVaccine; + this.primaryVaccineDetails = primaryVaccineDetails; + this.investigationStatus = investigationStatus; + this.aefiClassification = aefiClassification; + this.deletionReason = deletionReason; + this.otherDeletionReason = otherDeletionReason; + this.isInJurisdiction = isInJurisdiction; + } + + public String getAefiReportUuid() { + return aefiReportUuid; + } + + public void setAefiReportUuid(String aefiReportUuid) { + this.aefiReportUuid = aefiReportUuid; + } + + public String getInvestigationCaseId() { + return investigationCaseId; + } + + public void setInvestigationCaseId(String investigationCaseId) { + this.investigationCaseId = investigationCaseId; + } + + public Disease getDisease() { + return disease; + } + + public void setDisease(Disease disease) { + this.disease = disease; + } + + public String getPersonFirstName() { + return personFirstName; + } + + public void setPersonFirstName(String personFirstName) { + this.personFirstName = personFirstName; + } + + public String getPersonLastName() { + return personLastName; + } + + public void setPersonLastName(String personLastName) { + this.personLastName = personLastName; + } + + public AgeAndBirthDateDto getAgeAndBirthDate() { + return ageAndBirthDate; + } + + public void setAgeAndBirthDate(AgeAndBirthDateDto ageAndBirthDate) { + this.ageAndBirthDate = ageAndBirthDate; + } + + public Sex getSex() { + return sex; + } + + public void setSex(Sex sex) { + this.sex = sex; + } + + public String getRegion() { + return region; + } + + public void setRegion(String region) { + this.region = region; + } + + public String getDistrict() { + return district; + } + + public void setDistrict(String district) { + this.district = district; + } + + public PlaceOfVaccination getPlaceOfVaccination() { + return placeOfVaccination; + } + + public void setPlaceOfVaccination(PlaceOfVaccination placeOfVaccination) { + this.placeOfVaccination = placeOfVaccination; + } + + public VaccinationActivity getVaccinationActivity() { + return vaccinationActivity; + } + + public void setVaccinationActivity(VaccinationActivity vaccinationActivity) { + this.vaccinationActivity = vaccinationActivity; + } + + public Date getAefiReportDate() { + return aefiReportDate; + } + + public void setAefiReportDate(Date aefiReportDate) { + this.aefiReportDate = aefiReportDate; + } + + public Date getReportDate() { + return reportDate; + } + + public void setReportDate(Date reportDate) { + this.reportDate = reportDate; + } + + public Date getInvestigationDate() { + return investigationDate; + } + + public void setInvestigationDate(Date investigationDate) { + this.investigationDate = investigationDate; + } + + public AefiInvestigationStage getInvestigationStage() { + return investigationStage; + } + + public void setInvestigationStage(AefiInvestigationStage investigationStage) { + this.investigationStage = investigationStage; + } + + public VaccinationSite getTypeOfSite() { + return typeOfSite; + } + + public void setTypeOfSite(VaccinationSite typeOfSite) { + this.typeOfSite = typeOfSite; + } + + public Date getKeySymptomDateTime() { + return keySymptomDateTime; + } + + public void setKeySymptomDateTime(Date keySymptomDateTime) { + this.keySymptomDateTime = keySymptomDateTime; + } + + public Date getHospitalizationDate() { + return hospitalizationDate; + } + + public void setHospitalizationDate(Date hospitalizationDate) { + this.hospitalizationDate = hospitalizationDate; + } + + public Date getReportedToHealthAuthorityDate() { + return reportedToHealthAuthorityDate; + } + + public void setReportedToHealthAuthorityDate(Date reportedToHealthAuthorityDate) { + this.reportedToHealthAuthorityDate = reportedToHealthAuthorityDate; + } + + public PatientStatusAtAefiInvestigation getStatusOnDateOfInvestigation() { + return statusOnDateOfInvestigation; + } + + public void setStatusOnDateOfInvestigation(PatientStatusAtAefiInvestigation statusOnDateOfInvestigation) { + this.statusOnDateOfInvestigation = statusOnDateOfInvestigation; + } + + public Vaccine getPrimaryVaccine() { + return primaryVaccine; + } + + public void setPrimaryVaccine(Vaccine primaryVaccine) { + this.primaryVaccine = primaryVaccine; + } + + public String getPrimaryVaccineDetails() { + return primaryVaccineDetails; + } + + public void setPrimaryVaccineDetails(String primaryVaccineDetails) { + this.primaryVaccineDetails = primaryVaccineDetails; + } + + public AefiInvestigationStatus getInvestigationStatus() { + return investigationStatus; + } + + public void setInvestigationStatus(AefiInvestigationStatus investigationStatus) { + this.investigationStatus = investigationStatus; + } + + public AefiClassification getAefiClassification() { + return aefiClassification; + } + + public void setAefiClassification(AefiClassification aefiClassification) { + this.aefiClassification = aefiClassification; + } + + public DeletionReason getDeletionReason() { + return deletionReason; + } + + public void setDeletionReason(DeletionReason deletionReason) { + this.deletionReason = deletionReason; + } + + public String getOtherDeletionReason() { + return otherDeletionReason; + } + + public void setOtherDeletionReason(String otherDeletionReason) { + this.otherDeletionReason = otherDeletionReason; + } + + @Override + public boolean isInJurisdiction() { + return isInJurisdiction; + } + + @Override + public void setInJurisdiction(boolean inJurisdiction) { + isInJurisdiction = inJurisdiction; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListCriteria.java new file mode 100644 index 00000000000..c150fb25e81 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListCriteria.java @@ -0,0 +1,44 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.utils.criteria.BaseCriteria; + +public class AefiInvestigationListCriteria extends BaseCriteria { + + private final AefiReferenceDto aefiReport; + + public static class Builder { + + private final AefiReferenceDto aefiReferenceDto; + + public Builder(AefiReferenceDto aefiReferenceDto) { + this.aefiReferenceDto = aefiReferenceDto; + } + + public AefiInvestigationListCriteria build() { + return new AefiInvestigationListCriteria(this); + } + } + + private AefiInvestigationListCriteria(AefiInvestigationListCriteria.Builder builder) { + this.aefiReport = builder.aefiReferenceDto; + } + + public AefiReferenceDto getAefiReport() { + return aefiReport; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListEntryDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListEntryDto.java new file mode 100644 index 00000000000..7282075e79a --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationListEntryDto.java @@ -0,0 +1,156 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import java.io.Serializable; +import java.util.Date; + +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableIndexDto; + +public class AefiInvestigationListEntryDto extends PseudonymizableIndexDto implements Serializable, Cloneable { + + public static final String I18N_PREFIX = "AefiInvestigationListEntry"; + + public static final String UUID = "uuid"; + public static final String INVESTIGATION_CASE_ID = "investigationCaseId"; + public static final String INVESTIGATION_DATE = "investigationDate"; + public static final String INVESTIGATION_STAGE = "investigationStage"; + public static final String STATUS_ON_DATE_OF_INVESTIGATION = "statusOnDateOfInvestigation"; + public static final String PRIMARY_VACCINE_NAME = "primaryVaccine"; + public static final String PRIMARY_VACCINE_DETAILS = "primaryVaccineDetails"; + public static final String PRIMARY_VACCINE_DOSE = "primaryVaccineDose"; + public static final String PRIMARY_VACCINE_VACCINATION_DATE = "primaryVaccineVaccinationDate"; + public static final String INVESTIGATION_STATUS = "investigationStatus"; + public static final String AEFI_CLASSIFICATION = "aefiClassification"; + + private String investigationCaseId; + private Date investigationDate; + private AefiInvestigationStage investigationStage; + private PatientStatusAtAefiInvestigation statusOnDateOfInvestigation; + private Vaccine primaryVaccine; + private String primaryVaccineDetails; + private String primaryVaccineDose; + private Date primaryVaccineVaccinationDate; + private AefiInvestigationStatus investigationStatus; + private AefiClassification aefiClassification; + + public AefiInvestigationListEntryDto( + String uuid, + String investigationCaseId, + Date investigationDate, + AefiInvestigationStage investigationStage, + PatientStatusAtAefiInvestigation statusOnDateOfInvestigation, + Vaccine primaryVaccine, + String primaryVaccineDetails, + String primaryVaccineDose, + Date primaryVaccineVaccinationDate, + AefiInvestigationStatus investigationStatus, + AefiClassification aefiClassification) { + + super(uuid); + this.investigationCaseId = investigationCaseId; + this.investigationDate = investigationDate; + this.investigationStage = investigationStage; + this.statusOnDateOfInvestigation = statusOnDateOfInvestigation; + this.primaryVaccine = primaryVaccine; + this.primaryVaccineDetails = primaryVaccineDetails; + this.primaryVaccineDose = primaryVaccineDose; + this.primaryVaccineVaccinationDate = primaryVaccineVaccinationDate; + this.investigationStatus = investigationStatus; + this.aefiClassification = aefiClassification; + } + + public String getInvestigationCaseId() { + return investigationCaseId; + } + + public void setInvestigationCaseId(String investigationCaseId) { + this.investigationCaseId = investigationCaseId; + } + + public Date getInvestigationDate() { + return investigationDate; + } + + public void setInvestigationDate(Date investigationDate) { + this.investigationDate = investigationDate; + } + + public AefiInvestigationStage getInvestigationStage() { + return investigationStage; + } + + public void setInvestigationStage(AefiInvestigationStage investigationStage) { + this.investigationStage = investigationStage; + } + + public PatientStatusAtAefiInvestigation getStatusOnDateOfInvestigation() { + return statusOnDateOfInvestigation; + } + + public void setStatusOnDateOfInvestigation(PatientStatusAtAefiInvestigation statusOnDateOfInvestigation) { + this.statusOnDateOfInvestigation = statusOnDateOfInvestigation; + } + + public Vaccine getPrimaryVaccine() { + return primaryVaccine; + } + + public void setPrimaryVaccine(Vaccine primaryVaccine) { + this.primaryVaccine = primaryVaccine; + } + + public String getPrimaryVaccineDetails() { + return primaryVaccineDetails; + } + + public void setPrimaryVaccineDetails(String primaryVaccineDetails) { + this.primaryVaccineDetails = primaryVaccineDetails; + } + + public String getPrimaryVaccineDose() { + return primaryVaccineDose; + } + + public void setPrimaryVaccineDose(String primaryVaccineDose) { + this.primaryVaccineDose = primaryVaccineDose; + } + + public Date getPrimaryVaccineVaccinationDate() { + return primaryVaccineVaccinationDate; + } + + public void setPrimaryVaccineVaccinationDate(Date primaryVaccineVaccinationDate) { + this.primaryVaccineVaccinationDate = primaryVaccineVaccinationDate; + } + + public AefiInvestigationStatus getInvestigationStatus() { + return investigationStatus; + } + + public void setInvestigationStatus(AefiInvestigationStatus investigationStatus) { + this.investigationStatus = investigationStatus; + } + + public AefiClassification getAefiClassification() { + return aefiClassification; + } + + public void setAefiClassification(AefiClassification aefiClassification) { + this.aefiClassification = aefiClassification; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationReferenceDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationReferenceDto.java new file mode 100644 index 00000000000..10d9c77a3a5 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationReferenceDto.java @@ -0,0 +1,42 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.ReferenceDto; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.utils.DependingOnFeatureType; + +@DependingOnFeatureType(featureType = FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT) +public class AefiInvestigationReferenceDto extends ReferenceDto { + + private String externalId; + + public AefiInvestigationReferenceDto() { + } + + public AefiInvestigationReferenceDto(String uuid, String externalId) { + super(uuid); + this.externalId = externalId; + } + + public String getExternalId() { + return externalId; + } + + public void setExternalId(String externalId) { + this.externalId = externalId; + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStage.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStage.java new file mode 100644 index 00000000000..e6a8d26526e --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStage.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiInvestigationStage { + + FIRST, + INTERIM, + FINAL; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStatus.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStatus.java new file mode 100644 index 00000000000..1506a3c1138 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiInvestigationStatus.java @@ -0,0 +1,33 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiInvestigationStatus { + + DONE, + DISCARDED; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } + + public String toShortString() { + return I18nProperties.getEnumCaptionShort(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java index 51acac53ec1..9f992ad8963 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiReferenceDto.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -30,8 +27,8 @@ public class AefiReferenceDto extends ReferenceDto { public AefiReferenceDto() { } - public AefiReferenceDto(String uuid, String caption, String externalId) { - super(uuid, caption); + public AefiReferenceDto(String uuid, String externalId) { + super(uuid); this.externalId = externalId; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiVaccinationPeriod.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiVaccinationPeriod.java new file mode 100644 index 00000000000..67bee292daa --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/AefiVaccinationPeriod.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum AefiVaccinationPeriod { + + WITHIN_FIRST_FEW_DOSES, + WITHIN_LAST_DOSES, + UNKNOWN; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/BirthTerm.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/BirthTerm.java new file mode 100644 index 00000000000..11abb53e8d8 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/BirthTerm.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum BirthTerm { + + FULL_TERM, + PRE_TERM, + POST_TERM; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/DeliveryProcedure.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/DeliveryProcedure.java new file mode 100644 index 00000000000..998861a1e6a --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/DeliveryProcedure.java @@ -0,0 +1,31 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum DeliveryProcedure { + + NORMAL, + CAESAREAN, + ASSISTED, + WITH_COMPLICATION; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PatientStatusAtAefiInvestigation.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PatientStatusAtAefiInvestigation.java new file mode 100644 index 00000000000..6067cb39260 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PatientStatusAtAefiInvestigation.java @@ -0,0 +1,32 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum PatientStatusAtAefiInvestigation { + + DIED, + DISABLED, + RECOVERED, + RECOVERED_COMPLETELY, + UNKNOWN; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PlaceOfVaccination.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PlaceOfVaccination.java new file mode 100644 index 00000000000..772ca5641a1 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/PlaceOfVaccination.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum PlaceOfVaccination { + + GOVERNMENT_HEALTH_FACILITY, + PRIVATE_HEALTH_FACILITY, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiInfoSource.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiInfoSource.java new file mode 100644 index 00000000000..aa21ff6588d --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SeriousAefiInfoSource.java @@ -0,0 +1,31 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum SeriousAefiInfoSource { + + EXAMINATION, + DOCUMENTS, + VERBAL_AUTOPSY, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SyringeType.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SyringeType.java new file mode 100644 index 00000000000..a8161b6c21a --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/SyringeType.java @@ -0,0 +1,31 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum SyringeType { + + GLASS, + DISPOSABLE, + RECYCLED_DISPOSABLE, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationActivity.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationActivity.java new file mode 100644 index 00000000000..d7724fb2fc8 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationActivity.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum VaccinationActivity { + + CAMPAIGN, + ROUTINE, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationSite.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationSite.java new file mode 100644 index 00000000000..c4625e1ceed --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccinationSite.java @@ -0,0 +1,32 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum VaccinationSite { + + FIXED, + MOBILE, + OUTREACH, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } + +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccineCarrier.java b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccineCarrier.java new file mode 100644 index 00000000000..86ebe5d47ab --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/adverseeventsfollowingimmunization/VaccineCarrier.java @@ -0,0 +1,30 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.adverseeventsfollowingimmunization; + +import de.symeda.sormas.api.i18n.I18nProperties; + +public enum VaccineCarrier { + + SHORT_RANGE, + LONG_RANGE, + OTHER; + + @Override + public String toString() { + return I18nProperties.getEnumCaption(this); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/Vaccine.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/Vaccine.java index d1a41a5e01b..9071192bc8d 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/Vaccine.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/Vaccine.java @@ -59,6 +59,18 @@ public enum Vaccine { @Diseases(value = { Disease.CORONAVIRUS }) SANOFI_GSK(VaccineManufacturer.SANOFI_GSK), + @Diseases(value = { + Disease.CSM }) + MenABCWY(VaccineManufacturer.PFIZER), + @Diseases(value = { + Disease.MONKEYPOX }) + ACAM2000(VaccineManufacturer.SANOFI_PASTEUR_BIOLOGICS), + @Diseases(value = { + Disease.MONKEYPOX }) + LC_16(VaccineManufacturer.KM_BIOLOGICS), + @Diseases(value = { + Disease.MONKEYPOX }) + MVA_BN(VaccineManufacturer.BAVARIAN_NORDIC), UNKNOWN, OTHER; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/VaccineManufacturer.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/VaccineManufacturer.java index 9d384da94a7..286293d076b 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/VaccineManufacturer.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/VaccineManufacturer.java @@ -24,6 +24,12 @@ public enum VaccineManufacturer { @Diseases(value = { Disease.CORONAVIRUS }) BIONTECH_PFIZER, + @Diseases(value = { + Disease.CSM }) + PFIZER, + @Diseases(value = { + Disease.MONKEYPOX }) + BAVARIAN_NORDIC, @Diseases(value = { Disease.CORONAVIRUS }) MODERNA, @@ -33,12 +39,18 @@ public enum VaccineManufacturer { @Diseases(value = { Disease.CORONAVIRUS }) JOHNSON_JOHNSON, + @Diseases(value = { + Disease.MONKEYPOX }) + KM_BIOLOGICS, @Diseases(value = { Disease.CORONAVIRUS }) NOVAVAX, @Diseases(value = { Disease.CORONAVIRUS }) SANOFI_GSK, + @Diseases(value = { + Disease.MONKEYPOX }) + SANOFI_PASTEUR_BIOLOGICS, VALNEVA, UNKNOWN, OTHER; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java index 2e61f221bc5..2e6e9c7a087 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacade.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -23,6 +20,8 @@ import javax.ejb.Remote; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; import de.symeda.sormas.api.caze.Vaccine; import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; @@ -32,6 +31,10 @@ public interface AefiDashboardFacade { Map getAefiCountsByType(AefiDashboardCriteria dashboardCriteria); + Map> getAefiInvestigationCountsByInvestigationStatus(AefiDashboardCriteria dashboardCriteria); + + Map> getAefiInvestigationCountsByAefiClassification(AefiDashboardCriteria dashboardCriteria); + Map> getAefiCountsByVaccine(AefiDashboardCriteria dashboardCriteria); AefiChartData getAefiByVaccineDoseChartData(AefiDashboardCriteria dashboardCriteria); diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index ac9fd632af3..233647e5b4f 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -237,15 +237,26 @@ public interface Captions { String Aefi_uuid = "Aefi.uuid"; String Aefi_worldwideId = "Aefi.worldwideId"; String aefiActiveAdverseEvents = "aefiActiveAdverseEvents"; + String aefiActiveInvestigations = "aefiActiveInvestigations"; String aefiAefiDataView = "aefiAefiDataView"; + String aefiAefiInvestigationDataView = "aefiAefiInvestigationDataView"; + String aefiAefiInvestigationList = "aefiAefiInvestigationList"; String aefiAefiList = "aefiAefiList"; String aefiAllActiveAndArchivedAdverseEvents = "aefiAllActiveAndArchivedAdverseEvents"; + String aefiAllActiveAndArchivedInvestigations = "aefiAllActiveAndArchivedInvestigations"; String aefiArchivedAdverseEvents = "aefiArchivedAdverseEvents"; + String aefiArchivedInvestigations = "aefiArchivedInvestigations"; String AefiCriteria_aefiType = "AefiCriteria.aefiType"; String AefiCriteria_outcome = "AefiCriteria.outcome"; String AefiCriteria_vaccineManufacturer = "AefiCriteria.vaccineManufacturer"; String AefiCriteria_vaccineName = "AefiCriteria.vaccineName"; + String aefiDashboardAefiClassificationCoincidentalAdverseEvent = "aefiDashboardAefiClassificationCoincidentalAdverseEvent"; + String aefiDashboardAefiClassificationRelatedToVaccination = "aefiDashboardAefiClassificationRelatedToVaccination"; + String aefiDashboardAefiClassificationUndetermined = "aefiDashboardAefiClassificationUndetermined"; + String aefiDashboardAefiInvestigationDiscarded = "aefiDashboardAefiInvestigationDiscarded"; + String aefiDashboardAefiInvestigationDone = "aefiDashboardAefiInvestigationDone"; String aefiDashboardAllAefi = "aefiDashboardAllAefi"; + String aefiDashboardAllAefiInvestigation = "aefiDashboardAllAefiInvestigation"; String aefiDashboardNonSerious = "aefiDashboardNonSerious"; String aefiDashboardNonSeriousAefi = "aefiDashboardNonSeriousAefi"; String aefiDashboardSerious = "aefiDashboardSerious"; @@ -253,6 +264,50 @@ public interface Captions { String aefiDashboardShowNonSeriousAefi = "aefiDashboardShowNonSeriousAefi"; String aefiDashboardShowSeriousAefi = "aefiDashboardShowSeriousAefi"; String aefiDeletedAdverseEvents = "aefiDeletedAdverseEvents"; + String aefiDeletedInvestigations = "aefiDeletedInvestigations"; + String AefiExport_aefiDescription = "AefiExport.aefiDescription"; + String AefiExport_birthDate = "AefiExport.birthDate"; + String AefiExport_firstName = "AefiExport.firstName"; + String AefiExport_lastName = "AefiExport.lastName"; + String AefiExport_nationalLevelComment = "AefiExport.nationalLevelComment"; + String AefiExport_onsetAgeDays = "AefiExport.onsetAgeDays"; + String AefiExport_onsetAgeGroup = "AefiExport.onsetAgeGroup"; + String AefiExport_onsetAgeMonths = "AefiExport.onsetAgeMonths"; + String AefiExport_onsetAgeYears = "AefiExport.onsetAgeYears"; + String AefiExport_outcome = "AefiExport.outcome"; + String AefiExport_patientAddressCommunity = "AefiExport.patientAddressCommunity"; + String AefiExport_patientAddressDetails = "AefiExport.patientAddressDetails"; + String AefiExport_patientAddressDistrict = "AefiExport.patientAddressDistrict"; + String AefiExport_patientAddressRegion = "AefiExport.patientAddressRegion"; + String AefiExport_primarySuspectVaccineBatchNumber = "AefiExport.primarySuspectVaccineBatchNumber"; + String AefiExport_primarySuspectVaccineBrand = "AefiExport.primarySuspectVaccineBrand"; + String AefiExport_primarySuspectVaccineDiluentBatchNumber = "AefiExport.primarySuspectVaccineDiluentBatchNumber"; + String AefiExport_primarySuspectVaccineDose = "AefiExport.primarySuspectVaccineDose"; + String AefiExport_primarySuspectVaccineManufacturer = "AefiExport.primarySuspectVaccineManufacturer"; + String AefiExport_primarySuspectVaccineName = "AefiExport.primarySuspectVaccineName"; + String AefiExport_primarySuspectVaccineOtherName = "AefiExport.primarySuspectVaccineOtherName"; + String AefiExport_primarySuspectVaccineVaccinationDate = "AefiExport.primarySuspectVaccineVaccinationDate"; + String AefiExport_receivedAtNationalLevelDate = "AefiExport.receivedAtNationalLevelDate"; + String AefiExport_reportDate = "AefiExport.reportDate"; + String AefiExport_reportingIdNumber = "AefiExport.reportingIdNumber"; + String AefiExport_reportingOfficerAddressCountryName = "AefiExport.reportingOfficerAddressCountryName"; + String AefiExport_reportingOfficerDepartment = "AefiExport.reportingOfficerDepartment"; + String AefiExport_reportingOfficerDesignation = "AefiExport.reportingOfficerDesignation"; + String AefiExport_reportingOfficerEmail = "AefiExport.reportingOfficerEmail"; + String AefiExport_reportingOfficerFacilityCommunity = "AefiExport.reportingOfficerFacilityCommunity"; + String AefiExport_reportingOfficerFacilityDistrict = "AefiExport.reportingOfficerFacilityDistrict"; + String AefiExport_reportingOfficerFacilityName = "AefiExport.reportingOfficerFacilityName"; + String AefiExport_reportingOfficerFacilityRegion = "AefiExport.reportingOfficerFacilityRegion"; + String AefiExport_reportingOfficerName = "AefiExport.reportingOfficerName"; + String AefiExport_reportingOfficerPhoneNumber = "AefiExport.reportingOfficerPhoneNumber"; + String AefiExport_serious = "AefiExport.serious"; + String AefiExport_sex = "AefiExport.sex"; + String AefiExport_startDateTime = "AefiExport.startDateTime"; + String AefiExport_vaccinationFacilityCommunity = "AefiExport.vaccinationFacilityCommunity"; + String AefiExport_vaccinationFacilityDistrict = "AefiExport.vaccinationFacilityDistrict"; + String AefiExport_vaccinationFacilityName = "AefiExport.vaccinationFacilityName"; + String AefiExport_vaccinationFacilityRegion = "AefiExport.vaccinationFacilityRegion"; + String AefiExport_worldWideId = "AefiExport.worldWideId"; String AefiIndex_adverseEvents = "AefiIndex.adverseEvents"; String AefiIndex_ageAndBirthDate = "AefiIndex.ageAndBirthDate"; String AefiIndex_disease = "AefiIndex.disease"; @@ -270,7 +325,195 @@ public interface Captions { String AefiIndex_startDateTime = "AefiIndex.startDateTime"; String AefiIndex_uuid = "AefiIndex.uuid"; String AefiIndex_vaccinationDate = "AefiIndex.vaccinationDate"; + String AefiInvestigation_adSyringesUsedForImmunization = "AefiInvestigation.adSyringesUsedForImmunization"; + String AefiInvestigation_adverseEventAfterPreviousVaccinations = "AefiInvestigation.adverseEventAfterPreviousVaccinations"; + String AefiInvestigation_adverseEventAfterPreviousVaccinationsDetails = "AefiInvestigation.adverseEventAfterPreviousVaccinationsDetails"; + String AefiInvestigation_aefiClassification = "AefiInvestigation.aefiClassification"; + String AefiInvestigation_aefiClassificationDetails = "AefiInvestigation.aefiClassificationDetails"; + String AefiInvestigation_aefiClassificationSubType = "AefiInvestigation.aefiClassificationSubType"; + String AefiInvestigation_allCasesInClusterReceivedVaccineFromSameVial = "AefiInvestigation.allCasesInClusterReceivedVaccineFromSameVial"; + String AefiInvestigation_allCasesInClusterReceivedVaccineFromSameVialDetails = "AefiInvestigation.allCasesInClusterReceivedVaccineFromSameVialDetails"; + String AefiInvestigation_anyOtherItemInRefrigerator = "AefiInvestigation.anyOtherItemInRefrigerator"; + String AefiInvestigation_anyStorageTemperatureDeviationOutsideTwoToEightDegrees = "AefiInvestigation.anyStorageTemperatureDeviationOutsideTwoToEightDegrees"; + String AefiInvestigation_autopsyDate = "AefiInvestigation.autopsyDate"; + String AefiInvestigation_autopsyDone = "AefiInvestigation.autopsyDone"; + String AefiInvestigation_autopsyPlannedDateTime = "AefiInvestigation.autopsyPlannedDateTime"; + String AefiInvestigation_birthTerm = "AefiInvestigation.birthTerm"; + String AefiInvestigation_birthWeight = "AefiInvestigation.birthWeight"; + String AefiInvestigation_caseIsPartOfACluster = "AefiInvestigation.caseIsPartOfACluster"; + String AefiInvestigation_caseIsPartOfAClusterDetails = "AefiInvestigation.caseIsPartOfAClusterDetails"; + String AefiInvestigation_causality = "AefiInvestigation.causality"; + String AefiInvestigation_causalityDetails = "AefiInvestigation.causalityDetails"; + String AefiInvestigation_changeDate = "AefiInvestigation.changeDate"; + String AefiInvestigation_clinicalDetailsDateTime = "AefiInvestigation.clinicalDetailsDateTime"; + String AefiInvestigation_clinicalDetailsOfficerDesignation = "AefiInvestigation.clinicalDetailsOfficerDesignation"; + String AefiInvestigation_clinicalDetailsOfficerEmail = "AefiInvestigation.clinicalDetailsOfficerEmail"; + String AefiInvestigation_clinicalDetailsOfficerName = "AefiInvestigation.clinicalDetailsOfficerName"; + String AefiInvestigation_clinicalDetailsOfficerPhoneNumber = "AefiInvestigation.clinicalDetailsOfficerPhoneNumber"; + String AefiInvestigation_communityInvestigationAdditionalDetails = "AefiInvestigation.communityInvestigationAdditionalDetails"; + String AefiInvestigation_conditionedIcepackUsed = "AefiInvestigation.conditionedIcepackUsed"; + String AefiInvestigation_contraIndicationScreenedPriorToVaccination = "AefiInvestigation.contraIndicationScreenedPriorToVaccination"; + String AefiInvestigation_correctDoseOrRoute = "AefiInvestigation.correctDoseOrRoute"; + String AefiInvestigation_correctProcedureForStorageFollowed = "AefiInvestigation.correctProcedureForStorageFollowed"; + String AefiInvestigation_country = "AefiInvestigation.country"; + String AefiInvestigation_creationDate = "AefiInvestigation.creationDate"; + String AefiInvestigation_currentlyOnConcomitantMedication = "AefiInvestigation.currentlyOnConcomitantMedication"; + String AefiInvestigation_currentlyOnConcomitantMedicationDetails = "AefiInvestigation.currentlyOnConcomitantMedicationDetails"; + String AefiInvestigation_deathDateTime = "AefiInvestigation.deathDateTime"; + String AefiInvestigation_deletionReason = "AefiInvestigation.deletionReason"; + String AefiInvestigation_deliveryProcedure = "AefiInvestigation.deliveryProcedure"; + String AefiInvestigation_deliveryProcedureDetails = "AefiInvestigation.deliveryProcedureDetails"; + String AefiInvestigation_errorInVaccineHandling = "AefiInvestigation.errorInVaccineHandling"; + String AefiInvestigation_errorInVaccineHandlingDetails = "AefiInvestigation.errorInVaccineHandlingDetails"; + String AefiInvestigation_errorInVaccineReconstitution = "AefiInvestigation.errorInVaccineReconstitution"; + String AefiInvestigation_errorInVaccineReconstitutionDetails = "AefiInvestigation.errorInVaccineReconstitutionDetails"; + String AefiInvestigation_errorPrescribingVaccine = "AefiInvestigation.errorPrescribingVaccine"; + String AefiInvestigation_errorPrescribingVaccineDetails = "AefiInvestigation.errorPrescribingVaccineDetails"; + String AefiInvestigation_eventIsAStressResponseRelatedToImmunization = "AefiInvestigation.eventIsAStressResponseRelatedToImmunization"; + String AefiInvestigation_eventIsAStressResponseRelatedToImmunizationDetails = "AefiInvestigation.eventIsAStressResponseRelatedToImmunizationDetails"; + String AefiInvestigation_externalId = "AefiInvestigation.externalId"; + String AefiInvestigation_familyHistoryOfDiseaseOrAllergy = "AefiInvestigation.familyHistoryOfDiseaseOrAllergy"; + String AefiInvestigation_familyHistoryOfDiseaseOrAllergyDetails = "AefiInvestigation.familyHistoryOfDiseaseOrAllergyDetails"; + String AefiInvestigation_firstCaregiversName = "AefiInvestigation.firstCaregiversName"; + String AefiInvestigation_formCompletionDate = "AefiInvestigation.formCompletionDate"; + String AefiInvestigation_historyOfAllergyToVaccineDrugOrFood = "AefiInvestigation.historyOfAllergyToVaccineDrugOrFood"; + String AefiInvestigation_historyOfAllergyToVaccineDrugOrFoodDetails = "AefiInvestigation.historyOfAllergyToVaccineDrugOrFoodDetails"; + String AefiInvestigation_historyOfHospitalizationInLastThirtyDaysWithCause = "AefiInvestigation.historyOfHospitalizationInLastThirtyDaysWithCause"; + String AefiInvestigation_historyOfHospitalizationInLastThirtyDaysWithCauseDetails = "AefiInvestigation.historyOfHospitalizationInLastThirtyDaysWithCauseDetails"; + String AefiInvestigation_hospitalizationDate = "AefiInvestigation.hospitalizationDate"; + String AefiInvestigation_injectionTechniqueAdditionalDetails = "AefiInvestigation.injectionTechniqueAdditionalDetails"; + String AefiInvestigation_investigationCaseId = "AefiInvestigation.investigationCaseId"; + String AefiInvestigation_investigationCompletionDate = "AefiInvestigation.investigationCompletionDate"; + String AefiInvestigation_investigationDate = "AefiInvestigation.investigationDate"; + String AefiInvestigation_investigationStage = "AefiInvestigation.investigationStage"; + String AefiInvestigation_investigationStatus = "AefiInvestigation.investigationStatus"; + String AefiInvestigation_investigationStatusDetails = "AefiInvestigation.investigationStatusDetails"; + String AefiInvestigation_keySymptomDateTime = "AefiInvestigation.keySymptomDateTime"; + String AefiInvestigation_lastTrainingReceivedByVaccinatorDate = "AefiInvestigation.lastTrainingReceivedByVaccinatorDate"; + String AefiInvestigation_nonTouchTechniqueFollowed = "AefiInvestigation.nonTouchTechniqueFollowed"; + String AefiInvestigation_numberImmunizedConcernedVaccineSameBatchNumberLocationDetails = "AefiInvestigation.numberImmunizedConcernedVaccineSameBatchNumberLocationDetails"; + String AefiInvestigation_numberImmunizedConcernedVaccineSameBatchNumberOtherLocations = "AefiInvestigation.numberImmunizedConcernedVaccineSameBatchNumberOtherLocations"; + String AefiInvestigation_numberImmunizedFromConcernedVaccineVial = "AefiInvestigation.numberImmunizedFromConcernedVaccineVial"; + String AefiInvestigation_numberImmunizedWithConcernedVaccineInSameSession = "AefiInvestigation.numberImmunizedWithConcernedVaccineInSameSession"; + String AefiInvestigation_numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays = "AefiInvestigation.numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays"; + String AefiInvestigation_numberOfCasesDetectedInCluster = "AefiInvestigation.numberOfCasesDetectedInCluster"; + String AefiInvestigation_numberOfSimilarEventsReportedSamePeriodAndLocality = "AefiInvestigation.numberOfSimilarEventsReportedSamePeriodAndLocality"; + String AefiInvestigation_numberOfThoseAffectedNotVaccinated = "AefiInvestigation.numberOfThoseAffectedNotVaccinated"; + String AefiInvestigation_numberOfThoseAffectedVaccinated = "AefiInvestigation.numberOfThoseAffectedVaccinated"; + String AefiInvestigation_numberOfThoseAffectedVaccinatedUnknown = "AefiInvestigation.numberOfThoseAffectedVaccinatedUnknown"; + String AefiInvestigation_numberOfVialsUsedInCluster = "AefiInvestigation.numberOfVialsUsedInCluster"; + String AefiInvestigation_numberOfVialsUsedInClusterDetails = "AefiInvestigation.numberOfVialsUsedInClusterDetails"; + String AefiInvestigation_numberOfWeeksPregnant = "AefiInvestigation.numberOfWeeksPregnant"; + String AefiInvestigation_otherCaregiversNames = "AefiInvestigation.otherCaregiversNames"; + String AefiInvestigation_otherDeletionReason = "AefiInvestigation.otherDeletionReason"; + String AefiInvestigation_otherInvestigationFindings = "AefiInvestigation.otherInvestigationFindings"; + String AefiInvestigation_otherSourcesWhoProvidedInfo = "AefiInvestigation.otherSourcesWhoProvidedInfo"; + String AefiInvestigation_partiallyUsedReconstitutedVaccinesInRefrigerator = "AefiInvestigation.partiallyUsedReconstitutedVaccinesInRefrigerator"; + String AefiInvestigation_pastHistoryOfSimilarEvent = "AefiInvestigation.pastHistoryOfSimilarEvent"; + String AefiInvestigation_pastHistoryOfSimilarEventDetails = "AefiInvestigation.pastHistoryOfSimilarEventDetails"; + String AefiInvestigation_patientImmunizedPeriod = "AefiInvestigation.patientImmunizedPeriod"; + String AefiInvestigation_patientImmunizedPeriodDetails = "AefiInvestigation.patientImmunizedPeriodDetails"; + String AefiInvestigation_patientReceivedMedicalCare = "AefiInvestigation.patientReceivedMedicalCare"; + String AefiInvestigation_patientReceivedMedicalCareDetails = "AefiInvestigation.patientReceivedMedicalCareDetails"; + String AefiInvestigation_placeOfVaccination = "AefiInvestigation.placeOfVaccination"; + String AefiInvestigation_placeOfVaccinationDetails = "AefiInvestigation.placeOfVaccinationDetails"; + String AefiInvestigation_preExistingIllnessThirtyDaysOrCongenitalDisorder = "AefiInvestigation.preExistingIllnessThirtyDaysOrCongenitalDisorder"; + String AefiInvestigation_preExistingIllnessThirtyDaysOrCongenitalDisorderDetails = "AefiInvestigation.preExistingIllnessThirtyDaysOrCongenitalDisorderDetails"; + String AefiInvestigation_provisionalOrFinalDiagnosis = "AefiInvestigation.provisionalOrFinalDiagnosis"; + String AefiInvestigation_reconstitutionAdditionalDetails = "AefiInvestigation.reconstitutionAdditionalDetails"; + String AefiInvestigation_reportDate = "AefiInvestigation.reportDate"; + String AefiInvestigation_reportedToHealthAuthorityDate = "AefiInvestigation.reportedToHealthAuthorityDate"; + String AefiInvestigation_reportingOfficerAddress = "AefiInvestigation.reportingOfficerAddress"; + String AefiInvestigation_reportingOfficerDepartment = "AefiInvestigation.reportingOfficerDepartment"; + String AefiInvestigation_reportingOfficerDesignation = "AefiInvestigation.reportingOfficerDesignation"; + String AefiInvestigation_reportingOfficerEmail = "AefiInvestigation.reportingOfficerEmail"; + String AefiInvestigation_reportingOfficerFacility = "AefiInvestigation.reportingOfficerFacility"; + String AefiInvestigation_reportingOfficerFacilityDetails = "AefiInvestigation.reportingOfficerFacilityDetails"; + String AefiInvestigation_reportingOfficerLandlinePhoneNumber = "AefiInvestigation.reportingOfficerLandlinePhoneNumber"; + String AefiInvestigation_reportingOfficerMobilePhoneNumber = "AefiInvestigation.reportingOfficerMobilePhoneNumber"; + String AefiInvestigation_reportingOfficerName = "AefiInvestigation.reportingOfficerName"; + String AefiInvestigation_reportingUser = "AefiInvestigation.reportingUser"; + String AefiInvestigation_responsibleCommunity = "AefiInvestigation.responsibleCommunity"; + String AefiInvestigation_responsibleDistrict = "AefiInvestigation.responsibleDistrict"; + String AefiInvestigation_responsibleRegion = "AefiInvestigation.responsibleRegion"; + String AefiInvestigation_sameReconstitutionSyringeForEachVaccination = "AefiInvestigation.sameReconstitutionSyringeForEachVaccination"; + String AefiInvestigation_sameReconstitutionSyringeForEachVaccineVial = "AefiInvestigation.sameReconstitutionSyringeForEachVaccineVial"; + String AefiInvestigation_sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine = "AefiInvestigation.sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine"; + String AefiInvestigation_sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines = "AefiInvestigation.sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines"; + String AefiInvestigation_seriousAefiInfoSource = "AefiInvestigation.seriousAefiInfoSource"; + String AefiInvestigation_seriousAefiInfoSourceDetails = "AefiInvestigation.seriousAefiInfoSourceDetails"; + String AefiInvestigation_seriousAefiVerbalAutopsyInfoSourceDetails = "AefiInvestigation.seriousAefiVerbalAutopsyInfoSourceDetails"; + String AefiInvestigation_signsAndSymptomsFromTimeOfVaccination = "AefiInvestigation.signsAndSymptomsFromTimeOfVaccination"; + String AefiInvestigation_similarEventsReportedSamePeriodAndLocality = "AefiInvestigation.similarEventsReportedSamePeriodAndLocality"; + String AefiInvestigation_similarEventsReportedSamePeriodAndLocalityDetails = "AefiInvestigation.similarEventsReportedSamePeriodAndLocalityDetails"; + String AefiInvestigation_statusOnDateOfInvestigation = "AefiInvestigation.statusOnDateOfInvestigation"; + String AefiInvestigation_storageTemperatureMonitoringAdditionalDetails = "AefiInvestigation.storageTemperatureMonitoringAdditionalDetails"; + String AefiInvestigation_syringesUsedAdditionalDetails = "AefiInvestigation.syringesUsedAdditionalDetails"; + String AefiInvestigation_timeOfReconstitutionMentionedOnTheVial = "AefiInvestigation.timeOfReconstitutionMentionedOnTheVial"; + String AefiInvestigation_trainingReceivedByVaccinator = "AefiInvestigation.trainingReceivedByVaccinator"; + String AefiInvestigation_typeOfSite = "AefiInvestigation.typeOfSite"; + String AefiInvestigation_typeOfSiteDetails = "AefiInvestigation.typeOfSiteDetails"; + String AefiInvestigation_typeOfSyringesUsed = "AefiInvestigation.typeOfSyringesUsed"; + String AefiInvestigation_typeOfSyringesUsedDetails = "AefiInvestigation.typeOfSyringesUsedDetails"; + String AefiInvestigation_unusableDiluentsInStore = "AefiInvestigation.unusableDiluentsInStore"; + String AefiInvestigation_unusableVaccinesInRefrigerator = "AefiInvestigation.unusableVaccinesInRefrigerator"; + String AefiInvestigation_uuid = "AefiInvestigation.uuid"; + String AefiInvestigation_vaccinationActivity = "AefiInvestigation.vaccinationActivity"; + String AefiInvestigation_vaccinationActivityDetails = "AefiInvestigation.vaccinationActivityDetails"; + String AefiInvestigation_vaccinationFacility = "AefiInvestigation.vaccinationFacility"; + String AefiInvestigation_vaccinationFacilityDetails = "AefiInvestigation.vaccinationFacilityDetails"; + String AefiInvestigation_vaccineAdministeredIncorrectly = "AefiInvestigation.vaccineAdministeredIncorrectly"; + String AefiInvestigation_vaccineAdministeredIncorrectlyDetails = "AefiInvestigation.vaccineAdministeredIncorrectlyDetails"; + String AefiInvestigation_vaccineCarrierReturnedFromSiteOnSameDateAsVaccination = "AefiInvestigation.vaccineCarrierReturnedFromSiteOnSameDateAsVaccination"; + String AefiInvestigation_vaccineCarrierSentToSiteOnSameDateAsVaccination = "AefiInvestigation.vaccineCarrierSentToSiteOnSameDateAsVaccination"; + String AefiInvestigation_vaccineCarrierType = "AefiInvestigation.vaccineCarrierType"; + String AefiInvestigation_vaccineCarrierTypeDetails = "AefiInvestigation.vaccineCarrierTypeDetails"; + String AefiInvestigation_vaccineCouldHaveBeenUnSterile = "AefiInvestigation.vaccineCouldHaveBeenUnSterile"; + String AefiInvestigation_vaccineCouldHaveBeenUnSterileDetails = "AefiInvestigation.vaccineCouldHaveBeenUnSterileDetails"; + String AefiInvestigation_vaccineGivenPeriod = "AefiInvestigation.vaccineGivenPeriod"; + String AefiInvestigation_vaccineGivenPeriodDetails = "AefiInvestigation.vaccineGivenPeriodDetails"; + String AefiInvestigation_vaccineHasQualityDefect = "AefiInvestigation.vaccineHasQualityDefect"; + String AefiInvestigation_vaccineHasQualityDefectDetails = "AefiInvestigation.vaccineHasQualityDefectDetails"; + String AefiInvestigation_vaccinePhysicalConditionAbnormal = "AefiInvestigation.vaccinePhysicalConditionAbnormal"; + String AefiInvestigation_vaccinePhysicalConditionAbnormalDetails = "AefiInvestigation.vaccinePhysicalConditionAbnormalDetails"; + String AefiInvestigation_vaccinesAndDiluentsUsedRecommendedByManufacturer = "AefiInvestigation.vaccinesAndDiluentsUsedRecommendedByManufacturer"; + String AefiInvestigation_vaccineStoragePointAdditionalDetails = "AefiInvestigation.vaccineStoragePointAdditionalDetails"; + String AefiInvestigation_vaccineStorageRefrigeratorTemperatureMonitored = "AefiInvestigation.vaccineStorageRefrigeratorTemperatureMonitored"; + String AefiInvestigation_vaccineTransportationAdditionalDetails = "AefiInvestigation.vaccineTransportationAdditionalDetails"; + String aefiInvestigationClinicalDetailsOfficer = "aefiInvestigationClinicalDetailsOfficer"; + String AefiInvestigationCriteria_aefiClassification = "AefiInvestigationCriteria.aefiClassification"; + String AefiInvestigationCriteria_statusAtAefiInvestigation = "AefiInvestigationCriteria.statusAtAefiInvestigation"; + String AefiInvestigationCriteria_vaccineManufacturer = "AefiInvestigationCriteria.vaccineManufacturer"; + String AefiInvestigationCriteria_vaccineName = "AefiInvestigationCriteria.vaccineName"; + String aefiInvestigationForAdultWomen = "aefiInvestigationForAdultWomen"; + String aefiInvestigationForInfants = "aefiInvestigationForInfants"; + String AefiInvestigationIndex_aefiClassification = "AefiInvestigationIndex.aefiClassification"; + String AefiInvestigationIndex_aefiReportUuid = "AefiInvestigationIndex.aefiReportUuid"; + String AefiInvestigationIndex_ageAndBirthDate = "AefiInvestigationIndex.ageAndBirthDate"; + String AefiInvestigationIndex_disease = "AefiInvestigationIndex.disease"; + String AefiInvestigationIndex_district = "AefiInvestigationIndex.district"; + String AefiInvestigationIndex_investigationCaseId = "AefiInvestigationIndex.investigationCaseId"; + String AefiInvestigationIndex_investigationDate = "AefiInvestigationIndex.investigationDate"; + String AefiInvestigationIndex_investigationStatus = "AefiInvestigationIndex.investigationStatus"; + String AefiInvestigationIndex_personFirstName = "AefiInvestigationIndex.personFirstName"; + String AefiInvestigationIndex_personLastName = "AefiInvestigationIndex.personLastName"; + String AefiInvestigationIndex_primaryVaccine = "AefiInvestigationIndex.primaryVaccine"; + String AefiInvestigationIndex_region = "AefiInvestigationIndex.region"; + String AefiInvestigationIndex_reportDate = "AefiInvestigationIndex.reportDate"; + String AefiInvestigationIndex_sex = "AefiInvestigationIndex.sex"; + String AefiInvestigationIndex_statusOnDateOfInvestigation = "AefiInvestigationIndex.statusOnDateOfInvestigation"; + String AefiInvestigationIndex_uuid = "AefiInvestigationIndex.uuid"; + String AefiInvestigationListEntry_aefiClassification = "AefiInvestigationListEntry.aefiClassification"; + String AefiInvestigationListEntry_investigationDate = "AefiInvestigationListEntry.investigationDate"; + String AefiInvestigationListEntry_investigationStage = "AefiInvestigationListEntry.investigationStage"; + String AefiInvestigationListEntry_statusOnDateOfInvestigation = "AefiInvestigationListEntry.statusOnDateOfInvestigation"; + String aefiInvestigationMedicalCareDetailsInstruction = "aefiInvestigationMedicalCareDetailsInstruction"; + String aefiInvestigationOfThoseAffected = "aefiInvestigationOfThoseAffected"; + String aefiInvestigationReconstitutionProcedure = "aefiInvestigationReconstitutionProcedure"; + String aefiInvestigationSourceOfInformation = "aefiInvestigationSourceOfInformation"; String aefiNewAdverseEvent = "aefiNewAdverseEvent"; + String aefiNewAefiInvestigation = "aefiNewAefiInvestigation"; + String aefiNewAefiInvestigationStageTitle = "aefiNewAefiInvestigationStageTitle"; String aefiVaccinationsDiluentBatchLotNumber = "aefiVaccinationsDiluentBatchLotNumber"; String aefiVaccinationsDiluentExpiryDate = "aefiVaccinationsDiluentExpiryDate"; String aefiVaccinationsDiluentInformation = "aefiVaccinationsDiluentInformation"; @@ -2694,6 +2937,23 @@ public interface Captions { String TestReport_testLabPostalCode = "TestReport.testLabPostalCode"; String TestReport_testResult = "TestReport.testResult"; String TestReport_testType = "TestReport.testType"; + String titleAefiInvestigationBasicDetails = "titleAefiInvestigationBasicDetails"; + String titleAefiInvestigationColdChainAndTransport = "titleAefiInvestigationColdChainAndTransport"; + String titleAefiInvestigationColdChainAndTransportLastVaccineStoragePoint = "titleAefiInvestigationColdChainAndTransportLastVaccineStoragePoint"; + String titleAefiInvestigationColdChainAndTransportSubTitle = "titleAefiInvestigationColdChainAndTransportSubTitle"; + String titleAefiInvestigationColdChainAndTransportVaccineTransportation = "titleAefiInvestigationColdChainAndTransportVaccineTransportation"; + String titleAefiInvestigationCommunityInvestigation = "titleAefiInvestigationCommunityInvestigation"; + String titleAefiInvestigationCommunityInvestigationThoseAffected = "titleAefiInvestigationCommunityInvestigationThoseAffected"; + String titleAefiInvestigationFirstExaminationDetails = "titleAefiInvestigationFirstExaminationDetails"; + String titleAefiInvestigationImmunizationPractices = "titleAefiInvestigationImmunizationPractices"; + String titleAefiInvestigationImmunizationPracticesInjectionTechnique = "titleAefiInvestigationImmunizationPracticesInjectionTechnique"; + String titleAefiInvestigationImmunizationPracticesReconstitution = "titleAefiInvestigationImmunizationPracticesReconstitution"; + String titleAefiInvestigationImmunizationPracticesSubTitle = "titleAefiInvestigationImmunizationPracticesSubTitle"; + String titleAefiInvestigationImmunizationPracticesSyringesAndNeedlesUsed = "titleAefiInvestigationImmunizationPracticesSyringesAndNeedlesUsed"; + String titleAefiInvestigationInvestigationStatus = "titleAefiInvestigationInvestigationStatus"; + String titleAefiInvestigationOtherFindings = "titleAefiInvestigationOtherFindings"; + String titleAefiInvestigationRelevantPatientInformation = "titleAefiInvestigationRelevantPatientInformation"; + String titleAefiInvestigationVaccinesDetails = "titleAefiInvestigationVaccinesDetails"; String to = "to"; String total = "total"; String travelEntriesNoTravelEntriesForPerson = "travelEntriesNoTravelEntriesForPerson"; @@ -2852,6 +3112,7 @@ public interface Captions { String versionIsMissing = "versionIsMissing"; String view = "view"; String View_actions = "View.actions"; + String View_adverseeventinvestigations = "View.adverseeventinvestigations"; String View_adverseevents = "View.adverseevents"; String View_aggregatereports = "View.aggregatereports"; String View_aggregatereports_aggregatereporting = "View.aggregatereports.aggregatereporting"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index 42c2dd68365..f6c1ce4c730 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -54,6 +54,7 @@ public interface Strings { String comparedTo = "comparedTo"; String confirmationAlsoAdjustQuarantine = "confirmationAlsoAdjustQuarantine"; String confirmationArchiveAdverseEvent = "confirmationArchiveAdverseEvent"; + String confirmationArchiveAdverseEventInvestigation = "confirmationArchiveAdverseEventInvestigation"; String confirmationArchiveArea = "confirmationArchiveArea"; String confirmationArchiveAreas = "confirmationArchiveAreas"; String confirmationArchiveCampaign = "confirmationArchiveCampaign"; @@ -98,6 +99,7 @@ public interface Strings { String confirmationChangeCaseDisease = "confirmationChangeCaseDisease"; String confirmationContactSourceCaseDiscardUnsavedChanges = "confirmationContactSourceCaseDiscardUnsavedChanges"; String confirmationDearchiveAdverseEvent = "confirmationDearchiveAdverseEvent"; + String confirmationDearchiveAdverseEventInvestigation = "confirmationDearchiveAdverseEventInvestigation"; String confirmationDearchiveArea = "confirmationDearchiveArea"; String confirmationDearchiveAreas = "confirmationDearchiveAreas"; String confirmationDearchiveCampaign = "confirmationDearchiveCampaign"; @@ -220,6 +222,8 @@ public interface Strings { String entityAdditionalTest = "entityAdditionalTest"; String entityAdditionalTests = "entityAdditionalTests"; String entityAdverseEvent = "entityAdverseEvent"; + String entityAdverseEventInvestigation = "entityAdverseEventInvestigation"; + String entityAdverseEventInvestigations = "entityAdverseEventInvestigations"; String entityAdverseEvents = "entityAdverseEvents"; String entityAggregateReports = "entityAggregateReports"; String entityAreas = "entityAreas"; @@ -302,6 +306,7 @@ public interface Strings { String entityWeeklyReports = "entityWeeklyReports"; String epiWeek = "epiWeek"; String errorAccessDenied = "errorAccessDenied"; + String errorAdverseEventInvestigationNotEditable = "errorAdverseEventInvestigationNotEditable"; String errorAdverseEventNotEditable = "errorAdverseEventNotEditable"; String errorCampaignDiagramTotalsCalculationError = "errorCampaignDiagramTotalsCalculationError"; String errorCampaignNotEditable = "errorCampaignNotEditable"; @@ -421,17 +426,21 @@ public interface Strings { String headingAefiDashboardEpiCurve = "headingAefiDashboardEpiCurve"; String headingAefiDashboardMap = "headingAefiDashboardMap"; String headingAefiFirstDecisionLevel = "headingAefiFirstDecisionLevel"; + String headingAefiInvestigationFormSubHeading = "headingAefiInvestigationFormSubHeading"; + String headingAefiInvestigationSelectConcernedVaccine = "headingAefiInvestigationSelectConcernedVaccine"; String headingAefiNationalDecisionLevel = "headingAefiNationalDecisionLevel"; String headingAefiPatientsAgeAtOnset = "headingAefiPatientsAgeAtOnset"; String headingAefiPatientsIdentification = "headingAefiPatientsIdentification"; - String headingAefiPickPrimarySuspectVaccine = "headingAefiPickPrimarySuspectVaccine"; String headingAefiReportersInformation = "headingAefiReportersInformation"; String headingAefiReportingInformation = "headingAefiReportingInformation"; + String headingAefiReportInvestigations = "headingAefiReportInvestigations"; + String headingAefiSelectPrimarySuspectVaccine = "headingAefiSelectPrimarySuspectVaccine"; String headingAefiVaccinations = "headingAefiVaccinations"; String headingAllContacts = "headingAllContacts"; String headingAnimalContactDetails = "headingAnimalContactDetails"; String headingAnimalContacts = "headingAnimalContacts"; String headingArchiveAdverseEvent = "headingArchiveAdverseEvent"; + String headingArchiveAdverseEventInvestigation = "headingArchiveAdverseEventInvestigation"; String headingArchiveCampaign = "headingArchiveCampaign"; String headingArchiveCase = "headingArchiveCase"; String headingArchiveContact = "headingArchiveContact"; @@ -553,6 +562,7 @@ public interface Strings { String headingDatabaseExportFailed = "headingDatabaseExportFailed"; String headingDataImport = "headingDataImport"; String headingDearchiveAdverseEvent = "headingDearchiveAdverseEvent"; + String headingDearchiveAdverseEventInvestigation = "headingDearchiveAdverseEventInvestigation"; String headingDearchiveCampaign = "headingDearchiveCampaign"; String headingDearchiveCase = "headingDearchiveCase"; String headingDearchiveContact = "headingDearchiveContact"; @@ -950,6 +960,7 @@ public interface Strings { String infoCountryNotEditableEventParticipantsWithoutJurisdiction = "infoCountryNotEditableEventParticipantsWithoutJurisdiction"; String infoCreateEntry = "infoCreateEntry"; String infoCreateNewContactDiscardsChanges = "infoCreateNewContactDiscardsChanges"; + String infoCreateNewSampleDiscardsChangesCase = "infoCreateNewSampleDiscardsChangesCase"; String infoCreateNewSampleDiscardsChangesContact = "infoCreateNewSampleDiscardsChangesContact"; String infoCreateNewSampleDiscardsChangesEventParticipant = "infoCreateNewSampleDiscardsChangesEventParticipant"; String infoCustomExport = "infoCustomExport"; @@ -1015,6 +1026,7 @@ public interface Strings { String infoMoreDetailsAboutHospitalization = "infoMoreDetailsAboutHospitalization"; String infoNoAccessToPersonEntities = "infoNoAccessToPersonEntities"; String infoNoAdditionalTests = "infoNoAdditionalTests"; + String infoNoAefiInvestigations = "infoNoAefiInvestigations"; String infoNoCasesFoundStatistics = "infoNoCasesFoundStatistics"; String infoNoCustomizableEnumTranslations = "infoNoCustomizableEnumTranslations"; String infoNoDiseaseSelected = "infoNoDiseaseSelected"; @@ -1120,6 +1132,9 @@ public interface Strings { String messageAdditionalTestSaved = "messageAdditionalTestSaved"; String messageAdverseEventArchived = "messageAdverseEventArchived"; String messageAdverseEventDearchived = "messageAdverseEventDearchived"; + String messageAdverseEventInvestigationArchived = "messageAdverseEventInvestigationArchived"; + String messageAdverseEventInvestigationDearchived = "messageAdverseEventInvestigationDearchived"; + String messageAdverseEventInvestigationSaved = "messageAdverseEventInvestigationSaved"; String messageAdverseEventSaved = "messageAdverseEventSaved"; String messageAggregatedReportEpiWeekFilterNotFilled = "messageAggregatedReportEpiWeekFilterNotFilled"; String messageAggregateReportDelete = "messageAggregateReportDelete"; @@ -1650,6 +1665,12 @@ public interface Strings { String promptAefiDateType = "promptAefiDateType"; String promptAefiEpiWeekFrom = "promptAefiEpiWeekFrom"; String promptAefiEpiWeekTo = "promptAefiEpiWeekTo"; + String promptAefiInvestigationDateFrom = "promptAefiInvestigationDateFrom"; + String promptAefiInvestigationDateTo = "promptAefiInvestigationDateTo"; + String promptAefiInvestigationDateType = "promptAefiInvestigationDateType"; + String promptAefiInvestigationEpiWeekFrom = "promptAefiInvestigationEpiWeekFrom"; + String promptAefiInvestigationEpiWeekTo = "promptAefiInvestigationEpiWeekTo"; + String promptAefiInvestigationValidFrom = "promptAefiInvestigationValidFrom"; String promptAefiValidFrom = "promptAefiValidFrom"; String promptAllAreas = "promptAllAreas"; String promptAllCommunities = "promptAllCommunities"; @@ -1676,6 +1697,7 @@ public interface Strings { String promptDateTo = "promptDateTo"; String promptDisease = "promptDisease"; String promptDistrict = "promptDistrict"; + String promptEmail = "promptEmail"; String promptEnvironmentDateFrom = "promptEnvironmentDateFrom"; String promptEnvironmentDateTo = "promptEnvironmentDateTo"; String promptEnvironmentEpiWeekFrom = "promptEnvironmentEpiWeekFrom"; @@ -1738,6 +1760,7 @@ public interface Strings { String promptPrescriptionTextFilter = "promptPrescriptionTextFilter"; String promptRegion = "promptRegion"; String promptRelatedPersonLikeField = "promptRelatedPersonLikeField"; + String promptRemarks = "promptRemarks"; String promptSampleDashboardFilterDateType = "promptSampleDashboardFilterDateType"; String promptSampleDateFrom = "promptSampleDateFrom"; String promptSampleDateTo = "promptSampleDateTo"; @@ -1757,6 +1780,7 @@ public interface Strings { String promptTaskEpiWeekFrom = "promptTaskEpiWeekFrom"; String promptTaskEpiWeekTo = "promptTaskEpiWeekTo"; String promptTaskSearchField = "promptTaskSearchField"; + String promptTelephoneNumber = "promptTelephoneNumber"; String promptTravelEntryDateFrom = "promptTravelEntryDateFrom"; String promptTravelEntryDateTo = "promptTravelEntryDateTo"; String promptTravelEntryEpiWeekFrom = "promptTravelEntryEpiWeekFrom"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java index 68a9fbe69a4..acf04abd059 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Validations.java @@ -10,9 +10,11 @@ public interface Validations { * 1. java:S115: Violation of name convention for constants of this class is accepted: Close as false positive. */ + String aefiInvestigationWithoutPrimarySuspectVaccine = "aefiInvestigationWithoutPrimarySuspectVaccine"; + String aefiInvestigationWithoutSuspectVaccines = "aefiInvestigationWithoutSuspectVaccines"; String aefiWithoutAdverseEvents = "aefiWithoutAdverseEvents"; String aefiWithoutPrimarySuspectVaccine = "aefiWithoutPrimarySuspectVaccine"; - String aefiWithoutSuspectVaccine = "aefiWithoutSuspectVaccine"; + String aefiWithoutSuspectVaccines = "aefiWithoutSuspectVaccines"; String afterDate = "afterDate"; String afterDateSoft = "afterDateSoft"; String afterDateWithDate = "afterDateWithDate"; @@ -266,6 +268,7 @@ public interface Validations { String userNameNotUnique = "userNameNotUnique"; String uuidPatternNotMatching = "uuidPatternNotMatching"; String vaccineDosesFormat = "vaccineDosesFormat"; + String validAefiReport = "validAefiReport"; String validCaseContactOrEventParticipant = "validCaseContactOrEventParticipant"; String validCommunity = "validCommunity"; String validDateOfArrival = "validDateOfArrival"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java b/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java index 64684f3a800..806a73d1cb6 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/user/UserRight.java @@ -68,6 +68,7 @@ public enum UserRight { ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT(UserRightGroup.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION), PERSON_VIEW(UserRightGroup.PERSON), PERSON_EDIT(UserRightGroup.PERSON, UserRight._PERSON_VIEW), @@ -336,6 +337,7 @@ public enum UserRight { public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT"; public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE"; public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE"; + public static final String _ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT = "ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT"; public static final String _PERSON_VIEW = "PERSON_VIEW"; public static final String _PERSON_EDIT = "PERSON_EDIT"; public static final String _PERSON_DELETE = "PERSON_DELETE"; diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index 70623db15e1..6c5a3bfde2c 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -1022,13 +1022,19 @@ sampleDashboardShowContactSamples=Show Contact Samples sampleDashboardShowEventParticipantSamples=Show Event Participant Samples sampleDashboardShowEnvironmentSamples=Show Environment Samples #AefiDashboard -aefiDashboardAllAefi=All AEFI +aefiDashboardAllAefi=AEFI aefiDashboardSerious=Serious aefiDashboardNonSerious=Non Serious aefiDashboardSeriousAefi=Serious AEFI aefiDashboardNonSeriousAefi=Non-Serious AEFI aefiDashboardShowSeriousAefi=Show Serious AEFI aefiDashboardShowNonSeriousAefi=Show Non-Serious AEFI +aefiDashboardAllAefiInvestigation=Investigations +aefiDashboardAefiInvestigationDone=Investigation Done +aefiDashboardAefiInvestigationDiscarded=Investigation Discarded +aefiDashboardAefiClassificationRelatedToVaccination=Related to vaccine or vaccination +aefiDashboardAefiClassificationCoincidentalAdverseEvent=Coincidental adverse event +aefiDashboardAefiClassificationUndetermined=Undetermined captionDefault=Default defaultRegion=Default Region @@ -2333,11 +2339,270 @@ AefiIndex.outcome=Outcome AefiIndex.vaccinationDate=Date of vaccination AefiIndex.startDateTime=Date of AEFI onset AefiIndex.adverseEvents=Adverse events -# Advers Events Criteria +# Adverse Events Export +AefiExport.receivedAtNationalLevelDate=Date AEFI report first received at national centre +AefiExport.vaccinationFacilityName=Place of vaccination +AefiExport.vaccinationFacilityRegion=Place of vaccination region +AefiExport.vaccinationFacilityDistrict=Place of vaccination district +AefiExport.vaccinationFacilityCommunity=Place of vaccination community +AefiExport.reportingOfficerAddressCountryName=Country where this AEFI reported +AefiExport.patientAddressRegion=Location (address) region +AefiExport.patientAddressDistrict=Location (address) district +AefiExport.patientAddressCommunity=Location (address) community +AefiExport.patientAddressDetails=Location (address) +AefiExport.reportingIdNumber=AEFI reporting id number +AefiExport.worldWideId=Worldwide unique number +AefiExport.firstName=Patient identifier (First Name) +AefiExport.lastName=Patient identifier (Last Name) +AefiExport.birthDate=Date of birth +AefiExport.onsetAgeYears=Age (years) at time of onset +AefiExport.onsetAgeMonths=Age (months) at time of onset +AefiExport.onsetAgeDays=Age (days) at time of onset +AefiExport.onsetAgeGroup=Age group at onset +AefiExport.sex=Sex +AefiExport.aefiDescription=History of Event +AefiExport.primarySuspectVaccineName=Primary suspect vaccine name (generic) +AefiExport.primarySuspectVaccineOtherName=Other primary suspect vaccine name (generic) +AefiExport.primarySuspectVaccineBrand=Primary suspect vaccine name (brand) +AefiExport.primarySuspectVaccineManufacturer=Primary suspect vaccine name (manufacturer) +AefiExport.primarySuspectVaccineBatchNumber=Primary supect vaccine batch number +AefiExport.primarySuspectVaccineDose=Primary suspect vaccine dose number for this particular vaccinee +AefiExport.primarySuspectVaccineDiluentBatchNumber=Diluent batch/ lot number +AefiExport.primarySuspectVaccineVaccinationDate=Date and time of primary vaccination +AefiExport.startDateTime=Date and time of AEFI onset +AefiExport.outcome=Outcome of AEFI +AefiExport.serious=Serious +AefiExport.reportingOfficerName=Name of first reporter of AEFI +AefiExport.reportingOfficerFacilityName=Institution/location +AefiExport.reportingOfficerFacilityRegion=Reporter institution/location region +AefiExport.reportingOfficerFacilityDistrict=Reporter institution/location district +AefiExport.reportingOfficerFacilityCommunity=Reporter institution/location community +AefiExport.reportingOfficerDesignation=Position +AefiExport.reportingOfficerDepartment=Department +AefiExport.reportingOfficerEmail=E-mail Id +AefiExport.reportingOfficerPhoneNumber=Telephone number +AefiExport.reportDate=Date of report +AefiExport.nationalLevelComment=Comments (if any) +# Adverse Events Criteria AefiCriteria.aefiType=AEFI Type AefiCriteria.outcome=Outcome AefiCriteria.vaccineName=Vaccine AefiCriteria.vaccineManufacturer=Manufacturer +# Adverse Events List Entry +AefiInvestigationListEntry.investigationDate=Investigation Date +AefiInvestigationListEntry.investigationStage=Stage +AefiInvestigationListEntry.statusOnDateOfInvestigation=Status +AefiInvestigationListEntry.aefiClassification=Classification +# Adverse Events Following Immunization Investigations +AefiInvestigation.uuid=AEFI Investigation UUID +AefiInvestigation.reportDate=Date of report +AefiInvestigation.reportingUser=Reporting user +AefiInvestigation.externalId=External ID +AefiInvestigation.responsibleRegion=Responsible region +AefiInvestigation.responsibleDistrict=Responsible district +AefiInvestigation.responsibleCommunity=Responsible community +AefiInvestigation.country=Country +AefiInvestigation.investigationCaseId=Investigation Case ID +AefiInvestigation.placeOfVaccination=Place of vaccination +AefiInvestigation.placeOfVaccinationDetails=Place of vaccination details +AefiInvestigation.vaccinationActivity=Vaccination in +AefiInvestigation.vaccinationActivityDetails=Vaccination activity details +AefiInvestigation.vaccinationFacility=Facility +AefiInvestigation.vaccinationFacilityDetails=Facility name & description +AefiInvestigation.reportingOfficerName=Name of reporting officer +AefiInvestigation.reportingOfficerFacility=Facility +AefiInvestigation.reportingOfficerFacilityDetails=Facility name & description +AefiInvestigation.reportingOfficerDesignation=Designation / Position +AefiInvestigation.reportingOfficerDepartment=Department +AefiInvestigation.reportingOfficerAddress=Address of reporting officer +AefiInvestigation.reportingOfficerLandlinePhoneNumber=Telephone # landline (with code): +AefiInvestigation.reportingOfficerMobilePhoneNumber=Mobile +AefiInvestigation.reportingOfficerEmail=E-mail +AefiInvestigation.investigationDate=Date of investigation +AefiInvestigation.formCompletionDate=Date of filling this form +AefiInvestigation.investigationStage=This report is +AefiInvestigation.typeOfSite=Type of site +AefiInvestigation.typeOfSiteDetails=Type of site details +AefiInvestigation.keySymptomDateTime=Date & time of first/key symptom +AefiInvestigation.hospitalizationDate=Date of hospitalization +AefiInvestigation.reportedToHealthAuthorityDate=Date first reported to the health authority +AefiInvestigation.statusOnDateOfInvestigation=Status on the date of investigation +AefiInvestigation.deathDateTime=Date and time of death +AefiInvestigation.autopsyDone=Autopsy done? +AefiInvestigation.autopsyDate=Autopsy date +AefiInvestigation.autopsyPlannedDateTime=Autopsy planned on +AefiInvestigation.pastHistoryOfSimilarEvent=Past history of similar event +AefiInvestigation.pastHistoryOfSimilarEventDetails=Additional information +AefiInvestigation.adverseEventAfterPreviousVaccinations=Adverse event after previous vaccination(s) +AefiInvestigation.adverseEventAfterPreviousVaccinationsDetails=Additional information +AefiInvestigation.historyOfAllergyToVaccineDrugOrFood=History of allergy to vaccine, drug or food +AefiInvestigation.historyOfAllergyToVaccineDrugOrFoodDetails=Additional information +AefiInvestigation.preExistingIllnessThirtyDaysOrCongenitalDisorder=Pre-existing illness (30 days) / congenital disorder +AefiInvestigation.preExistingIllnessThirtyDaysOrCongenitalDisorderDetails=Additional information +AefiInvestigation.historyOfHospitalizationInLastThirtyDaysWithCause=History of hospitalization in last 30 days, with cause +AefiInvestigation.historyOfHospitalizationInLastThirtyDaysWithCauseDetails=Additional information +AefiInvestigation.currentlyOnConcomitantMedication=Patient currently on concomitant medication? +AefiInvestigation.currentlyOnConcomitantMedicationDetails=Additional information +AefiInvestigation.familyHistoryOfDiseaseOrAllergy=Family history of any disease (relevant to AEFI) or allergy +AefiInvestigation.familyHistoryOfDiseaseOrAllergyDetails=Additional information +AefiInvestigation.numberOfWeeksPregnant=Number of weeks pregnant +AefiInvestigation.birthTerm=The birth was +AefiInvestigation.birthWeight=Birth weight: +AefiInvestigation.deliveryProcedure=Delivery procedure +AefiInvestigation.deliveryProcedureDetails=Delivery procedure details +AefiInvestigation.seriousAefiInfoSource=Source of information +AefiInvestigation.seriousAefiInfoSourceDetails=Other source of information details +AefiInvestigation.seriousAefiVerbalAutopsyInfoSourceDetails=Please mention verbal autopsy source +AefiInvestigation.firstCaregiversName=Name of the person who first examined/treated the patient +AefiInvestigation.otherCaregiversNames=Name of other persons treating the patient +AefiInvestigation.otherSourcesWhoProvidedInfo=Other sources who provided information +AefiInvestigation.signsAndSymptomsFromTimeOfVaccination=Signs and symptoms in chronological order from the time of vaccination +AefiInvestigation.clinicalDetailsOfficerName=Name of person completing these clinical details +AefiInvestigation.clinicalDetailsOfficerPhoneNumber=Phone number +AefiInvestigation.clinicalDetailsOfficerEmail=E-mail +AefiInvestigation.clinicalDetailsOfficerDesignation=Designation +AefiInvestigation.clinicalDetailsDateTime=Date/time +AefiInvestigation.patientReceivedMedicalCare=Patient has received medical care +AefiInvestigation.patientReceivedMedicalCareDetails=Complete additional medical care information NOT AVAILABLE in existing documents +AefiInvestigation.provisionalOrFinalDiagnosis=Provisional / Final diagnosis +AefiInvestigation.patientImmunizedPeriod=When was the patient immunized? +AefiInvestigation.patientImmunizedPeriodDetails=Additional information +AefiInvestigation.vaccineGivenPeriod=In case of multidose vials, was the vaccine given +AefiInvestigation.vaccineGivenPeriodDetails=Additional information +AefiInvestigation.errorPrescribingVaccine=Was there an error in prescribing or non-adherence to recommendations for use of this vaccine? +AefiInvestigation.errorPrescribingVaccineDetails=Additional information +AefiInvestigation.vaccineCouldHaveBeenUnSterile=Based on your investigation, do you feel that the vaccine (ingredients) administered could have been unsterile? +AefiInvestigation.vaccineCouldHaveBeenUnSterileDetails=Additional information +AefiInvestigation.vaccinePhysicalConditionAbnormal=Based on your investigation, do you feel that the vaccine's physical condition (e.g. colour, turbidity, foreign substances etc.) was abnormal at the time of administration? +AefiInvestigation.vaccinePhysicalConditionAbnormalDetails=Additional information +AefiInvestigation.errorInVaccineReconstitution=Based on your investigation, do you feel that there was an error in vaccine reconstitution/preparation by the vaccinator (e.g. wrong product, wrong diluent, improper mixing, improper syringe filling etc.)? +AefiInvestigation.errorInVaccineReconstitutionDetails=Additional information +AefiInvestigation.errorInVaccineHandling=Based on your investigation, do you feel that there was an error in vaccine handling (e.g. break in cold chain during transport, storage and/or immunization session etc.)? +AefiInvestigation.errorInVaccineHandlingDetails=Additional information +AefiInvestigation.vaccineAdministeredIncorrectly=Based on your investigation, do you feel that the vaccine was administered incorrectly (e.g. wrong dose, site or route of administration, wrong needle size, not following good injection practice etc.)? +AefiInvestigation.vaccineAdministeredIncorrectlyDetails=Additional information +AefiInvestigation.numberImmunizedFromConcernedVaccineVial=Number immunized from the concerned vaccine vial/ampoule +AefiInvestigation.numberImmunizedWithConcernedVaccineInSameSession=Number immunized with the concerned vaccine in the same session +AefiInvestigation.numberImmunizedConcernedVaccineSameBatchNumberOtherLocations=Number immunized with the concerned vaccine having the same batch number in other locations +AefiInvestigation.numberImmunizedConcernedVaccineSameBatchNumberLocationDetails=Specify locations +AefiInvestigation.vaccineHasQualityDefect=Could the vaccine given to this patient have a quality defect or is substandard or falsified? +AefiInvestigation.vaccineHasQualityDefectDetails=Additional information +AefiInvestigation.eventIsAStressResponseRelatedToImmunization=Could this event be a stress response related to immunization (e.g. acute stress response, vasovagal reaction, hyperventilation, dissociative neurological symptom reaction etc.)? +AefiInvestigation.eventIsAStressResponseRelatedToImmunizationDetails=Additional information +AefiInvestigation.caseIsPartOfACluster=Is this case a part of a cluster? +AefiInvestigation.caseIsPartOfAClusterDetails=Additional information +AefiInvestigation.numberOfCasesDetectedInCluster=How many other cases have been detected in the cluster? +AefiInvestigation.allCasesInClusterReceivedVaccineFromSameVial=Did all the cases in the cluster receive vaccine from the same vial? +AefiInvestigation.allCasesInClusterReceivedVaccineFromSameVialDetails=Additional information +AefiInvestigation.numberOfVialsUsedInCluster=Number of vials used in the cluster +AefiInvestigation.numberOfVialsUsedInClusterDetails=Number of vials used in the cluster details +AefiInvestigation.adSyringesUsedForImmunization=Are AD syringes used for immunization? +AefiInvestigation.typeOfSyringesUsed=Type of syringes used +AefiInvestigation.typeOfSyringesUsedDetails=Additional information +AefiInvestigation.syringesUsedAdditionalDetails=Specific key findings/additional observations and comments +AefiInvestigation.sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine=Same reconstitution syringe used for multiple vials of same vaccine? +AefiInvestigation.sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines=Same reconstitution syringe used for reconstituting different vaccines? +AefiInvestigation.sameReconstitutionSyringeForEachVaccineVial=Separate reconstitution syringe for each vaccine vial? +AefiInvestigation.sameReconstitutionSyringeForEachVaccination=Separate reconstitution syringe for each vaccination? +AefiInvestigation.vaccinesAndDiluentsUsedRecommendedByManufacturer=Are the vaccines and diluents used the same as those recommended by the manufacturer? +AefiInvestigation.reconstitutionAdditionalDetails=Specific key findings/additional observations and comments +AefiInvestigation.correctDoseOrRoute=Correct dose and route? +AefiInvestigation.timeOfReconstitutionMentionedOnTheVial=Time of reconstitution mentioned on the vial? (in case of freeze dried vaccines) +AefiInvestigation.nonTouchTechniqueFollowed=Non-touch technique followed? +AefiInvestigation.contraIndicationScreenedPriorToVaccination=Contraindications screened prior to vaccination? +AefiInvestigation.numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays=How many AEFI were reported from the centre that distributed the vaccine in the last 30 days? +AefiInvestigation.trainingReceivedByVaccinator=Training received by the vaccinator? +AefiInvestigation.lastTrainingReceivedByVaccinatorDate=Date of last training +AefiInvestigation.injectionTechniqueAdditionalDetails=Specific key findings/ additional observations and comments? +AefiInvestigation.vaccineStorageRefrigeratorTemperatureMonitored=Is the temperature of the vaccine storage refrigerator monitored? +AefiInvestigation.anyStorageTemperatureDeviationOutsideTwoToEightDegrees=Was there any deviation outside of 2-8 (degrees C) after the vaccine was placed inside? +AefiInvestigation.storageTemperatureMonitoringAdditionalDetails=Provide details of monitoring separately +AefiInvestigation.correctProcedureForStorageFollowed=Was the correct procedure for storing vaccines, diluents and syringes followed? +AefiInvestigation.anyOtherItemInRefrigerator=Was any other item (other than EPI vaccines and diluents) in the refrigerator or freezer? +AefiInvestigation.partiallyUsedReconstitutedVaccinesInRefrigerator=Were any partially used reconstituted vaccines in the refrigerator? +AefiInvestigation.unusableVaccinesInRefrigerator=Were any unusable vaccines (expired, no label, VVM at stages 3 or 4, frozen) in the refrigerator? +AefiInvestigation.unusableDiluentsInStore=Were any unusable diluents (expired, manufacturer not matched, cracked, dirty ampoule) in the store? +AefiInvestigation.vaccineStoragePointAdditionalDetails=Specific key findings/additional observations and comments: +AefiInvestigation.vaccineCarrierType=Type of vaccine carrier used +AefiInvestigation.vaccineCarrierTypeDetails=Additional information +AefiInvestigation.vaccineCarrierSentToSiteOnSameDateAsVaccination=Was the vaccine carrier sent to the site on the same day as vaccination? +AefiInvestigation.vaccineCarrierReturnedFromSiteOnSameDateAsVaccination=Was the vaccine carrier returned from the site on the same day as vaccination? +AefiInvestigation.conditionedIcepackUsed=Was a conditioned ice-pack used? +AefiInvestigation.vaccineTransportationAdditionalDetails=Specific key findings/additional observations and comments +AefiInvestigation.similarEventsReportedSamePeriodAndLocality=Were any similar events reported within a time period similar to when the adverse event occurred and in the same locality? +AefiInvestigation.similarEventsReportedSamePeriodAndLocalityDetails=Additional information +AefiInvestigation.numberOfSimilarEventsReportedSamePeriodAndLocality=How many events/episodes? +AefiInvestigation.numberOfThoseAffectedVaccinated=Vaccinated +AefiInvestigation.numberOfThoseAffectedNotVaccinated=Not vaccinated +AefiInvestigation.numberOfThoseAffectedVaccinatedUnknown=Unknown +AefiInvestigation.communityInvestigationAdditionalDetails=Other comments: +AefiInvestigation.otherInvestigationFindings=Other findings/observations/comments +AefiInvestigation.investigationStatus=Investigation status +AefiInvestigation.investigationStatusDetails=Investigation status details +AefiInvestigation.aefiClassification=Classification of AEFI +AefiInvestigation.aefiClassificationSubType=Reason for classification +AefiInvestigation.aefiClassificationDetails=Classification of AEFI details +AefiInvestigation.causality=Causality +AefiInvestigation.causalityDetails=Causality details +AefiInvestigation.investigationCompletionDate=Date investigation completed +AefiInvestigation.deletionReason=Reason for deletion +AefiInvestigation.otherDeletionReason=Reason for deletion details +AefiInvestigation.creationDate=Creation date +AefiInvestigation.changeDate=Date of last change +aefiInvestigationForAdultWomen=For adult women +aefiInvestigationForInfants=For infants +aefiInvestigationSourceOfInformation=Source of information +aefiInvestigationClinicalDetailsOfficer=Clinical Details Officer +aefiInvestigationMedicalCareDetailsInstruction=**Instructions ? Attach copies of ALL available documents (including case sheet, discharge summary, case notes, laboratory reports and autopsy reports) and then complete additional information NOT AVAILABLE in existing documents, +aefiInvestigationReconstitutionProcedure=Reconstitution procedure +aefiInvestigationOfThoseAffected=Of those effected, how many are +aefiAefiInvestigationList=Adverse Event Investigations +aefiNewAefiInvestigation = New Investigation +aefiAefiInvestigationDataView=Adverse Event Investigation +aefiNewAefiInvestigationStageTitle=Investigation +aefiActiveInvestigations = Active AEFI investigations +aefiArchivedInvestigations = Archived AEFI investigations +aefiAllActiveAndArchivedInvestigations = All active and archived AEFI investigations +aefiDeletedInvestigations = Deleted AEFI investigations +titleAefiInvestigationBasicDetails=Section A: Basic details +titleAefiInvestigationRelevantPatientInformation=Section B: Relevant patient information prior to immunization +titleAefiInvestigationFirstExaminationDetails=Section C: Details of first examination** of serious AEFI case +titleAefiInvestigationVaccinesDetails=Section D: Details of vaccines provided at the site linked to AEFI on the corresponding day +titleAefiInvestigationImmunizationPractices=Section E: Immunization practices at the place(s) where concerned vaccine was used +titleAefiInvestigationImmunizationPracticesSubTitle=(Complete this section by asking and/or observing practice) +titleAefiInvestigationImmunizationPracticesSyringesAndNeedlesUsed=Syringes and needles used +titleAefiInvestigationImmunizationPracticesReconstitution=Reconstitution: (complete only if applicable) +titleAefiInvestigationImmunizationPracticesInjectionTechnique=Injection technique in vaccinator(s): (Observe another session in the same locality ? same or different place) +titleAefiInvestigationColdChainAndTransport=Section F: Cold chain and transport +titleAefiInvestigationColdChainAndTransportSubTitle=(Complete this section by asking and/or observing practice) +titleAefiInvestigationColdChainAndTransportLastVaccineStoragePoint=Last vaccine storage point +titleAefiInvestigationColdChainAndTransportVaccineTransportation=Vaccine transportation +titleAefiInvestigationCommunityInvestigation=Section G: Community investigation (Please visit locality and interview parents/others) +titleAefiInvestigationCommunityInvestigationThoseAffected=Of those effected, how many are +titleAefiInvestigationOtherFindings=Section H: Other findings/observations/comments +titleAefiInvestigationInvestigationStatus=Section I: Investigation Status +# Adverse Events Investigation Criteria +AefiInvestigationCriteria.statusAtAefiInvestigation=Patient status +AefiInvestigationCriteria.aefiClassification=Classification +AefiInvestigationCriteria.vaccineName=Vaccine +AefiInvestigationCriteria.vaccineManufacturer=Manufacturer +# Adverse Events Investigation Index +AefiInvestigationIndex.uuid=Investigation ID +AefiInvestigationIndex.aefiReportUuid=AEFI ID +AefiInvestigationIndex.investigationCaseId=Case ID +AefiInvestigationIndex.personFirstName=First Name +AefiInvestigationIndex.personLastName=Last Name +AefiInvestigationIndex.reportDate=Date of report +AefiInvestigationIndex.investigationDate=Date of investigation +AefiInvestigationIndex.disease=Disease +AefiInvestigationIndex.ageAndBirthDate=Age and birth date +AefiInvestigationIndex.sex=Sex +AefiInvestigationIndex.region=Region +AefiInvestigationIndex.district=District +AefiInvestigationIndex.primaryVaccine=Concerned Vaccine +AefiInvestigationIndex.statusOnDateOfInvestigation=Patient Status +AefiInvestigationIndex.investigationStatus=Status +AefiInvestigationIndex.aefiClassification=Classification # Statistics statisticsAddFilter=Add filter statisticsAttribute=Attribute @@ -2840,6 +3105,7 @@ View.samples.sub= View.travelEntries=Travel Entries Directory View.immunizations=Immunization Directory View.adverseevents=Adverse Events Directory +View.adverseeventinvestigations=AEFI Investigations Directory View.statistics=Statistics View.statistics.database-export=Database export View.tasks=Task Management diff --git a/sormas-api/src/main/resources/enum.properties b/sormas-api/src/main/resources/enum.properties index 5880c9e0f00..4bf5a3190d5 100644 --- a/sormas-api/src/main/resources/enum.properties +++ b/sormas-api/src/main/resources/enum.properties @@ -1556,6 +1556,7 @@ UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE = Create new adverse even UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT = Edit existing adverse event following immunization UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE = Delete adverse events following immunization from the system UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE = Archive adverse events following immunization +UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT = Export adverse events following immunization UserRight.PERSON_EXPORT = Export persons UserRight.CONTACT_MERGE = Merge contacts UserRight.EVENTGROUP_CREATE = Create new event groups @@ -1779,6 +1780,7 @@ UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE = Able to create new UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT = Able to edit existing adverse events following immunization UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE = Able to delete adverse events following immunization from the system UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE = Able to archive adverse events following immunization +UserRight.Desc.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT = Able to export adverse events following immunization UserRight.Desc.PERSON_EXPORT = Able to export persons UserRight.Desc.CONTACT_MERGE = Able to merge contacts UserRight.Desc.EVENTGROUP_CREATE = Able to create new event groups @@ -2073,16 +2075,24 @@ Vaccine.VALNEVA=inactivated (Valneva) Vaccine.NUVAXOVID=protein-based, recombinant (Novavax) Vaccine.MRNA_BIVALENT_BA_4_5_BIONTECH_PFIZER=mRNA/bivalent BA.4/5 (BioNTech/Pfizer) Vaccine.MRNA_BIVALENT_BA_4_5_MODERNA=mRNA/bivalent BA.4/5 (Moderna) +Vaccine.MenABCWY=PENBRAYA +Vaccine.ACAM2000=ACAM2000 +Vaccine.LC_16=LC-16 +Vaccine.MVA_BN=JYNNEOS Vaccine.UNKNOWN=Unknown Vaccine.OTHER=Other # VaccineManufacturer VaccineManufacturer.BIONTECH_PFIZER=BioNTech/Pfizer +VaccineManufacturer.PFIZER=Pfizer +VaccineManufacturer.BAVARIAN_NORDIC=Bavarian Nordic VaccineManufacturer.MODERNA=Moderna VaccineManufacturer.ASTRA_ZENECA=AstraZeneca VaccineManufacturer.JOHNSON_JOHNSON=Johnson & Johnson +VaccineManufacturer.KM_BIOLOGICS=KM Biologics VaccineManufacturer.NOVAVAX=Novavax VaccineManufacturer.SANOFI_GSK=Sanofi-GSK +VaccineManufacturer.SANOFI_PASTEUR_BIOLOGICS=Sanofi Pasteur Biologics VaccineManufacturer.ASTRA_ZENECA_BIONTECH_PFIZER=AstraZeneca & BioNTech/Pfizer VaccineManufacturer.ASTRA_ZENECA_MODERNA=AstraZeneca & Moderna VaccineManufacturer.VALNEVA=Valneva @@ -2384,4 +2394,94 @@ AefiDateType.VACCINATION_DATE=Date of vaccination # AefiDashboardFilterDateType AefiDashboardFilterDateType.REPORT_DATE=Date of report -AefiDashboardFilterDateType.START_DATE=Date of onset \ No newline at end of file +AefiDashboardFilterDateType.START_DATE=Date of onset + +# AefiInvestigationDateType +AefiInvestigationDateType.REPORT_DATE=Date of report +AefiInvestigationDateType.INVESTIGATION_DATE=Date of investigation +AefiInvestigationDateType.VACCINATION_DATE=Date of vaccination + +# PlaceOfVaccination +PlaceOfVaccination.GOVERNMENT_HEALTH_FACILITY=Government health facility +PlaceOfVaccination.PRIVATE_HEALTH_FACILITY=Private health facility +PlaceOfVaccination.OTHER=Other + +# VaccinationActivity +VaccinationActivity.CAMPAIGN=Campaign +VaccinationActivity.ROUTINE=Routine +VaccinationActivity.OTHER=Other + +# AefiInvestigationStage +AefiInvestigationStage.FIRST=First +AefiInvestigationStage.INTERIM=Interim +AefiInvestigationStage.FINAL=Final + +# VaccinationSite +VaccinationSite.FIXED=Fixed +VaccinationSite.MOBILE=Mobile +VaccinationSite.OUTREACH=Outreach +VaccinationSite.OTHER=Other + +# PatientStatusAtAefiInvestigation +PatientStatusAtAefiInvestigation.DIED=Died +PatientStatusAtAefiInvestigation.DISABLED=Disabled +PatientStatusAtAefiInvestigation.RECOVERED=Recovering +PatientStatusAtAefiInvestigation.RECOVERED_COMPLETELY=Recovered completely +PatientStatusAtAefiInvestigation.UNKNOWN=Unknown + +# BirthTerm +BirthTerm.FULL_TERM=Full-term +BirthTerm.PRE_TERM=Pre-term +BirthTerm.POST_TERM=Post-term + +# DeliveryProcedure +DeliveryProcedure.NORMAL=Normal +DeliveryProcedure.CAESAREAN=Caesarean +DeliveryProcedure.ASSISTED=Assisted (forceps, vacuum etc.) +DeliveryProcedure.WITH_COMPLICATION=with complication + +# SeriousAefiInfoSource +SeriousAefiInfoSource.EXAMINATION=Examination by the investigator +SeriousAefiInfoSource.DOCUMENTS=Documents +SeriousAefiInfoSource.VERBAL_AUTOPSY=Verbal autopsy +SeriousAefiInfoSource.OTHER=Other + +# AefiImmunizationPeriod +AefiImmunizationPeriod.WITHIN_FIRST_VACCINATIONS=Within the first vaccinations of the session +AefiImmunizationPeriod.WITHIN_LAST_VACCINATIONS=Within the last vaccinations of the session +AefiImmunizationPeriod.UNKNOWN=Unknown + +# AefiVaccinationPeriod +AefiVaccinationPeriod.WITHIN_FIRST_FEW_DOSES=Within the first few doses of the vial administered +AefiVaccinationPeriod.WITHIN_LAST_DOSES=Within the last doses of the vial administered +AefiVaccinationPeriod.UNKNOWN=Unknown + +# SyringeType +SyringeType.GLASS=Glass +SyringeType.DISPOSABLE=Disposable +SyringeType.RECYCLED_DISPOSABLE=Recycled disposable +SyringeType.OTHER=Other + +# VaccineCarrier +VaccineCarrier.SHORT_RANGE=Short range +VaccineCarrier.LONG_RANGE=Long range +VaccineCarrier.OTHER=Other + +# AefiInvestigationStatus +AefiInvestigationStatus.DONE=Done +AefiInvestigationStatus.DISCARDED=Discarded + +# AefiCausality +AefiCausality.CONFIRMED=Confirmed +AefiCausality.INCONCLUSIVE=Inconclusive + +# AefiClassification +AefiClassification.RELATED_TO_VACCINE_OR_VACCINATION=Related to vaccine or vaccination +AefiClassification.COINCIDENTAL_ADVERSE_EVENT=Coincidental adverse event +AefiClassification.UNDETERMINED=Undetermined + +# AefiClassificationSubType +AefiClassificationSubType.VACCINE_PRODUCT_RELATED=Vaccine product related +AefiClassificationSubType.VACCINE_QUALITY_DEFECT_RELATED=Vaccine quality defect related +AefiClassificationSubType.IMMUNIZATION_ERROR_RELATED=Immunization error related +AefiClassificationSubType.IMMUNIZATION_ANXIETY_RELATED=Immunization anxiety related \ No newline at end of file diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index 845fd0c6f1f..ea0e48ed95c 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -137,6 +137,7 @@ confirmationArchiveEvents = Are you sure you want to archive all %d selected eve confirmationArchiveEventParticipant = Are you sure you want to archive this event participant? This will not remove it from the system or any statistics, but only hide it from the list of event participants. confirmationArchiveImmunization = Are you sure you want to archive this immunization? This will not remove it from the system or any statistics, but only hide it from the normal immunization directory. confirmationArchiveAdverseEvent = Are you sure you want to archive this adverse event? This will not remove it from the system or any statistics, but only hide it from the normal adverse events directory. +confirmationArchiveAdverseEventInvestigation = Are you sure you want to archive this adverse event investigation? This will not remove it from the system or any statistics, but only hide it from the normal adverse events investigation directory. confirmationArchiveTask = Are you sure you want to archive this task? This will not remove it from the system or any statistics, but only hide it from the normal task management. confirmationArchiveTasks = Are you sure you want to archive all %d selected tasks? confirmationArchiveTravelEntry = Are you sure you want to archive this travel entry? This will not remove it from the system or any statistics, but only hide it from the normal travel entry directory. @@ -219,6 +220,7 @@ confirmationArchiveFacilities = Are you sure you want to archive all %d selected confirmationDearchiveFacilities = Are you sure you want to de-archive all %d selected facilities? confirmationDearchiveImmunization = Are you sure you want to de-archive this immunization? This will make it appear in the normal immunization directory again. confirmationDearchiveAdverseEvent = Are you sure you want to de-archive this adverse event? This will make it appear in the normal adverse events directory again. +confirmationDearchiveAdverseEventInvestigation = Are you sure you want to de-archive this adverse event investigation? This will make it appear in the normal adverse events investigation directory again. confirmationArchiveLaboratories = Are you sure you want to archive all %d selected laboratories? confirmationDearchiveLaboratories = Are you sure you want to de-archive all %d selected laboratories? confirmationDearchiveTask = Are you sure you want to de-archive this task? This will make it appear in the normal task directory again. @@ -347,6 +349,8 @@ entityCampaignFormMeta = Campaign form meta entityCampaignFormData = Campaign form data entityAdverseEvent = Adverse event entityAdverseEvents = Adverse events +entityAdverseEventInvestigation = Adverse event investigation +entityAdverseEventInvestigations = Adverse event investigations # Error Messages errorAccessDenied=You do not have the required rights to view this page. @@ -435,6 +439,7 @@ errorEnvironmentSampleNoReceivalRight = You do not have the necessary user right errorSendingExternalEmail = Email could not be sent. Please contact an admin and notify them about this problem. errorExternalEmailAttachmentCannotEncrypt=Can't send email with attachments. The person has no national health id or primary phone number to send the password to or the SMS service is not set up in the system. errorAdverseEventNotEditable = This adverse event is not editable anymore +errorAdverseEventInvestigationNotEditable = This adverse event investigation is not editable anymore errorExternalEmailMissingPersonEmailAddress=This person does not have an email address # headings @@ -452,6 +457,7 @@ headingArchiveEventParticipant = Archive event participant headingArchiveEventGroup = Archive event group headingArchiveImmunization = Archive immunization headingArchiveAdverseEvent = Archive adverse event +headingArchiveAdverseEventInvestigation = Archive adverse event investigation headingArchiveTravelEntry = Archive travel entry headingCampaignBasics = Campaign basics headingCampaignData = Campaign data @@ -527,6 +533,8 @@ headingDearchiveEvent = De-Archive event headingDearchiveEventParticipant = De-Archive event participant headingDearchiveEventGroup = De-Archive event group headingDearchiveImmunization = De-Archive immunization +headingDearchiveAdverseEvent = De-Archive adverse event +headingDearchiveAdverseEventInvestigation = De-Archive adverse event investigation headingDearchiveTravelEntry = De-Archive travel entry headingDefineOutbreakDistricts = Define which districts currently are affected by an outbreak. headingDeleteConfirmation = Confirm deletion @@ -884,6 +892,8 @@ headingExternalEmailSend=Send email headingExternalEmailDetails=Email details headingCustomizableEnumConfigurationInfo=Customizable enum configuration headingImmunizationAdverseEvents=Adverse events +headingAefiDashboardEpiCurve=Adverse Events Type Chart +headingAefiDashboardMap=Adverse Events Status Map headingAefiReportingInformation=Reporting information headingAefiPatientsIdentification=Patients identification headingAefiPatientsAgeAtOnset=Age at onset @@ -892,7 +902,10 @@ headingAefiAdverseEvents=Adverse events headingAefiFirstDecisionLevel=First decision Level headingAefiNationalDecisionLevel=National Decision Level headingAefiReportersInformation=Reporter's information -headingAefiPickPrimarySuspectVaccine=Select primary suspect vaccine +headingAefiSelectPrimarySuspectVaccine=Select primary suspect vaccine +headingAefiReportInvestigations=Investigations +headingAefiInvestigationSelectConcernedVaccine=Select concerned vaccine +headingAefiInvestigationFormSubHeading=(Only for Serious Adverse Events Following Immunization ? Death / Disability / Hospitalization / Cluster) # Info texts infoActivityAsCaseInvestigation = Please document ALL relevant activities after infection: @@ -1098,6 +1111,8 @@ infoCustomizableEnumConfigurationInfo = Customizable enums are value sets that c infoNoImmunizationAdverseEvents = No adverse events have been created for this immunization infoAefiSelectPrimarySuspectVaccine = The list below contains all vaccinations of the immunization. Please select the suspect vaccination related to this adverse event. infoArchivedAefiEntries = Adverse event entries are automatically archived after %d days without changes to the data. +infoNoAefiInvestigations = No investigations have been created for this adverse event +infoHeadingAefiDashboardMap=Adverse events are shown using the GPS coordinate of the facility or person's home address. # Messages messageActionOutsideJurisdictionDeletionDenied = The action outside user's jurisdiction cannot be deleted @@ -1281,6 +1296,8 @@ messageImmunizationSaved = Immunization data saved. messageImmunizationSavedVaccinationStatusUpdated = Immunization data saved. The vaccination status of matching cases, contacts, and event participants of the immunization person has been updated to vaccinated. messageAdverseEventArchived = Adverse event has been archived messageAdverseEventDearchived = Adverse event has been de-archived +messageAdverseEventInvestigationArchived = Adverse event investigation has been archived +messageAdverseEventInvestigationDearchived = Adverse event investigation has been de-archived messageImportCanceled = Import canceled!
The import has been canceled. All already processed rows have been successfully imported. You can now close this window. messageImportCanceledErrors = Import canceled!
The import has been canceled. Some of the already processed rows could not be imported due to malformed data.
Please close this window and download the error report. messageImportError = Could not import file. @@ -1547,6 +1564,7 @@ messageCustomizableEnumValueSaved = Customizable enum value saved messageExternalEmailAttachmentPassword=Please use this password to open the documents sent to you via email from SORMAS: %s messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messageAdverseEventSaved=Adverse event saved +messageAdverseEventInvestigationSaved=Adverse event investigation saved messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. @@ -1715,12 +1733,20 @@ promptAllCommunities=All communities promptExternalIdExternalSurveillanceTool=Will adopt external reporting tool GUID promptExternalJournalForceDeletion=Do you want to force the cancellation in SORMAS? This would mark the person as deleted from the external journal in SORMAS, while there is a high probability of personal data still remaining in the external journal. promptPersonDuplicateSearchIdExternalId=First Name, Last Name, ID, External ID, External token +promptAefiDashboardFilterDateType=AEFI reference date promptAefiDateType = Aefi reference date promptAefiValidFrom = New adverse events valid from... promptAefiDateFrom = New adverse events from... promptAefiDateTo = ... to promptAefiEpiWeekFrom = New adverse events from epi week... promptAefiEpiWeekTo = ... to epi week +promptAefiInvestigationDateType = Aefi investigation reference date +promptAefiInvestigationValidFrom = New Aefi investigations valid from... +promptAefiInvestigationDateFrom = New Aefi investigations from... +promptAefiInvestigationDateTo = ... to +promptAefiInvestigationEpiWeekFrom = New Aefi investigations from epi week... +promptAefiInvestigationEpiWeekTo = ... to epi week +promptRemarks = Remarks #DiseaseNetworkDiagram DiseaseNetworkDiagram.Classification.HEALTHY = Healthy DiseaseNetworkDiagram.heading = Disease network diagram @@ -1833,11 +1859,6 @@ headingSampleDashboardMap=Sample Status Map infoHeadingSampleDashboardMap=Samples are shown using the GPS coordinate of the person's home address. infoHeadingEnvironmentSampleDashboardMap=Environment samples are shown using the GPS coordinates of those samples or, if not available, their associated environment. -promptAefiDashboardFilterDateType=AEFI reference date -headingAefiDashboardEpiCurve=Adverse Events Type Chart -headingAefiDashboardMap=Adverse Events Status Map -infoHeadingAefiDashboardMap=Adverse events are shown using the GPS coordinate of the facility or person's home address. - headingSpecailCaseAccess = Grant special access headingCreateSpecailCaseAccess = Create new special access headingEditSpecailCaseAccess = Edit special access diff --git a/sormas-api/src/main/resources/validations.properties b/sormas-api/src/main/resources/validations.properties index ac49b31d7e3..f1030a60038 100644 --- a/sormas-api/src/main/resources/validations.properties +++ b/sormas-api/src/main/resources/validations.properties @@ -297,9 +297,12 @@ customizableEnumValueEmptyTranslations = Please select languages and enter capti customizableEnumValueDuplicateLanguage = Please only add one translation per language. customizableEnumValueDuplicateValue = The value %s already exists for data type %s. Enum values have to be unique. attachedDocumentNotRelatedToEntity=The attached document is not related to the entity. -aefiWithoutSuspectVaccine=You have to select at least one suspect vaccine +aefiWithoutSuspectVaccines=You have to select at least one suspect vaccine aefiWithoutPrimarySuspectVaccine=You have to select the primary suspect vaccine aefiWithoutAdverseEvents=You have to select at least one adverse event +validAefiReport=You have to link a valid adverse event report +aefiInvestigationWithoutSuspectVaccines=You have to select at least one suspect vaccine +aefiInvestigationWithoutPrimarySuspectVaccine=You have to select the concerned suspect vaccine invalidNationalHealthId=This value does not seem to be a correct national health ID invalidSelfReportType = Invalid SelfReport type selfReportAlreadyProcessedError = The self report was processed by another user in the meantime. diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java index 0d91053215e..672edfd0f34 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiFacadeEjb.java @@ -16,6 +16,7 @@ package de.symeda.sormas.backend.adverseeventsfollowingimmunization; import java.util.ArrayList; +import java.util.Collection; import java.util.Date; import java.util.List; @@ -34,6 +35,7 @@ import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiExportDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiFacade; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListCriteria; @@ -173,6 +175,15 @@ public List getIndexList(AefiCriteria criteria, Integer first, Int return resultsList; } + @Override + @RightsAllowed(UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT) + public List getExportList(AefiCriteria criteria, Collection selectedRows, int first, int max) { + List resultsList = service.getExportList(criteria, selectedRows, first, max); + Pseudonymizer pseudonymizer = createGenericPlaceholderPseudonymizer(); + pseudonymizer.pseudonymizeDtoCollection(AefiExportDto.class, resultsList, AefiExportDto::getInJurisdiction, null); + return resultsList; + } + @Override public List getEntriesList(AefiListCriteria criteria, Integer first, Integer max) { Long immunizationId = immunizationService.getIdByUuid(criteria.getImmunization().getUuid()); @@ -295,13 +306,14 @@ protected Aefi fillOrBuildEntity(@NotNull AefiDto source, Aefi target, boolean c target.setAgeGroup(source.getAgeGroup()); target.setHealthFacility(facilityService.getByReferenceDto(source.getHealthFacility())); target.setHealthFacilityDetails(source.getHealthFacilityDetails()); - target.setReporterName(source.getReporterName()); - target.setReporterInstitution(facilityService.getByReferenceDto(source.getReporterInstitution())); - target.setReporterDesignation(source.getReporterDesignation()); - target.setReporterDepartment(source.getReporterDepartment()); - target.setReporterAddress(locationFacade.fillOrBuildEntity(source.getReporterAddress(), target.getReporterAddress(), checkChangeDate)); - target.setReporterPhone(source.getReporterPhone()); - target.setReporterEmail(source.getReporterEmail()); + target.setReportingOfficerName(source.getReportingOfficerName()); + target.setReportingOfficerFacility(facilityService.getByReferenceDto(source.getReportingOfficerFacility())); + target.setReportingOfficerDesignation(source.getReportingOfficerDesignation()); + target.setReportingOfficerDepartment(source.getReportingOfficerDepartment()); + target.setReportingOfficerAddress( + locationFacade.fillOrBuildEntity(source.getReportingOfficerAddress(), target.getReportingOfficerAddress(), checkChangeDate)); + target.setReportingOfficerPhoneNumber(source.getReportingOfficerPhoneNumber()); + target.setReportingOfficerEmail(source.getReportingOfficerEmail()); target.setTodaysDate(source.getTodaysDate()); target.setStartDateTime(source.getStartDateTime()); target.setAefiDescription(source.getAefiDescription()); @@ -369,13 +381,13 @@ public static AefiDto toAefiDto(Aefi entity) { dto.setAgeGroup(entity.getAgeGroup()); dto.setHealthFacility(FacilityFacadeEjb.toReferenceDto(entity.getHealthFacility())); dto.setHealthFacilityDetails(entity.getHealthFacilityDetails()); - dto.setReporterName(entity.getReporterName()); - dto.setReporterInstitution(FacilityFacadeEjb.toReferenceDto(entity.getReporterInstitution())); - dto.setReporterDesignation(entity.getReporterDesignation()); - dto.setReporterDepartment(entity.getReporterDepartment()); - dto.setReporterAddress(LocationFacadeEjb.toDto(entity.getReporterAddress())); - dto.setReporterPhone(entity.getReporterPhone()); - dto.setReporterEmail(entity.getReporterEmail()); + dto.setReportingOfficerName(entity.getReportingOfficerName()); + dto.setReportingOfficerFacility(FacilityFacadeEjb.toReferenceDto(entity.getReportingOfficerFacility())); + dto.setReportingOfficerDesignation(entity.getReportingOfficerDesignation()); + dto.setReportingOfficerDepartment(entity.getReportingOfficerDepartment()); + dto.setReportingOfficerAddress(LocationFacadeEjb.toDto(entity.getReportingOfficerAddress())); + dto.setReportingOfficerPhoneNumber(entity.getReportingOfficerPhoneNumber()); + dto.setReportingOfficerEmail(entity.getReportingOfficerEmail()); dto.setTodaysDate(entity.getTodaysDate()); dto.setStartDateTime(entity.getStartDateTime()); dto.setAefiDescription(entity.getAefiDescription()); @@ -410,7 +422,7 @@ public static AefiReferenceDto toReferenceDto(Aefi entity) { return null; } - return new AefiReferenceDto(entity.getUuid(), "", ""); + return new AefiReferenceDto(entity.getUuid(), ""); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationFacadeEjb.java new file mode 100644 index 00000000000..8cef80f6506 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationFacadeEjb.java @@ -0,0 +1,641 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import java.util.ArrayList; +import java.util.Date; +import java.util.List; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; +import javax.inject.Inject; +import javax.validation.Valid; +import javax.validation.constraints.NotNull; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationFacade; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationIndexDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationReferenceDto; +import de.symeda.sormas.api.common.DeletableEntityType; +import de.symeda.sormas.api.common.DeletionDetails; +import de.symeda.sormas.api.common.progress.ProcessedEntity; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.api.utils.AccessDeniedException; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.api.utils.SortProperty; +import de.symeda.sormas.api.utils.ValidationRuntimeException; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.backend.FacadeHelper; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; +import de.symeda.sormas.backend.common.AbstractCoreFacadeEjb; +import de.symeda.sormas.backend.infrastructure.community.CommunityFacadeEjb; +import de.symeda.sormas.backend.infrastructure.community.CommunityService; +import de.symeda.sormas.backend.infrastructure.country.CountryFacadeEjb; +import de.symeda.sormas.backend.infrastructure.country.CountryService; +import de.symeda.sormas.backend.infrastructure.district.DistrictFacadeEjb; +import de.symeda.sormas.backend.infrastructure.district.DistrictService; +import de.symeda.sormas.backend.infrastructure.facility.FacilityFacadeEjb; +import de.symeda.sormas.backend.infrastructure.facility.FacilityService; +import de.symeda.sormas.backend.infrastructure.region.RegionFacadeEjb; +import de.symeda.sormas.backend.infrastructure.region.RegionService; +import de.symeda.sormas.backend.location.LocationFacadeEjb; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.user.UserFacadeEjb; +import de.symeda.sormas.backend.util.DtoHelper; +import de.symeda.sormas.backend.util.Pseudonymizer; +import de.symeda.sormas.backend.util.RightsAllowed; +import de.symeda.sormas.backend.vaccination.Vaccination; +import de.symeda.sormas.backend.vaccination.VaccinationFacadeEjb; +import de.symeda.sormas.backend.vaccination.VaccinationService; + +@Stateless(name = "AefiInvestigationFacade") +@RightsAllowed(UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW) +public class AefiInvestigationFacadeEjb + extends + AbstractCoreFacadeEjb + implements AefiInvestigationFacade { + + private final Logger logger = LoggerFactory.getLogger(AefiInvestigationFacadeEjb.class); + + @EJB + private AefiService aefiService; + @EJB + private LocationFacadeEjb.LocationFacadeEjbLocal locationFacade; + @EJB + private VaccinationFacadeEjb.VaccinationFacadeEjbLocal vaccinationFacade; + @EJB + private VaccinationService vaccinationService; + @EJB + private RegionService regionService; + @EJB + private DistrictService districtService; + @EJB + private CommunityService communityService; + @EJB + private FacilityService facilityService; + @EJB + private CountryService countryService; + + public AefiInvestigationFacadeEjb() { + } + + @Inject + public AefiInvestigationFacadeEjb(AefiInvestigationService service) { + super(AefiInvestigation.class, AefiInvestigationDto.class, service); + } + + @Override + @RightsAllowed({ + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT }) + public AefiInvestigationDto save(@Valid @NotNull AefiInvestigationDto dto) { + return save(dto, true, true); + } + + @RightsAllowed({ + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + UserRight._ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT }) + public AefiInvestigationDto save(@Valid @NotNull AefiInvestigationDto dto, boolean checkChangeDate, boolean internal) { + AefiInvestigation existingAefiInvestigation = service.getByUuid(dto.getUuid()); + + FacadeHelper.checkCreateAndEditRights( + existingAefiInvestigation, + userService, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT); + + if (internal && existingAefiInvestigation != null && !service.isEditAllowed(existingAefiInvestigation)) { + throw new AccessDeniedException(I18nProperties.getString(Strings.errorAdverseEventInvestigationNotEditable)); + } + + AefiInvestigationDto existingDto = toDto(existingAefiInvestigation); + + Pseudonymizer pseudonymizer = createPseudonymizer(existingAefiInvestigation); + restorePseudonymizedDto(dto, existingDto, existingAefiInvestigation, pseudonymizer); + + validate(dto); + + AefiInvestigation aefiInvestigation = fillOrBuildEntity(dto, existingAefiInvestigation, checkChangeDate); + + service.ensurePersisted(aefiInvestigation); + + return toPseudonymizedDto(aefiInvestigation, pseudonymizer); + } + + @Override + public long count(AefiInvestigationCriteria criteria) { + return service.count(criteria); + } + + @Override + public List getIndexList( + AefiInvestigationCriteria criteria, + Integer first, + Integer max, + List sortProperties) { + List resultsList = service.getIndexList(criteria, first, max, sortProperties); + Pseudonymizer pseudonymizer = createGenericPlaceholderPseudonymizer(); + pseudonymizer.pseudonymizeDtoCollection(AefiInvestigationIndexDto.class, resultsList, AefiInvestigationIndexDto::isInJurisdiction, null); + return resultsList; + } + + @Override + public List getEntriesList(AefiInvestigationListCriteria criteria, Integer first, Integer max) { + Long aefiId = aefiService.getIdByUuid(criteria.getAefiReport().getUuid()); + return service.getEntriesList(aefiId, first, max); + } + + @Override + public void validate(AefiInvestigationDto aefiInvestigationDto) throws ValidationRuntimeException { + if (DateHelper.isDateAfter(aefiInvestigationDto.getInvestigationDate(), aefiInvestigationDto.getReportDate())) { + String validationError = String.format( + I18nProperties.getValidationError(Validations.afterDate), + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, AefiInvestigationDto.INVESTIGATION_DATE), + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, AefiInvestigationDto.REPORT_DATE)); + throw new ValidationRuntimeException(validationError); + } + + // Check whether any required field that does not have a not null constraint in the database is empty + if (aefiInvestigationDto.getAefiReport() == null) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.validAefiReport)); + } + + /* + * if (aefiInvestigationDto.getPrimarySuspectVaccine() == null) { + * throw new + * ValidationRuntimeException(I18nProperties.getValidationError(Validations.aefiInvestigationWithoutPrimarySuspectVaccine)); + * } + */ + + if (aefiInvestigationDto.getReportingUser() == null && !aefiInvestigationDto.isPseudonymized()) { + throw new ValidationRuntimeException(I18nProperties.getValidationError(Validations.validReportingUser)); + } + } + + @Override + public List getArchivedUuidsSince(Date since) { + return null; + } + + @Override + public List delete(List uuids, DeletionDetails deletionDetails) { + return null; + } + + @Override + public List restore(List uuids) { + return null; + } + + @Override + protected AefiInvestigation fillOrBuildEntity(AefiInvestigationDto source, AefiInvestigation target, boolean checkChangeDate) { + return fillOrBuildEntity(source, target, checkChangeDate, false); + } + + protected AefiInvestigation fillOrBuildEntity( + @NotNull AefiInvestigationDto source, + AefiInvestigation target, + boolean checkChangeDate, + boolean includeVaccinations) { + + target = DtoHelper.fillOrBuildEntity(source, target, AefiInvestigation::build, checkChangeDate); + + target.setAefiReport(aefiService.getByReferenceDto(source.getAefiReport())); + target.setAddress(locationFacade.fillOrBuildEntity(source.getAddress(), target.getAddress(), checkChangeDate)); + + if (includeVaccinations) { + List vaccinationEntities = new ArrayList<>(); + for (VaccinationDto vaccinationDto : source.getVaccinations()) { + Vaccination vaccination = vaccinationService.getByUuid(vaccinationDto.getUuid()); + vaccination = vaccinationFacade.fillOrBuildEntity(vaccinationDto, vaccination, checkChangeDate); + vaccinationEntities.add(vaccination); + } + target.getVaccinations().clear(); + target.getVaccinations().addAll(vaccinationEntities); + } + + if (source.getPrimarySuspectVaccine() != null) { + target.setPrimarySuspectVaccine(vaccinationService.getByUuid(source.getPrimarySuspectVaccine().getUuid())); + } + + target.setReportDate(source.getReportDate()); + target.setReportingUser(userService.getByReferenceDto(source.getReportingUser())); + target.setExternalId(source.getExternalId()); + target.setResponsibleRegion(regionService.getByReferenceDto(source.getResponsibleRegion())); + target.setResponsibleDistrict(districtService.getByReferenceDto(source.getResponsibleDistrict())); + target.setResponsibleCommunity(communityService.getByReferenceDto(source.getResponsibleCommunity())); + target.setCountry(countryService.getByReferenceDto(source.getCountry())); + target.setInvestigationCaseId(source.getInvestigationCaseId()); + target.setPlaceOfVaccination(source.getPlaceOfVaccination()); + target.setPlaceOfVaccinationDetails(source.getPlaceOfVaccinationDetails()); + target.setVaccinationActivity(source.getVaccinationActivity()); + target.setVaccinationActivityDetails(source.getVaccinationActivityDetails()); + target.setVaccinationFacility(facilityService.getByReferenceDto(source.getVaccinationFacility())); + target.setVaccinationFacilityDetails(source.getVaccinationFacilityDetails()); + target.setReportingOfficerName(source.getReportingOfficerName()); + target.setReportingOfficerFacility(facilityService.getByReferenceDto(source.getReportingOfficerFacility())); + target.setReportingOfficerFacilityDetails(source.getReportingOfficerFacilityDetails()); + target.setReportingOfficerDesignation(source.getReportingOfficerDesignation()); + target.setReportingOfficerDepartment(source.getReportingOfficerDepartment()); + target.setReportingOfficerAddress( + locationFacade.fillOrBuildEntity(source.getReportingOfficerAddress(), target.getReportingOfficerAddress(), checkChangeDate)); + target.setReportingOfficerLandlinePhoneNumber(source.getReportingOfficerLandlinePhoneNumber()); + target.setReportingOfficerEmail(source.getReportingOfficerEmail()); + target.setReportingOfficerEmail(source.getReportingOfficerEmail()); + target.setInvestigationDate(source.getInvestigationDate()); + target.setFormCompletionDate(source.getFormCompletionDate()); + target.setInvestigationStage(source.getInvestigationStage()); + target.setTypeOfSite(source.getTypeOfSite()); + target.setTypeOfSiteDetails(source.getTypeOfSiteDetails()); + target.setKeySymptomDateTime(source.getKeySymptomDateTime()); + target.setHospitalizationDate(source.getHospitalizationDate()); + target.setReportedToHealthAuthorityDate(source.getReportedToHealthAuthorityDate()); + target.setStatusOnDateOfInvestigation(source.getStatusOnDateOfInvestigation()); + target.setDeathDateTime(source.getDeathDateTime()); + target.setAutopsyDone(source.getAutopsyDone()); + target.setAutopsyPlannedDateTime(source.getAutopsyPlannedDateTime()); + target.setPastHistoryOfSimilarEvent(source.getPastHistoryOfSimilarEvent()); + target.setPastHistoryOfSimilarEventDetails(source.getPastHistoryOfSimilarEventDetails()); + target.setAdverseEventAfterPreviousVaccinations(source.getAdverseEventAfterPreviousVaccinations()); + target.setAdverseEventAfterPreviousVaccinationsDetails(source.getAdverseEventAfterPreviousVaccinationsDetails()); + target.setHistoryOfAllergyToVaccineDrugOrFood(source.getHistoryOfAllergyToVaccineDrugOrFood()); + target.setHistoryOfAllergyToVaccineDrugOrFoodDetails(source.getHistoryOfAllergyToVaccineDrugOrFoodDetails()); + target.setPreExistingIllnessThirtyDaysOrCongenitalDisorder(source.getPreExistingIllnessThirtyDaysOrCongenitalDisorder()); + target.setPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails(source.getPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails()); + target.setHistoryOfHospitalizationInLastThirtyDaysWithCause(source.getHistoryOfHospitalizationInLastThirtyDaysWithCause()); + target.setHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails(source.getHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails()); + target.setCurrentlyOnConcomitantMedication(source.getCurrentlyOnConcomitantMedication()); + target.setCurrentlyOnConcomitantMedicationDetails(source.getCurrentlyOnConcomitantMedicationDetails()); + target.setFamilyHistoryOfDiseaseOrAllergy(source.getFamilyHistoryOfDiseaseOrAllergy()); + target.setFamilyHistoryOfDiseaseOrAllergyDetails(source.getFamilyHistoryOfDiseaseOrAllergyDetails()); + target.setBirthTerm(source.getBirthTerm()); + target.setBirthWeight(source.getBirthWeight()); + target.setDeliveryProcedure(source.getDeliveryProcedure()); + target.setDeliveryProcedureDetails(source.getDeliveryProcedureDetails()); + target.setSeriousAefiInfoSource(source.getSeriousAefiInfoSource()); + target.setSeriousAefiInfoSourceDetails(source.getSeriousAefiInfoSourceDetails()); + target.setSeriousAefiVerbalAutopsyInfoSourceDetails(source.getSeriousAefiVerbalAutopsyInfoSourceDetails()); + target.setFirstCaregiversName(source.getFirstCaregiversName()); + target.setOtherCaregiversNames(source.getOtherCaregiversNames()); + target.setOtherSourcesWhoProvidedInfo(source.getOtherSourcesWhoProvidedInfo()); + target.setSignsAndSymptomsFromTimeOfVaccination(source.getSignsAndSymptomsFromTimeOfVaccination()); + target.setClinicalDetailsOfficerName(source.getClinicalDetailsOfficerName()); + target.setClinicalDetailsOfficerPhoneNumber(source.getClinicalDetailsOfficerPhoneNumber()); + target.setClinicalDetailsOfficerEmail(source.getClinicalDetailsOfficerEmail()); + target.setClinicalDetailsOfficerDesignation(source.getClinicalDetailsOfficerDesignation()); + target.setClinicalDetailsDateTime(source.getClinicalDetailsDateTime()); + target.setPatientReceivedMedicalCare(source.getPatientReceivedMedicalCare()); + target.setPatientReceivedMedicalCareDetails(source.getPatientReceivedMedicalCareDetails()); + target.setProvisionalOrFinalDiagnosis(source.getProvisionalOrFinalDiagnosis()); + target.setPatientImmunizedPeriod(source.getPatientImmunizedPeriod()); + target.setPatientImmunizedPeriodDetails(source.getPatientImmunizedPeriodDetails()); + target.setVaccineGivenPeriod(source.getVaccineGivenPeriod()); + target.setVaccineGivenPeriodDetails(source.getVaccineGivenPeriodDetails()); + target.setErrorPrescribingVaccine(source.getErrorPrescribingVaccine()); + target.setErrorPrescribingVaccineDetails(source.getErrorPrescribingVaccineDetails()); + target.setVaccineCouldHaveBeenUnSterile(source.getVaccineCouldHaveBeenUnSterile()); + target.setVaccineCouldHaveBeenUnSterileDetails(source.getVaccineCouldHaveBeenUnSterileDetails()); + target.setVaccinePhysicalConditionAbnormal(source.getVaccinePhysicalConditionAbnormal()); + target.setVaccinePhysicalConditionAbnormalDetails(source.getVaccinePhysicalConditionAbnormalDetails()); + target.setErrorInVaccineReconstitution(source.getErrorInVaccineReconstitution()); + target.setErrorInVaccineReconstitutionDetails(source.getErrorInVaccineReconstitutionDetails()); + target.setErrorInVaccineHandling(source.getErrorInVaccineHandling()); + target.setErrorInVaccineHandlingDetails(source.getErrorInVaccineHandlingDetails()); + target.setVaccineAdministeredIncorrectly(source.getVaccineAdministeredIncorrectly()); + target.setVaccineAdministeredIncorrectlyDetails(source.getVaccineAdministeredIncorrectlyDetails()); + target.setNumberImmunizedFromConcernedVaccineVial(source.getNumberImmunizedFromConcernedVaccineVial()); + target.setNumberImmunizedWithConcernedVaccineInSameSession(source.getNumberImmunizedWithConcernedVaccineInSameSession()); + target.setNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations( + source.getNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations()); + target.setNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails( + source.getNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails()); + target.setVaccineHasQualityDefect(source.getVaccineHasQualityDefect()); + target.setVaccineHasQualityDefectDetails(source.getVaccineHasQualityDefectDetails()); + target.setEventIsAStressResponseRelatedToImmunization(source.getEventIsAStressResponseRelatedToImmunization()); + target.setEventIsAStressResponseRelatedToImmunizationDetails(source.getEventIsAStressResponseRelatedToImmunizationDetails()); + target.setCaseIsPartOfACluster(source.getCaseIsPartOfACluster()); + target.setCaseIsPartOfAClusterDetails(source.getCaseIsPartOfAClusterDetails()); + target.setNumberOfCasesDetectedInCluster(source.getNumberOfCasesDetectedInCluster()); + target.setAllCasesInClusterReceivedVaccineFromSameVial(source.getAllCasesInClusterReceivedVaccineFromSameVial()); + target.setAllCasesInClusterReceivedVaccineFromSameVialDetails(source.getAllCasesInClusterReceivedVaccineFromSameVialDetails()); + target.setNumberOfVialsUsedInCluster(source.getNumberOfVialsUsedInCluster()); + target.setNumberOfVialsUsedInClusterDetails(source.getNumberOfVialsUsedInClusterDetails()); + target.setAdSyringesUsedForImmunization(source.getAdSyringesUsedForImmunization()); + target.setTypeOfSyringesUsed(source.getTypeOfSyringesUsed()); + target.setTypeOfSyringesUsedDetails(source.getTypeOfSyringesUsedDetails()); + target.setSyringesUsedAdditionalDetails(source.getSyringesUsedAdditionalDetails()); + target.setSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine(source.getSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine()); + target.setSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines( + source.getSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines()); + target.setSameReconstitutionSyringeForEachVaccineVial(source.getSameReconstitutionSyringeForEachVaccineVial()); + target.setSameReconstitutionSyringeForEachVaccination(source.getSameReconstitutionSyringeForEachVaccination()); + target.setVaccinesAndDiluentsUsedRecommendedByManufacturer(source.getVaccinesAndDiluentsUsedRecommendedByManufacturer()); + target.setReconstitutionAdditionalDetails(source.getReconstitutionAdditionalDetails()); + target.setCorrectDoseOrRoute(source.getCorrectDoseOrRoute()); + target.setTimeOfReconstitutionMentionedOnTheVial(source.getTimeOfReconstitutionMentionedOnTheVial()); + target.setNonTouchTechniqueFollowed(source.getNonTouchTechniqueFollowed()); + target.setContraIndicationScreenedPriorToVaccination(source.getContraIndicationScreenedPriorToVaccination()); + target.setNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays( + source.getNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays()); + target.setTrainingReceivedByVaccinator(source.getTrainingReceivedByVaccinator()); + target.setLastTrainingReceivedByVaccinatorDate(source.getLastTrainingReceivedByVaccinatorDate()); + target.setInjectionTechniqueAdditionalDetails(source.getInjectionTechniqueAdditionalDetails()); + target.setVaccineStorageRefrigeratorTemperatureMonitored(source.getVaccineStorageRefrigeratorTemperatureMonitored()); + target.setAnyStorageTemperatureDeviationOutsideTwoToEightDegrees(source.getAnyStorageTemperatureDeviationOutsideTwoToEightDegrees()); + target.setStorageTemperatureMonitoringAdditionalDetails(source.getStorageTemperatureMonitoringAdditionalDetails()); + target.setCorrectProcedureForStorageFollowed(source.getCorrectProcedureForStorageFollowed()); + target.setAnyOtherItemInRefrigerator(source.getAnyOtherItemInRefrigerator()); + target.setPartiallyUsedReconstitutedVaccinesInRefrigerator(source.getPartiallyUsedReconstitutedVaccinesInRefrigerator()); + target.setUnusableVaccinesInRefrigerator(source.getUnusableVaccinesInRefrigerator()); + target.setUnusableDiluentsInStore(source.getUnusableDiluentsInStore()); + target.setVaccineStoragePointAdditionalDetails(source.getVaccineStoragePointAdditionalDetails()); + target.setVaccineCarrierType(source.getVaccineCarrierType()); + target.setVaccineCarrierTypeDetails(source.getVaccineCarrierTypeDetails()); + target.setVaccineCarrierSentToSiteOnSameDateAsVaccination(source.getVaccineCarrierSentToSiteOnSameDateAsVaccination()); + target.setVaccineCarrierReturnedFromSiteOnSameDateAsVaccination(source.getVaccineCarrierReturnedFromSiteOnSameDateAsVaccination()); + target.setConditionedIcepackUsed(source.getConditionedIcepackUsed()); + target.setVaccineTransportationAdditionalDetails(source.getVaccineTransportationAdditionalDetails()); + target.setSimilarEventsReportedSamePeriodAndLocality(source.getSimilarEventsReportedSamePeriodAndLocality()); + target.setSimilarEventsReportedSamePeriodAndLocalityDetails(source.getSimilarEventsReportedSamePeriodAndLocalityDetails()); + target.setNumberOfSimilarEventsReportedSamePeriodAndLocality(source.getNumberOfSimilarEventsReportedSamePeriodAndLocality()); + target.setNumberOfThoseAffectedVaccinated(source.getNumberOfThoseAffectedVaccinated()); + target.setNumberOfThoseAffectedNotVaccinated(source.getNumberOfThoseAffectedNotVaccinated()); + target.setNumberOfThoseAffectedVaccinatedUnknown(source.getNumberOfThoseAffectedVaccinatedUnknown()); + target.setCommunityInvestigationAdditionalDetails(source.getCommunityInvestigationAdditionalDetails()); + target.setOtherInvestigationFindings(source.getOtherInvestigationFindings()); + target.setInvestigationStatus(source.getInvestigationStatus()); + target.setAefiClassification(source.getAefiClassification()); + target.setArchived(source.isArchived()); + target.setDeleted(source.isDeleted()); + target.setDeletionReason(source.getDeletionReason()); + target.setOtherDeletionReason(source.getOtherDeletionReason()); + + return target; + } + + @Override + protected AefiInvestigationDto toDto(AefiInvestigation entity) { + return toAefiInvestigationDto(entity); + } + + public static AefiInvestigationDto toAefiInvestigationDto(AefiInvestigation entity) { + + if (entity == null) { + return null; + } + AefiInvestigationDto dto = new AefiInvestigationDto(); + DtoHelper.fillDto(dto, entity); + + dto.setAefiReport(AefiFacadeEjb.toReferenceDto(entity.getAefiReport())); + dto.setAddress(LocationFacadeEjb.toDto(entity.getAddress())); + + List vaccinationDtos = new ArrayList<>(); + for (Vaccination vaccination : entity.getAefiReport().getImmunization().getVaccinations()) { + VaccinationDto vaccinationDto = VaccinationFacadeEjb.toVaccinationDto(vaccination); + vaccinationDtos.add(vaccinationDto); + } + dto.setVaccinations(vaccinationDtos); + + if (entity.getPrimarySuspectVaccine() != null) { + dto.setPrimarySuspectVaccine(VaccinationFacadeEjb.toVaccinationDto(entity.getPrimarySuspectVaccine())); + } + + dto.setReportDate(entity.getReportDate()); + dto.setReportingUser(UserFacadeEjb.toReferenceDto(entity.getReportingUser())); + dto.setExternalId(entity.getExternalId()); + dto.setResponsibleRegion(RegionFacadeEjb.toReferenceDto(entity.getResponsibleRegion())); + dto.setResponsibleDistrict(DistrictFacadeEjb.toReferenceDto(entity.getResponsibleDistrict())); + dto.setResponsibleCommunity(CommunityFacadeEjb.toReferenceDto(entity.getResponsibleCommunity())); + dto.setCountry(CountryFacadeEjb.toReferenceDto(entity.getCountry())); + dto.setInvestigationCaseId(entity.getInvestigationCaseId()); + dto.setPlaceOfVaccination(entity.getPlaceOfVaccination()); + dto.setPlaceOfVaccinationDetails(entity.getPlaceOfVaccinationDetails()); + dto.setVaccinationActivity(entity.getVaccinationActivity()); + dto.setVaccinationActivityDetails(entity.getVaccinationActivityDetails()); + dto.setVaccinationFacility(FacilityFacadeEjb.toReferenceDto(entity.getVaccinationFacility())); + dto.setVaccinationFacilityDetails(entity.getVaccinationFacilityDetails()); + dto.setReportingOfficerName(entity.getReportingOfficerName()); + dto.setReportingOfficerFacility(FacilityFacadeEjb.toReferenceDto(entity.getReportingOfficerFacility())); + dto.setReportingOfficerFacilityDetails(entity.getReportingOfficerFacilityDetails()); + dto.setReportingOfficerDesignation(entity.getReportingOfficerDesignation()); + dto.setReportingOfficerDepartment(entity.getReportingOfficerDepartment()); + dto.setReportingOfficerAddress(LocationFacadeEjb.toDto(entity.getReportingOfficerAddress())); + dto.setReportingOfficerLandlinePhoneNumber(entity.getReportingOfficerLandlinePhoneNumber()); + dto.setReportingOfficerEmail(entity.getReportingOfficerEmail()); + dto.setReportingOfficerEmail(entity.getReportingOfficerEmail()); + dto.setInvestigationDate(entity.getInvestigationDate()); + dto.setFormCompletionDate(entity.getFormCompletionDate()); + dto.setInvestigationStage(entity.getInvestigationStage()); + dto.setTypeOfSite(entity.getTypeOfSite()); + dto.setTypeOfSiteDetails(entity.getTypeOfSiteDetails()); + dto.setKeySymptomDateTime(entity.getKeySymptomDateTime()); + dto.setHospitalizationDate(entity.getHospitalizationDate()); + dto.setReportedToHealthAuthorityDate(entity.getReportedToHealthAuthorityDate()); + dto.setStatusOnDateOfInvestigation(entity.getStatusOnDateOfInvestigation()); + dto.setDeathDateTime(entity.getDeathDateTime()); + dto.setAutopsyDone(entity.getAutopsyDone()); + dto.setAutopsyPlannedDateTime(entity.getAutopsyPlannedDateTime()); + dto.setPastHistoryOfSimilarEvent(entity.getPastHistoryOfSimilarEvent()); + dto.setPastHistoryOfSimilarEventDetails(entity.getPastHistoryOfSimilarEventDetails()); + dto.setAdverseEventAfterPreviousVaccinations(entity.getAdverseEventAfterPreviousVaccinations()); + dto.setAdverseEventAfterPreviousVaccinationsDetails(entity.getAdverseEventAfterPreviousVaccinationsDetails()); + dto.setHistoryOfAllergyToVaccineDrugOrFood(entity.getHistoryOfAllergyToVaccineDrugOrFood()); + dto.setHistoryOfAllergyToVaccineDrugOrFoodDetails(entity.getHistoryOfAllergyToVaccineDrugOrFoodDetails()); + dto.setPreExistingIllnessThirtyDaysOrCongenitalDisorder(entity.getPreExistingIllnessThirtyDaysOrCongenitalDisorder()); + dto.setPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails(entity.getPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails()); + dto.setHistoryOfHospitalizationInLastThirtyDaysWithCause(entity.getHistoryOfHospitalizationInLastThirtyDaysWithCause()); + dto.setHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails(entity.getHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails()); + dto.setCurrentlyOnConcomitantMedication(entity.getCurrentlyOnConcomitantMedication()); + dto.setCurrentlyOnConcomitantMedicationDetails(entity.getCurrentlyOnConcomitantMedicationDetails()); + dto.setFamilyHistoryOfDiseaseOrAllergy(entity.getFamilyHistoryOfDiseaseOrAllergy()); + dto.setFamilyHistoryOfDiseaseOrAllergyDetails(entity.getFamilyHistoryOfDiseaseOrAllergyDetails()); + dto.setBirthTerm(entity.getBirthTerm()); + dto.setBirthWeight(entity.getBirthWeight()); + dto.setDeliveryProcedure(entity.getDeliveryProcedure()); + dto.setDeliveryProcedureDetails(entity.getDeliveryProcedureDetails()); + dto.setSeriousAefiInfoSource(entity.getSeriousAefiInfoSource()); + dto.setSeriousAefiInfoSourceDetails(entity.getSeriousAefiInfoSourceDetails()); + dto.setSeriousAefiVerbalAutopsyInfoSourceDetails(entity.getSeriousAefiVerbalAutopsyInfoSourceDetails()); + dto.setFirstCaregiversName(entity.getFirstCaregiversName()); + dto.setOtherCaregiversNames(entity.getOtherCaregiversNames()); + dto.setOtherSourcesWhoProvidedInfo(entity.getOtherSourcesWhoProvidedInfo()); + dto.setSignsAndSymptomsFromTimeOfVaccination(entity.getSignsAndSymptomsFromTimeOfVaccination()); + dto.setClinicalDetailsOfficerName(entity.getClinicalDetailsOfficerName()); + dto.setClinicalDetailsOfficerPhoneNumber(entity.getClinicalDetailsOfficerPhoneNumber()); + dto.setClinicalDetailsOfficerEmail(entity.getClinicalDetailsOfficerEmail()); + dto.setClinicalDetailsOfficerDesignation(entity.getClinicalDetailsOfficerDesignation()); + dto.setClinicalDetailsDateTime(entity.getClinicalDetailsDateTime()); + dto.setPatientReceivedMedicalCare(entity.getPatientReceivedMedicalCare()); + dto.setPatientReceivedMedicalCareDetails(entity.getPatientReceivedMedicalCareDetails()); + dto.setProvisionalOrFinalDiagnosis(entity.getProvisionalOrFinalDiagnosis()); + dto.setPatientImmunizedPeriod(entity.getPatientImmunizedPeriod()); + dto.setPatientImmunizedPeriodDetails(entity.getPatientImmunizedPeriodDetails()); + dto.setVaccineGivenPeriod(entity.getVaccineGivenPeriod()); + dto.setVaccineGivenPeriodDetails(entity.getVaccineGivenPeriodDetails()); + dto.setErrorPrescribingVaccine(entity.getErrorPrescribingVaccine()); + dto.setErrorPrescribingVaccineDetails(entity.getErrorPrescribingVaccineDetails()); + dto.setVaccineCouldHaveBeenUnSterile(entity.getVaccineCouldHaveBeenUnSterile()); + dto.setVaccineCouldHaveBeenUnSterileDetails(entity.getVaccineCouldHaveBeenUnSterileDetails()); + dto.setVaccinePhysicalConditionAbnormal(entity.getVaccinePhysicalConditionAbnormal()); + dto.setVaccinePhysicalConditionAbnormalDetails(entity.getVaccinePhysicalConditionAbnormalDetails()); + dto.setErrorInVaccineReconstitution(entity.getErrorInVaccineReconstitution()); + dto.setErrorInVaccineReconstitutionDetails(entity.getErrorInVaccineReconstitutionDetails()); + dto.setErrorInVaccineHandling(entity.getErrorInVaccineHandling()); + dto.setErrorInVaccineHandlingDetails(entity.getErrorInVaccineHandlingDetails()); + dto.setVaccineAdministeredIncorrectly(entity.getVaccineAdministeredIncorrectly()); + dto.setVaccineAdministeredIncorrectlyDetails(entity.getVaccineAdministeredIncorrectlyDetails()); + dto.setNumberImmunizedFromConcernedVaccineVial(entity.getNumberImmunizedFromConcernedVaccineVial()); + dto.setNumberImmunizedWithConcernedVaccineInSameSession(entity.getNumberImmunizedWithConcernedVaccineInSameSession()); + dto.setNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations(entity.getNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations()); + dto.setNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails( + entity.getNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails()); + dto.setVaccineHasQualityDefect(entity.getVaccineHasQualityDefect()); + dto.setVaccineHasQualityDefectDetails(entity.getVaccineHasQualityDefectDetails()); + dto.setEventIsAStressResponseRelatedToImmunization(entity.getEventIsAStressResponseRelatedToImmunization()); + dto.setEventIsAStressResponseRelatedToImmunizationDetails(entity.getEventIsAStressResponseRelatedToImmunizationDetails()); + dto.setCaseIsPartOfACluster(entity.getCaseIsPartOfACluster()); + dto.setCaseIsPartOfAClusterDetails(entity.getCaseIsPartOfAClusterDetails()); + dto.setNumberOfCasesDetectedInCluster(entity.getNumberOfCasesDetectedInCluster()); + dto.setAllCasesInClusterReceivedVaccineFromSameVial(entity.getAllCasesInClusterReceivedVaccineFromSameVial()); + dto.setAllCasesInClusterReceivedVaccineFromSameVialDetails(entity.getAllCasesInClusterReceivedVaccineFromSameVialDetails()); + dto.setNumberOfVialsUsedInCluster(entity.getNumberOfVialsUsedInCluster()); + dto.setNumberOfVialsUsedInClusterDetails(entity.getNumberOfVialsUsedInClusterDetails()); + dto.setAdSyringesUsedForImmunization(entity.getAdSyringesUsedForImmunization()); + dto.setTypeOfSyringesUsed(entity.getTypeOfSyringesUsed()); + dto.setTypeOfSyringesUsedDetails(entity.getTypeOfSyringesUsedDetails()); + dto.setSyringesUsedAdditionalDetails(entity.getSyringesUsedAdditionalDetails()); + dto.setSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine(entity.getSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine()); + dto.setSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines( + entity.getSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines()); + dto.setSameReconstitutionSyringeForEachVaccineVial(entity.getSameReconstitutionSyringeForEachVaccineVial()); + dto.setSameReconstitutionSyringeForEachVaccination(entity.getSameReconstitutionSyringeForEachVaccination()); + dto.setVaccinesAndDiluentsUsedRecommendedByManufacturer(entity.getVaccinesAndDiluentsUsedRecommendedByManufacturer()); + dto.setReconstitutionAdditionalDetails(entity.getReconstitutionAdditionalDetails()); + dto.setCorrectDoseOrRoute(entity.getCorrectDoseOrRoute()); + dto.setTimeOfReconstitutionMentionedOnTheVial(entity.getTimeOfReconstitutionMentionedOnTheVial()); + dto.setNonTouchTechniqueFollowed(entity.getNonTouchTechniqueFollowed()); + dto.setContraIndicationScreenedPriorToVaccination(entity.getContraIndicationScreenedPriorToVaccination()); + dto.setNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays( + entity.getNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays()); + dto.setTrainingReceivedByVaccinator(entity.getTrainingReceivedByVaccinator()); + dto.setLastTrainingReceivedByVaccinatorDate(entity.getLastTrainingReceivedByVaccinatorDate()); + dto.setInjectionTechniqueAdditionalDetails(entity.getInjectionTechniqueAdditionalDetails()); + dto.setVaccineStorageRefrigeratorTemperatureMonitored(entity.getVaccineStorageRefrigeratorTemperatureMonitored()); + dto.setAnyStorageTemperatureDeviationOutsideTwoToEightDegrees(entity.getAnyStorageTemperatureDeviationOutsideTwoToEightDegrees()); + dto.setStorageTemperatureMonitoringAdditionalDetails(entity.getStorageTemperatureMonitoringAdditionalDetails()); + dto.setCorrectProcedureForStorageFollowed(entity.getCorrectProcedureForStorageFollowed()); + dto.setAnyOtherItemInRefrigerator(entity.getAnyOtherItemInRefrigerator()); + dto.setPartiallyUsedReconstitutedVaccinesInRefrigerator(entity.getPartiallyUsedReconstitutedVaccinesInRefrigerator()); + dto.setUnusableVaccinesInRefrigerator(entity.getUnusableVaccinesInRefrigerator()); + dto.setUnusableDiluentsInStore(entity.getUnusableDiluentsInStore()); + dto.setVaccineStoragePointAdditionalDetails(entity.getVaccineStoragePointAdditionalDetails()); + dto.setVaccineCarrierType(entity.getVaccineCarrierType()); + dto.setVaccineCarrierTypeDetails(entity.getVaccineCarrierTypeDetails()); + dto.setVaccineCarrierSentToSiteOnSameDateAsVaccination(entity.getVaccineCarrierSentToSiteOnSameDateAsVaccination()); + dto.setVaccineCarrierReturnedFromSiteOnSameDateAsVaccination(entity.getVaccineCarrierReturnedFromSiteOnSameDateAsVaccination()); + dto.setConditionedIcepackUsed(entity.getConditionedIcepackUsed()); + dto.setVaccineTransportationAdditionalDetails(entity.getVaccineTransportationAdditionalDetails()); + dto.setSimilarEventsReportedSamePeriodAndLocality(entity.getSimilarEventsReportedSamePeriodAndLocality()); + dto.setSimilarEventsReportedSamePeriodAndLocalityDetails(entity.getSimilarEventsReportedSamePeriodAndLocalityDetails()); + dto.setNumberOfSimilarEventsReportedSamePeriodAndLocality(entity.getNumberOfSimilarEventsReportedSamePeriodAndLocality()); + dto.setNumberOfThoseAffectedVaccinated(entity.getNumberOfThoseAffectedVaccinated()); + dto.setNumberOfThoseAffectedNotVaccinated(entity.getNumberOfThoseAffectedNotVaccinated()); + dto.setNumberOfThoseAffectedVaccinatedUnknown(entity.getNumberOfThoseAffectedVaccinatedUnknown()); + dto.setCommunityInvestigationAdditionalDetails(entity.getCommunityInvestigationAdditionalDetails()); + dto.setOtherInvestigationFindings(entity.getOtherInvestigationFindings()); + dto.setInvestigationStatus(entity.getInvestigationStatus()); + dto.setAefiClassification(entity.getAefiClassification()); + dto.setArchived(entity.isArchived()); + dto.setDeleted(entity.isDeleted()); + dto.setDeletionReason(entity.getDeletionReason()); + dto.setOtherDeletionReason(entity.getOtherDeletionReason()); + + return dto; + } + + @Override + protected AefiInvestigationReferenceDto toRefDto(AefiInvestigation aefiInvestigation) { + return toReferenceDto(aefiInvestigation); + } + + public static AefiInvestigationReferenceDto toReferenceDto(AefiInvestigation entity) { + + if (entity == null) { + return null; + } + + return new AefiInvestigationReferenceDto(entity.getUuid(), ""); + } + + @Override + protected void pseudonymizeDto( + AefiInvestigation source, + AefiInvestigationDto dto, + Pseudonymizer pseudonymizer, + boolean inJurisdiction) { + + if (dto != null) { + pseudonymizer.pseudonymizeDto(AefiInvestigationDto.class, dto, inJurisdiction, c -> { + pseudonymizer.pseudonymizeUser(source.getReportingUser(), userService.getCurrentUser(), dto::setReportingUser, dto); + }); + } + } + + @Override + protected void restorePseudonymizedDto( + AefiInvestigationDto dto, + AefiInvestigationDto existingDto, + AefiInvestigation entity, + Pseudonymizer pseudonymizer) { + + if (existingDto != null) { + final boolean inJurisdiction = service.inJurisdictionOrOwned(entity); + final User currentUser = userService.getCurrentUser(); + pseudonymizer.restoreUser(entity.getReportingUser(), currentUser, dto, dto::setReportingUser); + pseudonymizer.restorePseudonymizedValues(AefiInvestigationDto.class, dto, existingDto, inJurisdiction); + } + } + + @Override + protected DeletableEntityType getDeletableEntityType() { + return null; + } + + @LocalBean + @Stateless + public static class AefiInvestigationFacadeEjbLocal extends AefiInvestigationFacadeEjb { + + public AefiInvestigationFacadeEjbLocal() { + super(); + } + + @Inject + public AefiInvestigationFacadeEjbLocal(AefiInvestigationService service) { + super(service); + } + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationJurisdictionPredicateValidator.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationJurisdictionPredicateValidator.java new file mode 100644 index 00000000000..bf377ed8fb7 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationJurisdictionPredicateValidator.java @@ -0,0 +1,116 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import java.util.List; + +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.Path; +import javax.persistence.criteria.Predicate; + +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigationJoins; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.util.PredicateJurisdictionValidator; + +public class AefiInvestigationJurisdictionPredicateValidator extends PredicateJurisdictionValidator { + + private final AefiInvestigationJoins joins; + + public AefiInvestigationJurisdictionPredicateValidator( + CriteriaBuilder cb, + AefiInvestigationJoins joins, + User user, + List jurisdictionValidators) { + super(cb, user, null, jurisdictionValidators); + this.joins = joins; + } + + public static AefiInvestigationJurisdictionPredicateValidator of(AefiInvestigationQueryContext qc, User user) { + return new AefiInvestigationJurisdictionPredicateValidator(qc.getCriteriaBuilder(), qc.getJoins(), user, null); + } + + @Override + public Predicate isRootInJurisdiction() { + return isInJurisdictionByJurisdictionLevel(user.getJurisdictionLevel()); + } + + @Override + public Predicate isRootInJurisdictionOrOwned() { + final Path reportingUserPath = joins.getRoot().get(AefiInvestigation.REPORTING_USER); + final Predicate reportedByCurrentUser = cb.and(cb.isNotNull(reportingUserPath), cb.equal(reportingUserPath.get(User.ID), user.getId())); + return cb.or(reportedByCurrentUser, this.isRootInJurisdiction()); + } + + @Override + public Predicate isRootInJurisdictionForRestrictedAccess() { + return null; + } + + @Override + protected Predicate whenNotAllowed() { + return cb.disjunction(); + } + + @Override + protected Predicate whenNationalLevel() { + return cb.conjunction(); + } + + @Override + protected Predicate whenRegionalLevel() { + return cb.equal( + joins.getRoot().get(AefiInvestigation.AEFI_REPORT).get(Aefi.IMMUNIZATION).get(Immunization.RESPONSIBLE_REGION).get(Region.ID), + user.getRegion().getId()); + } + + @Override + protected Predicate whenDistrictLevel() { + return cb.equal( + joins.getRoot().get(AefiInvestigation.AEFI_REPORT).get(Aefi.IMMUNIZATION).get(Immunization.RESPONSIBLE_DISTRICT).get(District.ID), + user.getDistrict().getId()); + } + + @Override + protected Predicate whenCommunityLevel() { + return cb.equal( + joins.getRoot().get(AefiInvestigation.AEFI_REPORT).get(Aefi.IMMUNIZATION).get(Immunization.RESPONSIBLE_COMMUNITY).get(Community.ID), + user.getCommunity().getId()); + } + + @Override + protected Predicate whenFacilityLevel() { + return cb.equal( + joins.getRoot().get(AefiInvestigation.AEFI_REPORT).get(Aefi.IMMUNIZATION).get(Immunization.HEALTH_FACILITY).get(Facility.ID), + user.getHealthFacility().getId()); + } + + @Override + protected Predicate whenPointOfEntryLevel() { + return cb.disjunction(); + } + + @Override + protected Predicate whenLaboratoryLevel() { + return cb.disjunction(); + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationQueryContext.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationQueryContext.java new file mode 100644 index 00000000000..81defa6406f --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationQueryContext.java @@ -0,0 +1,45 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.Expression; +import javax.persistence.criteria.From; + +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigationJoins; +import de.symeda.sormas.backend.common.QueryContext; + +public class AefiInvestigationQueryContext extends QueryContext { + + public AefiInvestigationQueryContext(CriteriaBuilder cb, CriteriaQuery query, From root) { + super(cb, query, root, new AefiInvestigationJoins(root)); + } + + public AefiInvestigationQueryContext(CriteriaBuilder cb, CriteriaQuery query, AefiInvestigationJoins joins) { + super(cb, query, joins.getRoot(), joins); + } + + public AefiInvestigationQueryContext(CriteriaBuilder cb, CriteriaQuery query, From root, AefiInvestigationJoins joins) { + super(cb, query, root, joins); + } + + @Override + protected Expression createExpression(String name) { + return null; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationService.java new file mode 100644 index 00000000000..dfa5f010a5f --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiInvestigationService.java @@ -0,0 +1,548 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization; + +import static de.symeda.sormas.backend.common.CriteriaBuilderHelper.andEquals; + +import java.util.ArrayList; +import java.util.List; +import java.util.stream.Collectors; +import java.util.stream.Stream; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; +import javax.persistence.Tuple; +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.CriteriaQuery; +import javax.persistence.criteria.Expression; +import javax.persistence.criteria.From; +import javax.persistence.criteria.Join; +import javax.persistence.criteria.Order; +import javax.persistence.criteria.Path; +import javax.persistence.criteria.Predicate; +import javax.persistence.criteria.Root; +import javax.persistence.criteria.Selection; + +import org.apache.commons.collections4.CollectionUtils; + +import de.symeda.sormas.api.EntityRelevanceStatus; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationIndexDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.common.DeletableEntityType; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.feature.FeatureTypeProperty; +import de.symeda.sormas.api.person.PersonIndexDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.SortProperty; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigationJoins; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers.AefiInvestigationIndexDtoResultTransformer; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers.AefiInvestigationListEntryDtoResultTransformer; +import de.symeda.sormas.backend.common.AbstractCoreAdoService; +import de.symeda.sormas.backend.common.CriteriaBuilderHelper; +import de.symeda.sormas.backend.feature.FeatureConfigurationFacadeEjb; +import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.person.Person; +import de.symeda.sormas.backend.person.PersonJoins; +import de.symeda.sormas.backend.person.PersonJurisdictionPredicateValidator; +import de.symeda.sormas.backend.person.PersonQueryContext; +import de.symeda.sormas.backend.person.PersonService; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.user.UserService; +import de.symeda.sormas.backend.util.IterableHelper; +import de.symeda.sormas.backend.util.JurisdictionHelper; +import de.symeda.sormas.backend.util.ModelConstants; +import de.symeda.sormas.backend.util.QueryHelper; +import de.symeda.sormas.backend.vaccination.Vaccination; + +@Stateless +@LocalBean +public class AefiInvestigationService extends AbstractCoreAdoService { + + @EJB + private PersonService personService; + @EJB + private UserService userService; + @EJB + private FeatureConfigurationFacadeEjb.FeatureConfigurationFacadeEjbLocal featureConfigurationFacade; + + public AefiInvestigationService() { + super(AefiInvestigation.class, DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION); + } + + public List getEntriesList(Long aefiReportId, Integer first, Integer max) { + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Tuple.class); + final Root aefiInvestigation = cq.from(AefiInvestigation.class); + + AefiInvestigationQueryContext queryContext = new AefiInvestigationQueryContext(cb, cq, aefiInvestigation); + AefiInvestigationJoins joins = queryContext.getJoins(); + + Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + + cq.multiselect( + aefiInvestigation.get(AefiInvestigation.UUID), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_CASE_ID), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_DATE), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_STAGE), + aefiInvestigation.get(AefiInvestigation.STATUS_ON_DATE_OF_INVESTIGATION), + primarySuspectVaccine.get(Vaccination.VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.OTHER_VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.VACCINE_DOSE), + primarySuspectVaccine.get(Vaccination.VACCINATION_DATE), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_STATUS), + aefiInvestigation.get(AefiInvestigation.AEFI_CLASSIFICATION), + aefiInvestigation.get(Aefi.CHANGE_DATE), + JurisdictionHelper.booleanSelector(cb, isInJurisdictionOrOwned(queryContext))); + + final Predicate criteriaFilter = buildCriteriaFilter(aefiReportId, queryContext); + if (criteriaFilter != null) { + cq.where(criteriaFilter); + } + + cq.orderBy(cb.desc(aefiInvestigation.get(AefiInvestigation.CHANGE_DATE))); + + cq.distinct(true); + + return QueryHelper.getResultList(em, cq, new AefiInvestigationListEntryDtoResultTransformer(), first, max); + } + + private Predicate buildCriteriaFilter(Long aefiReportId, AefiInvestigationQueryContext queryContext) { + + final CriteriaBuilder cb = queryContext.getCriteriaBuilder(); + final From from = queryContext.getRoot(); + + Predicate filter = cb.equal(from.get(AefiInvestigation.AEFI_REPORT_ID), aefiReportId); + + filter = CriteriaBuilderHelper.and(cb, filter, cb.isFalse(from.get(AefiInvestigation.DELETED))); + + return filter; + } + + public long count(AefiInvestigationCriteria criteria) { + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Long.class); + final Root aefiInvestigation = cq.from(AefiInvestigation.class); + + AefiInvestigationQueryContext queryContext = new AefiInvestigationQueryContext(cb, cq, aefiInvestigation); + + buildWhereCondition(criteria, cb, cq, queryContext, null); + + cq.select(cb.countDistinct(aefiInvestigation)); + return em.createQuery(cq).getSingleResult(); + } + + public List getIndexList( + AefiInvestigationCriteria criteria, + Integer first, + Integer max, + List sortProperties) { + + List indexListIds = getIndexListIds(criteria, first, max, sortProperties); + List aefiInvestigationIndexDtos = new ArrayList<>(); + + IterableHelper.executeBatched(indexListIds, ModelConstants.PARAMETER_LIMIT, batchedIds -> { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(Tuple.class); + final Root aefiInvestigation = cq.from(AefiInvestigation.class); + + AefiInvestigationQueryContext queryContext = new AefiInvestigationQueryContext(cb, cq, aefiInvestigation); + AefiInvestigationJoins joins = queryContext.getJoins(); + + final Join aefi = joins.getAefi(); + final Join immunization = joins.getAefiJoins().getImmunization(); + final Join person = joins.getAefiJoins().getImmunizationJoins().getPerson(); + + final Join responsibleRegion = joins.getAefiJoins().getImmunizationJoins().getResponsibleRegion(); + final Join responsibleDistrict = joins.getAefiJoins().getImmunizationJoins().getResponsibleDistrict(); + + final Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + + cq.multiselect( + Stream + .concat( + Stream.of( + aefiInvestigation.get(AefiInvestigation.UUID), + aefi.get(Aefi.UUID), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_CASE_ID), + immunization.get(Immunization.DISEASE), + person.get(Person.FIRST_NAME), + person.get(Person.LAST_NAME), + person.get(Person.APPROXIMATE_AGE), + person.get(Person.APPROXIMATE_AGE_TYPE), + person.get(Person.BIRTHDATE_DD), + person.get(Person.BIRTHDATE_MM), + person.get(Person.BIRTHDATE_YYYY), + person.get(Person.SEX), + responsibleRegion.get(Region.NAME), + responsibleDistrict.get(District.NAME), + aefiInvestigation.get(AefiInvestigation.PLACE_OF_VACCINATION), + aefiInvestigation.get(AefiInvestigation.VACCINATION_ACTIVITY), + aefi.get(Aefi.REPORT_DATE), + aefiInvestigation.get(Aefi.REPORT_DATE), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_DATE), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_STAGE), + aefiInvestigation.get(AefiInvestigation.TYPE_OF_SITE), + aefiInvestigation.get(AefiInvestigation.KEY_SYMPTOM_DATE_TIME), + aefiInvestigation.get(AefiInvestigation.HOSPITALIZATION_DATE), + aefiInvestigation.get(AefiInvestigation.REPORTED_TO_HEALTH_AUTHORITY_DATE), + aefiInvestigation.get(AefiInvestigation.STATUS_ON_DATE_OF_INVESTIGATION), + primarySuspectVaccine.get(Vaccination.VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.OTHER_VACCINE_NAME), + aefiInvestigation.get(AefiInvestigation.INVESTIGATION_STATUS), + aefiInvestigation.get(AefiInvestigation.AEFI_CLASSIFICATION), + aefiInvestigation.get(AefiInvestigation.DELETION_REASON), + aefiInvestigation.get(AefiInvestigation.OTHER_DELETION_REASON), + JurisdictionHelper.booleanSelector(cb, isInJurisdictionOrOwned(queryContext)), + aefiInvestigation.get(AefiInvestigation.CHANGE_DATE)), + // add sort properties to select + sortBy(sortProperties, queryContext).stream()) + .collect(Collectors.toList())); + + buildWhereCondition(criteria, cb, cq, queryContext, aefiInvestigation.get(AefiInvestigation.ID).in(batchedIds)); + cq.distinct(true); + + aefiInvestigationIndexDtos.addAll(QueryHelper.getResultList(em, cq, new AefiInvestigationIndexDtoResultTransformer(), null, null)); + }); + + return aefiInvestigationIndexDtos; + } + + private List getIndexListIds(AefiInvestigationCriteria criteria, Integer first, Integer max, List sortProperties) { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createTupleQuery(); + final Root aefiInvestigation = cq.from(AefiInvestigation.class); + + AefiInvestigationQueryContext queryContext = new AefiInvestigationQueryContext(cb, cq, aefiInvestigation); + + List> selections = new ArrayList<>(); + selections.add(aefiInvestigation.get(AefiInvestigation.ID)); + selections.addAll(sortBy(sortProperties, queryContext)); + + cq.multiselect(selections); + + buildWhereCondition(criteria, cb, cq, queryContext, null); + cq.distinct(true); + + List aefiInvestigationsResultList = QueryHelper.getResultList(em, cq, first, max); + return aefiInvestigationsResultList.stream().map(t -> t.get(0, Long.class)).collect(Collectors.toList()); + } + + private List> sortBy(List sortProperties, AefiInvestigationQueryContext queryContext) { + + List> selections = new ArrayList<>(); + CriteriaBuilder cb = queryContext.getCriteriaBuilder(); + CriteriaQuery cq = queryContext.getQuery(); + + if (CollectionUtils.isNotEmpty(sortProperties)) { + List order = new ArrayList<>(sortProperties.size()); + for (SortProperty sortProperty : sortProperties) { + Expression expression; + switch (sortProperty.propertyName) { + case AefiInvestigationIndexDto.UUID: + case AefiInvestigationIndexDto.DISEASE: + case AefiInvestigationIndexDto.INVESTIGATION_DATE: + case AefiInvestigationIndexDto.STATUS_ON_DATE_OF_INVESTIGATION: + case AefiInvestigationIndexDto.AEFI_CLASSIFICATION: + expression = queryContext.getRoot().get(sortProperty.propertyName); + break; + case AefiInvestigationIndexDto.AEFI_REPORT_UUID: + expression = queryContext.getJoins().getAefi().get(Aefi.UUID); + break; + case AefiInvestigationIndexDto.PERSON_FIRST_NAME: + expression = cb.lower(queryContext.getJoins().getAefiJoins().getImmunizationJoins().getPerson().get(Person.FIRST_NAME)); + break; + case AefiInvestigationIndexDto.PERSON_LAST_NAME: + expression = cb.lower(queryContext.getJoins().getAefiJoins().getImmunizationJoins().getPerson().get(Person.LAST_NAME)); + break; + case AefiInvestigationIndexDto.AGE_AND_BIRTH_DATE: + expression = queryContext.getJoins().getAefiJoins().getImmunizationJoins().getPerson().get(Person.APPROXIMATE_AGE); + break; + case AefiInvestigationIndexDto.SEX: + expression = queryContext.getJoins().getAefiJoins().getImmunizationJoins().getPerson().get(Person.SEX); + break; + case AefiInvestigationIndexDto.REGION: + expression = cb.lower( + queryContext.getJoins() + .getAefiJoins() + .getImmunizationJoins() + .getPersonJoins() + .getAddressJoins() + .getRegion() + .get(Region.NAME)); + break; + case AefiInvestigationIndexDto.DISTRICT: + expression = cb.lower( + queryContext.getJoins() + .getAefiJoins() + .getImmunizationJoins() + .getPersonJoins() + .getAddressJoins() + .getDistrict() + .get(District.NAME)); + break; + case AefiInvestigationIndexDto.PRIMARY_VACCINE_NAME: + expression = queryContext.getJoins().getPrimarySuspectVaccination().get(Vaccination.VACCINE_NAME); + break; + default: + throw new IllegalArgumentException(sortProperty.propertyName); + } + order.add(sortProperty.ascending ? cb.asc(expression) : cb.desc(expression)); + selections.add(expression); + } + cq.orderBy(order); + } else { + Path changeDate = queryContext.getRoot().get(AefiInvestigation.CHANGE_DATE); + cq.orderBy(cb.desc(changeDate)); + selections.add(changeDate); + } + + return selections; + } + + private void buildWhereCondition( + AefiInvestigationCriteria criteria, + CriteriaBuilder cb, + CriteriaQuery cq, + AefiInvestigationQueryContext queryContext, + Predicate additionalFilter) { + + Predicate filter = createUserFilter(queryContext); + if (additionalFilter != null) { + filter = CriteriaBuilderHelper.and(cb, additionalFilter, filter); + } + + if (criteria != null) { + final Predicate criteriaFilter = buildCriteriaFilter(criteria, queryContext); + filter = CriteriaBuilderHelper.and(cb, filter, criteriaFilter); + } + + if (filter != null) { + cq.where(filter); + } + } + + public Predicate buildCriteriaFilter(AefiInvestigationCriteria criteria, AefiInvestigationQueryContext queryContext) { + + final AefiInvestigationJoins joins = queryContext.getJoins(); + final CriteriaBuilder cb = queryContext.getCriteriaBuilder(); + final From from = queryContext.getRoot(); + final Join aefi = joins.getAefi(); + final Join immunization = joins.getAefiJoins().getImmunization(); + final Join person = joins.getAefiJoins().getImmunizationJoins().getPerson(); + final Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + + final Join location = joins.getAefiJoins().getImmunizationJoins().getPersonJoins().getAddress(); + + Predicate filter = null; + if (criteria.getDisease() != null) { + filter = CriteriaBuilderHelper.and(cb, null, cb.equal(immunization.get(Immunization.DISEASE), criteria.getDisease())); + } + + if (!DataHelper.isNullOrEmpty(criteria.getAefiReportLike())) { + + String[] textFilters = criteria.getAefiReportLike().split("\\s+"); + + for (String textFilter : textFilters) { + if (DataHelper.isNullOrEmpty(textFilter)) { + continue; + } + + Predicate likeFilters = cb.or(CriteriaBuilderHelper.ilike(cb, aefi.get(Aefi.UUID), textFilter)); + filter = CriteriaBuilderHelper.and(cb, filter, likeFilters); + } + } + + if (!DataHelper.isNullOrEmpty(criteria.getPersonLike())) { + final CriteriaQuery cq = cb.createQuery(PersonIndexDto.class); + final PersonQueryContext personQueryContext = + new PersonQueryContext(cb, cq, joins.getAefiJoins().getImmunizationJoins().getPersonJoins()); + + String[] textFilters = criteria.getPersonLike().split("\\s+"); + + for (String textFilter : textFilters) { + if (DataHelper.isNullOrEmpty(textFilter)) { + continue; + } + + Predicate likeFilters = cb.or( + CriteriaBuilderHelper.ilike(cb, from.get(AefiInvestigation.UUID), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, person.get(Person.FIRST_NAME), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, person.get(Person.LAST_NAME), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.UUID), textFilter), + CriteriaBuilderHelper.ilike(cb, personQueryContext.getSubqueryExpression(PersonQueryContext.PERSON_EMAIL_SUBQUERY), textFilter), + phoneNumberPredicate(cb, personQueryContext.getSubqueryExpression(PersonQueryContext.PERSON_PHONE_SUBQUERY), textFilter), + CriteriaBuilderHelper + .ilike(cb, personQueryContext.getSubqueryExpression(PersonQueryContext.PERSON_PRIMARY_OTHER_SUBQUERY), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, location.get(Location.STREET), textFilter), + CriteriaBuilderHelper.unaccentedIlike(cb, location.get(Location.CITY), textFilter), + CriteriaBuilderHelper.ilike(cb, location.get(Location.POSTAL_CODE), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.INTERNAL_TOKEN), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.EXTERNAL_ID), textFilter), + CriteriaBuilderHelper.ilike(cb, person.get(Person.EXTERNAL_TOKEN), textFilter)); + filter = CriteriaBuilderHelper.and(cb, filter, likeFilters); + } + } + if (criteria.getAefiType() != null) { + if (criteria.getAefiType() == AefiType.SERIOUS) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(aefi.get(Aefi.SERIOUS), YesNoUnknown.YES)); + } else { + filter = CriteriaBuilderHelper.and(cb, filter, cb.notEqual(aefi.get(Aefi.SERIOUS), YesNoUnknown.YES)); + } + } + if (criteria.getStatusAtAefiInvestigation() != null) { + filter = CriteriaBuilderHelper + .and(cb, filter, cb.equal(from.get(AefiInvestigation.STATUS_ON_DATE_OF_INVESTIGATION), criteria.getStatusAtAefiInvestigation())); + } + if (criteria.getAefiClassification() != null) { + filter = + CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(AefiInvestigation.AEFI_CLASSIFICATION), criteria.getAefiClassification())); + } + if (criteria.getVaccineName() != null) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(primarySuspectVaccine.get(Vaccination.VACCINE_NAME), criteria.getVaccineName())); + } + if (criteria.getVaccineManufacturer() != null) { + filter = CriteriaBuilderHelper + .and(cb, filter, cb.equal(primarySuspectVaccine.get(Vaccination.VACCINE_MANUFACTURER), criteria.getVaccineManufacturer())); + } + filter = andEquals(cb, () -> joins.getAefiJoins().getImmunizationJoins().getResponsibleRegion(), filter, criteria.getRegion()); + filter = andEquals(cb, () -> joins.getAefiJoins().getImmunizationJoins().getResponsibleDistrict(), filter, criteria.getDistrict()); + filter = andEquals(cb, () -> joins.getAefiJoins().getImmunizationJoins().getResponsibleCommunity(), filter, criteria.getCommunity()); + if (criteria.getFacilityType() != null) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(immunization.get(Immunization.FACILITY_TYPE), criteria.getFacilityType())); + } + filter = andEquals(cb, () -> joins.getAefiJoins().getImmunizationJoins().getHealthFacility(), filter, criteria.getHealthFacility()); + if (criteria.getAefiInvestigationDateType() != null) { + Path path = buildPathForDateFilter(criteria.getAefiInvestigationDateType(), queryContext); + if (path != null) { + filter = CriteriaBuilderHelper.applyDateFilter(cb, filter, path, criteria.getFromDate(), criteria.getToDate()); + } + } + + if (criteria.getRelevanceStatus() != null) { + if (criteria.getRelevanceStatus() == EntityRelevanceStatus.ACTIVE) { + filter = CriteriaBuilderHelper + .and(cb, filter, cb.or(cb.equal(from.get(AefiInvestigation.ARCHIVED), false), cb.isNull(from.get(AefiInvestigation.ARCHIVED)))); + } else if (criteria.getRelevanceStatus() == EntityRelevanceStatus.ARCHIVED) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(AefiInvestigation.ARCHIVED), true)); + } else if (criteria.getRelevanceStatus() == EntityRelevanceStatus.DELETED) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(AefiInvestigation.DELETED), true)); + } + } + if (criteria.getRelevanceStatus() != EntityRelevanceStatus.DELETED) { + filter = CriteriaBuilderHelper.and(cb, filter, cb.isFalse(from.get(AefiInvestigation.DELETED))); + } + + return filter; + } + + private Path buildPathForDateFilter(AefiInvestigationDateType aefiInvestigationDateType, AefiInvestigationQueryContext queryContext) { + Path path = null; + String dateField = getDateFieldFromDateType(aefiInvestigationDateType); + if (dateField != null) { + if (Vaccination.VACCINATION_DATE.equals(dateField)) { + final Join primarySuspectVaccination = queryContext.getJoins().getPrimarySuspectVaccination(); + path = primarySuspectVaccination.get(Vaccination.VACCINATION_DATE); + } else { + path = queryContext.getRoot().get(dateField); + } + } + return path; + } + + private String getDateFieldFromDateType(AefiInvestigationDateType aefiInvestigationDateType) { + switch (aefiInvestigationDateType) { + case REPORT_DATE: + return AefiInvestigation.REPORT_DATE; + case INVESTIGATION_DATE: + return AefiInvestigation.INVESTIGATION_DATE; + case VACCINATION_DATE: + return Vaccination.VACCINATION_DATE; + } + return null; + } + + public Predicate createUserFilter(AefiInvestigationQueryContext qc) { + + User currentUser = getCurrentUser(); + if (currentUser == null) { + return null; + } + final CriteriaBuilder cb = qc.getCriteriaBuilder(); + + Predicate filter = isInJurisdictionOrOwned(qc); + + filter = CriteriaBuilderHelper.and( + cb, + filter, + CriteriaBuilderHelper.limitedDiseasePredicate( + cb, + currentUser, + qc.getRoot().get(AefiInvestigation.AEFI_REPORT).get(Aefi.IMMUNIZATION).get(Immunization.DISEASE))); + + return filter; + } + + private Predicate isInJurisdictionOrOwned(AefiInvestigationQueryContext qc) { + + final User currentUser = userService.getCurrentUser(); + CriteriaBuilder cb = qc.getCriteriaBuilder(); + Predicate filter; + if (!featureConfigurationFacade + .isPropertyValueTrue(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT, FeatureTypeProperty.REDUCED)) { + filter = AefiInvestigationJurisdictionPredicateValidator.of(qc, currentUser).inJurisdictionOrOwned(); + } else { + filter = CriteriaBuilderHelper.or( + cb, + cb.equal(qc.getRoot().get(AefiInvestigation.REPORTING_USER), currentUser), + PersonJurisdictionPredicateValidator + .of( + qc.getQuery(), + cb, + new PersonJoins(qc.getJoins().getAefiJoins().getImmunizationJoins().getPerson()), + currentUser, + personService.getPermittedAssociations()) + .inJurisdictionOrOwned()); + } + return filter; + } + + @Override + protected Predicate createUserFilterInternal(CriteriaBuilder cb, CriteriaQuery cq, From from) { + return createUserFilter(new AefiInvestigationQueryContext(cb, cq, from)); + } + + @Override + protected AefiInvestigationJoins toJoins(From adoPath) { + return new AefiInvestigationJoins(adoPath); + } + + @Override + public Predicate inJurisdictionOrOwned(CriteriaBuilder cb, CriteriaQuery query, From from) { + return isInJurisdictionOrOwned(new AefiInvestigationQueryContext(cb, query, from)); + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java index 215572a9d48..776d84268c9 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiJurisdictionPredicateValidator.java @@ -55,7 +55,7 @@ public Predicate isRootInJurisdiction() { @Override public Predicate isRootInJurisdictionOrOwned() { - final Path reportingUserPath = joins.getRoot().get(Immunization.REPORTING_USER); + final Path reportingUserPath = joins.getRoot().get(Aefi.REPORTING_USER); final Predicate reportedByCurrentUser = cb.and(cb.isNotNull(reportingUserPath), cb.equal(reportingUserPath.get(User.ID), user.getId())); return cb.or(reportedByCurrentUser, this.isRootInJurisdiction()); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java index 5d18a99795b..62736e5f1b5 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/AefiService.java @@ -18,6 +18,7 @@ import static de.symeda.sormas.backend.common.CriteriaBuilderHelper.andEquals; import java.util.ArrayList; +import java.util.Collection; import java.util.List; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -26,22 +27,27 @@ import javax.ejb.LocalBean; import javax.ejb.Stateless; import javax.persistence.Tuple; +import javax.persistence.TypedQuery; import javax.persistence.criteria.CriteriaBuilder; import javax.persistence.criteria.CriteriaQuery; import javax.persistence.criteria.Expression; import javax.persistence.criteria.From; import javax.persistence.criteria.Join; +import javax.persistence.criteria.JoinType; import javax.persistence.criteria.Order; +import javax.persistence.criteria.ParameterExpression; import javax.persistence.criteria.Path; import javax.persistence.criteria.Predicate; import javax.persistence.criteria.Root; import javax.persistence.criteria.Selection; +import javax.validation.constraints.NotNull; import org.apache.commons.collections4.CollectionUtils; import de.symeda.sormas.api.EntityRelevanceStatus; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiExportDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; @@ -58,10 +64,14 @@ import de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers.AefiIndexDtoResultTransformer; import de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers.AefiListEntryDtoResultTransformer; import de.symeda.sormas.backend.common.AbstractCoreAdoService; +import de.symeda.sormas.backend.common.AbstractDomainObject; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.feature.FeatureConfigurationFacadeEjb; import de.symeda.sormas.backend.immunization.entity.Immunization; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.country.Country; import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; import de.symeda.sormas.backend.infrastructure.region.Region; import de.symeda.sormas.backend.location.Location; import de.symeda.sormas.backend.person.Person; @@ -92,6 +102,24 @@ public AefiService() { super(Aefi.class, DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION); } + public Long getIdByUuid(@NotNull String uuid) { + + if (uuid == null) { + return null; + } + + CriteriaBuilder cb = em.getCriteriaBuilder(); + ParameterExpression uuidParam = cb.parameter(String.class, AbstractDomainObject.UUID); + CriteriaQuery cq = cb.createQuery(Long.class); + Root from = cq.from(Aefi.class); + cq.select(from.get(AbstractDomainObject.ID)); + cq.where(cb.equal(from.get(AbstractDomainObject.UUID), uuidParam)); + + TypedQuery q = em.createQuery(cq).setParameter(uuidParam, uuid); + + return q.getResultList().stream().findFirst().orElse(null); + } + public List getEntriesList(Long immunizationId, Integer first, Integer max) { final CriteriaBuilder cb = em.getCriteriaBuilder(); final CriteriaQuery cq = cb.createQuery(Tuple.class); @@ -205,6 +233,7 @@ public List getIndexList(AefiCriteria criteria, Integer first, Int responsibleDistrict.get(District.NAME), aefi.get(Aefi.SERIOUS), primarySuspectVaccine.get(Vaccination.VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.OTHER_VACCINE_NAME), aefi.get(Aefi.OUTCOME), primarySuspectVaccine.get(Vaccination.VACCINATION_DATE), aefi.get(Aefi.REPORT_DATE), @@ -260,6 +289,121 @@ private List getIndexListIds(AefiCriteria criteria, Integer first, Integer return aefiResultList.stream().map(t -> t.get(0, Long.class)).collect(Collectors.toList()); } + public List getExportList(AefiCriteria criteria, Collection selectedRows, int first, int max) { + + final CriteriaBuilder cb = em.getCriteriaBuilder(); + final CriteriaQuery cq = cb.createQuery(AefiExportDto.class); + final Root aefi = cq.from(Aefi.class); + + final AefiQueryContext aefiQueryContext = new AefiQueryContext(cb, cq, aefi); + AefiJoins joins = aefiQueryContext.getJoins(); + + final Join immunization = joins.getImmunization(); + final Join person = joins.getImmunizationJoins().getPerson(); + final Join immunizationFacility = joins.getImmunizationJoins().getHealthFacility(); + final Join immunizationFacilityRegion = immunizationFacility.join(Facility.REGION, JoinType.LEFT); + final Join immunizationFacilityDistrict = immunizationFacility.join(Facility.DISTRICT, JoinType.LEFT); + final Join immunizationFacilityCommunity = immunizationFacility.join(Facility.COMMUNITY, JoinType.LEFT); + + final Join personLocation = person.join(Person.ADDRESS, JoinType.LEFT); + final Join personLocationRegion = personLocation.join(Location.REGION, JoinType.LEFT); + final Join personLocationDistrict = personLocation.join(Location.DISTRICT, JoinType.LEFT); + final Join personLocationCommunity = personLocation.join(Location.COMMUNITY, JoinType.LEFT); + + final Join responsibleRegion = joins.getImmunizationJoins().getResponsibleRegion(); + final Join responsibleDistrict = joins.getImmunizationJoins().getResponsibleDistrict(); + + final Join primarySuspectVaccine = joins.getPrimarySuspectVaccination(); + final Join adverseEvents = joins.getAdverseEvents(); + + final Join reportingUser = joins.getReportingUser(); + final Join reportingUserLocation = reportingUser.join(User.ADDRESS, JoinType.LEFT); + final Join reportingUserLocationCountry = reportingUserLocation.join(Location.COUNTRY, JoinType.LEFT); + final Join reportingUserFacility = reportingUser.join(User.HEALTH_FACILITY, JoinType.LEFT); + final Join reportingUserFacilityRegion = reportingUserFacility.join(Facility.REGION, JoinType.LEFT); + final Join reportingUserFacilityDistrict = reportingUserFacility.join(Facility.DISTRICT, JoinType.LEFT); + final Join reportingUserFacilityCommunity = reportingUserFacility.join(Facility.COMMUNITY, JoinType.LEFT); + + cq.multiselect( + aefi.get(Aefi.UUID), + aefi.get(Aefi.RECEIVED_AT_NATIONAL_LEVEL_DATE), + immunizationFacility.get(Facility.NAME), + immunizationFacilityRegion.get(Region.NAME), + immunizationFacilityDistrict.get(District.NAME), + immunizationFacilityCommunity.get(Community.NAME), + reportingUserLocationCountry.get(Country.DEFAULT_NAME), + personLocationRegion.get(Region.NAME), + personLocationDistrict.get(District.NAME), + personLocationCommunity.get(Community.NAME), + personLocation.get(Location.STREET), + personLocation.get(Location.HOUSE_NUMBER), + personLocation.get(Location.POSTAL_CODE), + personLocation.get(Location.CITY), + aefi.get(Aefi.REPORTINGID_NUMBER), + aefi.get(Aefi.WORLD_WIDE_ID), + person.get(Person.FIRST_NAME), + person.get(Person.LAST_NAME), + person.get(Person.BIRTHDATE_DD), + person.get(Person.BIRTHDATE_MM), + person.get(Person.BIRTHDATE_YYYY), + aefi.get(Aefi.ONSET_AGE_YEARS), + aefi.get(Aefi.ONSET_AGE_MONTHS), + aefi.get(Aefi.ONSET_AGE_DAYS), + aefi.get(Aefi.AGE_GROUP), + person.get(Person.SEX), + aefi.get(Aefi.AEFI_DESCRIPTION), + primarySuspectVaccine.get(Vaccination.VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.OTHER_VACCINE_NAME), + primarySuspectVaccine.get(Vaccination.VACCINE_MANUFACTURER), + primarySuspectVaccine.get(Vaccination.VACCINE_BATCH_NUMBER), + primarySuspectVaccine.get(Vaccination.VACCINE_DOSE), + primarySuspectVaccine.get(Vaccination.VACCINATION_DATE), + aefi.get(Aefi.START_DATE_TIME), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION_MORE_THAN_THREE_DAYS), + adverseEvents.get(AdverseEvents.SEVERE_LOCAL_REACTION_BEYOND_NEAREST_JOINT), + adverseEvents.get(AdverseEvents.SEIZURES), + adverseEvents.get(AdverseEvents.SEIZURE_TYPE), + adverseEvents.get(AdverseEvents.ABSCESS), + adverseEvents.get(AdverseEvents.SEPSIS), + adverseEvents.get(AdverseEvents.ENCEPHALOPATHY), + adverseEvents.get(AdverseEvents.TOXIC_SHOCK_SYNDROME), + adverseEvents.get(AdverseEvents.THROMBOCYTOPENIA), + adverseEvents.get(AdverseEvents.ANAPHYLAXIS), + adverseEvents.get(AdverseEvents.FEVERISH_FEELING), + adverseEvents.get(AdverseEvents.OTHER_ADVERSE_EVENT_DETAILS), + aefi.get(Aefi.OUTCOME), + aefi.get(Aefi.SERIOUS), + reportingUser.get(User.FIRST_NAME), + reportingUser.get(User.LAST_NAME), + reportingUserFacility.get(Facility.NAME), + reportingUserFacilityRegion.get(Region.NAME), + reportingUserFacilityDistrict.get(District.NAME), + reportingUserFacilityCommunity.get(Community.NAME), + reportingUser.get(User.USER_EMAIL), + reportingUser.get(User.PHONE), + aefi.get((Aefi.REPORT_DATE)), + aefi.get((Aefi.NATIONAL_LEVEL_COMMENT)), + JurisdictionHelper.booleanSelector(cb, isInJurisdictionOrOwned(aefiQueryContext))); + + /* + * buildWhereCondition(criteria, cb, cq, aefiQueryContext, null); + * cq.distinct(true); + */ + + Predicate filter = buildExportListWhereCondition(criteria, cb, cq, aefiQueryContext); + if (selectedRows != null && !selectedRows.isEmpty()) { + filter = CriteriaBuilderHelper.andInValues(selectedRows, filter, cb, aefi.get(Aefi.UUID)); + } + + if (filter != null) { + cq.where(filter); + } + cq.distinct(true); + + return QueryHelper.getResultList(em, cq, first, max); + } + private List> sortBy(List sortProperties, AefiQueryContext aefiQueryContext) { List> selections = new ArrayList<>(); @@ -349,6 +493,22 @@ private void buildWhereCondition( } } + private Predicate buildExportListWhereCondition( + AefiCriteria criteria, + CriteriaBuilder cb, + CriteriaQuery cq, + AefiQueryContext aefiQueryContext) { + + Predicate filter = createUserFilter(aefiQueryContext); + + if (criteria != null) { + final Predicate criteriaFilter = buildCriteriaFilter(criteria, aefiQueryContext); + filter = CriteriaBuilderHelper.and(cb, filter, criteriaFilter); + } + + return filter; + } + public Predicate buildCriteriaFilter(AefiCriteria criteria, AefiQueryContext aefiQueryContext) { final AefiJoins joins = aefiQueryContext.getJoins(); @@ -365,11 +525,11 @@ public Predicate buildCriteriaFilter(AefiCriteria criteria, AefiQueryContext aef filter = CriteriaBuilderHelper.and(cb, null, cb.equal(immunization.get(Immunization.DISEASE), criteria.getDisease())); } - if (!DataHelper.isNullOrEmpty(criteria.getNameAddressPhoneEmailLike())) { + if (!DataHelper.isNullOrEmpty(criteria.getPersonLike())) { final CriteriaQuery cq = cb.createQuery(PersonIndexDto.class); final PersonQueryContext personQueryContext = new PersonQueryContext(cb, cq, joins.getImmunizationJoins().getPersonJoins()); - String[] textFilters = criteria.getNameAddressPhoneEmailLike().split("\\s+"); + String[] textFilters = criteria.getPersonLike().split("\\s+"); for (String textFilter : textFilters) { if (DataHelper.isNullOrEmpty(textFilter)) { @@ -496,7 +656,7 @@ private Predicate isInJurisdictionOrOwned(AefiQueryContext qc) { } else { filter = CriteriaBuilderHelper.or( cb, - cb.equal(qc.getRoot().get(Immunization.REPORTING_USER), currentUser), + cb.equal(qc.getRoot().get(Aefi.REPORTING_USER), currentUser), PersonJurisdictionPredicateValidator .of(qc.getQuery(), cb, new PersonJoins(qc.getJoins().getPerson()), currentUser, personService.getPermittedAssociations()) .inJurisdictionOrOwned()); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java index 1b62575f5e4..92b65406c2b 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -91,13 +88,13 @@ public class Aefi extends CoreAdo { public static final String AGE_GROUP = "ageGroup"; public static final String HEALTH_FACILITY = "healthFacility"; public static final String HEALTH_FACILITY_DETAILS = "healthFacilityDetails"; - public static final String REPORTER_NAME = "reporterName"; - public static final String REPORTER_INSTITUTION = "reporterInstitution"; - public static final String REPORTER_DESIGNATION = "reporterDesignation"; - public static final String REPORTER_DEPARTMENT = "reporterDepartment"; - public static final String REPORTER_ADDRESS = "reporterAddress"; - public static final String REPORTER_PHONE = "reporterPhone"; - public static final String REPORTER_EMAIL = "reporterEmail"; + public static final String REPORTING_OFFICER_NAME = "reportingOfficerName"; + public static final String REPORTING_OFFICER_FACILITY = "reportingOfficerFacility"; + public static final String REPORTING_OFFICER_DESIGNATION = "reportingOfficerDesignation"; + public static final String REPORTING_OFFICER_DEPARTMENT = "reportingOfficerDepartment"; + public static final String REPORTING_OFFICER_ADDRESS = "reportingOfficerAddress"; + public static final String REPORTING_OFFICER_PHONE_NUMBER = "reportingOfficerPhoneNumber"; + public static final String REPORTING_OFFICER_EMAIL = "reportingOfficerEmail"; public static final String TODAYS_DATE = "todaysDate"; public static final String START_DATE_TIME = "startDateTime"; public static final String AEFI_DESCRIPTION = "aefiDescription"; @@ -140,13 +137,13 @@ public class Aefi extends CoreAdo { private AefiAgeGroup ageGroup; private Facility healthFacility; private String healthFacilityDetails; - private String reporterName; - private Facility reporterInstitution; - private String reporterDesignation; - private String reporterDepartment; - private Location reporterAddress; - private String reporterPhone; - private String reporterEmail; + private String reportingOfficerName; + private Facility reportingOfficerFacility; + private String reportingOfficerDesignation; + private String reportingOfficerDepartment; + private Location reportingOfficerAddress; + private String reportingOfficerPhoneNumber; + private String reportingOfficerEmail; private Date todaysDate; private Date startDateTime; private String aefiDescription; @@ -323,6 +320,7 @@ public void setReportingIdNumber(String reportingIdNumber) { this.reportingIdNumber = reportingIdNumber; } + @Column public String getPhoneNumber() { return phoneNumber; } @@ -358,6 +356,7 @@ public void setLactating(YesNoUnknown lactating) { this.lactating = lactating; } + @Column public Integer getOnsetAgeYears() { return onsetAgeYears; } @@ -366,6 +365,7 @@ public void setOnsetAgeYears(Integer onsetAgeYears) { this.onsetAgeYears = onsetAgeYears; } + @Column public Integer getOnsetAgeMonths() { return onsetAgeMonths; } @@ -374,6 +374,7 @@ public void setOnsetAgeMonths(Integer onsetAgeMonths) { this.onsetAgeMonths = onsetAgeMonths; } + @Column public Integer getOnsetAgeDays() { return onsetAgeDays; } @@ -409,63 +410,68 @@ public void setHealthFacilityDetails(String healthFacilityDetails) { this.healthFacilityDetails = healthFacilityDetails; } - public String getReporterName() { - return reporterName; + @Column + public String getReportingOfficerName() { + return reportingOfficerName; } - public void setReporterName(String reporterName) { - this.reporterName = reporterName; + public void setReportingOfficerName(String reporterName) { + this.reportingOfficerName = reporterName; } @ManyToOne(fetch = FetchType.LAZY) - public Facility getReporterInstitution() { - return reporterInstitution; + public Facility getReportingOfficerFacility() { + return reportingOfficerFacility; } - public void setReporterInstitution(Facility reporterInstitution) { - this.reporterInstitution = reporterInstitution; + public void setReportingOfficerFacility(Facility reporterInstitution) { + this.reportingOfficerFacility = reporterInstitution; } - public String getReporterDesignation() { - return reporterDesignation; + @Column + public String getReportingOfficerDesignation() { + return reportingOfficerDesignation; } - public void setReporterDesignation(String reporterDesignation) { - this.reporterDesignation = reporterDesignation; + public void setReportingOfficerDesignation(String reporterDesignation) { + this.reportingOfficerDesignation = reporterDesignation; } - public String getReporterDepartment() { - return reporterDepartment; + @Column + public String getReportingOfficerDepartment() { + return reportingOfficerDepartment; } - public void setReporterDepartment(String reporterDepartment) { - this.reporterDepartment = reporterDepartment; + public void setReportingOfficerDepartment(String reporterDepartment) { + this.reportingOfficerDepartment = reporterDepartment; } @ManyToOne(cascade = CascadeType.ALL, fetch = FetchType.LAZY) - @JoinColumn(name = "reporteraddress_id") - public Location getReporterAddress() { - return reporterAddress; + @JoinColumn(name = "reportingofficeraddress_id") + public Location getReportingOfficerAddress() { + return reportingOfficerAddress; } - public void setReporterAddress(Location reporterAddress) { - this.reporterAddress = reporterAddress; + public void setReportingOfficerAddress(Location reporterAddress) { + this.reportingOfficerAddress = reporterAddress; } - public String getReporterPhone() { - return reporterPhone; + @Column + public String getReportingOfficerPhoneNumber() { + return reportingOfficerPhoneNumber; } - public void setReporterPhone(String reporterPhone) { - this.reporterPhone = reporterPhone; + public void setReportingOfficerPhoneNumber(String reporterPhone) { + this.reportingOfficerPhoneNumber = reporterPhone; } - public String getReporterEmail() { - return reporterEmail; + @Column + public String getReportingOfficerEmail() { + return reportingOfficerEmail; } - public void setReporterEmail(String reporterEmail) { - this.reporterEmail = reporterEmail; + public void setReportingOfficerEmail(String reporterEmail) { + this.reportingOfficerEmail = reporterEmail; } @Temporal(TemporalType.TIMESTAMP) @@ -587,6 +593,7 @@ public void setReceivedAtNationalLevelDate(Date receivedAtNationalLevelDate) { this.receivedAtNationalLevelDate = receivedAtNationalLevelDate; } + @Column public String getWorldwideId() { return worldwideId; } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java new file mode 100644 index 00000000000..e4157a4c6c0 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java @@ -0,0 +1,1832 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity; + +import java.util.Arrays; +import java.util.Date; +import java.util.HashSet; +import java.util.Set; +import java.util.stream.Collectors; + +import javax.persistence.CascadeType; +import javax.persistence.Column; +import javax.persistence.Entity; +import javax.persistence.EnumType; +import javax.persistence.Enumerated; +import javax.persistence.FetchType; +import javax.persistence.JoinColumn; +import javax.persistence.JoinTable; +import javax.persistence.ManyToMany; +import javax.persistence.ManyToOne; +import javax.persistence.OneToOne; +import javax.persistence.Table; +import javax.persistence.Temporal; +import javax.persistence.TemporalType; +import javax.persistence.Transient; + +import org.apache.commons.lang3.StringUtils; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCausality; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassificationSubType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiImmunizationPeriod; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStage; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiVaccinationPeriod; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.BirthTerm; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.DeliveryProcedure; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PatientStatusAtAefiInvestigation; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PlaceOfVaccination; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeriousAefiInfoSource; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SyringeType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccinationActivity; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccinationSite; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccineCarrier; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.backend.common.CoreAdo; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.country.Country; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.vaccination.Vaccination; + +@Entity +@Table(name = "adverseeventsfollowingimmunizationinvestigation") +public class AefiInvestigation extends CoreAdo { + + private static final long serialVersionUID = 6128204752074963848L; + + public static final String TABLE_NAME = "adverseeventsfollowingimmunizationinvestigation"; + public static final String AEFI_INVESTIGATION_VACCINATIONS_TABLE_NAME = "adverseeventsfollowingimmunizationinvestigation_vaccinations"; + + public static final String AEFI_REPORT = "aefiReport"; + public static final String AEFI_REPORT_ID = "aefiReportId"; + public static final String ADDRESS = "address"; + public static final String VACCINATIONS = "vaccinations"; + public static final String PRIMARY_SUSPECT_VACCINE = "primarySuspectVaccine"; + public static final String REPORT_DATE = "reportDate"; + public static final String REPORTING_USER = "reportingUser"; + public static final String EXTERNAL_ID = "externalId"; + public static final String RESPONSIBLE_REGION = "responsibleRegion"; + public static final String RESPONSIBLE_DISTRICT = "responsibleDistrict"; + public static final String RESPONSIBLE_COMMUNITY = "responsibleCommunity"; + public static final String COUNTRY = "country"; + public static final String INVESTIGATION_CASE_ID = "investigationCaseId"; + public static final String PLACE_OF_VACCINATION = "placeOfVaccination"; + public static final String PLACE_OF_VACCINATION_DETAILS = "placeOfVaccinationDetails"; + public static final String VACCINATION_ACTIVITY = "vaccinationActivity"; + public static final String VACCINATION_ACTIVITY_DETAILS = "vaccinationActivityDetails"; + public static final String VACCINATION_FACILITY = "vaccinationFacility"; + public static final String VACCINATION_FACILITY_DETAILS = "vaccinationFacilityDetails"; + public static final String REPORTING_OFFICER_NAME = "reportingOfficerName"; + public static final String REPORTING_OFFICER_FACILITY = "reportingOfficerFacility"; + public static final String REPORTING_OFFICER_FACILITY_DETAILS = "reportingOfficerFacilityDetails"; + public static final String REPORTING_OFFICER_DESIGNATION = "reportingOfficerDesignation"; + public static final String REPORTING_OFFICER_DEPARTMENT = "reportingOfficerDepartment"; + public static final String REPORTING_OFFICER_ADDRESS = "reportingOfficerAddress"; + public static final String REPORTING_OFFICER_LANDLINE_PHONE_NUMBER = "reportingOfficerLandlinePhoneNumber"; + public static final String REPORTING_OFFICER_MOBILE_PHONE_NUMBER = "reportingOfficerMobilePhoneNumber"; + public static final String REPORTING_OFFICER_EMAIL = "reportingOfficerEmail"; + public static final String INVESTIGATION_DATE = "investigationDate"; + public static final String FORM_COMPLETION_DATE = "formCompletionDate"; + public static final String INVESTIGATION_STAGE = "investigationStage"; + public static final String TYPE_OF_SITE = "typeOfSite"; + public static final String TYPE_OF_SITE_DETAILS = "typeOfSiteDetails"; + public static final String KEY_SYMPTOM_DATE_TIME = "keySymptomDateTime"; + public static final String HOSPITALIZATION_DATE = "hospitalizationDate"; + public static final String REPORTED_TO_HEALTH_AUTHORITY_DATE = "reportedToHealthAuthorityDate"; + public static final String STATUS_ON_DATE_OF_INVESTIGATION = "statusOnDateOfInvestigation"; + public static final String DEATH_DATE_TIME = "deathDateTime"; + public static final String AUTOPSY_DONE = "autopsyDone"; + public static final String AUTOPSY_DATE = "autopsyDate"; + public static final String AUTOPSY_PLANNED_DATE_TIME = "autopsyPlannedDateTime"; + public static final String PAST_HISTORY_OF_SIMILAR_EVENT = "pastHistoryOfSimilarEvent"; + public static final String PAST_HISTORY_OF_SIMILAR_EVENT_DETAILS = "pastHistoryOfSimilarEventDetails"; + public static final String ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS = "adverseEventAfterPreviousVaccinations"; + public static final String ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS_DETAILS = "adverseEventAfterPreviousVaccinationsDetails"; + public static final String HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD = "historyOfAllergyToVaccineDrugOrFood"; + public static final String HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD_DETAILS = "historyOfAllergyToVaccineDrugOrFoodDetails"; + public static final String PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER = "preExistingIllnessThirtyDaysOrCongenitalDisorder"; + public static final String PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER_DETAILS = + "preExistingIllnessThirtyDaysOrCongenitalDisorderDetails"; + public static final String HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE = "historyOfHospitalizationInLastThirtyDaysWithCause"; + public static final String HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE_DETAILS = + "historyOfHospitalizationInLastThirtyDaysWithCauseDetails"; + public static final String CURRENTLY_ON_CONCOMITANT_MEDICATION = "currentlyOnConcomitantMedication"; + public static final String CURRENTLY_ON_CONCOMITANT_MEDICATION_DETAILS = "currentlyOnConcomitantMedicationDetails"; + public static final String FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY = "familyHistoryOfDiseaseOrAllergy"; + public static final String FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY_DETAILS = "familyHistoryOfDiseaseOrAllergyDetails"; + public static final String NUMBER_OF_WEEKS_PREGNANT = "numberOfWeeksPregnant"; + public static final String BIRTH_TERM = "birthTerm"; + public static final String BIRTH_WEIGHT = "birthWeight"; + public static final String DELIVERY_PROCEDURE = "deliveryProcedure"; + public static final String DELIVERY_PROCEDURE_DETAILS = "deliveryProcedureDetails"; + public static final String SERIOUS_AEFI_INFO_SOURCE_STRING = "seriousAefiInfoSourceString"; + public static final String SERIOUS_AEFI_INFO_SOURCE_DETAILS = "seriousAefiInfoSourceDetails"; + public static final String SERIOUS_AEFI_VERBAL_AUTOPSY_INFO_SOURCE_DETAILS = "seriousAefiVerbalAutopsyInfoSourceDetails"; + public static final String FIRST_CAREGIVERS_NAME = "firstCaregiversName"; + public static final String OTHER_CAREGIVERS_NAMES = "otherCaregiversNames"; + public static final String OTHER_SOURCES_WHO_PROVIDED_INFO = "otherSourcesWhoProvidedInfo"; + public static final String SIGNS_AND_SYMPTOMS_FROM_TIME_OF_VACCINATION = "signsAndSymptomsFromTimeOfVaccination"; + public static final String CLINICAL_DETAILS_OFFICER_NAME = "clinicalDetailsOfficerName"; + public static final String CLINICAL_DETAILS_OFFICER_PHONE_NUMBER = "clinicalDetailsOfficerPhoneNumber"; + public static final String CLINICAL_DETAILS_OFFICER_EMAIL = "clinicalDetailsOfficerEmail"; + public static final String CLINICAL_DETAILS_OFFICER_DESIGNATION = "clinicalDetailsOfficerDesignation"; + public static final String CLINICAL_DETAILS_DATE_TIME = "clinicalDetailsDateTime"; + public static final String PATIENT_RECEIVED_MEDICAL_CARE = "patientReceivedMedicalCare"; + public static final String PATIENT_RECEIVED_MEDICAL_CARE_DETAILS = "patientReceivedMedicalCareDetails"; + public static final String PROVISIONAL_OR_FINAL_DIAGNOSIS = "provisionalOrFinalDiagnosis"; + public static final String PATIENT_IMMUNIZED_PERIOD = "patientImmunizedPeriod"; + public static final String PATIENT_IMMUNIZED_PERIOD_DETAILS = "patientImmunizedPeriodDetails"; + public static final String VACCINE_GIVEN_PERIOD = "vaccineGivenPeriod"; + public static final String VACCINE_GIVEN_PERIOD_DETAILS = "vaccineGivenPeriodDetails"; + public static final String ERROR_PRESCRIBING_VACCINE = "errorPrescribingVaccine"; + public static final String ERROR_PRESCRIBING_VACCINE_DETAILS = "errorPrescribingVaccineDetails"; + public static final String VACCINE_COULD_HAVE_BEEN_UNSTERILE = "vaccineCouldHaveBeenUnSterile"; + public static final String VACCINE_COULD_HAVE_BEEN_UNSTERILE_DETAILS = "vaccineCouldHaveBeenUnSterileDetails"; + public static final String VACCINE_PHYSICAL_CONDITION_ABNORMAL = "vaccinePhysicalConditionAbnormal"; + public static final String VACCINE_PHYSICAL_CONDITION_ABNORMAL_DETAILS = "vaccinePhysicalConditionAbnormalDetails"; + public static final String ERROR_IN_VACCINE_RECONSTITUTION = "errorInVaccineReconstitution"; + public static final String ERROR_IN_VACCINE_RECONSTITUTION_DETAILS = "errorInVaccineReconstitutionDetails"; + public static final String ERROR_IN_VACCINE_HANDLING = "errorInVaccineHandling"; + public static final String ERROR_IN_VACCINE_HANDLING_DETAILS = "errorInVaccineHandlingDetails"; + public static final String VACCINE_ADMINISTERED_INCORRECTLY = "vaccineAdministeredIncorrectly"; + public static final String VACCINE_ADMINISTERED_INCORRECTLY_DETAILS = "vaccineAdministeredIncorrectlyDetails"; + public static final String NUMBER_IMMUNIZED_FROM_CONCERNED_VACCINE_VIAL = "numberImmunizedFromConcernedVaccineVial"; + public static final String NUMBER_IMMUNIZED_WITH_CONCERNED_VACCINE_IN_SAME_SESSION = "numberImmunizedWithConcernedVaccineInSameSession"; + public static final String NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_OTHER_LOCATIONS = + "numberImmunizedConcernedVaccineSameBatchNumberOtherLocations"; + public static final String NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_LOCATION_DETAILS = + "numberImmunizedConcernedVaccineSameBatchNumberLocationDetails"; + public static final String VACCINE_HAS_QUALITY_DEFECT = "vaccineHasQualityDefect"; + public static final String VACCINE_HAS_QUALITY_DEFECT_DETAILS = "vaccineHasQualityDefectDetails"; + public static final String EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION = "eventIsAStressResponseRelatedToImmunization"; + public static final String EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION_DETAILS = "eventIsAStressResponseRelatedToImmunizationDetails"; + public static final String CASE_IS_PART_OF_A_CLUSTER = "caseIsPartOfACluster"; + public static final String CASE_IS_PART_OF_A_CLUSTER_DETAILS = "caseIsPartOfAClusterDetails"; + public static final String NUMBER_OF_CASES_DETECTED_IN_CLUSTER = "numberOfCasesDetectedInCluster"; + public static final String ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL = "allCasesInClusterReceivedVaccineFromSameVial"; + public static final String ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL_DETAILS = "allCasesInClusterReceivedVaccineFromSameVialDetails"; + public static final String NUMBER_OF_VIALS_USED_IN_CLUSTER = "numberOfVialsUsedInCluster"; + public static final String NUMBER_OF_VIALS_USED_IN_CLUSTER_DETAILS = "numberOfVialsUsedInClusterDetails"; + public static final String AD_SYRINGES_USED_FOR_IMMUNIZATION = "adSyringesUsedForImmunization"; + public static final String TYPE_OF_SYRINGES_USED = "typeOfSyringesUsed"; + public static final String TYPE_OF_SYRINGES_USED_DETAILS = "typeOfSyringesUsedDetails"; + public static final String SYRINGES_USED_ADDITIONAL_DETAILS = "syringesUsedAdditionalDetails"; + public static final String SAME_RECONSTITUTION_SYRINGE_USED_FOR_MULTIPLE_VIALS_OF_SAME_VACCINE = + "sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine"; + public static final String SAME_RECONSTITUTION_SYRINGE_USED_FOR_RECONSTITUTING_DIFFERENT_VACCINES = + "sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines"; + public static final String SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINE_VIAL = "sameReconstitutionSyringeForEachVaccineVial"; + public static final String SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINATION = "sameReconstitutionSyringeForEachVaccination"; + public static final String VACCINES_AND_DILUENTS_USED_RECOMMENDED_BY_MANUFACTURER = "vaccinesAndDiluentsUsedRecommendedByManufacturer"; + public static final String RECONSTITUTION_ADDITIONAL_DETAILS = "reconstitutionAdditionalDetails"; + public static final String CORRECT_DOSE_OR_ROUTE = "correctDoseOrRoute"; + public static final String TIME_OF_RECONSTITUTION_MENTIONED_ON_THE_VIAL = "timeOfReconstitutionMentionedOnTheVial"; + public static final String NON_TOUCH_TECHNIQUE_FOLLOWED = "nonTouchTechniqueFollowed"; + public static final String CONTRAINDICATION_SCREENED_PRIOR_TO_VACCINATION = "contraIndicationScreenedPriorToVaccination"; + public static final String NUMBER_OF_AEFI_REPORTED_FROM_VACCINE_DISTRIBUTION_CENTER_LAST_THIRTY_DAYS = + "numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays"; + public static final String TRAINING_RECEIVED_BY_VACCINATOR = "trainingReceivedByVaccinator"; + public static final String LAST_TRAINING_RECEIVED_BY_VACCINATOR_DATE = "lastTrainingReceivedByVaccinatorDate"; + public static final String INJECTION_TECHNIQUE_ADDITIONAL_DETAILS = "injectionTechniqueAdditionalDetails"; + public static final String VACCINE_STORAGE_REFRIGERATOR_TEMPERATURE_MONITORED = "vaccineStorageRefrigeratorTemperatureMonitored"; + public static final String ANY_STORAGE_TEMPERATURE_DEVIATION_OUTSIDE_TWO_TO_EIGHT_DEGREES = + "anyStorageTemperatureDeviationOutsideTwoToEightDegrees"; + public static final String STORAGE_TEMPERATURE_MONITORING_ADDITIONAL_DETAILS = "storageTemperatureMonitoringAdditionalDetails"; + public static final String CORRECT_PROCEDURE_FOR_STORAGE_FOLLOWED = "correctProcedureForStorageFollowed"; + public static final String ANY_OTHER_ITEM_IN_REFRIGERATOR = "anyOtherItemInRefrigerator"; + public static final String PARTIALLY_USED_RECONSTITUTED_VACCINES_IN_REFRIGERATOR = "partiallyUsedReconstitutedVaccinesInRefrigerator"; + public static final String UNUSABLE_VACCINES_IN_REFRIGERATOR = "unusableVaccinesInRefrigerator"; + public static final String UNUSABLE_DILUENTS_IN_STORE = "unusableDiluentsInStore"; + public static final String VACCINE_STORAGE_POINT_ADDITIONAL_DETAILS = "vaccineStoragePointAdditionalDetails"; + public static final String VACCINE_CARRIER_TYPE = "vaccineCarrierType"; + public static final String VACCINE_CARRIER_TYPE_DETAILS = "vaccineCarrierTypeDetails"; + public static final String VACCINE_CARRIER_SENT_TO_SITE_ON_SAME_DATE_AS_VACCINATION = "vaccineCarrierSentToSiteOnSameDateAsVaccination"; + public static final String VACCINE_CARRIER_RETURNED_FROM_SITE_ON_SAME_DATE_AS_VACCINATION = + "vaccineCarrierReturnedFromSiteOnSameDateAsVaccination"; + public static final String CONDITIONED_ICE_PACK_USED = "conditionedIcepackUsed"; + public static final String VACCINE_TRANSPORTATION_ADDITIONAL_DETAILS = "vaccineTransportationAdditionalDetails"; + public static final String SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY = "similarEventsReportedSamePeriodAndLocality"; + public static final String SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY_DETAILS = "similarEventsReportedSamePeriodAndLocalityDetails"; + public static final String NUMBER_OF_SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY = "numberOfSimilarEventsReportedSamePeriodAndLocality"; + public static final String NUMBER_OF_THOSE_AFFECTED_VACCINATED = "numberOfThoseAffectedVaccinated"; + public static final String NUMBER_OF_THOSE_AFFECTED_NOT_VACCINATED = "numberOfThoseAffectedNotVaccinated"; + public static final String NUMBER_OF_THOSE_AFFECTED_VACCINATED_UNKNOWN = "numberOfThoseAffectedVaccinatedUnknown"; + public static final String COMMUNITY_INVESTIGATION_ADDITIONAL_DETAILS = "communityInvestigationAdditionalDetails"; + public static final String OTHER_INVESTIGATION_FINDINGS = "otherInvestigationFindings"; + public static final String INVESTIGATION_STATUS = "investigationStatus"; + public static final String INVESTIGATION_STATUS_DETAILS = "investigationStatusDetails"; + public static final String AEFI_CLASSIFICATION = "aefiClassification"; + public static final String AEFI_CLASSIFICATION_SUB_TYPE = "aefiClassificationSubType"; + public static final String AEFI_CLASSIFICATION_DETAILS = "aefiClassificationDetails"; + public static final String CAUSALITY = "causality"; + public static final String CAUSALITY_DETAILS = "causalityDetails"; + public static final String INVESTIGATION_COMPLETION_DATE = "investigationCompletionDate"; + + private Aefi aefiReport; + private Long aefiReportId; + private Location address; + private Set vaccinations; + private Vaccination primarySuspectVaccine; + private Date reportDate; + private User reportingUser; + private String externalId; + private Region responsibleRegion; + private District responsibleDistrict; + private Community responsibleCommunity; + private Country country; + private String investigationCaseId; + private PlaceOfVaccination placeOfVaccination; + private String placeOfVaccinationDetails; + private VaccinationActivity vaccinationActivity; + private String vaccinationActivityDetails; + private Facility vaccinationFacility; + private String vaccinationFacilityDetails; + private String reportingOfficerName; + private Facility reportingOfficerFacility; + private String reportingOfficerFacilityDetails; + private String reportingOfficerDesignation; + private String reportingOfficerDepartment; + private Location reportingOfficerAddress; + private String reportingOfficerLandlinePhoneNumber; + private String reportingOfficerMobilePhoneNumber; + private String reportingOfficerEmail; + private Date investigationDate; + private Date formCompletionDate; + private AefiInvestigationStage investigationStage; + private VaccinationSite typeOfSite; + private String typeOfSiteDetails; + private Date keySymptomDateTime; + private Date hospitalizationDate; + private Date reportedToHealthAuthorityDate; + private PatientStatusAtAefiInvestigation statusOnDateOfInvestigation; + private Date deathDateTime; + private YesNoUnknown autopsyDone; + private Date autopsyDate; + private Date autopsyPlannedDateTime; + private YesNoUnknown pastHistoryOfSimilarEvent; + private String pastHistoryOfSimilarEventDetails; + private YesNoUnknown adverseEventAfterPreviousVaccinations; + private String adverseEventAfterPreviousVaccinationsDetails; + private YesNoUnknown historyOfAllergyToVaccineDrugOrFood; + private String historyOfAllergyToVaccineDrugOrFoodDetails; + private YesNoUnknown preExistingIllnessThirtyDaysOrCongenitalDisorder; + private String preExistingIllnessThirtyDaysOrCongenitalDisorderDetails; + private YesNoUnknown historyOfHospitalizationInLastThirtyDaysWithCause; + private String historyOfHospitalizationInLastThirtyDaysWithCauseDetails; + private YesNoUnknown currentlyOnConcomitantMedication; + private String currentlyOnConcomitantMedicationDetails; + private YesNoUnknown familyHistoryOfDiseaseOrAllergy; + private String familyHistoryOfDiseaseOrAllergyDetails; + private Integer numberOfWeeksPregnant; + private BirthTerm birthTerm; + private Float birthWeight; + private DeliveryProcedure deliveryProcedure; + private String deliveryProcedureDetails; + private Set seriousAefiInfoSource; + private String seriousAefiInfoSourceString; + private String seriousAefiInfoSourceDetails; + private String seriousAefiVerbalAutopsyInfoSourceDetails; + private String firstCaregiversName; + private String otherCaregiversNames; + private String otherSourcesWhoProvidedInfo; + private String signsAndSymptomsFromTimeOfVaccination; + private String clinicalDetailsOfficerName; + private String clinicalDetailsOfficerPhoneNumber; + private String clinicalDetailsOfficerEmail; + private String clinicalDetailsOfficerDesignation; + private Date clinicalDetailsDateTime; + private YesNoUnknown patientReceivedMedicalCare; + private String patientReceivedMedicalCareDetails; + private String provisionalOrFinalDiagnosis; + private AefiImmunizationPeriod patientImmunizedPeriod; + private String patientImmunizedPeriodDetails; + private AefiVaccinationPeriod vaccineGivenPeriod; + private String vaccineGivenPeriodDetails; + private YesNoUnknown errorPrescribingVaccine; + private String errorPrescribingVaccineDetails; + private YesNoUnknown vaccineCouldHaveBeenUnSterile; + private String vaccineCouldHaveBeenUnSterileDetails; + private YesNoUnknown vaccinePhysicalConditionAbnormal; + private String vaccinePhysicalConditionAbnormalDetails; + private YesNoUnknown errorInVaccineReconstitution; + private String errorInVaccineReconstitutionDetails; + private YesNoUnknown errorInVaccineHandling; + private String errorInVaccineHandlingDetails; + private YesNoUnknown vaccineAdministeredIncorrectly; + private String vaccineAdministeredIncorrectlyDetails; + private Integer numberImmunizedFromConcernedVaccineVial; + private Integer numberImmunizedWithConcernedVaccineInSameSession; + private Integer numberImmunizedConcernedVaccineSameBatchNumberOtherLocations; + private String numberImmunizedConcernedVaccineSameBatchNumberLocationDetails; + private YesNoUnknown vaccineHasQualityDefect; + private String vaccineHasQualityDefectDetails; + private YesNoUnknown eventIsAStressResponseRelatedToImmunization; + private String eventIsAStressResponseRelatedToImmunizationDetails; + private YesNoUnknown caseIsPartOfACluster; + private String caseIsPartOfAClusterDetails; + private Integer numberOfCasesDetectedInCluster; + private YesNoUnknown allCasesInClusterReceivedVaccineFromSameVial; + private String allCasesInClusterReceivedVaccineFromSameVialDetails; + private Integer numberOfVialsUsedInCluster; + private String numberOfVialsUsedInClusterDetails; + private YesNoUnknown adSyringesUsedForImmunization; + private SyringeType typeOfSyringesUsed; + private String typeOfSyringesUsedDetails; + private String syringesUsedAdditionalDetails; + private YesNoUnknown sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine; + private YesNoUnknown sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines; + private YesNoUnknown sameReconstitutionSyringeForEachVaccineVial; + private YesNoUnknown sameReconstitutionSyringeForEachVaccination; + private YesNoUnknown vaccinesAndDiluentsUsedRecommendedByManufacturer; + private String reconstitutionAdditionalDetails; + private YesNoUnknown correctDoseOrRoute; + private YesNoUnknown timeOfReconstitutionMentionedOnTheVial; + private YesNoUnknown nonTouchTechniqueFollowed; + private YesNoUnknown contraIndicationScreenedPriorToVaccination; + private Integer numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays; + private YesNoUnknown trainingReceivedByVaccinator; + private Date lastTrainingReceivedByVaccinatorDate; + private String injectionTechniqueAdditionalDetails; + private YesNoUnknown vaccineStorageRefrigeratorTemperatureMonitored; + private YesNoUnknown anyStorageTemperatureDeviationOutsideTwoToEightDegrees; + private String storageTemperatureMonitoringAdditionalDetails; + private YesNoUnknown correctProcedureForStorageFollowed; + private YesNoUnknown anyOtherItemInRefrigerator; + private YesNoUnknown partiallyUsedReconstitutedVaccinesInRefrigerator; + private YesNoUnknown unusableVaccinesInRefrigerator; + private YesNoUnknown unusableDiluentsInStore; + private String vaccineStoragePointAdditionalDetails; + private VaccineCarrier vaccineCarrierType; + private String vaccineCarrierTypeDetails; + private YesNoUnknown vaccineCarrierSentToSiteOnSameDateAsVaccination; + private YesNoUnknown vaccineCarrierReturnedFromSiteOnSameDateAsVaccination; + private YesNoUnknown conditionedIcepackUsed; + private String vaccineTransportationAdditionalDetails; + private YesNoUnknown similarEventsReportedSamePeriodAndLocality; + private String similarEventsReportedSamePeriodAndLocalityDetails; + private Integer numberOfSimilarEventsReportedSamePeriodAndLocality; + private Integer numberOfThoseAffectedVaccinated; + private Integer numberOfThoseAffectedNotVaccinated; + private Integer numberOfThoseAffectedVaccinatedUnknown; + private String communityInvestigationAdditionalDetails; + private String otherInvestigationFindings; + private AefiInvestigationStatus investigationStatus; + private String investigationStatusDetails; + private AefiClassification aefiClassification; + private AefiClassificationSubType aefiClassificationSubType; + private String aefiClassificationDetails; + private AefiCausality causality; + private String causalityDetails; + private Date investigationCompletionDate; + + public static AefiInvestigation build() { + AefiInvestigation aefiInvestigation = new AefiInvestigation(); + return aefiInvestigation; + } + + @ManyToOne + @JoinColumn(name = "adverseeventsfollowingimmunization_id", nullable = false) + public Aefi getAefiReport() { + return aefiReport; + } + + public void setAefiReport(Aefi aefiReport) { + this.aefiReport = aefiReport; + } + + @Column(name = "adverseeventsfollowingimmunization_id", updatable = false, insertable = false) + public Long getAefiReportId() { + return aefiReportId; + } + + public void setAefiReportId(Long aefiReportId) { + this.aefiReportId = aefiReportId; + } + + @ManyToOne(cascade = CascadeType.ALL, fetch = FetchType.LAZY) + public Location getAddress() { + return address; + } + + public void setAddress(Location address) { + this.address = address; + } + + @ManyToMany(fetch = FetchType.LAZY) + @JoinTable(name = AEFI_INVESTIGATION_VACCINATIONS_TABLE_NAME, + joinColumns = @JoinColumn(name = "adverseeventsfollowingimmunizationinvestigation_id"), + inverseJoinColumns = @JoinColumn(name = "vaccination_id")) + public Set getVaccinations() { + return vaccinations; + } + + public void setVaccinations(Set vaccinations) { + this.vaccinations = vaccinations; + } + + @OneToOne + public Vaccination getPrimarySuspectVaccine() { + return primarySuspectVaccine; + } + + public void setPrimarySuspectVaccine(Vaccination primarySuspectVaccine) { + this.primarySuspectVaccine = primarySuspectVaccine; + } + + @Temporal(TemporalType.TIMESTAMP) + @Column(nullable = false) + public Date getReportDate() { + return reportDate; + } + + public void setReportDate(Date reportDate) { + this.reportDate = reportDate; + } + + @ManyToOne(fetch = FetchType.LAZY) + @JoinColumn(nullable = false) + public User getReportingUser() { + return reportingUser; + } + + public void setReportingUser(User reportingUser) { + this.reportingUser = reportingUser; + } + + public String getExternalId() { + return externalId; + } + + public void setExternalId(String externalId) { + this.externalId = externalId; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Region getResponsibleRegion() { + return responsibleRegion; + } + + public void setResponsibleRegion(Region responsibleRegion) { + this.responsibleRegion = responsibleRegion; + } + + @ManyToOne(fetch = FetchType.LAZY) + public District getResponsibleDistrict() { + return responsibleDistrict; + } + + public void setResponsibleDistrict(District responsibleDistrict) { + this.responsibleDistrict = responsibleDistrict; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Community getResponsibleCommunity() { + return responsibleCommunity; + } + + public void setResponsibleCommunity(Community responsibleCommunity) { + this.responsibleCommunity = responsibleCommunity; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Country getCountry() { + return country; + } + + public void setCountry(Country country) { + this.country = country; + } + + public String getInvestigationCaseId() { + return investigationCaseId; + } + + public void setInvestigationCaseId(String investigationCaseId) { + this.investigationCaseId = investigationCaseId; + } + + @Enumerated(EnumType.STRING) + public PlaceOfVaccination getPlaceOfVaccination() { + return placeOfVaccination; + } + + public void setPlaceOfVaccination(PlaceOfVaccination placeOfVaccination) { + this.placeOfVaccination = placeOfVaccination; + } + + public String getPlaceOfVaccinationDetails() { + return placeOfVaccinationDetails; + } + + public void setPlaceOfVaccinationDetails(String placeOfVaccinationDetails) { + this.placeOfVaccinationDetails = placeOfVaccinationDetails; + } + + @Enumerated(EnumType.STRING) + public VaccinationActivity getVaccinationActivity() { + return vaccinationActivity; + } + + public void setVaccinationActivity(VaccinationActivity vaccinationActivity) { + this.vaccinationActivity = vaccinationActivity; + } + + public String getVaccinationActivityDetails() { + return vaccinationActivityDetails; + } + + public void setVaccinationActivityDetails(String vaccinationActivityDetails) { + this.vaccinationActivityDetails = vaccinationActivityDetails; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Facility getVaccinationFacility() { + return vaccinationFacility; + } + + public void setVaccinationFacility(Facility vaccinationFacility) { + this.vaccinationFacility = vaccinationFacility; + } + + public String getVaccinationFacilityDetails() { + return vaccinationFacilityDetails; + } + + public void setVaccinationFacilityDetails(String vaccinationFacilityDetails) { + this.vaccinationFacilityDetails = vaccinationFacilityDetails; + } + + public String getReportingOfficerName() { + return reportingOfficerName; + } + + public void setReportingOfficerName(String reportingOfficerName) { + this.reportingOfficerName = reportingOfficerName; + } + + @ManyToOne(fetch = FetchType.LAZY) + public Facility getReportingOfficerFacility() { + return reportingOfficerFacility; + } + + public void setReportingOfficerFacility(Facility reportingOfficerFacility) { + this.reportingOfficerFacility = reportingOfficerFacility; + } + + public String getReportingOfficerFacilityDetails() { + return reportingOfficerFacilityDetails; + } + + public void setReportingOfficerFacilityDetails(String reportingOfficerFacilityDetails) { + this.reportingOfficerFacilityDetails = reportingOfficerFacilityDetails; + } + + public String getReportingOfficerDesignation() { + return reportingOfficerDesignation; + } + + public void setReportingOfficerDesignation(String reportingOfficerDesignation) { + this.reportingOfficerDesignation = reportingOfficerDesignation; + } + + public String getReportingOfficerDepartment() { + return reportingOfficerDepartment; + } + + public void setReportingOfficerDepartment(String reportingOfficerDepartment) { + this.reportingOfficerDepartment = reportingOfficerDepartment; + } + + @ManyToOne(cascade = CascadeType.ALL, fetch = FetchType.LAZY) + @JoinColumn(name = "reportingofficeraddress_id") + public Location getReportingOfficerAddress() { + return reportingOfficerAddress; + } + + public void setReportingOfficerAddress(Location reportingOfficerAddress) { + this.reportingOfficerAddress = reportingOfficerAddress; + } + + public String getReportingOfficerLandlinePhoneNumber() { + return reportingOfficerLandlinePhoneNumber; + } + + public void setReportingOfficerLandlinePhoneNumber(String reportingOfficerLandlinePhoneNumber) { + this.reportingOfficerLandlinePhoneNumber = reportingOfficerLandlinePhoneNumber; + } + + public String getReportingOfficerMobilePhoneNumber() { + return reportingOfficerMobilePhoneNumber; + } + + public void setReportingOfficerMobilePhoneNumber(String reportingOfficerMobilePhoneNumber) { + this.reportingOfficerMobilePhoneNumber = reportingOfficerMobilePhoneNumber; + } + + public String getReportingOfficerEmail() { + return reportingOfficerEmail; + } + + public void setReportingOfficerEmail(String reportingOfficerEmail) { + this.reportingOfficerEmail = reportingOfficerEmail; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getInvestigationDate() { + return investigationDate; + } + + public void setInvestigationDate(Date investigationDate) { + this.investigationDate = investigationDate; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getFormCompletionDate() { + return formCompletionDate; + } + + public void setFormCompletionDate(Date formCompletionDate) { + this.formCompletionDate = formCompletionDate; + } + + @Enumerated(EnumType.STRING) + public AefiInvestigationStage getInvestigationStage() { + return investigationStage; + } + + public void setInvestigationStage(AefiInvestigationStage investigationStage) { + this.investigationStage = investigationStage; + } + + @Enumerated(EnumType.STRING) + public VaccinationSite getTypeOfSite() { + return typeOfSite; + } + + public void setTypeOfSite(VaccinationSite typeOfSite) { + this.typeOfSite = typeOfSite; + } + + @Column + public String getTypeOfSiteDetails() { + return typeOfSiteDetails; + } + + public void setTypeOfSiteDetails(String typeOfSiteDetails) { + this.typeOfSiteDetails = typeOfSiteDetails; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getKeySymptomDateTime() { + return keySymptomDateTime; + } + + public void setKeySymptomDateTime(Date keySymptomDateTime) { + this.keySymptomDateTime = keySymptomDateTime; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getHospitalizationDate() { + return hospitalizationDate; + } + + public void setHospitalizationDate(Date hospitalizationDate) { + this.hospitalizationDate = hospitalizationDate; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getReportedToHealthAuthorityDate() { + return reportedToHealthAuthorityDate; + } + + public void setReportedToHealthAuthorityDate(Date reportedToHealthAuthorityDate) { + this.reportedToHealthAuthorityDate = reportedToHealthAuthorityDate; + } + + @Enumerated(EnumType.STRING) + public PatientStatusAtAefiInvestigation getStatusOnDateOfInvestigation() { + return statusOnDateOfInvestigation; + } + + public void setStatusOnDateOfInvestigation(PatientStatusAtAefiInvestigation statusOnDateOfInvestigation) { + this.statusOnDateOfInvestigation = statusOnDateOfInvestigation; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getDeathDateTime() { + return deathDateTime; + } + + public void setDeathDateTime(Date deathDateTime) { + this.deathDateTime = deathDateTime; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAutopsyDone() { + return autopsyDone; + } + + public void setAutopsyDone(YesNoUnknown autopsyDone) { + this.autopsyDone = autopsyDone; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getAutopsyDate() { + return autopsyDate; + } + + public void setAutopsyDate(Date autopsyDate) { + this.autopsyDate = autopsyDate; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getAutopsyPlannedDateTime() { + return autopsyPlannedDateTime; + } + + public void setAutopsyPlannedDateTime(Date autopsyPlannedDateTime) { + this.autopsyPlannedDateTime = autopsyPlannedDateTime; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getPastHistoryOfSimilarEvent() { + return pastHistoryOfSimilarEvent; + } + + public void setPastHistoryOfSimilarEvent(YesNoUnknown pastHistoryOfSimilarEvent) { + this.pastHistoryOfSimilarEvent = pastHistoryOfSimilarEvent; + } + + @Column + public String getPastHistoryOfSimilarEventDetails() { + return pastHistoryOfSimilarEventDetails; + } + + public void setPastHistoryOfSimilarEventDetails(String pastHistoryOfSimilarEventDetails) { + this.pastHistoryOfSimilarEventDetails = pastHistoryOfSimilarEventDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAdverseEventAfterPreviousVaccinations() { + return adverseEventAfterPreviousVaccinations; + } + + public void setAdverseEventAfterPreviousVaccinations(YesNoUnknown adverseEventAfterPreviousVaccinations) { + this.adverseEventAfterPreviousVaccinations = adverseEventAfterPreviousVaccinations; + } + + @Column + public String getAdverseEventAfterPreviousVaccinationsDetails() { + return adverseEventAfterPreviousVaccinationsDetails; + } + + public void setAdverseEventAfterPreviousVaccinationsDetails(String adverseEventAfterPreviousVaccinationsDetails) { + this.adverseEventAfterPreviousVaccinationsDetails = adverseEventAfterPreviousVaccinationsDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getHistoryOfAllergyToVaccineDrugOrFood() { + return historyOfAllergyToVaccineDrugOrFood; + } + + public void setHistoryOfAllergyToVaccineDrugOrFood(YesNoUnknown historyOfAllergyToVaccineDrugOrFood) { + this.historyOfAllergyToVaccineDrugOrFood = historyOfAllergyToVaccineDrugOrFood; + } + + @Column + public String getHistoryOfAllergyToVaccineDrugOrFoodDetails() { + return historyOfAllergyToVaccineDrugOrFoodDetails; + } + + public void setHistoryOfAllergyToVaccineDrugOrFoodDetails(String historyOfAllergyToVaccineDrugOrFoodDetails) { + this.historyOfAllergyToVaccineDrugOrFoodDetails = historyOfAllergyToVaccineDrugOrFoodDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getPreExistingIllnessThirtyDaysOrCongenitalDisorder() { + return preExistingIllnessThirtyDaysOrCongenitalDisorder; + } + + public void setPreExistingIllnessThirtyDaysOrCongenitalDisorder(YesNoUnknown preExistingIllnessThirtyDaysOrCongenitalDisorder) { + this.preExistingIllnessThirtyDaysOrCongenitalDisorder = preExistingIllnessThirtyDaysOrCongenitalDisorder; + } + + @Column + public String getPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails() { + return preExistingIllnessThirtyDaysOrCongenitalDisorderDetails; + } + + public void setPreExistingIllnessThirtyDaysOrCongenitalDisorderDetails(String preExistingIllnessThirtyDaysOrCongenitalDisorderDetails) { + this.preExistingIllnessThirtyDaysOrCongenitalDisorderDetails = preExistingIllnessThirtyDaysOrCongenitalDisorderDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getHistoryOfHospitalizationInLastThirtyDaysWithCause() { + return historyOfHospitalizationInLastThirtyDaysWithCause; + } + + public void setHistoryOfHospitalizationInLastThirtyDaysWithCause(YesNoUnknown historyOfHospitalizationInLastThirtyDaysWithCause) { + this.historyOfHospitalizationInLastThirtyDaysWithCause = historyOfHospitalizationInLastThirtyDaysWithCause; + } + + @Column + public String getHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails() { + return historyOfHospitalizationInLastThirtyDaysWithCauseDetails; + } + + public void setHistoryOfHospitalizationInLastThirtyDaysWithCauseDetails(String historyOfHospitalizationInLastThirtyDaysWithCauseDetails) { + this.historyOfHospitalizationInLastThirtyDaysWithCauseDetails = historyOfHospitalizationInLastThirtyDaysWithCauseDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getCurrentlyOnConcomitantMedication() { + return currentlyOnConcomitantMedication; + } + + public void setCurrentlyOnConcomitantMedication(YesNoUnknown currentlyOnConcomitantMedication) { + this.currentlyOnConcomitantMedication = currentlyOnConcomitantMedication; + } + + @Column + public String getCurrentlyOnConcomitantMedicationDetails() { + return currentlyOnConcomitantMedicationDetails; + } + + public void setCurrentlyOnConcomitantMedicationDetails(String currentlyOnConcomitantMedicationDetails) { + this.currentlyOnConcomitantMedicationDetails = currentlyOnConcomitantMedicationDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getFamilyHistoryOfDiseaseOrAllergy() { + return familyHistoryOfDiseaseOrAllergy; + } + + public void setFamilyHistoryOfDiseaseOrAllergy(YesNoUnknown familyHistoryOfDiseaseOrAllergy) { + this.familyHistoryOfDiseaseOrAllergy = familyHistoryOfDiseaseOrAllergy; + } + + @Column + public String getFamilyHistoryOfDiseaseOrAllergyDetails() { + return familyHistoryOfDiseaseOrAllergyDetails; + } + + public void setFamilyHistoryOfDiseaseOrAllergyDetails(String familyHistoryOfDiseaseOrAllergyDetails) { + this.familyHistoryOfDiseaseOrAllergyDetails = familyHistoryOfDiseaseOrAllergyDetails; + } + + @Column + public Integer getNumberOfWeeksPregnant() { + return numberOfWeeksPregnant; + } + + public void setNumberOfWeeksPregnant(Integer numberOfWeeksPregnant) { + this.numberOfWeeksPregnant = numberOfWeeksPregnant; + } + + @Enumerated(EnumType.STRING) + public BirthTerm getBirthTerm() { + return birthTerm; + } + + public void setBirthTerm(BirthTerm birthTerm) { + this.birthTerm = birthTerm; + } + + @Column + public Float getBirthWeight() { + return birthWeight; + } + + public void setBirthWeight(Float birthWeight) { + this.birthWeight = birthWeight; + } + + @Enumerated(EnumType.STRING) + public DeliveryProcedure getDeliveryProcedure() { + return deliveryProcedure; + } + + public void setDeliveryProcedure(DeliveryProcedure deliveryProcedure) { + this.deliveryProcedure = deliveryProcedure; + } + + @Column + public String getDeliveryProcedureDetails() { + return deliveryProcedureDetails; + } + + public void setDeliveryProcedureDetails(String deliveryProcedureDetails) { + this.deliveryProcedureDetails = deliveryProcedureDetails; + } + + @Transient + public Set getSeriousAefiInfoSource() { + if (seriousAefiInfoSource == null) { + if (StringUtils.isEmpty(seriousAefiInfoSourceString)) { + seriousAefiInfoSource = new HashSet<>(); + } else { + seriousAefiInfoSource = + Arrays.stream(seriousAefiInfoSourceString.split(",")).map(SeriousAefiInfoSource::valueOf).collect(Collectors.toSet()); + } + } + return seriousAefiInfoSource; + } + + public void setSeriousAefiInfoSource(Set seriousAefiInfoSource) { + this.seriousAefiInfoSource = seriousAefiInfoSource; + + if (this.seriousAefiInfoSource == null) { + return; + } + + StringBuilder sb = new StringBuilder(); + seriousAefiInfoSource.stream().forEach(t -> { + sb.append(t.name()); + sb.append(","); + }); + if (sb.length() > 0) { + sb.substring(0, sb.lastIndexOf(",")); + } + seriousAefiInfoSourceString = sb.toString(); + } + + @Column + public String getSeriousAefiInfoSourceString() { + return seriousAefiInfoSourceString; + } + + public void setSeriousAefiInfoSourceString(String seriousAefiInfoSourceString) { + this.seriousAefiInfoSourceString = seriousAefiInfoSourceString; + } + + @Column + public String getSeriousAefiInfoSourceDetails() { + return seriousAefiInfoSourceDetails; + } + + public void setSeriousAefiInfoSourceDetails(String seriousAefiInfoSourceDetails) { + this.seriousAefiInfoSourceDetails = seriousAefiInfoSourceDetails; + } + + @Column + public String getSeriousAefiVerbalAutopsyInfoSourceDetails() { + return seriousAefiVerbalAutopsyInfoSourceDetails; + } + + public void setSeriousAefiVerbalAutopsyInfoSourceDetails(String seriousAefiVerbalAutopsyInfoSourceDetails) { + this.seriousAefiVerbalAutopsyInfoSourceDetails = seriousAefiVerbalAutopsyInfoSourceDetails; + } + + @Column + public String getFirstCaregiversName() { + return firstCaregiversName; + } + + public void setFirstCaregiversName(String firstCaregiversName) { + this.firstCaregiversName = firstCaregiversName; + } + + @Column + public String getOtherCaregiversNames() { + return otherCaregiversNames; + } + + public void setOtherCaregiversNames(String otherCaregiversNames) { + this.otherCaregiversNames = otherCaregiversNames; + } + + @Column + public String getOtherSourcesWhoProvidedInfo() { + return otherSourcesWhoProvidedInfo; + } + + public void setOtherSourcesWhoProvidedInfo(String otherSourcesWhoProvidedInfo) { + this.otherSourcesWhoProvidedInfo = otherSourcesWhoProvidedInfo; + } + + @Column + public String getSignsAndSymptomsFromTimeOfVaccination() { + return signsAndSymptomsFromTimeOfVaccination; + } + + public void setSignsAndSymptomsFromTimeOfVaccination(String signsAndSymptomsFromTimeOfVaccination) { + this.signsAndSymptomsFromTimeOfVaccination = signsAndSymptomsFromTimeOfVaccination; + } + + @Column + public String getClinicalDetailsOfficerName() { + return clinicalDetailsOfficerName; + } + + public void setClinicalDetailsOfficerName(String clinicalDetailsOfficerName) { + this.clinicalDetailsOfficerName = clinicalDetailsOfficerName; + } + + @Column + public String getClinicalDetailsOfficerPhoneNumber() { + return clinicalDetailsOfficerPhoneNumber; + } + + public void setClinicalDetailsOfficerPhoneNumber(String clinicalDetailsOfficerPhoneNumber) { + this.clinicalDetailsOfficerPhoneNumber = clinicalDetailsOfficerPhoneNumber; + } + + @Column + public String getClinicalDetailsOfficerEmail() { + return clinicalDetailsOfficerEmail; + } + + public void setClinicalDetailsOfficerEmail(String clinicalDetailsOfficerEmail) { + this.clinicalDetailsOfficerEmail = clinicalDetailsOfficerEmail; + } + + @Column + public String getClinicalDetailsOfficerDesignation() { + return clinicalDetailsOfficerDesignation; + } + + public void setClinicalDetailsOfficerDesignation(String clinicalDetailsOfficerDesignation) { + this.clinicalDetailsOfficerDesignation = clinicalDetailsOfficerDesignation; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getClinicalDetailsDateTime() { + return clinicalDetailsDateTime; + } + + public void setClinicalDetailsDateTime(Date clinicalDetailsDateTime) { + this.clinicalDetailsDateTime = clinicalDetailsDateTime; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getPatientReceivedMedicalCare() { + return patientReceivedMedicalCare; + } + + public void setPatientReceivedMedicalCare(YesNoUnknown patientReceivedMedicalCare) { + this.patientReceivedMedicalCare = patientReceivedMedicalCare; + } + + @Column + public String getPatientReceivedMedicalCareDetails() { + return patientReceivedMedicalCareDetails; + } + + public void setPatientReceivedMedicalCareDetails(String patientReceivedMedicalCareDetails) { + this.patientReceivedMedicalCareDetails = patientReceivedMedicalCareDetails; + } + + @Column + public String getProvisionalOrFinalDiagnosis() { + return provisionalOrFinalDiagnosis; + } + + public void setProvisionalOrFinalDiagnosis(String provisionalOrFinalDiagnosis) { + this.provisionalOrFinalDiagnosis = provisionalOrFinalDiagnosis; + } + + @Enumerated(EnumType.STRING) + public AefiImmunizationPeriod getPatientImmunizedPeriod() { + return patientImmunizedPeriod; + } + + public void setPatientImmunizedPeriod(AefiImmunizationPeriod patientImmunizedPeriod) { + this.patientImmunizedPeriod = patientImmunizedPeriod; + } + + @Column + public String getPatientImmunizedPeriodDetails() { + return patientImmunizedPeriodDetails; + } + + public void setPatientImmunizedPeriodDetails(String patientImmunizedPeriodDetails) { + this.patientImmunizedPeriodDetails = patientImmunizedPeriodDetails; + } + + @Enumerated(EnumType.STRING) + public AefiVaccinationPeriod getVaccineGivenPeriod() { + return vaccineGivenPeriod; + } + + public void setVaccineGivenPeriod(AefiVaccinationPeriod vaccineGivenPeriod) { + this.vaccineGivenPeriod = vaccineGivenPeriod; + } + + @Column + public String getVaccineGivenPeriodDetails() { + return vaccineGivenPeriodDetails; + } + + public void setVaccineGivenPeriodDetails(String vaccineGivenPeriodDetails) { + this.vaccineGivenPeriodDetails = vaccineGivenPeriodDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getErrorPrescribingVaccine() { + return errorPrescribingVaccine; + } + + public void setErrorPrescribingVaccine(YesNoUnknown errorPrescribingVaccine) { + this.errorPrescribingVaccine = errorPrescribingVaccine; + } + + @Column + public String getErrorPrescribingVaccineDetails() { + return errorPrescribingVaccineDetails; + } + + public void setErrorPrescribingVaccineDetails(String errorPrescribingVaccineDetails) { + this.errorPrescribingVaccineDetails = errorPrescribingVaccineDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccineCouldHaveBeenUnSterile() { + return vaccineCouldHaveBeenUnSterile; + } + + public void setVaccineCouldHaveBeenUnSterile(YesNoUnknown vaccineCouldHaveBeenUnSterile) { + this.vaccineCouldHaveBeenUnSterile = vaccineCouldHaveBeenUnSterile; + } + + @Column + public String getVaccineCouldHaveBeenUnSterileDetails() { + return vaccineCouldHaveBeenUnSterileDetails; + } + + public void setVaccineCouldHaveBeenUnSterileDetails(String vaccineCouldHaveBeenUnSterileDetails) { + this.vaccineCouldHaveBeenUnSterileDetails = vaccineCouldHaveBeenUnSterileDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccinePhysicalConditionAbnormal() { + return vaccinePhysicalConditionAbnormal; + } + + public void setVaccinePhysicalConditionAbnormal(YesNoUnknown vaccinePhysicalConditionAbnormal) { + this.vaccinePhysicalConditionAbnormal = vaccinePhysicalConditionAbnormal; + } + + @Column + public String getVaccinePhysicalConditionAbnormalDetails() { + return vaccinePhysicalConditionAbnormalDetails; + } + + public void setVaccinePhysicalConditionAbnormalDetails(String vaccinePhysicalConditionAbnormalDetails) { + this.vaccinePhysicalConditionAbnormalDetails = vaccinePhysicalConditionAbnormalDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getErrorInVaccineReconstitution() { + return errorInVaccineReconstitution; + } + + public void setErrorInVaccineReconstitution(YesNoUnknown errorInVaccineReconstitution) { + this.errorInVaccineReconstitution = errorInVaccineReconstitution; + } + + @Column + public String getErrorInVaccineReconstitutionDetails() { + return errorInVaccineReconstitutionDetails; + } + + public void setErrorInVaccineReconstitutionDetails(String errorInVaccineReconstitutionDetails) { + this.errorInVaccineReconstitutionDetails = errorInVaccineReconstitutionDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getErrorInVaccineHandling() { + return errorInVaccineHandling; + } + + public void setErrorInVaccineHandling(YesNoUnknown errorInVaccineHandling) { + this.errorInVaccineHandling = errorInVaccineHandling; + } + + @Column + public String getErrorInVaccineHandlingDetails() { + return errorInVaccineHandlingDetails; + } + + public void setErrorInVaccineHandlingDetails(String errorInVaccineHandlingDetails) { + this.errorInVaccineHandlingDetails = errorInVaccineHandlingDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccineAdministeredIncorrectly() { + return vaccineAdministeredIncorrectly; + } + + public void setVaccineAdministeredIncorrectly(YesNoUnknown vaccineAdministeredIncorrectly) { + this.vaccineAdministeredIncorrectly = vaccineAdministeredIncorrectly; + } + + @Column + public String getVaccineAdministeredIncorrectlyDetails() { + return vaccineAdministeredIncorrectlyDetails; + } + + public void setVaccineAdministeredIncorrectlyDetails(String vaccineAdministeredIncorrectlyDetails) { + this.vaccineAdministeredIncorrectlyDetails = vaccineAdministeredIncorrectlyDetails; + } + + @Column + public Integer getNumberImmunizedFromConcernedVaccineVial() { + return numberImmunizedFromConcernedVaccineVial; + } + + public void setNumberImmunizedFromConcernedVaccineVial(Integer numberImmunizedFromConcernedVaccineVial) { + this.numberImmunizedFromConcernedVaccineVial = numberImmunizedFromConcernedVaccineVial; + } + + @Column + public Integer getNumberImmunizedWithConcernedVaccineInSameSession() { + return numberImmunizedWithConcernedVaccineInSameSession; + } + + public void setNumberImmunizedWithConcernedVaccineInSameSession(Integer numberImmunizedWithConcernedVaccineInSameSession) { + this.numberImmunizedWithConcernedVaccineInSameSession = numberImmunizedWithConcernedVaccineInSameSession; + } + + @Column + public Integer getNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations() { + return numberImmunizedConcernedVaccineSameBatchNumberOtherLocations; + } + + public void setNumberImmunizedConcernedVaccineSameBatchNumberOtherLocations( + Integer numberImmunizedConcernedVaccineSameBatchNumberOtherLocations) { + this.numberImmunizedConcernedVaccineSameBatchNumberOtherLocations = numberImmunizedConcernedVaccineSameBatchNumberOtherLocations; + } + + @Column + public String getNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails() { + return numberImmunizedConcernedVaccineSameBatchNumberLocationDetails; + } + + public void setNumberImmunizedConcernedVaccineSameBatchNumberLocationDetails( + String numberImmunizedConcernedVaccineSameBatchNumberLocationDetails) { + this.numberImmunizedConcernedVaccineSameBatchNumberLocationDetails = numberImmunizedConcernedVaccineSameBatchNumberLocationDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccineHasQualityDefect() { + return vaccineHasQualityDefect; + } + + public void setVaccineHasQualityDefect(YesNoUnknown vaccineHasQualityDefect) { + this.vaccineHasQualityDefect = vaccineHasQualityDefect; + } + + @Column + public String getVaccineHasQualityDefectDetails() { + return vaccineHasQualityDefectDetails; + } + + public void setVaccineHasQualityDefectDetails(String vaccineHasQualityDefectDetails) { + this.vaccineHasQualityDefectDetails = vaccineHasQualityDefectDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getEventIsAStressResponseRelatedToImmunization() { + return eventIsAStressResponseRelatedToImmunization; + } + + public void setEventIsAStressResponseRelatedToImmunization(YesNoUnknown eventIsAStressResponseRelatedToImmunization) { + this.eventIsAStressResponseRelatedToImmunization = eventIsAStressResponseRelatedToImmunization; + } + + @Column + public String getEventIsAStressResponseRelatedToImmunizationDetails() { + return eventIsAStressResponseRelatedToImmunizationDetails; + } + + public void setEventIsAStressResponseRelatedToImmunizationDetails(String eventIsAStressResponseRelatedToImmunizationDetails) { + this.eventIsAStressResponseRelatedToImmunizationDetails = eventIsAStressResponseRelatedToImmunizationDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getCaseIsPartOfACluster() { + return caseIsPartOfACluster; + } + + public void setCaseIsPartOfACluster(YesNoUnknown caseIsPartOfACluster) { + this.caseIsPartOfACluster = caseIsPartOfACluster; + } + + @Column + public String getCaseIsPartOfAClusterDetails() { + return caseIsPartOfAClusterDetails; + } + + public void setCaseIsPartOfAClusterDetails(String caseIsPartOfAClusterDetails) { + this.caseIsPartOfAClusterDetails = caseIsPartOfAClusterDetails; + } + + @Column + public Integer getNumberOfCasesDetectedInCluster() { + return numberOfCasesDetectedInCluster; + } + + public void setNumberOfCasesDetectedInCluster(Integer numberOfCasesDetectedInCluster) { + this.numberOfCasesDetectedInCluster = numberOfCasesDetectedInCluster; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAllCasesInClusterReceivedVaccineFromSameVial() { + return allCasesInClusterReceivedVaccineFromSameVial; + } + + public void setAllCasesInClusterReceivedVaccineFromSameVial(YesNoUnknown allCasesInClusterReceivedVaccineFromSameVial) { + this.allCasesInClusterReceivedVaccineFromSameVial = allCasesInClusterReceivedVaccineFromSameVial; + } + + @Column + public String getAllCasesInClusterReceivedVaccineFromSameVialDetails() { + return allCasesInClusterReceivedVaccineFromSameVialDetails; + } + + public void setAllCasesInClusterReceivedVaccineFromSameVialDetails(String allCasesInClusterReceivedVaccineFromSameVialDetails) { + this.allCasesInClusterReceivedVaccineFromSameVialDetails = allCasesInClusterReceivedVaccineFromSameVialDetails; + } + + @Column + public Integer getNumberOfVialsUsedInCluster() { + return numberOfVialsUsedInCluster; + } + + public void setNumberOfVialsUsedInCluster(Integer numberOfVialsUsedInCluster) { + this.numberOfVialsUsedInCluster = numberOfVialsUsedInCluster; + } + + @Column + public String getNumberOfVialsUsedInClusterDetails() { + return numberOfVialsUsedInClusterDetails; + } + + public void setNumberOfVialsUsedInClusterDetails(String numberOfVialsUsedInClusterDetails) { + this.numberOfVialsUsedInClusterDetails = numberOfVialsUsedInClusterDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAdSyringesUsedForImmunization() { + return adSyringesUsedForImmunization; + } + + public void setAdSyringesUsedForImmunization(YesNoUnknown adSyringesUsedForImmunization) { + this.adSyringesUsedForImmunization = adSyringesUsedForImmunization; + } + + @Enumerated(EnumType.STRING) + public SyringeType getTypeOfSyringesUsed() { + return typeOfSyringesUsed; + } + + public void setTypeOfSyringesUsed(SyringeType typeOfSyringesUsed) { + this.typeOfSyringesUsed = typeOfSyringesUsed; + } + + @Column + public String getTypeOfSyringesUsedDetails() { + return typeOfSyringesUsedDetails; + } + + public void setTypeOfSyringesUsedDetails(String typeOfSyringesUsedDetails) { + this.typeOfSyringesUsedDetails = typeOfSyringesUsedDetails; + } + + @Column + public String getSyringesUsedAdditionalDetails() { + return syringesUsedAdditionalDetails; + } + + public void setSyringesUsedAdditionalDetails(String syringesUsedAdditionalDetails) { + this.syringesUsedAdditionalDetails = syringesUsedAdditionalDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine() { + return sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine; + } + + public void setSameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine( + YesNoUnknown sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine) { + this.sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine = sameReconstitutionSyringeUsedForMultipleVialsOfSameVaccine; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines() { + return sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines; + } + + public void setSameReconstitutionSyringeUsedForReconstitutingDifferentVaccines( + YesNoUnknown sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines) { + this.sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines = sameReconstitutionSyringeUsedForReconstitutingDifferentVaccines; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getSameReconstitutionSyringeForEachVaccineVial() { + return sameReconstitutionSyringeForEachVaccineVial; + } + + public void setSameReconstitutionSyringeForEachVaccineVial(YesNoUnknown sameReconstitutionSyringeForEachVaccineVial) { + this.sameReconstitutionSyringeForEachVaccineVial = sameReconstitutionSyringeForEachVaccineVial; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getSameReconstitutionSyringeForEachVaccination() { + return sameReconstitutionSyringeForEachVaccination; + } + + public void setSameReconstitutionSyringeForEachVaccination(YesNoUnknown sameReconstitutionSyringeForEachVaccination) { + this.sameReconstitutionSyringeForEachVaccination = sameReconstitutionSyringeForEachVaccination; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccinesAndDiluentsUsedRecommendedByManufacturer() { + return vaccinesAndDiluentsUsedRecommendedByManufacturer; + } + + public void setVaccinesAndDiluentsUsedRecommendedByManufacturer(YesNoUnknown vaccinesAndDiluentsUsedRecommendedByManufacturer) { + this.vaccinesAndDiluentsUsedRecommendedByManufacturer = vaccinesAndDiluentsUsedRecommendedByManufacturer; + } + + @Column + public String getReconstitutionAdditionalDetails() { + return reconstitutionAdditionalDetails; + } + + public void setReconstitutionAdditionalDetails(String reconstitutionAdditionalDetails) { + this.reconstitutionAdditionalDetails = reconstitutionAdditionalDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getCorrectDoseOrRoute() { + return correctDoseOrRoute; + } + + public void setCorrectDoseOrRoute(YesNoUnknown correctDoseOrRoute) { + this.correctDoseOrRoute = correctDoseOrRoute; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getTimeOfReconstitutionMentionedOnTheVial() { + return timeOfReconstitutionMentionedOnTheVial; + } + + public void setTimeOfReconstitutionMentionedOnTheVial(YesNoUnknown timeOfReconstitutionMentionedOnTheVial) { + this.timeOfReconstitutionMentionedOnTheVial = timeOfReconstitutionMentionedOnTheVial; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getNonTouchTechniqueFollowed() { + return nonTouchTechniqueFollowed; + } + + public void setNonTouchTechniqueFollowed(YesNoUnknown nonTouchTechniqueFollowed) { + this.nonTouchTechniqueFollowed = nonTouchTechniqueFollowed; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getContraIndicationScreenedPriorToVaccination() { + return contraIndicationScreenedPriorToVaccination; + } + + public void setContraIndicationScreenedPriorToVaccination(YesNoUnknown contraIndicationScreenedPriorToVaccination) { + this.contraIndicationScreenedPriorToVaccination = contraIndicationScreenedPriorToVaccination; + } + + @Column + public Integer getNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays() { + return numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays; + } + + public void setNumberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays( + Integer numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays) { + this.numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays = numberOfAefiReportedFromVaccineDistributionCenterLastThirtyDays; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getTrainingReceivedByVaccinator() { + return trainingReceivedByVaccinator; + } + + public void setTrainingReceivedByVaccinator(YesNoUnknown trainingReceivedByVaccinator) { + this.trainingReceivedByVaccinator = trainingReceivedByVaccinator; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getLastTrainingReceivedByVaccinatorDate() { + return lastTrainingReceivedByVaccinatorDate; + } + + public void setLastTrainingReceivedByVaccinatorDate(Date lastTrainingReceivedByVaccinatorDate) { + this.lastTrainingReceivedByVaccinatorDate = lastTrainingReceivedByVaccinatorDate; + } + + @Column + public String getInjectionTechniqueAdditionalDetails() { + return injectionTechniqueAdditionalDetails; + } + + public void setInjectionTechniqueAdditionalDetails(String injectionTechniqueAdditionalDetails) { + this.injectionTechniqueAdditionalDetails = injectionTechniqueAdditionalDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccineStorageRefrigeratorTemperatureMonitored() { + return vaccineStorageRefrigeratorTemperatureMonitored; + } + + public void setVaccineStorageRefrigeratorTemperatureMonitored(YesNoUnknown vaccineStorageRefrigeratorTemperatureMonitored) { + this.vaccineStorageRefrigeratorTemperatureMonitored = vaccineStorageRefrigeratorTemperatureMonitored; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAnyStorageTemperatureDeviationOutsideTwoToEightDegrees() { + return anyStorageTemperatureDeviationOutsideTwoToEightDegrees; + } + + public void setAnyStorageTemperatureDeviationOutsideTwoToEightDegrees(YesNoUnknown anyStorageTemperatureDeviationOutsideTwoToEightDegrees) { + this.anyStorageTemperatureDeviationOutsideTwoToEightDegrees = anyStorageTemperatureDeviationOutsideTwoToEightDegrees; + } + + @Column + public String getStorageTemperatureMonitoringAdditionalDetails() { + return storageTemperatureMonitoringAdditionalDetails; + } + + public void setStorageTemperatureMonitoringAdditionalDetails(String storageTemperatureMonitoringAdditionalDetails) { + this.storageTemperatureMonitoringAdditionalDetails = storageTemperatureMonitoringAdditionalDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getCorrectProcedureForStorageFollowed() { + return correctProcedureForStorageFollowed; + } + + public void setCorrectProcedureForStorageFollowed(YesNoUnknown correctProcedureForStorageFollowed) { + this.correctProcedureForStorageFollowed = correctProcedureForStorageFollowed; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAnyOtherItemInRefrigerator() { + return anyOtherItemInRefrigerator; + } + + public void setAnyOtherItemInRefrigerator(YesNoUnknown anyOtherItemInRefrigerator) { + this.anyOtherItemInRefrigerator = anyOtherItemInRefrigerator; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getPartiallyUsedReconstitutedVaccinesInRefrigerator() { + return partiallyUsedReconstitutedVaccinesInRefrigerator; + } + + public void setPartiallyUsedReconstitutedVaccinesInRefrigerator(YesNoUnknown partiallyUsedReconstitutedVaccinesInRefrigerator) { + this.partiallyUsedReconstitutedVaccinesInRefrigerator = partiallyUsedReconstitutedVaccinesInRefrigerator; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getUnusableVaccinesInRefrigerator() { + return unusableVaccinesInRefrigerator; + } + + public void setUnusableVaccinesInRefrigerator(YesNoUnknown unusableVaccinesInRefrigerator) { + this.unusableVaccinesInRefrigerator = unusableVaccinesInRefrigerator; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getUnusableDiluentsInStore() { + return unusableDiluentsInStore; + } + + public void setUnusableDiluentsInStore(YesNoUnknown unusableDiluentsInStore) { + this.unusableDiluentsInStore = unusableDiluentsInStore; + } + + @Column + public String getVaccineStoragePointAdditionalDetails() { + return vaccineStoragePointAdditionalDetails; + } + + public void setVaccineStoragePointAdditionalDetails(String vaccineStoragePointAdditionalDetails) { + this.vaccineStoragePointAdditionalDetails = vaccineStoragePointAdditionalDetails; + } + + @Enumerated(EnumType.STRING) + public VaccineCarrier getVaccineCarrierType() { + return vaccineCarrierType; + } + + public void setVaccineCarrierType(VaccineCarrier vaccineCarrierType) { + this.vaccineCarrierType = vaccineCarrierType; + } + + @Column + public String getVaccineCarrierTypeDetails() { + return vaccineCarrierTypeDetails; + } + + public void setVaccineCarrierTypeDetails(String vaccineCarrierTypeDetails) { + this.vaccineCarrierTypeDetails = vaccineCarrierTypeDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccineCarrierSentToSiteOnSameDateAsVaccination() { + return vaccineCarrierSentToSiteOnSameDateAsVaccination; + } + + public void setVaccineCarrierSentToSiteOnSameDateAsVaccination(YesNoUnknown vaccineCarrierSentToSiteOnSameDateAsVaccination) { + this.vaccineCarrierSentToSiteOnSameDateAsVaccination = vaccineCarrierSentToSiteOnSameDateAsVaccination; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getVaccineCarrierReturnedFromSiteOnSameDateAsVaccination() { + return vaccineCarrierReturnedFromSiteOnSameDateAsVaccination; + } + + public void setVaccineCarrierReturnedFromSiteOnSameDateAsVaccination(YesNoUnknown vaccineCarrierReturnedFromSiteOnSameDateAsVaccination) { + this.vaccineCarrierReturnedFromSiteOnSameDateAsVaccination = vaccineCarrierReturnedFromSiteOnSameDateAsVaccination; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getConditionedIcepackUsed() { + return conditionedIcepackUsed; + } + + public void setConditionedIcepackUsed(YesNoUnknown conditionedIcepackUsed) { + this.conditionedIcepackUsed = conditionedIcepackUsed; + } + + @Column + public String getVaccineTransportationAdditionalDetails() { + return vaccineTransportationAdditionalDetails; + } + + public void setVaccineTransportationAdditionalDetails(String vaccineTransportationAdditionalDetails) { + this.vaccineTransportationAdditionalDetails = vaccineTransportationAdditionalDetails; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getSimilarEventsReportedSamePeriodAndLocality() { + return similarEventsReportedSamePeriodAndLocality; + } + + public void setSimilarEventsReportedSamePeriodAndLocality(YesNoUnknown similarEventsReportedSamePeriodAndLocality) { + this.similarEventsReportedSamePeriodAndLocality = similarEventsReportedSamePeriodAndLocality; + } + + @Column + public String getSimilarEventsReportedSamePeriodAndLocalityDetails() { + return similarEventsReportedSamePeriodAndLocalityDetails; + } + + public void setSimilarEventsReportedSamePeriodAndLocalityDetails(String similarEventsReportedSamePeriodAndLocalityDetails) { + this.similarEventsReportedSamePeriodAndLocalityDetails = similarEventsReportedSamePeriodAndLocalityDetails; + } + + @Column + public Integer getNumberOfSimilarEventsReportedSamePeriodAndLocality() { + return numberOfSimilarEventsReportedSamePeriodAndLocality; + } + + public void setNumberOfSimilarEventsReportedSamePeriodAndLocality(Integer numberOfSimilarEventsReportedSamePeriodAndLocality) { + this.numberOfSimilarEventsReportedSamePeriodAndLocality = numberOfSimilarEventsReportedSamePeriodAndLocality; + } + + @Column + public Integer getNumberOfThoseAffectedVaccinated() { + return numberOfThoseAffectedVaccinated; + } + + public void setNumberOfThoseAffectedVaccinated(Integer numberOfThoseAffectedVaccinated) { + this.numberOfThoseAffectedVaccinated = numberOfThoseAffectedVaccinated; + } + + @Column + public Integer getNumberOfThoseAffectedNotVaccinated() { + return numberOfThoseAffectedNotVaccinated; + } + + public void setNumberOfThoseAffectedNotVaccinated(Integer numberOfThoseAffectedNotVaccinated) { + this.numberOfThoseAffectedNotVaccinated = numberOfThoseAffectedNotVaccinated; + } + + @Column + public Integer getNumberOfThoseAffectedVaccinatedUnknown() { + return numberOfThoseAffectedVaccinatedUnknown; + } + + public void setNumberOfThoseAffectedVaccinatedUnknown(Integer numberOfThoseAffectedVaccinatedUnknown) { + this.numberOfThoseAffectedVaccinatedUnknown = numberOfThoseAffectedVaccinatedUnknown; + } + + @Column + public String getCommunityInvestigationAdditionalDetails() { + return communityInvestigationAdditionalDetails; + } + + public void setCommunityInvestigationAdditionalDetails(String communityInvestigationAdditionalDetails) { + this.communityInvestigationAdditionalDetails = communityInvestigationAdditionalDetails; + } + + @Column + public String getOtherInvestigationFindings() { + return otherInvestigationFindings; + } + + public void setOtherInvestigationFindings(String otherInvestigationFindings) { + this.otherInvestigationFindings = otherInvestigationFindings; + } + + @Enumerated(EnumType.STRING) + public AefiInvestigationStatus getInvestigationStatus() { + return investigationStatus; + } + + public void setInvestigationStatus(AefiInvestigationStatus investigationStatus) { + this.investigationStatus = investigationStatus; + } + + @Column + public String getInvestigationStatusDetails() { + return investigationStatusDetails; + } + + public void setInvestigationStatusDetails(String investigationStatusDetails) { + this.investigationStatusDetails = investigationStatusDetails; + } + + @Enumerated(EnumType.STRING) + @Column(name = "adverseeventfollowingimmunizationclassification") + public AefiClassification getAefiClassification() { + return aefiClassification; + } + + public void setAefiClassification(AefiClassification aefiClassification) { + this.aefiClassification = aefiClassification; + } + + @Enumerated(EnumType.STRING) + @Column(name = "adverseeventfollowingimmunizationclassificationsubtype") + public AefiClassificationSubType getAefiClassificationSubType() { + return aefiClassificationSubType; + } + + public void setAefiClassificationSubType(AefiClassificationSubType aefiClassificationSubType) { + this.aefiClassificationSubType = aefiClassificationSubType; + } + + @Column(name = "adverseeventfollowingimmunizationclassificationdetails") + public String getAefiClassificationDetails() { + return aefiClassificationDetails; + } + + public void setAefiClassificationDetails(String aefiClassificationDetails) { + this.aefiClassificationDetails = aefiClassificationDetails; + } + + @Column + public AefiCausality getCausality() { + return causality; + } + + public void setCausality(AefiCausality causality) { + this.causality = causality; + } + + @Column + public String getCausalityDetails() { + return causalityDetails; + } + + public void setCausalityDetails(String causalityDetails) { + this.causalityDetails = causalityDetails; + } + + @Temporal(TemporalType.TIMESTAMP) + public Date getInvestigationCompletionDate() { + return investigationCompletionDate; + } + + public void setInvestigationCompletionDate(Date investigationCompletionDate) { + this.investigationCompletionDate = investigationCompletionDate; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigationJoins.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigationJoins.java new file mode 100644 index 00000000000..03495959759 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigationJoins.java @@ -0,0 +1,138 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity; + +import javax.persistence.criteria.From; +import javax.persistence.criteria.Join; +import javax.persistence.criteria.JoinType; + +import de.symeda.sormas.backend.common.QueryJoins; +import de.symeda.sormas.backend.infrastructure.community.Community; +import de.symeda.sormas.backend.infrastructure.country.Country; +import de.symeda.sormas.backend.infrastructure.district.District; +import de.symeda.sormas.backend.infrastructure.facility.Facility; +import de.symeda.sormas.backend.infrastructure.region.Region; +import de.symeda.sormas.backend.location.Location; +import de.symeda.sormas.backend.user.User; +import de.symeda.sormas.backend.vaccination.Vaccination; + +public class AefiInvestigationJoins extends QueryJoins { + + private Join aefi; + + private AefiJoins aefiJoins; + private Join address; + private Join primarySuspectVaccination; + private Join reportingUser; + private Join responsibleRegion; + private Join responsibleDistrict; + private Join responsibleCommunity; + private Join country; + private Join vaccinationFacility; + private Join reportingOfficerFacility; + + public AefiInvestigationJoins(From root) { + super(root); + } + + public Join getAefi() { + return getOrCreate(aefi, AefiInvestigation.AEFI_REPORT, JoinType.LEFT, this::setAefi); + } + + public void setAefi(Join aefi) { + this.aefi = aefi; + } + + public AefiJoins getAefiJoins() { + return getOrCreate(aefiJoins, () -> new AefiJoins(getAefi()), this::setAefiJoins); + } + + public void setAefiJoins(AefiJoins aefiJoins) { + this.aefiJoins = aefiJoins; + } + + public Join getAddress() { + return getOrCreate(address, AefiInvestigation.ADDRESS, JoinType.LEFT, this::setAddress); + } + + public void setAddress(Join address) { + this.address = address; + } + + public Join getPrimarySuspectVaccination() { + return getOrCreate(primarySuspectVaccination, AefiInvestigation.PRIMARY_SUSPECT_VACCINE, JoinType.LEFT, this::setPrimarySuspectVaccination); + } + + public void setPrimarySuspectVaccination(Join primarySuspectVaccination) { + this.primarySuspectVaccination = primarySuspectVaccination; + } + + public Join getReportingUser() { + return getOrCreate(reportingUser, AefiInvestigation.REPORTING_USER, JoinType.LEFT, this::setReportingUser); + } + + public void setReportingUser(Join reportingUser) { + this.reportingUser = reportingUser; + } + + public Join getResponsibleRegion() { + return getOrCreate(responsibleRegion, AefiInvestigation.RESPONSIBLE_REGION, JoinType.LEFT, this::setResponsibleRegion); + } + + public void setResponsibleRegion(Join responsibleRegion) { + this.responsibleRegion = responsibleRegion; + } + + public Join getResponsibleDistrict() { + return getOrCreate(responsibleDistrict, AefiInvestigation.RESPONSIBLE_DISTRICT, JoinType.LEFT, this::setResponsibleDistrict); + } + + public void setResponsibleDistrict(Join responsibleDistrict) { + this.responsibleDistrict = responsibleDistrict; + } + + public Join getResponsibleCommunity() { + return getOrCreate(responsibleCommunity, AefiInvestigation.RESPONSIBLE_COMMUNITY, JoinType.LEFT, this::setResponsibleCommunity); + } + + public void setResponsibleCommunity(Join responsibleCommunity) { + this.responsibleCommunity = responsibleCommunity; + } + + public Join getCountry() { + return getOrCreate(country, AefiInvestigation.COUNTRY, JoinType.LEFT, this::setCountry); + } + + public void setCountry(Join country) { + this.country = country; + } + + public Join getVaccinationFacility() { + return getOrCreate(vaccinationFacility, AefiInvestigation.VACCINATION_FACILITY, JoinType.LEFT, this::setVaccinationFacility); + } + + public void setVaccinationFacility(Join vaccinationFacility) { + this.vaccinationFacility = vaccinationFacility; + } + + public Join getReportingOfficerFacility() { + return getOrCreate(reportingOfficerFacility, AefiInvestigation.REPORTING_OFFICER_FACILITY, JoinType.LEFT, this::setReportingOfficerFacility); + } + + public void setReportingOfficerFacility(Join reportingOfficerFacility) { + this.reportingOfficerFacility = reportingOfficerFacility; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java index 33f71570b4f..e1f29ef65fd 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiJoins.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -153,7 +150,7 @@ public void setHealthFacility(Join healthFacility) { } public Join getReporterInstitution() { - return getOrCreate(reporterInstitution, Aefi.REPORTER_INSTITUTION, JoinType.LEFT, this::setReporterInstitution); + return getOrCreate(reporterInstitution, Aefi.REPORTING_OFFICER_FACILITY, JoinType.LEFT, this::setReporterInstitution); } public void setReporterInstitution(Join reporterInstitution) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java index 73594a78f5f..c2173e84654 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiIndexDtoResultTransformer.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -49,11 +46,11 @@ public Object transformTuple(Object[] objects, String[] strings) { //@formatter:off String adverseEvents = AefiHelper - .buildAdverseEventsString((AdverseEventState) objects[20], (boolean) objects[21], (boolean) objects[22], - (AdverseEventState) objects[23], (SeizureType) objects[24], (AdverseEventState) objects[25], - (AdverseEventState) objects[26], (AdverseEventState) objects[27], (AdverseEventState) objects[28], - (AdverseEventState) objects[29], (AdverseEventState) objects[30], (AdverseEventState) objects[31], - (String) objects[32]); + .buildAdverseEventsString((AdverseEventState) objects[21], (boolean) objects[22], (boolean) objects[23], + (AdverseEventState) objects[24], (SeizureType) objects[25], (AdverseEventState) objects[26], + (AdverseEventState) objects[27], (AdverseEventState) objects[28], (AdverseEventState) objects[29], + (AdverseEventState) objects[30], (AdverseEventState) objects[31], (AdverseEventState) objects[32], + (String) objects[33]); //@formatter:on return new AefiIndexDto( @@ -69,14 +66,15 @@ public Object transformTuple(Object[] objects, String[] strings) { (String) objects[13], (YesNoUnknown) objects[14], (Vaccine) objects[15], - (AefiOutcome) objects[16], - (Date) objects[17], + (String) objects[16], + (AefiOutcome) objects[17], (Date) objects[18], (Date) objects[19], + (Date) objects[20], adverseEvents, - (DeletionReason) objects[33], - (String) objects[34], - (boolean) objects[35]); + (DeletionReason) objects[34], + (String) objects[35], + (boolean) objects[36]); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationIndexDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationIndexDtoResultTransformer.java new file mode 100644 index 00000000000..51e2e29fa19 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationIndexDtoResultTransformer.java @@ -0,0 +1,84 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers; + +import java.util.Date; +import java.util.List; + +import org.hibernate.transform.ResultTransformer; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationIndexDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStage; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PatientStatusAtAefiInvestigation; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PlaceOfVaccination; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccinationActivity; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccinationSite; +import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.person.ApproximateAgeType; +import de.symeda.sormas.api.person.Sex; + +public class AefiInvestigationIndexDtoResultTransformer implements ResultTransformer { + + @Override + public Object transformTuple(Object[] objects, String[] strings) { + + Integer age = objects[6] != null ? (int) objects[6] : null; + ApproximateAgeType approximateAgeType = (ApproximateAgeType) objects[7]; + Integer birthdateDD = objects[8] != null ? (int) objects[8] : null; + Integer birthdateMM = objects[9] != null ? (int) objects[9] : null; + Integer birthdateYYYY = objects[10] != null ? (int) objects[10] : null; + + return new AefiInvestigationIndexDto( + (String) objects[0], + (String) objects[1], + (String) objects[2], + (Disease) objects[3], + (String) objects[4], + (String) objects[5], + new AgeAndBirthDateDto(age, approximateAgeType, birthdateDD, birthdateMM, birthdateYYYY), + (Sex) objects[11], + (String) objects[12], + (String) objects[13], + (PlaceOfVaccination) objects[14], + (VaccinationActivity) objects[15], + (Date) objects[16], + (Date) objects[17], + (Date) objects[18], + (AefiInvestigationStage) objects[19], + (VaccinationSite) objects[20], + (Date) objects[21], + (Date) objects[22], + (Date) objects[23], + (PatientStatusAtAefiInvestigation) objects[24], + (Vaccine) objects[25], + (String) objects[26], + (AefiInvestigationStatus) objects[27], + (AefiClassification) objects[28], + (DeletionReason) objects[29], + (String) objects[30], + (boolean) objects[31]); + } + + @Override + public List transformList(List list) { + return list; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationListEntryDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationListEntryDtoResultTransformer.java new file mode 100644 index 00000000000..e22f380b2d7 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/transformers/AefiInvestigationListEntryDtoResultTransformer.java @@ -0,0 +1,53 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.adverseeventsfollowingimmunization.transformers; + +import java.util.Date; +import java.util.List; + +import org.hibernate.transform.ResultTransformer; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStage; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PatientStatusAtAefiInvestigation; +import de.symeda.sormas.api.caze.Vaccine; + +public class AefiInvestigationListEntryDtoResultTransformer implements ResultTransformer { + + @Override + public Object transformTuple(Object[] objects, String[] strings) { + + return new AefiInvestigationListEntryDto( + (String) objects[0], + (String) objects[1], + (Date) objects[2], + (AefiInvestigationStage) objects[3], + (PatientStatusAtAefiInvestigation) objects[4], + (Vaccine) objects[5], + (String) objects[6], + (String) objects[7], + (Date) objects[8], + (AefiInvestigationStatus) objects[9], + (AefiClassification) objects[10]); + } + + @Override + public List transformList(List list) { + return list; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java index 91471089e92..9a3fc77ec3f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardFacadeEjb.java @@ -22,6 +22,8 @@ import javax.ejb.LocalBean; import javax.ejb.Stateless; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; import de.symeda.sormas.api.caze.Vaccine; import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; @@ -43,6 +45,17 @@ public Map getAefiCountsByType(AefiDashboardCriteria dashboardCr return aefiDashboardService.getAefiCountsByType(dashboardCriteria); } + @Override + public Map> getAefiInvestigationCountsByInvestigationStatus( + AefiDashboardCriteria dashboardCriteria) { + return aefiDashboardService.getAefiInvestigationCountsByInvestigationStatus(dashboardCriteria); + } + + @Override + public Map> getAefiInvestigationCountsByAefiClassification(AefiDashboardCriteria dashboardCriteria) { + return aefiDashboardService.getAefiInvestigationCountsByAefiClassification(dashboardCriteria); + } + @Override public Map> getAefiCountsByVaccine(AefiDashboardCriteria dashboardCriteria) { return aefiDashboardService.getAefiCountsByVaccine(dashboardCriteria); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java index 41a50bed7a5..213655e0049 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/dashboard/adverseeventsfollowingimmunization/AefiDashboardService.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -49,8 +46,10 @@ import org.slf4j.LoggerFactory; import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDashboardFilterDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; import de.symeda.sormas.api.caze.Vaccine; import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; @@ -62,6 +61,7 @@ import de.symeda.sormas.api.person.Sex; import de.symeda.sormas.api.utils.DateHelper; import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.AefiInvestigationService; import de.symeda.sormas.backend.adverseeventsfollowingimmunization.AefiQueryContext; import de.symeda.sormas.backend.adverseeventsfollowingimmunization.AefiService; import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents; @@ -97,6 +97,8 @@ public class AefiDashboardService { @EJB private AefiService aefiService; @EJB + private AefiInvestigationService aefiInvestigationService; + @EJB private RegionService regionService; @EJB private DistrictService districtService; @@ -113,6 +115,167 @@ public Map getAefiCountsByType(AefiDashboardCriteria dashboardCr return result; } + public Map> getAefiInvestigationCountsByInvestigationStatus( + AefiDashboardCriteria dashboardCriteria) { + + Map> countsByInvestigationStatus = new HashMap<>(); + + Map defaultValuesMap = new HashMap<>(); + defaultValuesMap.put("total", "0"); + defaultValuesMap.put("percent", "0"); + + for (AefiInvestigationStatus investigationStatus : AefiInvestigationStatus.values()) { + countsByInvestigationStatus.put(investigationStatus, new HashMap<>(defaultValuesMap)); + } + + Disease disease = dashboardCriteria.getDisease(); + RegionReferenceDto regionReference = dashboardCriteria.getRegion(); + DistrictReferenceDto districtReference = dashboardCriteria.getDistrict(); + + String whereConditions = createAefiNativeQueryFilter(dashboardCriteria); + if (StringUtils.isBlank(whereConditions)) { + whereConditions = " aefininvestigation.investigationstatus is not null"; + } else { + whereConditions = whereConditions + " AND aefininvestigation.investigationstatus is not null"; + } + + //@formatter:off + String queryString = "select aefininvestigation.investigationstatus, count(*) as total_status " + + " from adverseeventsfollowingimmunizationinvestigation aefininvestigation" + + " join adverseeventsfollowingimmunization aefi on aefininvestigation.adverseeventsfollowingimmunization_id = aefi.id" + + " join immunization on aefi.immunization_id = immunization.id" + + " where " + whereConditions + + " group by aefininvestigation.investigationstatus"; + //@formatter:on + + Query dataQuery = em.createNativeQuery(queryString); + + if (disease != null) { + dataQuery.setParameter("disease", disease.name()); + } + + if (regionReference != null) { + Region region = regionService.getByReferenceDto(regionReference); + dataQuery.setParameter("responsibleregion_id", region.getId()); + } + + if (districtReference != null) { + District district = districtService.getByReferenceDto(districtReference); + dataQuery.setParameter("responsibledistrict_id", district.getId()); + } + + if (dashboardCriteria.getDateFrom() != null && dashboardCriteria.getDateTo() != null) { + Date dateFrom = DateHelper.getStartOfDay(dashboardCriteria.getDateFrom()); + Date dateTo = DateHelper.getEndOfDay(dashboardCriteria.getDateTo()); + + SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); + String dateFromStr = DateHelper.formatLocalDate(dateFrom, simpleDateFormat); + String dateToStr = DateHelper.formatLocalDate(dateTo, simpleDateFormat); + + dataQuery.setParameter("dateFrom", dateFromStr); + dataQuery.setParameter("dateTo", dateToStr); + } + + @SuppressWarnings("unchecked") + List resultList = dataQuery.getResultList(); + + int totalInvestigations = 0; + for (Object[] result : resultList) { + totalInvestigations += ((BigInteger) result[1]).intValue(); + } + + int statusTotal; + int statusPercent; + for (Object[] result : resultList) { + statusTotal = ((BigInteger) result[1]).intValue(); + statusPercent = (totalInvestigations == 0) ? 0 : ((int) ((statusTotal * 100.0f) / totalInvestigations)); + countsByInvestigationStatus.get(AefiInvestigationStatus.valueOf((String) result[0])).put("total", String.valueOf(statusTotal)); + countsByInvestigationStatus.get(AefiInvestigationStatus.valueOf((String) result[0])).put("percent", String.valueOf(statusPercent)); + } + + return countsByInvestigationStatus; + } + + public Map> getAefiInvestigationCountsByAefiClassification(AefiDashboardCriteria dashboardCriteria) { + + Map> countsByAefiClassitication = new HashMap<>(); + + Map defaultValuesMap = new HashMap<>(); + defaultValuesMap.put("total", "0"); + defaultValuesMap.put("percent", "0"); + + for (AefiClassification aefiClassification : AefiClassification.values()) { + countsByAefiClassitication.put(aefiClassification, new HashMap<>(defaultValuesMap)); + } + + Disease disease = dashboardCriteria.getDisease(); + RegionReferenceDto regionReference = dashboardCriteria.getRegion(); + DistrictReferenceDto districtReference = dashboardCriteria.getDistrict(); + + String whereConditions = createAefiNativeQueryFilter(dashboardCriteria); + if (StringUtils.isBlank(whereConditions)) { + whereConditions = " aefininvestigation.adverseeventfollowingimmunizationclassification is not null"; + } else { + whereConditions = whereConditions + " AND aefininvestigation.adverseeventfollowingimmunizationclassification is not null"; + } + + //@formatter:off + String queryString = "select aefininvestigation.adverseeventfollowingimmunizationclassification, count(*) as total_classification " + + " from adverseeventsfollowingimmunizationinvestigation aefininvestigation" + + " join adverseeventsfollowingimmunization aefi on aefininvestigation.adverseeventsfollowingimmunization_id = aefi.id" + + " join immunization on aefi.immunization_id = immunization.id" + + " where " + whereConditions + + " group by aefininvestigation.adverseeventfollowingimmunizationclassification"; + //@formatter:on + + Query dataQuery = em.createNativeQuery(queryString); + + if (disease != null) { + dataQuery.setParameter("disease", disease.name()); + } + + if (regionReference != null) { + Region region = regionService.getByReferenceDto(regionReference); + dataQuery.setParameter("responsibleregion_id", region.getId()); + } + + if (districtReference != null) { + District district = districtService.getByReferenceDto(districtReference); + dataQuery.setParameter("responsibledistrict_id", district.getId()); + } + + if (dashboardCriteria.getDateFrom() != null && dashboardCriteria.getDateTo() != null) { + Date dateFrom = DateHelper.getStartOfDay(dashboardCriteria.getDateFrom()); + Date dateTo = DateHelper.getEndOfDay(dashboardCriteria.getDateTo()); + + SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd"); + String dateFromStr = DateHelper.formatLocalDate(dateFrom, simpleDateFormat); + String dateToStr = DateHelper.formatLocalDate(dateTo, simpleDateFormat); + + dataQuery.setParameter("dateFrom", dateFromStr); + dataQuery.setParameter("dateTo", dateToStr); + } + + @SuppressWarnings("unchecked") + List resultList = dataQuery.getResultList(); + + int totalInvestigations = 0; + for (Object[] result : resultList) { + totalInvestigations += ((BigInteger) result[1]).intValue(); + } + + int classificationTotal; + int classificationPercent; + for (Object[] result : resultList) { + classificationTotal = ((BigInteger) result[1]).intValue(); + classificationPercent = (totalInvestigations == 0) ? 0 : ((int) ((classificationTotal * 100.0f) / totalInvestigations)); + countsByAefiClassitication.get(AefiClassification.valueOf((String) result[0])).put("total", String.valueOf(classificationTotal)); + countsByAefiClassitication.get(AefiClassification.valueOf((String) result[0])).put("percent", String.valueOf(classificationPercent)); + } + + return countsByAefiClassitication; + } + public Map> getAefiCountsByVaccine(AefiDashboardCriteria dashboardCriteria) { Map> countsByVaccine = new HashMap<>(); @@ -349,7 +512,7 @@ public AefiChartData getAefiEventsByGenderChartData(AefiDashboardCriteria dashbo if (!resultList.isEmpty()) { Object[] firstResult = resultList.get(0); - maleSeries.addData(firstResult[1].toString()); + maleSeries.addData("-" + firstResult[1].toString()); femaleSeries.addData(firstResult[2].toString()); } else { maleSeries.addData("0"); @@ -500,10 +663,12 @@ private String createAefiNativeQueryFilter(AefiDashboardCriteria criteria) { switch (aefiDashboardFilterDateType) { case REPORT_DATE: - whereConditions.add("(aefi.reportdate >= cast(:dateFrom as timestamp) and aefi.reportdate <= cast(:dateTo as timestamp))"); + whereConditions + .add("(cast(aefi.reportdate as date) >= cast(:dateFrom as date) and cast(aefi.reportdate as date) <= cast(:dateTo as date))"); break; case START_DATE: - whereConditions.add("(aefi.startdatetime >= cast(:dateFrom as timestamp) and aefi.startdatetime <= cast(:dateTo as timestamp))"); + whereConditions.add( + "(cast(aefi.startdatetime as date) >= cast(:dateFrom as date) and cast(aefi.startdatetime as date) <= cast(:dateTo as date))"); break; default: throw new RuntimeException("Unhandled date type [" + aefiDashboardFilterDateType + "]"); diff --git a/sormas-backend/src/main/resources/META-INF/persistence.xml b/sormas-backend/src/main/resources/META-INF/persistence.xml index e0dd26ad9b2..002827d82d3 100644 --- a/sormas-backend/src/main/resources/META-INF/persistence.xml +++ b/sormas-backend/src/main/resources/META-INF/persistence.xml @@ -88,12 +88,12 @@ de.symeda.sormas.backend.externalmessage.labmessage.SampleReport de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents - de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess + de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess de.symeda.sormas.backend.selfreport.SelfReport - true + true ENABLE_SELECTIVE CALLBACK diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java index 7e2f4581fc1..1e8db100332 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/ControllerProvider.java @@ -19,6 +19,7 @@ import de.symeda.sormas.ui.action.ActionController; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.AefiController; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.AefiInvestigationController; import de.symeda.sormas.ui.campaign.CampaignController; import de.symeda.sormas.ui.caze.CaseController; import de.symeda.sormas.ui.caze.surveillancereport.SurveillanceReportController; @@ -91,6 +92,7 @@ public class ControllerProvider extends BaseControllerProvider { private final TravelEntryController travelEntryController; private final ImmunizationController immunizationController; private final AefiController aefiController; + private final AefiInvestigationController aefiInvestigationController; private final VaccinationController vaccinationController; private final ArchivingController archivingController; private final DeleteRestoreController deleteRestoreController; @@ -135,6 +137,7 @@ public ControllerProvider() { travelEntryController = new TravelEntryController(); immunizationController = new ImmunizationController(); aefiController = new AefiController(); + aefiInvestigationController = new AefiInvestigationController(); vaccinationController = new VaccinationController(); archivingController = new ArchivingController(); deleteRestoreController = new DeleteRestoreController(); @@ -271,6 +274,10 @@ public static AefiController getAefiController() { return get().aefiController; } + public static AefiInvestigationController getAefiInvestigationController() { + return get().aefiInvestigationController; + } + public static VaccinationController getVaccinationController() { return get().vaccinationController; } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiDataView.java new file mode 100644 index 00000000000..d021b2ac3b0 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiDataView.java @@ -0,0 +1,113 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import com.vaadin.navigator.ViewChangeListener.ViewChangeEvent; + +import de.symeda.sormas.api.EditPermissionFacade; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiReferenceDto; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.SubMenu; +import de.symeda.sormas.ui.immunization.ImmunizationDataView; +import de.symeda.sormas.ui.immunization.ImmunizationPersonView; +import de.symeda.sormas.ui.utils.AbstractEditAllowedDetailView; + +public abstract class AbstractAefiDataView extends AbstractEditAllowedDetailView { + + public static final String ROOT_VIEW_NAME = AefiView.VIEW_NAME; + + protected AbstractAefiDataView(String viewName) { + super(viewName); + } + + @Override + protected AefiReferenceDto getReferenceByUuid(String uuid) { + + final AefiReferenceDto reference; + + boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(uuid); + + if (isCreateAction) { + reference = new AefiReferenceDto(); + reference.setUuid(DataHelper.createUuid()); + } else { + if (FacadeProvider.getAefiFacade().exists(uuid)) { + reference = FacadeProvider.getAefiFacade().getReferenceByUuid(uuid); + } else { + reference = null; + } + } + return reference; + } + + @Override + protected String getRootViewName() { + return ROOT_VIEW_NAME; + } + + @Override + protected void initView(String params) { + + } + + @Override + protected EditPermissionFacade getEditPermissionFacade() { + return FacadeProvider.getAefiFacade(); + } + + @Override + public void enter(ViewChangeEvent event) { + + super.enter(event); + initOrRedirect(event); + } + + @Override + public void refreshMenu(SubMenu menu, String params) { + + if (!findReferenceByParams(params)) { + return; + } + + boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(params); + + String immunizationUuid = ""; + if (isCreateAction) { + immunizationUuid = ControllerProvider.getAefiController().getCreateActionImmunizationUuid(params); + } else { + AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(params); + immunizationUuid = aefiDto.getImmunization().getUuid(); + } + + menu.removeAllViews(); + menu.addView(AefiView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiList)); + menu.addView(ImmunizationDataView.VIEW_NAME, I18nProperties.getCaption(ImmunizationDto.I18N_PREFIX), immunizationUuid); + menu.addView( + ImmunizationPersonView.VIEW_NAME, + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, ImmunizationDto.PERSON), + immunizationUuid); + menu.addView(AefiDataView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiDataView), params); + + setMainHeaderComponent( + ControllerProvider.getAefiController().getAefiViewTitleLayout(getReference().getUuid(), immunizationUuid, isCreateAction)); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiInvestigationDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiInvestigationDataView.java new file mode 100644 index 00000000000..3b2ffe2cb7b --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiInvestigationDataView.java @@ -0,0 +1,122 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import com.vaadin.navigator.ViewChangeListener; + +import de.symeda.sormas.api.EditPermissionFacade; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationReferenceDto; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.SubMenu; +import de.symeda.sormas.ui.immunization.ImmunizationDataView; +import de.symeda.sormas.ui.immunization.ImmunizationPersonView; +import de.symeda.sormas.ui.utils.AbstractEditAllowedDetailView; + +public class AbstractAefiInvestigationDataView extends AbstractEditAllowedDetailView { + + public static final String ROOT_VIEW_NAME = AefiInvestigationView.VIEW_NAME; + + protected AbstractAefiInvestigationDataView(String viewName) { + super(viewName); + } + + @Override + protected AefiInvestigationReferenceDto getReferenceByUuid(String uuid) { + + final AefiInvestigationReferenceDto reference; + + boolean isCreateAction = ControllerProvider.getAefiInvestigationController().isCreateAction(uuid); + + if (isCreateAction) { + reference = new AefiInvestigationReferenceDto(); + reference.setUuid(DataHelper.createUuid()); + } else { + if (FacadeProvider.getAefiInvestigationFacade().exists(uuid)) { + reference = FacadeProvider.getAefiInvestigationFacade().getReferenceByUuid(uuid); + } else { + reference = null; + } + } + return reference; + } + + @Override + protected String getRootViewName() { + return ROOT_VIEW_NAME; + } + + @Override + protected void initView(String params) { + + } + + @Override + protected EditPermissionFacade getEditPermissionFacade() { + return FacadeProvider.getAefiInvestigationFacade(); + } + + @Override + public void enter(ViewChangeListener.ViewChangeEvent event) { + + super.enter(event); + initOrRedirect(event); + } + + @Override + public void refreshMenu(SubMenu menu, String params) { + + if (!findReferenceByParams(params)) { + return; + } + + boolean isCreateAction = ControllerProvider.getAefiInvestigationController().isCreateAction(params); + + String aefiReportUuid = ""; + String immunizationUuid = ""; + if (isCreateAction) { + aefiReportUuid = ControllerProvider.getAefiInvestigationController().getCreateActionAefiReportUuid(params); + AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(aefiReportUuid); + immunizationUuid = aefiDto.getImmunization().getUuid(); + } else { + AefiInvestigationDto aefiInvestigationDto = FacadeProvider.getAefiInvestigationFacade().getByUuid(params); + aefiReportUuid = aefiInvestigationDto.getAefiReport().getUuid(); + + AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(aefiReportUuid); + immunizationUuid = aefiDto.getImmunization().getUuid(); + } + + menu.removeAllViews(); + menu.addView(AefiView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiList)); + menu.addView(ImmunizationDataView.VIEW_NAME, I18nProperties.getCaption(ImmunizationDto.I18N_PREFIX), immunizationUuid); + menu.addView( + ImmunizationPersonView.VIEW_NAME, + I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, ImmunizationDto.PERSON), + immunizationUuid); + menu.addView(AefiDataView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiDataView), aefiReportUuid); + menu.addView(AefiInvestigationDataView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiInvestigationDataView), params); + + setMainHeaderComponent( + ControllerProvider.getAefiInvestigationController() + .getAefiInvestigationViewTitleLayout(getReference().getUuid(), aefiReportUuid, immunizationUuid, isCreateAction)); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java index 3e082e0657a..4499d4a7278 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java @@ -1,115 +1,38 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package de.symeda.sormas.ui.adverseeventsfollowingimmunization; -import com.vaadin.navigator.ViewChangeListener.ViewChangeEvent; +import com.vaadin.ui.Component; -import de.symeda.sormas.api.EditPermissionFacade; -import de.symeda.sormas.api.FacadeProvider; -import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; -import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiReferenceDto; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; -import de.symeda.sormas.api.immunization.ImmunizationDto; -import de.symeda.sormas.api.utils.DataHelper; -import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.SubMenu; -import de.symeda.sormas.ui.immunization.ImmunizationDataView; -import de.symeda.sormas.ui.immunization.ImmunizationPersonView; -import de.symeda.sormas.ui.utils.AbstractEditAllowedDetailView; +import de.symeda.sormas.ui.utils.AbstractSubNavigationView; -public abstract class AbstractAefiView extends AbstractEditAllowedDetailView { - - public static final String ROOT_VIEW_NAME = AefiView.VIEW_NAME; +public class AbstractAefiView extends AbstractSubNavigationView { protected AbstractAefiView(String viewName) { super(viewName); } - @Override - protected AefiReferenceDto getReferenceByUuid(String uuid) { - - final AefiReferenceDto reference; - - boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(uuid); - - if (isCreateAction) { - reference = new AefiReferenceDto(); - reference.setUuid(DataHelper.createUuid()); - } else { - if (FacadeProvider.getAefiFacade().exists(uuid)) { - reference = FacadeProvider.getAefiFacade().getReferenceByUuid(uuid); - } else { - reference = null; - } - } - return reference; - } - - @Override - protected String getRootViewName() { - return ROOT_VIEW_NAME; - } - - @Override - protected void initView(String params) { - - } - - @Override - protected EditPermissionFacade getEditPermissionFacade() { - return FacadeProvider.getAefiFacade(); - } - - @Override - public void enter(ViewChangeEvent event) { - - super.enter(event); - initOrRedirect(event); - } - @Override public void refreshMenu(SubMenu menu, String params) { - - if (!findReferenceByParams(params)) { - return; - } - - boolean isCreateAction = ControllerProvider.getAefiController().isCreateAction(params); - - String immunizationUuid = ""; - if (isCreateAction) { - immunizationUuid = ControllerProvider.getAefiController().getCreateActionImmunizationUuid(params); - } else { - AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(params); - immunizationUuid = aefiDto.getImmunization().getUuid(); - } - menu.removeAllViews(); - menu.addView(AefiView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiList)); - menu.addView(ImmunizationDataView.VIEW_NAME, I18nProperties.getCaption(ImmunizationDto.I18N_PREFIX), immunizationUuid); - menu.addView( - ImmunizationPersonView.VIEW_NAME, - I18nProperties.getPrefixCaption(ImmunizationDto.I18N_PREFIX, ImmunizationDto.PERSON), - immunizationUuid); - menu.addView(AefiDataView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiDataView), params); - setMainHeaderComponent(ControllerProvider.getAefiController().getAefiViewTitleLayout(params, immunizationUuid, isCreateAction)); + menu.addView(AefiView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiList), params); + menu.addView(AefiInvestigationView.VIEW_NAME, I18nProperties.getCaption(Captions.aefiAefiInvestigationList), params); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java index a3200c25c27..7a6f3ef13fe 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiController.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -28,6 +25,7 @@ import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.deletionconfiguration.DeletionInfoDto; import de.symeda.sormas.api.i18n.I18nProperties; @@ -36,6 +34,7 @@ import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.api.vaccination.VaccinationDto; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.SormasUI; @@ -44,8 +43,10 @@ import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.AefiDataForm; import de.symeda.sormas.ui.utils.ArchiveHandlers; import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; +import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.VaadinUiUtil; import de.symeda.sormas.ui.utils.components.automaticdeletion.DeletionLabel; +import de.symeda.sormas.ui.utils.components.page.title.RowLayout; import de.symeda.sormas.ui.utils.components.page.title.TitleLayout; import de.symeda.sormas.ui.utils.components.page.title.TitleLayoutHelper; @@ -53,9 +54,9 @@ public class AefiController { public void registerViews(Navigator navigator) { navigator.addView(AefiView.VIEW_NAME, AefiView.class); - //navigator.addView(ImmunizationDataView.VIEW_NAME, ImmunizationDataView.class); - //navigator.addView(ImmunizationPersonView.VIEW_NAME, ImmunizationPersonView.class); navigator.addView(AefiDataView.VIEW_NAME, AefiDataView.class); + navigator.addView(AefiInvestigationView.VIEW_NAME, AefiInvestigationView.class); + navigator.addView(AefiInvestigationDataView.VIEW_NAME, AefiInvestigationDataView.class); } public void selectPrimarySuspectVaccination(AefiDto aefiDto, Consumer commitCallback) { @@ -67,6 +68,7 @@ public void selectPrimarySuspectVaccination(AefiDto aefiDto, Consumer component = new CommitDiscardWrapperComponent<>(selectionField); component.addCommitListener(() -> { + VaccinationDto selectedVaccination = selectionField.getValue(); if (selectedVaccination != null) { aefiDto.setPrimarySuspectVaccine(selectedVaccination); @@ -78,7 +80,7 @@ public void selectPrimarySuspectVaccination(AefiDto aefiDto, Consumer component.getCommitButton().setEnabled(commitAllowed)); - VaadinUiUtil.showModalPopupWindow(component, I18nProperties.getString(Strings.headingAefiPickPrimarySuspectVaccine)); + VaadinUiUtil.showModalPopupWindow(component, I18nProperties.getString(Strings.headingAefiSelectPrimarySuspectVaccine)); } public void navigateToAefi(String uuid) { @@ -162,18 +164,30 @@ public CommitDiscardWrapperComponent getAefiDataEditComponent( public TitleLayout getAefiViewTitleLayout(String aefiUuid, String immunizationUuid, boolean isCreateAction) { + TitleLayout titleLayout = new TitleLayout(); + if (!isCreateAction) { AefiDto aefiDto = findAefi(aefiUuid); immunizationUuid = aefiDto.getImmunization().getUuid(); - } - TitleLayout titleLayout = new TitleLayout(); + if (aefiDto.getSerious() == YesNoUnknown.YES) { + RowLayout aefiTypeLayout = new RowLayout(); + aefiTypeLayout.addToLayout(AefiType.SERIOUS.toString(), CssStyles.LABEL_CRITICAL); + titleLayout.addRow(aefiTypeLayout); + } else { + titleLayout.addRow(AefiType.NON_SERIOUS.toString()); + } + } - String shortUuid = DataHelper.getShortUuid(immunizationUuid); + String shortUuid = DataHelper.getShortUuid(aefiUuid); ImmunizationDto immunizationDto = FacadeProvider.getImmunizationFacade().getByUuid(immunizationUuid); PersonDto person = FacadeProvider.getPersonFacade().getByUuid(immunizationDto.getPerson().getUuid()); StringBuilder mainRowText = TitleLayoutHelper.buildPersonString(person); - mainRowText.append(mainRowText.length() > 0 ? " (" + shortUuid + ")" : shortUuid); + + if (!StringUtils.isBlank(shortUuid)) { + mainRowText.append(mainRowText.length() > 0 ? " (" + shortUuid + ")" : shortUuid); + } + titleLayout.addMainRow(mainRowText.toString()); return titleLayout; @@ -189,7 +203,7 @@ public boolean isCreateAction(String params) { public String getCreateActionImmunizationUuid(String params) { return StringUtils.contains(params, "/") - ? StringUtils.substringBetween(params, "/", "/") - : StringUtils.substringBetween(params, "immunization", "create"); + ? StringUtils.substringBetween(params, "immunization/", "/adverseevent/create") + : StringUtils.substringBetween(params, "immunization", "adverseeventcreate"); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java index adc07afc106..44444a3b191 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiDataView.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -24,13 +21,16 @@ import de.symeda.sormas.api.EditPermissionType; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListCriteria; import de.symeda.sormas.api.feature.FeatureType; import de.symeda.sormas.api.feature.FeatureTypeProperty; import de.symeda.sormas.api.immunization.ImmunizationDto; import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefiinvestigationlink.AefiInvestigationListComponent; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.AefiDataForm; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiImmunizationInfo; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiPersonInfo; @@ -39,11 +39,10 @@ import de.symeda.sormas.ui.utils.DetailSubComponentWrapper; import de.symeda.sormas.ui.utils.LayoutWithSidePanel; -public class AefiDataView extends AbstractAefiView { +public class AefiDataView extends AbstractAefiDataView { public static final String VIEW_NAME = ROOT_VIEW_NAME + "/data"; - public static final String ADVERSE_EVENT_LOC = "adverseEventLoc"; public static final String PERSON_LOC = "personLoc"; public static final String IMMUNIZATION_LOC = "immunizationLoc"; public static final String INVESTIGATIONS_LOC = "investigationsLoc"; @@ -118,6 +117,18 @@ protected void initView(String params) { } } + if (!isCreateAction + && (aefi.getSerious() == YesNoUnknown.YES) + && currentUser.hasAllUserRights(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + + AefiInvestigationListCriteria listCriteria = new AefiInvestigationListCriteria.Builder(getReference()).build(); + AefiInvestigationListComponent aefiInvestigationListComponent = + new AefiInvestigationListComponent(listCriteria, this::showUnsavedChangesPopup, isEditAllowed(), isCreateAction); + CssStyles.style(aefiInvestigationListComponent, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + + layout.addSidePanelComponent(aefiInvestigationListComponent, INVESTIGATIONS_LOC); + } + if (!isCreateAction) { final String uuid = aefi.getUuid(); final EditPermissionType aefiEditAllowed = FacadeProvider.getAefiFacade().getEditPermissionType(uuid); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationController.java new file mode 100644 index 00000000000..0c44bb7a702 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationController.java @@ -0,0 +1,229 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import java.util.function.Consumer; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.navigator.Navigator; +import com.vaadin.server.Sizeable; +import com.vaadin.ui.Notification; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.deletionconfiguration.DeletionInfoDto; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.SormasUI; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiPrimarySuspectVaccinationSelectionField; +import de.symeda.sormas.ui.utils.ArchiveHandlers; +import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.VaadinUiUtil; +import de.symeda.sormas.ui.utils.components.automaticdeletion.DeletionLabel; +import de.symeda.sormas.ui.utils.components.page.title.RowLayout; +import de.symeda.sormas.ui.utils.components.page.title.TitleLayout; +import de.symeda.sormas.ui.utils.components.page.title.TitleLayoutHelper; + +public class AefiInvestigationController { + + public void registerViews(Navigator navigator) { + } + + public void selectPrimarySuspectVaccination(AefiInvestigationDto aefiInvestigationDto, Consumer commitCallback) { + + AefiPrimarySuspectVaccinationSelectionField selectionField = + new AefiPrimarySuspectVaccinationSelectionField(aefiInvestigationDto.getVaccinations(), aefiInvestigationDto.getPrimarySuspectVaccine()); + selectionField.setWidth(1024, Sizeable.Unit.PIXELS); + + final CommitDiscardWrapperComponent component = + new CommitDiscardWrapperComponent<>(selectionField); + component.addCommitListener(() -> { + VaccinationDto selectedVaccination = selectionField.getValue(); + if (selectedVaccination != null) { + aefiInvestigationDto.setPrimarySuspectVaccine(selectedVaccination); + + if (commitCallback != null) { + commitCallback.accept(selectedVaccination); + } + } + }); + + selectionField.setSelectionChangeCallback((commitAllowed) -> component.getCommitButton().setEnabled(commitAllowed)); + VaadinUiUtil.showModalPopupWindow(component, I18nProperties.getString(Strings.headingAefiInvestigationSelectConcernedVaccine)); + } + + public void navigateToAefiInvestigation(String uuid) { + navigateToView(AefiInvestigationDataView.VIEW_NAME, uuid); + } + + public void navigateToView(String viewName, String uuid) { + final String navigationState = viewName + "/" + uuid; + SormasUI.get().getNavigator().navigateTo(navigationState); + } + + public CommitDiscardWrapperComponent getAefiInvestigationDataEditComponent( + boolean isCreateAction, + AefiInvestigationDto aefiInvestigationDto, + Consumer actionCallback) { + + AefiInvestigationDataForm aefiInvestigationDataForm = new AefiInvestigationDataForm( + isCreateAction, + aefiInvestigationDto.isPseudonymized(), + aefiInvestigationDto.isInJurisdiction(), + actionCallback); + aefiInvestigationDataForm.setValue(aefiInvestigationDto); + + CommitDiscardWrapperComponent editComponent = + new CommitDiscardWrapperComponent<>(aefiInvestigationDataForm, true, aefiInvestigationDataForm.getFieldGroup()); + + if (!isCreateAction) { + DeletionInfoDto automaticDeletionInfoDto = + FacadeProvider.getAefiInvestigationFacade().getAutomaticDeletionInfo(aefiInvestigationDto.getUuid()); + DeletionInfoDto manuallyDeletionInfoDto = + FacadeProvider.getAefiInvestigationFacade().getManuallyDeletionInfo(aefiInvestigationDto.getUuid()); + + editComponent.getButtonsPanel() + .addComponentAsFirst( + new DeletionLabel( + automaticDeletionInfoDto, + manuallyDeletionInfoDto, + aefiInvestigationDto.isDeleted(), + AefiInvestigationDto.I18N_PREFIX)); + + if (aefiInvestigationDto.isDeleted()) { + editComponent.getWrappedComponent().getField(AefiInvestigationDto.DELETION_REASON).setVisible(true); + if (editComponent.getWrappedComponent().getField(AefiInvestigationDto.DELETION_REASON).getValue() == DeletionReason.OTHER_REASON) { + editComponent.getWrappedComponent().getField(AefiInvestigationDto.OTHER_DELETION_REASON).setVisible(true); + } + } + } + + editComponent.addCommitListener(() -> { + if (!aefiInvestigationDataForm.getFieldGroup().isModified()) { + AefiInvestigationDto aefiInvestigationDataFormValue = aefiInvestigationDataForm.getValue(); + + AefiInvestigationDto savedAefiInvestigationDto = FacadeProvider.getAefiInvestigationFacade().save(aefiInvestigationDataFormValue); + Notification.show(I18nProperties.getString(Strings.messageAdverseEventInvestigationSaved), Notification.Type.WARNING_MESSAGE); + + if (isCreateAction) { + navigateToAefiInvestigation(savedAefiInvestigationDto.getUuid()); + } else { + SormasUI.refreshView(); + } + } + }); + + if (!isCreateAction) { + // Initialize 'Delete' button + if (UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE)) { + editComponent.addDeleteWithReasonOrRestoreListener( + AefiInvestigationView.VIEW_NAME, + null, + I18nProperties.getString(Strings.entityAdverseEventInvestigation), + aefiInvestigationDto.getUuid(), + FacadeProvider.getAefiInvestigationFacade()); + } + + // Initialize 'Archive' button + if (UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE)) { + ControllerProvider.getArchiveController() + .addArchivingButton( + aefiInvestigationDto, + ArchiveHandlers.forAefiInvestigation(), + editComponent, + () -> navigateToAefiInvestigation(aefiInvestigationDto.getUuid())); + } + + editComponent.restrictEditableComponentsOnEditView( + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT, + null, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE, + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE, + FacadeProvider.getAefiInvestigationFacade().getEditPermissionType(aefiInvestigationDto.getUuid()), + aefiInvestigationDto.isInJurisdiction()); + } + + return editComponent; + } + + public TitleLayout getAefiInvestigationViewTitleLayout( + String aefiInvestigationUuid, + String aefiUuid, + String immunizationUuid, + boolean isCreateAction) { + + TitleLayout titleLayout = new TitleLayout(); + + String investigationStage = ""; + + if (isCreateAction) { + investigationStage = I18nProperties.getCaption(Captions.aefiNewAefiInvestigationStageTitle); + } else { + AefiInvestigationDto aefiInvestigationDto = findAefiInvestigation(aefiInvestigationUuid); + investigationStage = aefiInvestigationDto.getInvestigationStage().toString(); + } + + AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(aefiUuid); + if (aefiDto.getSerious() == YesNoUnknown.YES) { + RowLayout aefiTypeLayout = new RowLayout(); + aefiTypeLayout.addToLayout(investigationStage + ", "); + aefiTypeLayout.addToLayout(AefiType.SERIOUS.toString(), CssStyles.LABEL_CRITICAL); + titleLayout.addRow(aefiTypeLayout); + } else { + titleLayout.addRow(investigationStage + ", " + AefiType.NON_SERIOUS); + } + + String shortUuid = DataHelper.getShortUuid(aefiInvestigationUuid); + ImmunizationDto immunizationDto = FacadeProvider.getImmunizationFacade().getByUuid(immunizationUuid); + PersonDto person = FacadeProvider.getPersonFacade().getByUuid(immunizationDto.getPerson().getUuid()); + StringBuilder mainRowText = TitleLayoutHelper.buildPersonString(person); + + if (!StringUtils.isBlank(shortUuid)) { + mainRowText.append(mainRowText.length() > 0 ? " (" + shortUuid + ")" : shortUuid); + } + titleLayout.addMainRow(mainRowText.toString()); + + return titleLayout; + } + + private AefiInvestigationDto findAefiInvestigation(String uuid) { + return FacadeProvider.getAefiInvestigationFacade().getByUuid(uuid); + } + + public boolean isCreateAction(String params) { + return StringUtils.startsWith(params, "adverseevent") && StringUtils.endsWith(params, "create"); + } + + public String getCreateActionAefiReportUuid(String params) { + return StringUtils.contains(params, "/") + ? StringUtils.substringBetween(params, "adverseevent/", "/investigation/create") + : StringUtils.substringBetween(params, "adverseevent", "investigationcreate"); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java new file mode 100644 index 00000000000..1ba7beb0475 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java @@ -0,0 +1,950 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import static de.symeda.sormas.ui.utils.CssStyles.H3; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidColumnLocCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRow; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowLocs; +import static de.symeda.sormas.ui.utils.LayoutUtil.loc; +import static de.symeda.sormas.ui.utils.LayoutUtil.locCss; + +import java.util.Arrays; +import java.util.function.Consumer; + +import com.vaadin.ui.CustomLayout; +import com.vaadin.ui.Label; +import com.vaadin.v7.data.util.converter.Converter; +import com.vaadin.v7.ui.DateField; +import com.vaadin.v7.ui.OptionGroup; +import com.vaadin.v7.ui.PasswordField; +import com.vaadin.v7.ui.TextArea; +import com.vaadin.v7.ui.TextField; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.DeliveryProcedure; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PatientStatusAtAefiInvestigation; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.PlaceOfVaccination; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeriousAefiInfoSource; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.SyringeType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccinationActivity; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.VaccinationSite; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.Descriptions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.FormSectionAccordion; +import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldHelper; +import de.symeda.sormas.ui.utils.NullableOptionGroup; +import de.symeda.sormas.ui.utils.UserField; + +@SuppressWarnings("deprecation") +public class AefiInvestigationDataForm extends AbstractEditForm { + + public static final String FORM_HEADING_LOC = "formHeadingLoc"; + public static final String MAIN_ACCORDION_LOC = "mainAccordionLoc"; + public static final String FOR_INFANTS_HEADING_LOC = "forInfantsHeadingLoc"; + public static final String SOURCE_OF_INFORMATION_HEADING_LOC = "sourceOfInformationHeadingLoc"; + public static final String CLINICAL_DETAILS_OFFICER_HEADING_LOC = "clinicalDetailsOfficerHeadingLoc"; + public static final String SYRINGES_AND_NEEDLES_HEADING_LOC = "syringesAndNeedlesHeadingLoc"; + public static final String RECONSTITUTION_HEADING_LOC = "reconstitutionHeadingLoc"; + public static final String INJECTION_TECHNIQUE_HEADING_LOC = "injectionTechniqueHeadingLoc"; + public static final String VACCINE_STORAGE_POINT_HEADING_LOC = "vaccineStoragePointHeadingLoc"; + public static final String VACCINE_TRANSPORTATION_HEADING_LOC = "vaccineTransportationHeadingLoc"; + public static final String THOSE_AFFECTED_HEADING_LOC = "thoseAffectedHeadingLoc"; + + //@formatter:off + public static final String HTML_LAYOUT = + locCss(CssStyles.VSPACE_3, FORM_HEADING_LOC) + + fluidRowLocs(MAIN_ACCORDION_LOC); + //@formatter:on + + //@formatter:off + public static final String BASIC_DETAILS_HTML_LAYOUT = + fluidRowLocs(4, AefiInvestigationDto.UUID, 4, AefiInvestigationDto.REPORT_DATE, 3, AefiInvestigationDto.REPORTING_USER) + + fluidRowLocs(4, AefiInvestigationDto.RESPONSIBLE_REGION, 4, AefiInvestigationDto.RESPONSIBLE_DISTRICT, 3, AefiInvestigationDto.RESPONSIBLE_COMMUNITY) + + fluidRowLocs(4, AefiInvestigationDto.INVESTIGATION_CASE_ID) + + fluidRowLocs(8, AefiInvestigationDto.PLACE_OF_VACCINATION, 4, AefiInvestigationDto.PLACE_OF_VACCINATION_DETAILS) + + fluidRowLocs(4, AefiInvestigationDto.VACCINATION_ACTIVITY, 4, AefiInvestigationDto.VACCINATION_ACTIVITY_DETAILS) + + fluidRowLocs(AefiInvestigationDto.INVESTIGATION_DATE, AefiInvestigationDto.FORM_COMPLETION_DATE, AefiInvestigationDto.INVESTIGATION_STAGE) + + fluidRowLocs(AefiInvestigationDto.VACCINATIONS) + + fluidRowLocs(8, AefiInvestigationDto.TYPE_OF_SITE, 4, AefiInvestigationDto.TYPE_OF_SITE_DETAILS) + + fluidRowLocs(4, AefiInvestigationDto.KEY_SYMPTOM_DATE_TIME, 3, AefiInvestigationDto.HOSPITALIZATION_DATE, 5, AefiInvestigationDto.REPORTED_TO_HEALTH_AUTHORITY_DATE) + + fluidRowLocs(AefiInvestigationDto.STATUS_ON_DATE_OF_INVESTIGATION) + + fluidRowLocs(4, AefiInvestigationDto.DEATH_DATE_TIME, 3, AefiInvestigationDto.AUTOPSY_DONE, 5, AefiInvestigationDto.AUTOPSY_DATE) + + fluidRowLocs(4, "", 5, AefiInvestigationDto.AUTOPSY_PLANNED_DATE_TIME); + //@formatter:on + + //@formatter:off + public static final String RELEVANT_PATIENT_INFO_HTML_LAYOUT = + fluidRowLocs(8, AefiInvestigationDto.PAST_HISTORY_OF_SIMILAR_EVENT, 4, AefiInvestigationDto.PAST_HISTORY_OF_SIMILAR_EVENT_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS, 4, AefiInvestigationDto.ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD, 4, AefiInvestigationDto.HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER, 4, AefiInvestigationDto.PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE, 4, AefiInvestigationDto.HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.CURRENTLY_ON_CONCOMITANT_MEDICATION, 4, AefiInvestigationDto.CURRENTLY_ON_CONCOMITANT_MEDICATION_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY, 4, AefiInvestigationDto.FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY_DETAILS) + + loc(FOR_INFANTS_HEADING_LOC) + + fluidRowLocs(8, AefiInvestigationDto.BIRTH_TERM, 4, AefiInvestigationDto.BIRTH_WEIGHT) + + fluidRowLocs(8, AefiInvestigationDto.DELIVERY_PROCEDURE, 4, AefiInvestigationDto.DELIVERY_PROCEDURE_DETAILS); + //@formatter:on + + //@formatter:off + public static final String FIRST_EXAMINATION_HTML_LAYOUT = + loc(SOURCE_OF_INFORMATION_HEADING_LOC) + + fluidRowLocs(AefiInvestigationDto.SERIOUS_AEFI_INFO_SOURCE) + + fluidRowLocs(AefiInvestigationDto.SERIOUS_AEFI_INFO_SOURCE_DETAILS, AefiInvestigationDto.SERIOUS_AEFI_VERBAL_AUTOPSY_INFO_SOURCE_DETAILS) + + fluidRowLocs(AefiInvestigationDto.FIRST_CAREGIVERS_NAME, AefiInvestigationDto.OTHER_CAREGIVERS_NAMES) + + fluidRowLocs(6, AefiInvestigationDto.OTHER_SOURCES_WHO_PROVIDED_INFO) + + fluidRowLocs(AefiInvestigationDto.SIGNS_AND_SYMPTOMS_FROM_TIME_OF_VACCINATION) + + loc(CLINICAL_DETAILS_OFFICER_HEADING_LOC) + + fluidRowLocs(5, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_NAME, 4, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_PHONE_NUMBER, 3, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_EMAIL) + + fluidRowLocs(5, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_DESIGNATION, 4, AefiInvestigationDto.CLINICAL_DETAILS_DATE_TIME) + + fluidRowLocs(4, AefiInvestigationDto.PATIENT_RECEIVED_MEDICAL_CARE) + + fluidRowLocs(AefiInvestigationDto.PATIENT_RECEIVED_MEDICAL_CARE_DETAILS) + + fluidRowLocs(AefiInvestigationDto.PROVISIONAL_OR_FINAL_DIAGNOSIS); + //@formatter:on + + //@formatter:off + public static final String VACCINES_DETAILS_HTML_LAYOUT = + fluidRowLocs(AefiInvestigationDto.PATIENT_IMMUNIZED_PERIOD) + + fluidRowLocs(AefiInvestigationDto.VACCINE_GIVEN_PERIOD) + + fluidRowCss(CssStyles.VSPACE_TOP_4, + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.ERROR_PRESCRIBING_VACCINE), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.ERROR_PRESCRIBING_VACCINE_DETAILS) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_COULD_HAVE_BEEN_UNSTERILE), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.VACCINE_COULD_HAVE_BEEN_UNSTERILE_DETAILS) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_PHYSICAL_CONDITION_ABNORMAL), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.VACCINE_PHYSICAL_CONDITION_ABNORMAL_DETAILS) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.ERROR_IN_VACCINE_RECONSTITUTION), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.ERROR_IN_VACCINE_RECONSTITUTION_DETAILS) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.ERROR_IN_VACCINE_HANDLING), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.ERROR_IN_VACCINE_HANDLING_DETAILS) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_ADMINISTERED_INCORRECTLY), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.VACCINE_ADMINISTERED_INCORRECTLY_DETAILS) + ) + + fluidRowLocs(8, AefiInvestigationDto.NUMBER_IMMUNIZED_FROM_CONCERNED_VACCINE_VIAL) + + fluidRowLocs(8, AefiInvestigationDto.NUMBER_IMMUNIZED_WITH_CONCERNED_VACCINE_IN_SAME_SESSION) + + fluidRowLocs(12, AefiInvestigationDto.NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_OTHER_LOCATIONS) + + fluidRowLocs(12, AefiInvestigationDto.NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_LOCATION_DETAILS) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_HAS_QUALITY_DEFECT), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.VACCINE_HAS_QUALITY_DEFECT_DETAILS) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION), + fluidColumnLocCss("", 4, 0, AefiInvestigationDto.EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION_DETAILS) + ) + + fluidRowLocs(4, AefiInvestigationDto.CASE_IS_PART_OF_A_CLUSTER, 8, AefiInvestigationDto.CASE_IS_PART_OF_A_CLUSTER_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.NUMBER_OF_CASES_DETECTED_IN_CLUSTER) + + fluidRowLocs(8, AefiInvestigationDto.ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL, 4, AefiInvestigationDto.ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL_DETAILS) + + fluidRowLocs(8, AefiInvestigationDto.NUMBER_OF_VIALS_USED_IN_CLUSTER); + //@formatter:on + + //@formatter:off + public static final String IMMUNIZATION_PRACTICES_HTML_LAYOUT = + loc(SYRINGES_AND_NEEDLES_HEADING_LOC) + + fluidRowLocs(6, AefiInvestigationDto.AD_SYRINGES_USED_FOR_IMMUNIZATION) + + fluidRowLocs(8, AefiInvestigationDto.TYPE_OF_SYRINGES_USED, 4, AefiInvestigationDto.TYPE_OF_SYRINGES_USED_DETAILS) + + fluidRowLocs(AefiInvestigationDto.SYRINGES_USED_ADDITIONAL_DETAILS) + + loc(RECONSTITUTION_HEADING_LOC) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_USED_FOR_MULTIPLE_VIALS_OF_SAME_VACCINE) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_USED_FOR_RECONSTITUTING_DIFFERENT_VACCINES) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINE_VIAL) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINATION) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINES_AND_DILUENTS_USED_RECOMMENDED_BY_MANUFACTURER) + ) + + fluidRowLocs(AefiInvestigationDto.RECONSTITUTION_ADDITIONAL_DETAILS) + + loc(INJECTION_TECHNIQUE_HEADING_LOC) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.CORRECT_DOSE_OR_ROUTE) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.TIME_OF_RECONSTITUTION_MENTIONED_ON_THE_VIAL) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.NON_TOUCH_TECHNIQUE_FOLLOWED) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.CONTRAINDICATION_SCREENED_PRIOR_TO_VACCINATION) + ) + + fluidRowLocs(AefiInvestigationDto.NUMBER_OF_AEFI_REPORTED_FROM_VACCINE_DISTRIBUTION_CENTER_LAST_THIRTY_DAYS) + + fluidRowLocs(4, AefiInvestigationDto.TRAINING_RECEIVED_BY_VACCINATOR, 4, AefiInvestigationDto.LAST_TRAINING_RECEIVED_BY_VACCINATOR_DATE) + + fluidRowLocs(AefiInvestigationDto.INJECTION_TECHNIQUE_ADDITIONAL_DETAILS); + //@formatter:on + + //@formatter:off + public static final String COLD_CHAIN_AND_TRANSPORT_HTML_LAYOUT = + loc(VACCINE_STORAGE_POINT_HEADING_LOC) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_STORAGE_REFRIGERATOR_TEMPERATURE_MONITORED) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.ANY_STORAGE_TEMPERATURE_DEVIATION_OUTSIDE_TWO_TO_EIGHT_DEGREES) + ) + + fluidRowLocs(AefiInvestigationDto.STORAGE_TEMPERATURE_MONITORING_ADDITIONAL_DETAILS) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.CORRECT_PROCEDURE_FOR_STORAGE_FOLLOWED) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.ANY_OTHER_ITEM_IN_REFRIGERATOR) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.PARTIALLY_USED_RECONSTITUTED_VACCINES_IN_REFRIGERATOR) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.UNUSABLE_VACCINES_IN_REFRIGERATOR) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.UNUSABLE_DILUENTS_IN_STORE) + ) + + fluidRowLocs(AefiInvestigationDto.VACCINE_STORAGE_POINT_ADDITIONAL_DETAILS) + + loc(VACCINE_TRANSPORTATION_HEADING_LOC) + + fluidRowLocs(4, AefiInvestigationDto.VACCINE_CARRIER_TYPE) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_CARRIER_SENT_TO_SITE_ON_SAME_DATE_AS_VACCINATION) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.VACCINE_CARRIER_RETURNED_FROM_SITE_ON_SAME_DATE_AS_VACCINATION) + ) + + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.CONDITIONED_ICE_PACK_USED) + ) + + fluidRowLocs(AefiInvestigationDto.VACCINE_TRANSPORTATION_ADDITIONAL_DETAILS); + //@formatter:on + + //@formatter:off + public static final String COMMUNITY_INVESTIGATION_HTML_LAYOUT = + fluidRow( + fluidColumnLocCss(CssStyles.OPTIONGROUP_CAPTION_FLEX, 8, 0, AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY) + ) + + fluidRowLocs(AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY_DETAILS, AefiInvestigationDto.NUMBER_OF_SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY) + + loc(THOSE_AFFECTED_HEADING_LOC) + + fluidRowLocs(AefiInvestigationDto.NUMBER_OF_THOSE_AFFECTED_VACCINATED, AefiInvestigationDto.NUMBER_OF_THOSE_AFFECTED_NOT_VACCINATED, AefiInvestigationDto.NUMBER_OF_THOSE_AFFECTED_VACCINATED_UNKNOWN) + + fluidRowLocs(AefiInvestigationDto.COMMUNITY_INVESTIGATION_ADDITIONAL_DETAILS); + //@formatter:on + + //@formatter:off + public static final String OTHER_FINDINGS_HTML_LAYOUT = + fluidRowLocs(AefiInvestigationDto.OTHER_INVESTIGATION_FINDINGS); + //@formatter:on + + //@formatter:off + public static final String INVESTIGATION_STATUS_HTML_LAYOUT = + fluidRowLocs(4, AefiInvestigationDto.INVESTIGATION_STATUS, 8, AefiInvestigationDto.INVESTIGATION_STATUS_DETAILS) + + fluidRowLocs(AefiInvestigationDto.AEFI_CLASSIFICATION) + + fluidRowLocs(AefiInvestigationDto.AEFI_CLASSIFICATION_DETAILS); + //@formatter:on + + private boolean isCreateAction; + private final Consumer actionCallback; + private TextField responsibleRegion; + private TextField responsibleDistrict; + private TextField responsibleCommunity; + private AefiVaccinationsField vaccinationsField; + + public AefiInvestigationDataForm(boolean isCreateAction, boolean isPseudonymized, boolean inJurisdiction, Consumer actionCallback) { + super( + AefiInvestigationDto.class, + AefiInvestigationDto.I18N_PREFIX, + false, + FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), + UiFieldAccessCheckers.forDataAccessLevel(UserProvider.getCurrent().getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + + this.isCreateAction = isCreateAction; + this.actionCallback = actionCallback; + + if (isCreateAction) { + hideValidationUntilNextCommit(); + } + + addFields(); + } + + @Override + protected String createHtmlLayout() { + return HTML_LAYOUT; + } + + @Override + protected void addFields() { + + Label formHeadingLabel = new Label(I18nProperties.getString(Strings.headingAefiInvestigationFormSubHeading)); + formHeadingLabel.addStyleNames(H3, CssStyles.LABEL_CRITICAL, CssStyles.VSPACE_NONE, CssStyles.VSPACE_TOP_NONE); + getContent().addComponent(formHeadingLabel, FORM_HEADING_LOC); + + FormSectionAccordion accordion = new FormSectionAccordion(); + + //Basic details + CustomLayout basicDetailsLayout = new CustomLayout(); + basicDetailsLayout.setTemplateContents(BASIC_DETAILS_HTML_LAYOUT); + + if (isCreateAction) { + addField(basicDetailsLayout, AefiInvestigationDto.UUID, PasswordField.class); + } else { + addField(basicDetailsLayout, AefiInvestigationDto.UUID); + } + addField(basicDetailsLayout, AefiInvestigationDto.REPORT_DATE, DateField.class); + addField(basicDetailsLayout, AefiInvestigationDto.REPORTING_USER, UserField.class); + + responsibleRegion = new TextField(I18nProperties.getPrefixCaption(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.RESPONSIBLE_REGION)); + basicDetailsLayout.addComponent(responsibleRegion, AefiInvestigationDto.RESPONSIBLE_REGION); + responsibleDistrict = + new TextField(I18nProperties.getPrefixCaption(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.RESPONSIBLE_DISTRICT)); + basicDetailsLayout.addComponent(responsibleDistrict, AefiInvestigationDto.RESPONSIBLE_DISTRICT); + responsibleCommunity = + new TextField(I18nProperties.getPrefixCaption(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.RESPONSIBLE_COMMUNITY)); + basicDetailsLayout.addComponent(responsibleCommunity, AefiInvestigationDto.RESPONSIBLE_COMMUNITY); + + addField(basicDetailsLayout, AefiInvestigationDto.INVESTIGATION_CASE_ID); + addField(basicDetailsLayout, AefiInvestigationDto.PLACE_OF_VACCINATION, NullableOptionGroup.class); + addField(basicDetailsLayout, AefiInvestigationDto.PLACE_OF_VACCINATION_DETAILS); + addField(basicDetailsLayout, AefiInvestigationDto.VACCINATION_ACTIVITY, NullableOptionGroup.class); + addField(basicDetailsLayout, AefiInvestigationDto.VACCINATION_ACTIVITY_DETAILS); + addField(basicDetailsLayout, AefiInvestigationDto.INVESTIGATION_DATE, DateField.class); + addField(basicDetailsLayout, AefiInvestigationDto.FORM_COMPLETION_DATE, DateField.class); + addField(basicDetailsLayout, AefiInvestigationDto.INVESTIGATION_STAGE, NullableOptionGroup.class); + vaccinationsField = addField(basicDetailsLayout, AefiInvestigationDto.VACCINATIONS, AefiVaccinationsField.class); + addField(basicDetailsLayout, AefiInvestigationDto.TYPE_OF_SITE, NullableOptionGroup.class); + addField(basicDetailsLayout, AefiInvestigationDto.TYPE_OF_SITE_DETAILS); + addField(basicDetailsLayout, AefiInvestigationDto.KEY_SYMPTOM_DATE_TIME, DateTimeField.class); + addField(basicDetailsLayout, AefiInvestigationDto.HOSPITALIZATION_DATE, DateField.class); + addField(basicDetailsLayout, AefiInvestigationDto.REPORTED_TO_HEALTH_AUTHORITY_DATE, DateField.class); + addField(basicDetailsLayout, AefiInvestigationDto.STATUS_ON_DATE_OF_INVESTIGATION, NullableOptionGroup.class); + addField(basicDetailsLayout, AefiInvestigationDto.DEATH_DATE_TIME, DateTimeField.class); + addField(basicDetailsLayout, AefiInvestigationDto.AUTOPSY_DONE, NullableOptionGroup.class); + addField(basicDetailsLayout, AefiInvestigationDto.AUTOPSY_DATE, DateField.class); + addField(basicDetailsLayout, AefiInvestigationDto.AUTOPSY_PLANNED_DATE_TIME, DateTimeField.class); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationBasicDetails, true, basicDetailsLayout); + + CustomLayout relevantPatientInformationLayout = new CustomLayout(); + relevantPatientInformationLayout.setTemplateContents(RELEVANT_PATIENT_INFO_HTML_LAYOUT); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.PAST_HISTORY_OF_SIMILAR_EVENT); + TextField pastHistoryOfSimilarEventDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.PAST_HISTORY_OF_SIMILAR_EVENT_DETAILS); + pastHistoryOfSimilarEventDetails.setCaption(null); + pastHistoryOfSimilarEventDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS); + TextField adverseEventAfterPreviousVaccinationsDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS_DETAILS); + adverseEventAfterPreviousVaccinationsDetails.setCaption(null); + adverseEventAfterPreviousVaccinationsDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD); + TextField historyOfAllergyDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD_DETAILS); + historyOfAllergyDetails.setCaption(null); + historyOfAllergyDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER); + TextField preExistingIllnessDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER_DETAILS); + preExistingIllnessDetails.setCaption(null); + preExistingIllnessDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE); + TextField historyOfHospitalizationDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE_DETAILS); + historyOfHospitalizationDetails.setCaption(null); + historyOfHospitalizationDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.CURRENTLY_ON_CONCOMITANT_MEDICATION); + TextField currentlyOnConcomitantMedicationDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.CURRENTLY_ON_CONCOMITANT_MEDICATION_DETAILS); + currentlyOnConcomitantMedicationDetails.setCaption(null); + currentlyOnConcomitantMedicationDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY); + TextField familyHistoryOfDiseaseDetails = + addField(relevantPatientInformationLayout, AefiInvestigationDto.FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY_DETAILS); + familyHistoryOfDiseaseDetails.setCaption(null); + familyHistoryOfDiseaseDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + Label forInfantsLabel = new Label(I18nProperties.getCaption(Captions.aefiInvestigationForInfants)); + forInfantsLabel.addStyleName(CssStyles.H4); + relevantPatientInformationLayout.addComponent(forInfantsLabel, FOR_INFANTS_HEADING_LOC); + + addField(relevantPatientInformationLayout, AefiInvestigationDto.BIRTH_TERM, NullableOptionGroup.class); + addField(relevantPatientInformationLayout, AefiInvestigationDto.BIRTH_WEIGHT); + addField(relevantPatientInformationLayout, AefiInvestigationDto.DELIVERY_PROCEDURE, NullableOptionGroup.class); + addField(relevantPatientInformationLayout, AefiInvestigationDto.DELIVERY_PROCEDURE_DETAILS); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationRelevantPatientInformation, false, relevantPatientInformationLayout); + + //First examination + CustomLayout firstExaminationDetailsLayout = new CustomLayout(); + firstExaminationDetailsLayout.setTemplateContents(FIRST_EXAMINATION_HTML_LAYOUT); + + Label sourceOfInformationLabel = new Label(I18nProperties.getCaption(Captions.aefiInvestigationSourceOfInformation)); + sourceOfInformationLabel.addStyleName(CssStyles.H4); + firstExaminationDetailsLayout.addComponent(sourceOfInformationLabel, SOURCE_OF_INFORMATION_HEADING_LOC); + + OptionGroup seriousAefiSourcesOfInformation = + addField(firstExaminationDetailsLayout, AefiInvestigationDto.SERIOUS_AEFI_INFO_SOURCE, OptionGroup.class); + CssStyles.style(seriousAefiSourcesOfInformation, CssStyles.OPTIONGROUP_CHECKBOXES_HORIZONTAL); + seriousAefiSourcesOfInformation.setMultiSelect(true); + seriousAefiSourcesOfInformation.addItems((Object[]) SeriousAefiInfoSource.values()); + seriousAefiSourcesOfInformation.setCaption(null); + + addField(firstExaminationDetailsLayout, AefiInvestigationDto.SERIOUS_AEFI_INFO_SOURCE_DETAILS); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.SERIOUS_AEFI_VERBAL_AUTOPSY_INFO_SOURCE_DETAILS); + + addField(firstExaminationDetailsLayout, AefiInvestigationDto.FIRST_CAREGIVERS_NAME); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.OTHER_CAREGIVERS_NAMES); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.OTHER_SOURCES_WHO_PROVIDED_INFO); + + TextArea signsAndSymptomsFromTimeOfVaccination = + addField(firstExaminationDetailsLayout, AefiInvestigationDto.SIGNS_AND_SYMPTOMS_FROM_TIME_OF_VACCINATION, TextArea.class); + signsAndSymptomsFromTimeOfVaccination.setRows(6); + signsAndSymptomsFromTimeOfVaccination.setDescription( + I18nProperties + .getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.SIGNS_AND_SYMPTOMS_FROM_TIME_OF_VACCINATION, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + Label clinicalDetailsOfficerLabel = new Label(I18nProperties.getCaption(Captions.aefiInvestigationClinicalDetailsOfficer)); + clinicalDetailsOfficerLabel.addStyleName(CssStyles.H4); + firstExaminationDetailsLayout.addComponent(clinicalDetailsOfficerLabel, CLINICAL_DETAILS_OFFICER_HEADING_LOC); + + addField(firstExaminationDetailsLayout, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_NAME); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_PHONE_NUMBER); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_EMAIL); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.CLINICAL_DETAILS_OFFICER_DESIGNATION); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.CLINICAL_DETAILS_DATE_TIME, DateTimeField.class); + addField(firstExaminationDetailsLayout, AefiInvestigationDto.PATIENT_RECEIVED_MEDICAL_CARE, NullableOptionGroup.class); + + TextArea patientReceivedMedicalCareDetails = + addField(firstExaminationDetailsLayout, AefiInvestigationDto.PATIENT_RECEIVED_MEDICAL_CARE_DETAILS, TextArea.class); + patientReceivedMedicalCareDetails.setRows(6); + patientReceivedMedicalCareDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.PATIENT_RECEIVED_MEDICAL_CARE_DETAILS, "") + + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + TextArea provisionalOrFinalDiagnosis = + addField(firstExaminationDetailsLayout, AefiInvestigationDto.PROVISIONAL_OR_FINAL_DIAGNOSIS, TextArea.class); + provisionalOrFinalDiagnosis.setRows(6); + provisionalOrFinalDiagnosis.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.PROVISIONAL_OR_FINAL_DIAGNOSIS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationFirstExaminationDetails, false, firstExaminationDetailsLayout); + + //Vaccines details + CustomLayout vaccinesDetailsLayout = new CustomLayout(); + vaccinesDetailsLayout.setTemplateContents(VACCINES_DETAILS_HTML_LAYOUT); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.PATIENT_IMMUNIZED_PERIOD, NullableOptionGroup.class); + /* + * TextField patientImmunizedPeriodDetails = addField(vaccinesDetailsLayout, AefiInvestigationDto.PATIENT_IMMUNIZED_PERIOD_DETAILS); + * patientImmunizedPeriodDetails.setCaption(null); + * patientImmunizedPeriodDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + */ + + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_GIVEN_PERIOD, NullableOptionGroup.class); + /* + * TextField vaccineGivenPeriodDetails = addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_GIVEN_PERIOD_DETAILS); + * vaccineGivenPeriodDetails.setCaption(null); + * vaccineGivenPeriodDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + */ + + addField(vaccinesDetailsLayout, AefiInvestigationDto.ERROR_PRESCRIBING_VACCINE); + TextField errorPrescribingVaccineDetails = addField(vaccinesDetailsLayout, AefiInvestigationDto.ERROR_PRESCRIBING_VACCINE_DETAILS); + errorPrescribingVaccineDetails.setCaption(null); + errorPrescribingVaccineDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_COULD_HAVE_BEEN_UNSTERILE); + TextField vaccineCouldBeUnsterileDetails = addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_COULD_HAVE_BEEN_UNSTERILE_DETAILS); + vaccineCouldBeUnsterileDetails.setCaption(null); + vaccineCouldBeUnsterileDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_PHYSICAL_CONDITION_ABNORMAL); + TextField vaccinePhysicalConditionAbnormalDetails = + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_PHYSICAL_CONDITION_ABNORMAL_DETAILS); + vaccinePhysicalConditionAbnormalDetails.setCaption(null); + vaccinePhysicalConditionAbnormalDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.ERROR_IN_VACCINE_RECONSTITUTION); + TextField errorInVaccineReconstitutionAbnormalDetails = + addField(vaccinesDetailsLayout, AefiInvestigationDto.ERROR_IN_VACCINE_RECONSTITUTION_DETAILS); + errorInVaccineReconstitutionAbnormalDetails.setCaption(null); + errorInVaccineReconstitutionAbnormalDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.ERROR_IN_VACCINE_HANDLING); + TextField errorInVaccineHandlingDetails = addField(vaccinesDetailsLayout, AefiInvestigationDto.ERROR_IN_VACCINE_HANDLING_DETAILS); + errorInVaccineHandlingDetails.setCaption(null); + errorInVaccineHandlingDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_ADMINISTERED_INCORRECTLY); + TextField vaccineAdministeredIncorrectlyDetails = + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_ADMINISTERED_INCORRECTLY_DETAILS); + vaccineAdministeredIncorrectlyDetails.setCaption(null); + vaccineAdministeredIncorrectlyDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + TextField numberImmunizedFromConcernedVaccineVial = + addField(vaccinesDetailsLayout, AefiInvestigationDto.NUMBER_IMMUNIZED_FROM_CONCERNED_VACCINE_VIAL, TextField.class); + numberImmunizedFromConcernedVaccineVial.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberImmunizedFromConcernedVaccineVial.getCaption())); + + TextField numberImmunizedFromConcernedVaccineInSameSession = + addField(vaccinesDetailsLayout, AefiInvestigationDto.NUMBER_IMMUNIZED_WITH_CONCERNED_VACCINE_IN_SAME_SESSION, TextField.class); + numberImmunizedFromConcernedVaccineInSameSession.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberImmunizedFromConcernedVaccineInSameSession.getCaption())); + + TextField numberImmunizedFromConcernedVaccineSameBatchNumberOtherLocationsSession = addField( + vaccinesDetailsLayout, + AefiInvestigationDto.NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_OTHER_LOCATIONS, + TextField.class); + numberImmunizedFromConcernedVaccineSameBatchNumberOtherLocationsSession.setConversionError( + I18nProperties.getValidationError( + Validations.onlyIntegerNumbersAllowed, + numberImmunizedFromConcernedVaccineSameBatchNumberOtherLocationsSession.getCaption())); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.NUMBER_IMMUNIZED_CONCERNED_VACCINE_SAME_BATCH_NUMBER_LOCATION_DETAILS); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_HAS_QUALITY_DEFECT); + TextField vaccineHasQualityDefectDetails = addField(vaccinesDetailsLayout, AefiInvestigationDto.VACCINE_HAS_QUALITY_DEFECT_DETAILS); + vaccineHasQualityDefectDetails.setCaption(null); + vaccineHasQualityDefectDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION); + TextField eventIsStressResponseRelatedToImmunizationDetails = + addField(vaccinesDetailsLayout, AefiInvestigationDto.EVENT_IS_A_STRESS_RESPONSE_RELATED_TO_IMMUNIZATION_DETAILS); + eventIsStressResponseRelatedToImmunizationDetails.setCaption(null); + eventIsStressResponseRelatedToImmunizationDetails.setInputPrompt(I18nProperties.getString(Strings.promptRemarks)); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.CASE_IS_PART_OF_A_CLUSTER, NullableOptionGroup.class); + addField(vaccinesDetailsLayout, AefiInvestigationDto.CASE_IS_PART_OF_A_CLUSTER_DETAILS); + + TextField numberOfCasesDetectedInCluster = + addField(vaccinesDetailsLayout, AefiInvestigationDto.NUMBER_OF_CASES_DETECTED_IN_CLUSTER, TextField.class); + numberOfCasesDetectedInCluster.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfCasesDetectedInCluster.getCaption())); + + addField(vaccinesDetailsLayout, AefiInvestigationDto.ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL, NullableOptionGroup.class); + addField(vaccinesDetailsLayout, AefiInvestigationDto.ALL_CASES_IN_CLUSTER_RECEIVED_VACCINE_FROM_SAME_VIAL_DETAILS); + + TextField numberOfVialsUsedInCluster = addField(vaccinesDetailsLayout, AefiInvestigationDto.NUMBER_OF_VIALS_USED_IN_CLUSTER, TextField.class); + numberOfVialsUsedInCluster + .setConversionError(I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfVialsUsedInCluster.getCaption())); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationVaccinesDetails, false, vaccinesDetailsLayout); + + //Immunization practices + CustomLayout immunizationPracticesLayout = new CustomLayout(); + immunizationPracticesLayout.setTemplateContents(IMMUNIZATION_PRACTICES_HTML_LAYOUT); + + Label syringesAndNeedlesUsedLabel = + new Label(I18nProperties.getCaption(Captions.titleAefiInvestigationImmunizationPracticesSyringesAndNeedlesUsed)); + syringesAndNeedlesUsedLabel.addStyleName(CssStyles.H4); + immunizationPracticesLayout.addComponent(syringesAndNeedlesUsedLabel, SYRINGES_AND_NEEDLES_HEADING_LOC); + + addField(immunizationPracticesLayout, AefiInvestigationDto.AD_SYRINGES_USED_FOR_IMMUNIZATION, NullableOptionGroup.class); + addField(immunizationPracticesLayout, AefiInvestigationDto.TYPE_OF_SYRINGES_USED, NullableOptionGroup.class); + addField(immunizationPracticesLayout, AefiInvestigationDto.TYPE_OF_SYRINGES_USED_DETAILS); + + TextArea syringesUsedAdditionalDetails = + addField(immunizationPracticesLayout, AefiInvestigationDto.SYRINGES_USED_ADDITIONAL_DETAILS, TextArea.class); + syringesUsedAdditionalDetails.setRows(6); + syringesUsedAdditionalDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.SYRINGES_USED_ADDITIONAL_DETAILS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + Label reconstitutionLabel = new Label(I18nProperties.getCaption(Captions.titleAefiInvestigationImmunizationPracticesReconstitution)); + reconstitutionLabel.addStyleName(CssStyles.H4); + immunizationPracticesLayout.addComponent(reconstitutionLabel, RECONSTITUTION_HEADING_LOC); + + addField(immunizationPracticesLayout, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_USED_FOR_MULTIPLE_VIALS_OF_SAME_VACCINE); + addField(immunizationPracticesLayout, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_USED_FOR_RECONSTITUTING_DIFFERENT_VACCINES); + addField(immunizationPracticesLayout, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINE_VIAL); + addField(immunizationPracticesLayout, AefiInvestigationDto.SAME_RECONSTITUTION_SYRINGE_FOR_EACH_VACCINATION); + addField(immunizationPracticesLayout, AefiInvestigationDto.VACCINES_AND_DILUENTS_USED_RECOMMENDED_BY_MANUFACTURER); + + TextArea reconstitutionAdditionalDetails = + addField(immunizationPracticesLayout, AefiInvestigationDto.RECONSTITUTION_ADDITIONAL_DETAILS, TextArea.class); + reconstitutionAdditionalDetails.setRows(6); + reconstitutionAdditionalDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.RECONSTITUTION_ADDITIONAL_DETAILS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + Label injectionTechniqueLabel = new Label(I18nProperties.getCaption(Captions.titleAefiInvestigationImmunizationPracticesInjectionTechnique)); + injectionTechniqueLabel.addStyleName(CssStyles.H4); + immunizationPracticesLayout.addComponent(injectionTechniqueLabel, INJECTION_TECHNIQUE_HEADING_LOC); + + addField(immunizationPracticesLayout, AefiInvestigationDto.CORRECT_DOSE_OR_ROUTE); + addField(immunizationPracticesLayout, AefiInvestigationDto.TIME_OF_RECONSTITUTION_MENTIONED_ON_THE_VIAL); + addField(immunizationPracticesLayout, AefiInvestigationDto.NON_TOUCH_TECHNIQUE_FOLLOWED); + addField(immunizationPracticesLayout, AefiInvestigationDto.CONTRAINDICATION_SCREENED_PRIOR_TO_VACCINATION); + + TextField numberOfAefiFromDistributionCenter = addField( + immunizationPracticesLayout, + AefiInvestigationDto.NUMBER_OF_AEFI_REPORTED_FROM_VACCINE_DISTRIBUTION_CENTER_LAST_THIRTY_DAYS, + TextField.class); + numberOfAefiFromDistributionCenter.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfAefiFromDistributionCenter.getCaption())); + + addField(immunizationPracticesLayout, AefiInvestigationDto.TRAINING_RECEIVED_BY_VACCINATOR, NullableOptionGroup.class); + addField(immunizationPracticesLayout, AefiInvestigationDto.LAST_TRAINING_RECEIVED_BY_VACCINATOR_DATE, DateField.class); + + TextArea injectionTechniqueAdditionalDetails = + addField(immunizationPracticesLayout, AefiInvestigationDto.INJECTION_TECHNIQUE_ADDITIONAL_DETAILS, TextArea.class); + injectionTechniqueAdditionalDetails.setRows(6); + injectionTechniqueAdditionalDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.INJECTION_TECHNIQUE_ADDITIONAL_DETAILS, "") + + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationImmunizationPractices, false, immunizationPracticesLayout); + + //Cold chain and transport + CustomLayout coldChainAndTransportLayout = new CustomLayout(); + coldChainAndTransportLayout.setTemplateContents(COLD_CHAIN_AND_TRANSPORT_HTML_LAYOUT); + + Label lastVaccineStoragePointLabel = + new Label(I18nProperties.getCaption(Captions.titleAefiInvestigationColdChainAndTransportLastVaccineStoragePoint)); + lastVaccineStoragePointLabel.addStyleName(CssStyles.H4); + coldChainAndTransportLayout.addComponent(lastVaccineStoragePointLabel, VACCINE_STORAGE_POINT_HEADING_LOC); + + addField(coldChainAndTransportLayout, AefiInvestigationDto.VACCINE_STORAGE_REFRIGERATOR_TEMPERATURE_MONITORED); + addField(coldChainAndTransportLayout, AefiInvestigationDto.ANY_STORAGE_TEMPERATURE_DEVIATION_OUTSIDE_TWO_TO_EIGHT_DEGREES); + + TextArea storageTemperatureMonitoringDetails = + addField(coldChainAndTransportLayout, AefiInvestigationDto.STORAGE_TEMPERATURE_MONITORING_ADDITIONAL_DETAILS, TextArea.class); + storageTemperatureMonitoringDetails.setRows(6); + storageTemperatureMonitoringDetails.setDescription( + I18nProperties + .getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.STORAGE_TEMPERATURE_MONITORING_ADDITIONAL_DETAILS, "") + + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + addField(coldChainAndTransportLayout, AefiInvestigationDto.CORRECT_PROCEDURE_FOR_STORAGE_FOLLOWED); + addField(coldChainAndTransportLayout, AefiInvestigationDto.ANY_OTHER_ITEM_IN_REFRIGERATOR); + addField(coldChainAndTransportLayout, AefiInvestigationDto.PARTIALLY_USED_RECONSTITUTED_VACCINES_IN_REFRIGERATOR); + addField(coldChainAndTransportLayout, AefiInvestigationDto.UNUSABLE_VACCINES_IN_REFRIGERATOR); + addField(coldChainAndTransportLayout, AefiInvestigationDto.UNUSABLE_DILUENTS_IN_STORE); + + TextArea vaccineStoragePointAdditionalDetails = + addField(coldChainAndTransportLayout, AefiInvestigationDto.VACCINE_STORAGE_POINT_ADDITIONAL_DETAILS, TextArea.class); + vaccineStoragePointAdditionalDetails.setRows(6); + vaccineStoragePointAdditionalDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.VACCINE_STORAGE_POINT_ADDITIONAL_DETAILS, "") + + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + Label vaccineTransportationLabel = + new Label(I18nProperties.getCaption(Captions.titleAefiInvestigationColdChainAndTransportVaccineTransportation)); + vaccineTransportationLabel.addStyleName(CssStyles.H4); + coldChainAndTransportLayout.addComponent(vaccineTransportationLabel, VACCINE_TRANSPORTATION_HEADING_LOC); + + addField(coldChainAndTransportLayout, AefiInvestigationDto.VACCINE_CARRIER_TYPE, NullableOptionGroup.class); + addField(coldChainAndTransportLayout, AefiInvestigationDto.VACCINE_CARRIER_SENT_TO_SITE_ON_SAME_DATE_AS_VACCINATION); + addField(coldChainAndTransportLayout, AefiInvestigationDto.VACCINE_CARRIER_RETURNED_FROM_SITE_ON_SAME_DATE_AS_VACCINATION); + addField(coldChainAndTransportLayout, AefiInvestigationDto.CONDITIONED_ICE_PACK_USED); + + TextArea vaccineTransportationAdditionalDetails = + addField(coldChainAndTransportLayout, AefiInvestigationDto.VACCINE_TRANSPORTATION_ADDITIONAL_DETAILS, TextArea.class); + vaccineTransportationAdditionalDetails.setRows(6); + vaccineTransportationAdditionalDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.VACCINE_TRANSPORTATION_ADDITIONAL_DETAILS, "") + + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationColdChainAndTransport, false, coldChainAndTransportLayout); + + //Community investigation + CustomLayout communityInvestigationLayout = new CustomLayout(); + communityInvestigationLayout.setTemplateContents(COMMUNITY_INVESTIGATION_HTML_LAYOUT); + + addField(communityInvestigationLayout, AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY); + + TextArea similarEventsReportedSamePeriodDetails = + addField(communityInvestigationLayout, AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY_DETAILS, TextArea.class); + similarEventsReportedSamePeriodDetails.setRows(6); + similarEventsReportedSamePeriodDetails.setDescription( + I18nProperties.getPrefixDescription( + AefiInvestigationDto.I18N_PREFIX, + AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY_DETAILS, + "") + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + TextField numberOfSimilarEventsReportedInSamePeriod = + addField(communityInvestigationLayout, AefiInvestigationDto.NUMBER_OF_SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY, TextField.class); + numberOfSimilarEventsReportedInSamePeriod.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfSimilarEventsReportedInSamePeriod.getCaption())); + + Label thoseAffectedLabelLabel = new Label(I18nProperties.getCaption(Captions.titleAefiInvestigationCommunityInvestigationThoseAffected)); + thoseAffectedLabelLabel.addStyleName(CssStyles.H4); + communityInvestigationLayout.addComponent(thoseAffectedLabelLabel, THOSE_AFFECTED_HEADING_LOC); + + TextField numberOfThoseAffectedVaccinated = + addField(communityInvestigationLayout, AefiInvestigationDto.NUMBER_OF_THOSE_AFFECTED_VACCINATED, TextField.class); + numberOfThoseAffectedVaccinated.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfThoseAffectedVaccinated.getCaption())); + + TextField numberOfThoseAffectedNotVaccinated = + addField(communityInvestigationLayout, AefiInvestigationDto.NUMBER_OF_THOSE_AFFECTED_NOT_VACCINATED, TextField.class); + numberOfThoseAffectedNotVaccinated.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfThoseAffectedNotVaccinated.getCaption())); + + TextField numberOfThoseAffectedVaccinatedUnknown = + addField(communityInvestigationLayout, AefiInvestigationDto.NUMBER_OF_THOSE_AFFECTED_VACCINATED_UNKNOWN, TextField.class); + numberOfThoseAffectedVaccinatedUnknown.setConversionError( + I18nProperties.getValidationError(Validations.onlyIntegerNumbersAllowed, numberOfThoseAffectedVaccinatedUnknown.getCaption())); + + TextArea communityInvestigationAdditionalDetails = + addField(communityInvestigationLayout, AefiInvestigationDto.COMMUNITY_INVESTIGATION_ADDITIONAL_DETAILS, TextArea.class); + communityInvestigationAdditionalDetails.setRows(6); + communityInvestigationAdditionalDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.COMMUNITY_INVESTIGATION_ADDITIONAL_DETAILS, "") + + "\n" + I18nProperties.getDescription(Descriptions.descGdpr)); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationCommunityInvestigation, false, communityInvestigationLayout); + + //Other findings + CustomLayout otherFindingsLayout = new CustomLayout(); + otherFindingsLayout.setTemplateContents(OTHER_FINDINGS_HTML_LAYOUT); + + TextArea otherInvestigationFindingsDetails = addField(otherFindingsLayout, AefiInvestigationDto.OTHER_INVESTIGATION_FINDINGS, TextArea.class); + otherInvestigationFindingsDetails.setRows(6); + otherInvestigationFindingsDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.OTHER_INVESTIGATION_FINDINGS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationOtherFindings, true, otherFindingsLayout); + + //Investigation status + CustomLayout investigationStatusLayout = new CustomLayout(); + investigationStatusLayout.setTemplateContents(INVESTIGATION_STATUS_HTML_LAYOUT); + + addField(investigationStatusLayout, AefiInvestigationDto.INVESTIGATION_STATUS, NullableOptionGroup.class); + addField(investigationStatusLayout, AefiInvestigationDto.INVESTIGATION_STATUS_DETAILS); + addField(investigationStatusLayout, AefiInvestigationDto.AEFI_CLASSIFICATION, NullableOptionGroup.class); + TextArea aefiClassificationDetails = addField(investigationStatusLayout, AefiInvestigationDto.AEFI_CLASSIFICATION_DETAILS, TextArea.class); + aefiClassificationDetails.setRows(6); + aefiClassificationDetails.setDescription( + I18nProperties.getPrefixDescription(AefiInvestigationDto.I18N_PREFIX, AefiInvestigationDto.AEFI_CLASSIFICATION_DETAILS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + accordion.addFormSectionPanel(Captions.titleAefiInvestigationInvestigationStatus, true, investigationStatusLayout); + + getContent().addComponent(accordion, MAIN_ACCORDION_LOC); + + //set visibility, read only and required status + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.PLACE_OF_VACCINATION_DETAILS, + AefiInvestigationDto.PLACE_OF_VACCINATION, + Arrays.asList(PlaceOfVaccination.OTHER), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.VACCINATION_ACTIVITY_DETAILS, + AefiInvestigationDto.VACCINATION_ACTIVITY, + Arrays.asList(VaccinationActivity.OTHER), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.TYPE_OF_SITE_DETAILS, + AefiInvestigationDto.TYPE_OF_SITE, + Arrays.asList(VaccinationSite.OTHER), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + Arrays.asList(AefiInvestigationDto.DEATH_DATE_TIME, AefiInvestigationDto.AUTOPSY_DONE), + AefiInvestigationDto.STATUS_ON_DATE_OF_INVESTIGATION, + Arrays.asList(PatientStatusAtAefiInvestigation.DIED), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.AUTOPSY_DATE, + AefiInvestigationDto.AUTOPSY_DONE, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.AUTOPSY_PLANNED_DATE_TIME, + AefiInvestigationDto.AUTOPSY_DONE, + Arrays.asList(YesNoUnknown.NO), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.PAST_HISTORY_OF_SIMILAR_EVENT_DETAILS, + AefiInvestigationDto.PAST_HISTORY_OF_SIMILAR_EVENT, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS_DETAILS, + AefiInvestigationDto.ADVERSE_EVENT_AFTER_PREVIOUS_VACCINATIONS, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD_DETAILS, + AefiInvestigationDto.HISTORY_OF_ALLERGY_TO_VACCINE_DRUG_OR_FOOD, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER_DETAILS, + AefiInvestigationDto.PRE_EXISTING_ILLNESS_THIRTY_DAYS_OR_CONGENITAL_DISORDER, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE_DETAILS, + AefiInvestigationDto.HISTORY_OF_HOSPITALIZATION_IN_LAST_THIRTY_DAYS_WITH_CAUSE, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.CURRENTLY_ON_CONCOMITANT_MEDICATION_DETAILS, + AefiInvestigationDto.CURRENTLY_ON_CONCOMITANT_MEDICATION, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY_DETAILS, + AefiInvestigationDto.FAMILY_HISTORY_OF_DISEASE_OR_ALLERGY, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.DELIVERY_PROCEDURE_DETAILS, + AefiInvestigationDto.DELIVERY_PROCEDURE, + Arrays.asList(DeliveryProcedure.WITH_COMPLICATION), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.TYPE_OF_SYRINGES_USED_DETAILS, + AefiInvestigationDto.TYPE_OF_SYRINGES_USED, + Arrays.asList(SyringeType.OTHER), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + AefiInvestigationDto.LAST_TRAINING_RECEIVED_BY_VACCINATOR_DATE, + AefiInvestigationDto.TRAINING_RECEIVED_BY_VACCINATOR, + Arrays.asList(YesNoUnknown.YES), + true); + + FieldHelper.setVisibleWhen( + getFieldGroup(), + Arrays.asList( + AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY_DETAILS, + AefiInvestigationDto.NUMBER_OF_SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY), + AefiInvestigationDto.SIMILAR_EVENTS_REPORTED_SAME_PERIOD_AND_LOCALITY, + Arrays.asList(YesNoUnknown.YES), + true); + + setReadOnly(true, AefiDto.UUID, AefiDto.REPORTING_USER); + + setRequired(true, AefiInvestigationDto.REPORT_DATE, AefiInvestigationDto.INVESTIGATION_CASE_ID); + setRequired(true, AefiInvestigationDto.PLACE_OF_VACCINATION, AefiInvestigationDto.VACCINATION_ACTIVITY); + setRequired( + true, + AefiInvestigationDto.INVESTIGATION_DATE, + AefiInvestigationDto.FORM_COMPLETION_DATE, + AefiInvestigationDto.INVESTIGATION_STAGE); + setRequired(true, AefiInvestigationDto.TYPE_OF_SITE); + setRequired(true, AefiInvestigationDto.KEY_SYMPTOM_DATE_TIME); + setRequired(true, AefiInvestigationDto.STATUS_ON_DATE_OF_INVESTIGATION); + setRequired(true, AefiInvestigationDto.INVESTIGATION_STATUS, AefiInvestigationDto.AEFI_CLASSIFICATION); + } + + @Override + public void attach() { + super.attach(); + + AefiInvestigationDto dataFormValue = getValue(); + + AefiDto aefiDto = FacadeProvider.getAefiFacade().getByUuid(dataFormValue.getAefiReport().getUuid()); + ImmunizationDto immunizationDto = FacadeProvider.getImmunizationFacade().getByUuid(aefiDto.getImmunization().getUuid()); + + responsibleRegion.setValue(immunizationDto.getResponsibleRegion().getCaption()); + responsibleDistrict.setValue(immunizationDto.getResponsibleDistrict().getCaption()); + if (immunizationDto.getResponsibleCommunity() != null) { + responsibleCommunity.setValue(immunizationDto.getResponsibleCommunity().getCaption()); + } + + responsibleRegion.setReadOnly(true); + responsibleDistrict.setReadOnly(true); + responsibleCommunity.setReadOnly(true); + + vaccinationsField.applyAefiInvestigationContext(dataFormValue); + if (dataFormValue.getPrimarySuspectVaccine() != null) { + vaccinationsField.selectPrimarySuspectVaccination(dataFormValue.getPrimarySuspectVaccine()); + } + } + + @Override + public void setValue(AefiInvestigationDto newFieldValue) throws ReadOnlyException, Converter.ConversionException { + super.setValue(newFieldValue); + + getValue(); + + // HACK: Binding to the fields will call field listeners that may clear/modify the values of other fields. + // this hopefully resets everything to its correct value + discard(); + } + + @Override + public void discard() throws SourceException { + super.discard(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataView.java new file mode 100644 index 00000000000..30979b59ca1 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataView.java @@ -0,0 +1,132 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import java.util.ArrayList; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.EditPermissionType; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; +import de.symeda.sormas.api.immunization.ImmunizationDto; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiImmunizationInfo; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiInfo; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information.AefiPersonInfo; +import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DetailSubComponentWrapper; +import de.symeda.sormas.ui.utils.LayoutWithSidePanel; + +public class AefiInvestigationDataView extends AbstractAefiInvestigationDataView { + + public static final String VIEW_NAME = ROOT_VIEW_NAME + "/data"; + + public static final String ADVERSE_EVENT_LOC = "adverseEventLoc"; + public static final String PERSON_LOC = "personLoc"; + public static final String IMMUNIZATION_LOC = "immunizationLoc"; + + private CommitDiscardWrapperComponent editComponent; + + public AefiInvestigationDataView() { + super(VIEW_NAME); + } + + @Override + protected String getRootViewName() { + return super.getRootViewName(); + } + + @Override + protected void initView(String params) { + setHeightUndefined(); + + AefiInvestigationDto aefiInvestigationDto; + AefiDto aefiReport; + ImmunizationDto immunization; + + boolean isCreateAction = ControllerProvider.getAefiInvestigationController().isCreateAction(params); + if (isCreateAction) { + aefiInvestigationDto = AefiInvestigationDto.build(getReference()); + + String aefiReportUuid = ControllerProvider.getAefiInvestigationController().getCreateActionAefiReportUuid(params); + aefiReport = FacadeProvider.getAefiFacade().getByUuid(aefiReportUuid); + immunization = FacadeProvider.getImmunizationFacade().getByUuid(aefiReport.getImmunization().getUuid()); + + aefiInvestigationDto.setAefiReport(aefiReport.toReference()); + aefiInvestigationDto.setVaccinations(new ArrayList<>(immunization.getVaccinations())); + aefiInvestigationDto.setReportingUser(UserProvider.getCurrent().getUserReference()); + } else { + aefiInvestigationDto = FacadeProvider.getAefiInvestigationFacade().getByUuid(getReference().getUuid()); + aefiReport = FacadeProvider.getAefiFacade().getByUuid(aefiInvestigationDto.getAefiReport().getUuid()); + immunization = FacadeProvider.getImmunizationFacade().getByUuid(aefiReport.getImmunization().getUuid()); + } + + editComponent = ControllerProvider.getAefiInvestigationController() + .getAefiInvestigationDataEditComponent(isCreateAction, aefiInvestigationDto, this::showUnsavedChangesPopup); + + DetailSubComponentWrapper container = new DetailSubComponentWrapper(() -> editComponent); + container.setWidth(100, Unit.PERCENTAGE); + container.setMargin(true); + setSubComponent(container); + container.setEnabled(true); + + LayoutWithSidePanel layout = new LayoutWithSidePanel(editComponent, PERSON_LOC, ADVERSE_EVENT_LOC, IMMUNIZATION_LOC); + + container.addComponent(layout); + + UserProvider currentUser = UserProvider.getCurrent(); + if (currentUser.hasAllUserRights(UserRight.PERSON_VIEW)) { + + PersonDto personDto = FacadeProvider.getPersonFacade().getByUuid(immunization.getPerson().getUuid()); + Disease disease = immunization.getDisease(); + + AefiPersonInfo aefiPersonInfo = new AefiPersonInfo(personDto, disease); + CssStyles.style(aefiPersonInfo, CssStyles.VIEW_SECTION); + + layout.addSidePanelComponent(aefiPersonInfo, PERSON_LOC); + } + + if (currentUser.hasUserRight(UserRight.IMMUNIZATION_VIEW)) { + + AefiImmunizationInfo aefiImmunizationInfo = new AefiImmunizationInfo(immunization, (r) -> { + }); + CssStyles.style(aefiImmunizationInfo, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + + layout.addSidePanelComponent(aefiImmunizationInfo, IMMUNIZATION_LOC); + } + + if (currentUser.hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + + AefiInfo aefiInfo = new AefiInfo(aefiReport, (r) -> { + }); + CssStyles.style(aefiInfo, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + + layout.addSidePanelComponent(aefiInfo, ADVERSE_EVENT_LOC); + } + + if (!isCreateAction) { + final String uuid = aefiInvestigationDto.getUuid(); + final EditPermissionType aefiInvestigationEditAllowed = FacadeProvider.getAefiInvestigationFacade().getEditPermissionType(uuid); + final boolean deleted = FacadeProvider.getAefiInvestigationFacade().isDeleted(uuid); + layout.disableIfNecessary(deleted, aefiInvestigationEditAllowed); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java new file mode 100644 index 00000000000..ebc4814cfd8 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java @@ -0,0 +1,244 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization; + +import java.util.Collections; +import java.util.Objects; + +import org.vaadin.hene.popupbutton.PopupButton; + +import com.vaadin.icons.VaadinIcons; +import com.vaadin.navigator.ViewChangeListener; +import com.vaadin.server.FileDownloader; +import com.vaadin.server.StreamResource; +import com.vaadin.shared.ui.ContentMode; +import com.vaadin.shared.ui.MarginInfo; +import com.vaadin.ui.Alignment; +import com.vaadin.ui.Button; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; +import com.vaadin.ui.themes.ValoTheme; +import com.vaadin.v7.ui.ComboBox; + +import de.symeda.sormas.api.EntityRelevanceStatus; +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; +import de.symeda.sormas.api.common.DeletableEntityType; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.feature.FeatureTypeProperty; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.ViewModelProviders; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory.AefiInvestigationDataLayout; +import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory.AefiInvestigationFilterFormLayout; +import de.symeda.sormas.ui.utils.ButtonHelper; +import de.symeda.sormas.ui.utils.ComboBoxHelper; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.ExportEntityName; +import de.symeda.sormas.ui.utils.GridExportStreamResource; +import de.symeda.sormas.ui.utils.ViewConfiguration; + +public class AefiInvestigationView extends AbstractAefiView { + + public static final String VIEW_NAME = "adverseeventinvestigations"; + + private final AefiInvestigationCriteria criteria; + + private AefiInvestigationFilterFormLayout filterFormLayout; + private final AefiInvestigationDataLayout dataLayout; + + // Filters + private Label relevanceStatusInfoLabel; + private ComboBox relevanceStatusFilter; + private ViewConfiguration viewConfiguration; + + public AefiInvestigationView() { + super(VIEW_NAME); + + CssStyles.style(getViewTitleLabel(), CssStyles.PAGE_TITLE); + + viewConfiguration = ViewModelProviders.of(getClass()).get(ViewConfiguration.class); + + criteria = ViewModelProviders.of(AefiInvestigationView.class).get(AefiInvestigationCriteria.class); + if (criteria.getRelevanceStatus() == null) { + criteria.setRelevanceStatus(EntityRelevanceStatus.ACTIVE); + } + dataLayout = new AefiInvestigationDataLayout(criteria); + + if (UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT)) { + VerticalLayout exportLayout = new VerticalLayout(); + exportLayout.setSpacing(true); + exportLayout.setMargin(true); + exportLayout.addStyleName(CssStyles.LAYOUT_MINIMAL); + exportLayout.setWidth(200, Unit.PIXELS); + + PopupButton exportButton = ButtonHelper.createIconPopupButton(Captions.export, VaadinIcons.DOWNLOAD, exportLayout); + addHeaderComponent(exportButton); + + Button basicExportButton = ButtonHelper.createIconButton(Captions.exportBasic, VaadinIcons.TABLE, null, ValoTheme.BUTTON_PRIMARY); + basicExportButton.setDescription(I18nProperties.getString(Strings.infoBasicExport)); + basicExportButton.setWidth(100, Unit.PERCENTAGE); + exportLayout.addComponent(basicExportButton); + StreamResource streamResource = GridExportStreamResource.createStreamResourceWithSelectedItems( + dataLayout.getGrid(), + () -> viewConfiguration.isInEagerMode() ? dataLayout.getGrid().asMultiSelect().getSelectedItems() : Collections.emptySet(), + ExportEntityName.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_INVESTIGATION); + FileDownloader fileDownloader = new FileDownloader(streamResource); + fileDownloader.extend(basicExportButton); + } + + final VerticalLayout mainLayout = new VerticalLayout(); + mainLayout.addComponent(createFilterBar()); + + final VerticalLayout gridLayout = new VerticalLayout(); + gridLayout.setMargin(false); + gridLayout.setSpacing(false); + gridLayout.setSizeFull(); + CssStyles.style(gridLayout, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + + gridLayout.addComponent(createStatusFilterBar()); + gridLayout.addComponent(dataLayout); + gridLayout.setExpandRatio(dataLayout, 1); + + mainLayout.addComponent(gridLayout); + + mainLayout.setMargin(new MarginInfo(false, true, true, true)); + mainLayout.setSpacing(false); + mainLayout.setSizeFull(); + mainLayout.setExpandRatio(gridLayout, 1); + mainLayout.addStyleNames("crud-main-layout", CssStyles.VSPACE_TOP_4); + + addComponent(mainLayout); + } + + private void updateFilterComponents() { + // TODO replace with Vaadin 8 databinding + applyingCriteria = true; + + if (relevanceStatusFilter != null) { + relevanceStatusFilter.setValue(criteria.getRelevanceStatus()); + } + + filterFormLayout.setValue(criteria); + + applyingCriteria = false; + } + + private AefiInvestigationFilterFormLayout createFilterBar() { + filterFormLayout = new AefiInvestigationFilterFormLayout(); + + filterFormLayout.addResetHandler(clickEvent -> { + ViewModelProviders.of(AefiInvestigationView.class).remove(AefiInvestigationCriteria.class); + navigateTo(null, true); + }); + + filterFormLayout.addApplyHandler(clickEvent -> { + dataLayout.refreshGrid(); + }); + + return filterFormLayout; + } + + public HorizontalLayout createStatusFilterBar() { + HorizontalLayout statusFilterLayout = new HorizontalLayout(); + statusFilterLayout.setSpacing(true); + statusFilterLayout.setMargin(false); + statusFilterLayout.setWidth(100, Unit.PERCENTAGE); + statusFilterLayout.addStyleName(CssStyles.VSPACE_3); + + HorizontalLayout actionButtonsLayout = new HorizontalLayout(); + actionButtonsLayout.setSpacing(true); + + // Show active/archived/all dropdown + if (Objects.nonNull(UserProvider.getCurrent()) + && UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW)) { + + if (FacadeProvider.getFeatureConfigurationFacade() + .isFeatureEnabled(FeatureType.AUTOMATIC_ARCHIVING, DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION)) { + + int daysAfterAefiEntryGetsArchived = FacadeProvider.getFeatureConfigurationFacade() + .getProperty( + FeatureType.AUTOMATIC_ARCHIVING, + DeletableEntityType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION, + FeatureTypeProperty.THRESHOLD_IN_DAYS, + Integer.class); + if (daysAfterAefiEntryGetsArchived > 0) { + relevanceStatusInfoLabel = new Label( + VaadinIcons.INFO_CIRCLE.getHtml() + " " + + String.format(I18nProperties.getString(Strings.infoArchivedAefiEntries), daysAfterAefiEntryGetsArchived), + ContentMode.HTML); + relevanceStatusInfoLabel.setVisible(false); + relevanceStatusInfoLabel.addStyleName(CssStyles.LABEL_VERTICAL_ALIGN_SUPER); + actionButtonsLayout.addComponent(relevanceStatusInfoLabel); + actionButtonsLayout.setComponentAlignment(relevanceStatusInfoLabel, Alignment.MIDDLE_RIGHT); + } + } + relevanceStatusFilter = ComboBoxHelper.createComboBoxV7(); + relevanceStatusFilter.setId("relevanceStatus"); + relevanceStatusFilter.setWidth(260, Unit.PIXELS); + relevanceStatusFilter.setNullSelectionAllowed(false); + relevanceStatusFilter.setTextInputAllowed(false); + relevanceStatusFilter.addItems((Object[]) EntityRelevanceStatus.values()); + relevanceStatusFilter.setItemCaption(EntityRelevanceStatus.ACTIVE, I18nProperties.getCaption(Captions.aefiActiveInvestigations)); + relevanceStatusFilter.setItemCaption(EntityRelevanceStatus.ARCHIVED, I18nProperties.getCaption(Captions.aefiArchivedInvestigations)); + relevanceStatusFilter.setItemCaption( + EntityRelevanceStatus.ACTIVE_AND_ARCHIVED, + I18nProperties.getCaption(Captions.aefiAllActiveAndArchivedInvestigations)); + relevanceStatusFilter.setCaption(null); + relevanceStatusFilter.addStyleName(CssStyles.VSPACE_NONE); + + if (UserProvider.getCurrent().hasUserRight(UserRight.IMMUNIZATION_DELETE)) { + relevanceStatusFilter.setItemCaption(EntityRelevanceStatus.DELETED, I18nProperties.getCaption(Captions.aefiDeletedInvestigations)); + } else { + relevanceStatusFilter.removeItem(EntityRelevanceStatus.DELETED); + } + + relevanceStatusFilter.addValueChangeListener(e -> { + if (relevanceStatusInfoLabel != null) { + relevanceStatusInfoLabel.setVisible(EntityRelevanceStatus.ARCHIVED.equals(e.getProperty().getValue())); + } + criteria.setRelevanceStatus((EntityRelevanceStatus) e.getProperty().getValue()); + navigateTo(criteria); + }); + actionButtonsLayout.addComponent(relevanceStatusFilter); + } + + if (actionButtonsLayout.getComponentCount() > 0) { + statusFilterLayout.addComponent(actionButtonsLayout); + statusFilterLayout.setComponentAlignment(actionButtonsLayout, Alignment.TOP_RIGHT); + statusFilterLayout.setExpandRatio(actionButtonsLayout, 1); + } + + return statusFilterLayout; + } + + @Override + public void enter(ViewChangeListener.ViewChangeEvent event) { + + String params = event.getParameters().trim(); + if (params.startsWith("?")) { + params = params.substring(1); + criteria.fromUrlParams(params); + } + updateFilterComponents(); + + super.enter(event); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java index daa30665c50..ef58c7bc968 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java @@ -1,37 +1,45 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package de.symeda.sormas.ui.adverseeventsfollowingimmunization; +import java.util.Collections; import java.util.Objects; +import java.util.Set; +import java.util.stream.Collectors; + +import org.vaadin.hene.popupbutton.PopupButton; import com.vaadin.icons.VaadinIcons; import com.vaadin.navigator.ViewChangeListener; +import com.vaadin.server.FileDownloader; +import com.vaadin.server.StreamResource; import com.vaadin.shared.ui.ContentMode; +import com.vaadin.shared.ui.MarginInfo; import com.vaadin.ui.Alignment; +import com.vaadin.ui.Button; import com.vaadin.ui.HorizontalLayout; import com.vaadin.ui.Label; import com.vaadin.ui.VerticalLayout; +import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.ui.ComboBox; import de.symeda.sormas.api.EntityRelevanceStatus; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; import de.symeda.sormas.api.common.DeletableEntityType; import de.symeda.sormas.api.feature.FeatureType; import de.symeda.sormas.api.feature.FeatureTypeProperty; @@ -43,11 +51,15 @@ import de.symeda.sormas.ui.ViewModelProviders; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory.AefiDataLayout; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory.AefiFilterFormLayout; -import de.symeda.sormas.ui.utils.AbstractView; +import de.symeda.sormas.ui.utils.AefiDownloadUtil; +import de.symeda.sormas.ui.utils.ButtonHelper; import de.symeda.sormas.ui.utils.ComboBoxHelper; import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.ExportEntityName; +import de.symeda.sormas.ui.utils.GridExportStreamResource; +import de.symeda.sormas.ui.utils.ViewConfiguration; -public class AefiView extends AbstractView { +public class AefiView extends AbstractAefiView { public static final String VIEW_NAME = "adverseevents"; @@ -59,18 +71,53 @@ public class AefiView extends AbstractView { // Filters private Label relevanceStatusInfoLabel; private ComboBox relevanceStatusFilter; + private ViewConfiguration viewConfiguration; public AefiView() { super(VIEW_NAME); CssStyles.style(getViewTitleLabel(), CssStyles.PAGE_TITLE); + viewConfiguration = ViewModelProviders.of(getClass()).get(ViewConfiguration.class); + criteria = ViewModelProviders.of(AefiView.class).get(AefiCriteria.class); if (criteria.getRelevanceStatus() == null) { criteria.setRelevanceStatus(EntityRelevanceStatus.ACTIVE); } dataLayout = new AefiDataLayout(criteria); + if (UserProvider.getCurrent().hasUserRight(UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT)) { + VerticalLayout exportLayout = new VerticalLayout(); + exportLayout.setSpacing(true); + exportLayout.setMargin(true); + exportLayout.addStyleName(CssStyles.LAYOUT_MINIMAL); + exportLayout.setWidth(200, Unit.PIXELS); + + PopupButton exportButton = ButtonHelper.createIconPopupButton(Captions.export, VaadinIcons.DOWNLOAD, exportLayout); + addHeaderComponent(exportButton); + + Button basicExportButton = ButtonHelper.createIconButton(Captions.exportBasic, VaadinIcons.TABLE, null, ValoTheme.BUTTON_PRIMARY); + basicExportButton.setDescription(I18nProperties.getString(Strings.infoBasicExport)); + basicExportButton.setWidth(100, Unit.PERCENTAGE); + exportLayout.addComponent(basicExportButton); + StreamResource streamResource = GridExportStreamResource.createStreamResourceWithSelectedItems( + dataLayout.getGrid(), + () -> viewConfiguration.isInEagerMode() ? dataLayout.getGrid().asMultiSelect().getSelectedItems() : Collections.emptySet(), + ExportEntityName.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION); + FileDownloader fileDownloader = new FileDownloader(streamResource); + fileDownloader.extend(basicExportButton); + + StreamResource extendedExportStreamResource = + AefiDownloadUtil.createAefiExportResource(dataLayout.getGrid().getCriteria(), this::getSelectedRowUuids, null); + addExportButton( + extendedExportStreamResource, + exportButton, + exportLayout, + VaadinIcons.FILE_TEXT, + Captions.exportDetailed, + Strings.infoDetailedExport); + } + final VerticalLayout mainLayout = new VerticalLayout(); mainLayout.addComponent(createFilterBar()); @@ -86,11 +133,11 @@ public AefiView() { mainLayout.addComponent(gridLayout); - mainLayout.setMargin(true); + mainLayout.setMargin(new MarginInfo(false, true, true, true)); mainLayout.setSpacing(false); mainLayout.setSizeFull(); mainLayout.setExpandRatio(gridLayout, 1); - mainLayout.setStyleName("crud-main-layout"); + mainLayout.addStyleNames("crud-main-layout", CssStyles.VSPACE_TOP_4); addComponent(mainLayout); } @@ -195,6 +242,12 @@ public HorizontalLayout createStatusFilterBar() { return statusFilterLayout; } + private Set getSelectedRowUuids() { + return viewConfiguration.isInEagerMode() + ? dataLayout.getGrid().asMultiSelect().getSelectedItems().stream().map(AefiIndexDto::getUuid).collect(Collectors.toSet()) + : Collections.emptySet(); + } + @Override public void enter(ViewChangeListener.ViewChangeEvent event) { @@ -204,5 +257,7 @@ public void enter(ViewChangeListener.ViewChangeEvent event) { criteria.fromUrlParams(params); } updateFilterComponents(); + + super.enter(event); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java new file mode 100644 index 00000000000..7a78c37b890 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java @@ -0,0 +1,88 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefiinvestigationlink; + +import java.util.List; +import java.util.function.Consumer; + +import org.apache.commons.collections.CollectionUtils; + +import com.vaadin.ui.Button; +import com.vaadin.ui.Label; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListEntryDto; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.utils.PaginationList; + +@SuppressWarnings("serial") +public class AefiInvestigationList extends PaginationList { + + private static final int MAX_DISPLAYED_ENTRIES = 5; + + private final AefiInvestigationListCriteria listCriteria; + private final Consumer actionCallback; + private final boolean isEditable; + private final Label noInvestigationsLabel; + + public AefiInvestigationList(AefiInvestigationListCriteria listCriteria, Consumer actionCallback, boolean isEditable) { + + super(MAX_DISPLAYED_ENTRIES); + + this.listCriteria = listCriteria; + this.actionCallback = actionCallback; + this.isEditable = isEditable; + + noInvestigationsLabel = new Label(I18nProperties.getString(Strings.infoNoAefiInvestigations)); + } + + @Override + public void reload() { + + List listEntries = + FacadeProvider.getAefiInvestigationFacade().getEntriesList(listCriteria, 0, maxDisplayedEntries * 20); + + setEntries(listEntries); + if (CollectionUtils.isNotEmpty(listEntries)) { + showPage(1); + } else { + listLayout.removeAllComponents(); + updatePaginationLayout(); + listLayout.addComponent(noInvestigationsLabel); + } + } + + @Override + protected void drawDisplayedEntries() { + + List displayedEntries = getDisplayedEntries(); + for (AefiInvestigationListEntryDto listEntryDto : displayedEntries) { + AefiInvestigationListEntry listEntry = new AefiInvestigationListEntry(listEntryDto); + + String aefiInvestigationUuid = listEntryDto.getUuid(); + listEntry.addEditButton( + "edit-aefiinvestigation-" + aefiInvestigationUuid, + (Button.ClickListener) event -> ControllerProvider.getAefiInvestigationController() + .navigateToAefiInvestigation(aefiInvestigationUuid)); + + listEntry.setEnabled(isEditable); + listLayout.addComponent(listEntry); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListComponent.java new file mode 100644 index 00000000000..9faf3780c03 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListComponent.java @@ -0,0 +1,57 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefiinvestigationlink; + +import java.util.function.Consumer; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListCriteria; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.user.UserRight; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponent; + +@SuppressWarnings("serial") +public class AefiInvestigationListComponent extends SideComponent { + + public AefiInvestigationListComponent( + AefiInvestigationListCriteria listCriteria, + Consumer actionCallback, + boolean isEditAllowed, + boolean isCreateAction) { + super(I18nProperties.getString(Strings.headingAefiReportInvestigations), actionCallback); + + setMargin(false); + setWidth(100, Unit.PERCENTAGE); + + if (isEditAllowed) { + addCreateButton( + I18nProperties.getCaption(Captions.aefiNewAefiInvestigation), + () -> ControllerProvider.getAefiInvestigationController() + .navigateToAefiInvestigation("adverseevent/" + listCriteria.getAefiReport().getUuid() + "/investigation/create"), + UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE); + + if (isCreateAction) { + createButton.setEnabled(false); + } + } + + AefiInvestigationList aefiInvestigationList = new AefiInvestigationList(listCriteria, actionCallback, isEditAllowed); + addComponent(aefiInvestigationList); + aefiInvestigationList.reload(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListEntry.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListEntry.java new file mode 100644 index 00000000000..74e571ef1cd --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationListEntry.java @@ -0,0 +1,76 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.aefiinvestigationlink; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.ui.Label; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationListEntryDto; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponentField; + +@SuppressWarnings("serial") +public class AefiInvestigationListEntry extends SideComponentField { + + public static final String SEPARATOR = ": "; + + private final AefiInvestigationListEntryDto listEntry; + + public AefiInvestigationListEntry(AefiInvestigationListEntryDto listEntry) { + + this.listEntry = listEntry; + + if (!StringUtils.isBlank(listEntry.getInvestigationCaseId())) { + Label labelCaseId = new Label(listEntry.getInvestigationCaseId()); + CssStyles.style(labelCaseId, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelCaseId); + } + + Label labelInvestigationDate = new Label( + I18nProperties.getPrefixCaption(AefiInvestigationListEntryDto.I18N_PREFIX, AefiInvestigationListEntryDto.INVESTIGATION_DATE) + + SEPARATOR + + DateFormatHelper.formatLocalDate(listEntry.getInvestigationDate())); + addComponentToField(labelInvestigationDate); + + Label labelInvestigationStage = new Label( + I18nProperties.getPrefixCaption(AefiInvestigationListEntryDto.I18N_PREFIX, AefiInvestigationListEntryDto.INVESTIGATION_STAGE) + + SEPARATOR + + listEntry.getInvestigationStage()); + CssStyles.style(labelInvestigationStage, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelInvestigationStage); + + Label labelStatusOnInvestigation = new Label( + I18nProperties.getPrefixCaption(AefiInvestigationListEntryDto.I18N_PREFIX, AefiInvestigationListEntryDto.STATUS_ON_DATE_OF_INVESTIGATION) + + SEPARATOR + + listEntry.getStatusOnDateOfInvestigation()); + CssStyles.style(labelStatusOnInvestigation, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelStatusOnInvestigation); + + Label labelAefiClassification = new Label( + I18nProperties.getPrefixCaption(AefiInvestigationListEntryDto.I18N_PREFIX, AefiInvestigationListEntryDto.AEFI_CLASSIFICATION) + + SEPARATOR + + ((listEntry.getAefiClassification() != null) ? listEntry.getAefiClassification() : "")); + CssStyles.style(labelAefiClassification, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + addComponentToField(labelAefiClassification); + } + + public AefiInvestigationListEntryDto getListEntry() { + return listEntry; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java index 400715182cb..3503f05bbde 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiListComponent.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -41,7 +38,7 @@ public AefiListComponent(AefiListCriteria aefiListCriteria, Consumer a addCreateButton( I18nProperties.getCaption(Captions.aefiNewAdverseEvent), () -> ControllerProvider.getAefiController() - .navigateToAefi("immunization/" + aefiListCriteria.getImmunization().getUuid() + "/create"), + .navigateToAefi("immunization/" + aefiListCriteria.getImmunization().getUuid() + "/adverseevent/create"), UserRight.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE); if (totalVaccinations == 0) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java index fa96538ee1e..8b96dd8bc29 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiDataLayout.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -36,6 +33,10 @@ public AefiDataLayout(AefiCriteria criteria) { setExpandRatio(grid, 1); } + public AefiGrid getGrid() { + return grid; + } + public void refreshGrid() { grid.reload(); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java index 3f83e93210e..5f1f4a47288 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiFilterForm.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -77,7 +74,7 @@ public AefiFilterForm() { @Override protected String createHtmlLayout() { return divCss( - CssStyles.VIEW_SECTION, + "", filterLocs(ArrayUtils.addAll(getMainFilterLocators(), ACTION_BUTTONS_ID)) + locCss(CssStyles.VSPACE_TOP_NONE, MORE_FILTERS_ID)); } @@ -86,7 +83,7 @@ protected String createHtmlLayout() { protected String[] getMainFilterLocators() { return new String[] { AefiCriteria.DISEASE, - AefiCriteria.NAME_ADDRESS_PHONE_EMAIL_LIKE, + AefiCriteria.PERSON_LIKE, AefiCriteria.AEFI_TYPE, AefiCriteria.VACCINE_NAME, AefiCriteria.REGION, @@ -105,10 +102,7 @@ protected void addFields() { addField(FieldConfiguration.pixelSized(AefiCriteria.DISEASE, 140)); final TextField searchField = addField( - FieldConfiguration.withCaptionAndPixelSized( - AefiCriteria.NAME_ADDRESS_PHONE_EMAIL_LIKE, - I18nProperties.getString(Strings.promptPersonsSearchField), - 200)); + FieldConfiguration.withCaptionAndPixelSized(AefiCriteria.PERSON_LIKE, I18nProperties.getString(Strings.promptPersonsSearchField), 200)); searchField.setNullRepresentation(""); addFields(FieldConfiguration.pixelSized(AefiCriteria.AEFI_TYPE, 140), FieldConfiguration.pixelSized(AefiCriteria.VACCINE_NAME, 140)); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java index 8a27926187e..f89c5d7f240 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiGrid.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -20,6 +17,8 @@ import java.util.Date; +import org.apache.commons.lang3.StringUtils; + import com.vaadin.ui.renderers.DateRenderer; import com.vaadin.ui.renderers.TextRenderer; @@ -27,6 +26,7 @@ import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; import de.symeda.sormas.api.feature.FeatureType; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -42,12 +42,30 @@ public class AefiGrid extends FilteredGrid { + public static final String PRIMARY_VACCINE_COLUMN = "primaryVaccineColumn"; + public AefiGrid(AefiCriteria criteria) { super(AefiIndexDto.class); setSizeFull(); setLazyDataProvider(); setCriteria(criteria); + Column primaryVaccineColumn = addColumn(entry -> { + if (entry.getPrimaryVaccine() != null) { + if (entry.getPrimaryVaccine() != Vaccine.OTHER) { + return entry.getPrimaryVaccine().toString(); + } else { + return StringUtils.isBlank(entry.getPrimaryVaccineDetails()) + ? entry.getPrimaryVaccine().toString() + : (entry.getPrimaryVaccine() + " (" + entry.getPrimaryVaccineDetails() + ")"); + } + } else { + return "-"; + } + }); + primaryVaccineColumn.setId(PRIMARY_VACCINE_COLUMN); + primaryVaccineColumn.setCaption(I18nProperties.getPrefixCaption(AefiIndexDto.I18N_PREFIX, AefiIndexDto.PRIMARY_VACCINE_NAME)); + Column deleteColumn = addColumn(entry -> { if (entry.getDeletionReason() != null) { return entry.getDeletionReason() + (entry.getOtherDeletionReason() != null ? ": " + entry.getOtherDeletionReason() : ""); @@ -87,7 +105,7 @@ private void initColumns() { AefiIndexDto.REGION, AefiIndexDto.DISTRICT, AefiIndexDto.SERIOUS, - AefiIndexDto.PRIMARY_VACCINE_NAME, + PRIMARY_VACCINE_COLUMN, AefiIndexDto.OUTCOME, AefiIndexDto.VACCINATION_DATE, AefiIndexDto.REPORT_DATE, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationDataLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationDataLayout.java new file mode 100644 index 00000000000..e649ebc0473 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationDataLayout.java @@ -0,0 +1,43 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; + +public class AefiInvestigationDataLayout extends VerticalLayout { + + private final AefiInvestigationGrid grid; + + public AefiInvestigationDataLayout(AefiInvestigationCriteria criteria) { + grid = new AefiInvestigationGrid(criteria); + addComponent(grid); + + setMargin(false); + setSpacing(false); + setSizeFull(); + setExpandRatio(grid, 1); + } + + public AefiInvestigationGrid getGrid() { + return grid; + } + + public void refreshGrid() { + grid.reload(); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterForm.java new file mode 100644 index 00000000000..c10376e2785 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterForm.java @@ -0,0 +1,418 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import static de.symeda.sormas.ui.utils.LayoutUtil.divCss; +import static de.symeda.sormas.ui.utils.LayoutUtil.filterLocs; +import static de.symeda.sormas.ui.utils.LayoutUtil.loc; +import static de.symeda.sormas.ui.utils.LayoutUtil.locCss; + +import java.util.Date; +import java.util.stream.Stream; + +import org.apache.commons.lang3.ArrayUtils; + +import com.vaadin.ui.CustomLayout; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.v7.data.Property; +import com.vaadin.v7.ui.ComboBox; +import com.vaadin.v7.ui.Field; +import com.vaadin.v7.ui.TextField; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDateType; +import de.symeda.sormas.api.i18n.Descriptions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; +import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; +import de.symeda.sormas.api.infrastructure.facility.FacilityType; +import de.symeda.sormas.api.infrastructure.facility.FacilityTypeGroup; +import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; +import de.symeda.sormas.api.user.JurisdictionLevel; +import de.symeda.sormas.api.user.UserDto; +import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.api.utils.DateFilterOption; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.api.utils.EpiWeek; +import de.symeda.sormas.ui.UserProvider; +import de.symeda.sormas.ui.utils.AbstractFilterForm; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.EpiWeekAndDateFilterComponent; +import de.symeda.sormas.ui.utils.FieldConfiguration; +import de.symeda.sormas.ui.utils.FieldHelper; + +public class AefiInvestigationFilterForm extends AbstractFilterForm { + + private static final String ACTION_BUTTONS_ID = "actionButtons"; + private static final String MORE_FILTERS_ID = "moreFilters"; + private static final String WEEK_AND_DATE_FILTER = "weekAndDateFilter"; + + private static final String MORE_FILTERS_HTML = loc(WEEK_AND_DATE_FILTER); + + public AefiInvestigationFilterForm() { + super( + AefiInvestigationCriteria.class, + AefiInvestigationCriteria.I18N_PREFIX, + JurisdictionFieldConfig.of(AefiInvestigationCriteria.REGION, AefiInvestigationCriteria.DISTRICT, AefiInvestigationCriteria.COMMUNITY)); + } + + @Override + protected String createHtmlLayout() { + return divCss( + "", + filterLocs(ArrayUtils.addAll(getMainFilterLocators(), ACTION_BUTTONS_ID)) + locCss(CssStyles.VSPACE_TOP_NONE, MORE_FILTERS_ID)); + + } + + @Override + protected String[] getMainFilterLocators() { + return new String[] { + AefiInvestigationCriteria.DISEASE, + AefiInvestigationCriteria.PERSON_LIKE, + AefiInvestigationCriteria.VACCINE_NAME, + AefiInvestigationCriteria.REGION, + AefiInvestigationCriteria.DISTRICT, + AefiInvestigationCriteria.COMMUNITY, + AefiInvestigationCriteria.STATUS_ON_DATE_OF_INVESTIGATION, + AefiInvestigationCriteria.AEFI_CLASSIFICATION }; + } + + @Override + protected String createMoreFiltersHtmlLayout() { + return MORE_FILTERS_HTML; + } + + @Override + protected void addFields() { + addField(FieldConfiguration.pixelSized(AefiInvestigationCriteria.DISEASE, 140)); + + final TextField searchField = addField( + FieldConfiguration + .withCaptionAndPixelSized(AefiInvestigationCriteria.PERSON_LIKE, I18nProperties.getString(Strings.promptPersonsSearchField), 200)); + searchField.setNullRepresentation(""); + + addFields( + FieldConfiguration.pixelSized(AefiInvestigationCriteria.AEFI_TYPE, 140), + FieldConfiguration.pixelSized(AefiInvestigationCriteria.VACCINE_NAME, 140)); + + if (currentUserDto().getRegion() == null) { + ComboBox regionFilter = addField(getContent(), FieldConfiguration.pixelSized(AefiInvestigationCriteria.REGION, 140)); + regionFilter.addItems(FacadeProvider.getRegionFacade().getAllActiveByServerCountry()); + } + + ComboBox districtFilter = addField(getContent(), FieldConfiguration.pixelSized(AefiInvestigationCriteria.DISTRICT, 140)); + districtFilter.setDescription(I18nProperties.getDescription(Descriptions.descDistrictFilter)); + if (currentUserDto().getDistrict() != null) { + districtFilter.setVisible(false); + } + + addField(getContent(), FieldConfiguration.pixelSized(AefiInvestigationCriteria.COMMUNITY, 140)); + + addField(FieldConfiguration.pixelSized(AefiInvestigationCriteria.STATUS_ON_DATE_OF_INVESTIGATION, 140)); + addField(FieldConfiguration.pixelSized(AefiInvestigationCriteria.AEFI_CLASSIFICATION, 140)); + } + + @Override + public void addMoreFilters(CustomLayout moreFiltersContainer) { + moreFiltersContainer.addComponent(buildWeekAndDateFilter(), WEEK_AND_DATE_FILTER); + } + + @Override + protected void applyDependenciesOnFieldChange(String propertyId, Property.ValueChangeEvent event) { + + super.applyDependenciesOnFieldChange(propertyId, event); + + final AefiInvestigationCriteria criteria = getValue(); + + final ComboBox facilityTypeGroupField = getField(AefiInvestigationCriteria.FACILITY_TYPE_GROUP); + final ComboBox facilityTypeField = getField(AefiInvestigationCriteria.FACILITY_TYPE); + final ComboBox facilityField = getField(AefiInvestigationCriteria.HEALTH_FACILITY); + + final UserDto user = currentUserDto(); + final DistrictReferenceDto currentDistrict = + user.getDistrict() != null ? user.getDistrict() : (DistrictReferenceDto) districtFilter.getValue(); + + switch (propertyId) { + case AefiInvestigationCriteria.REGION: { + final RegionReferenceDto region = user.getRegion() != null ? user.getRegion() : (RegionReferenceDto) event.getProperty().getValue(); + + if (!DataHelper.equal(region, criteria.getRegion())) { + if (region != null) { + enableFields(districtFilter); + FieldHelper.updateItems(districtFilter, FacadeProvider.getDistrictFacade().getAllActiveByRegion(region.getUuid())); + } else { + clearAndDisableFields(districtFilter); + } + clearAndDisableFields(communityFilter, facilityField, facilityTypeField, facilityTypeGroupField); + } + + break; + } + case AefiInvestigationCriteria.DISTRICT: { + final DistrictReferenceDto newDistrict = (DistrictReferenceDto) event.getProperty().getValue(); + + if (!DataHelper.equal(newDistrict, criteria.getDistrict())) { + if (newDistrict != null) { + enableFields(communityFilter, facilityTypeGroupField); + + clearAndDisableFields(facilityField); + if (facilityTypeGroupField != null) { + if (facilityTypeGroupField.getValue() != null && facilityTypeField.getValue() != null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade() + .getActiveFacilitiesByDistrictAndType(newDistrict, (FacilityType) facilityTypeField.getValue(), true, false)); + enableFields(facilityField); + } else { + FieldHelper.updateEnumData(facilityTypeGroupField, FacilityTypeGroup.getAccomodationGroups()); + } + } + + FieldHelper.updateItems(communityFilter, FacadeProvider.getCommunityFacade().getAllActiveByDistrict(newDistrict.getUuid())); + } else { + clearAndDisableFields(communityFilter, facilityField, facilityTypeField, facilityTypeGroupField); + } + } + + break; + } + case AefiInvestigationCriteria.COMMUNITY: { + CommunityReferenceDto community = (CommunityReferenceDto) event.getProperty().getValue(); + if (!DataHelper.equal(community, criteria.getCommunity())) { + if (facilityField != null) { + facilityField.setValue(null); + } + + final FacilityType facilityType = facilityTypeField != null ? (FacilityType) facilityTypeField.getValue() : null; + + if (facilityType == null && facilityField != null) { + facilityField.removeAllItems(); + } else if (facilityField != null) { + if (community == null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByDistrictAndType(currentDistrict, facilityType, true, false)); + } else { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByCommunityAndType(community, facilityType, true, false)); + } + } + } + break; + } + case AefiInvestigationCriteria.FACILITY_TYPE_GROUP: { + FacilityTypeGroup typeGroup = (FacilityTypeGroup) event.getProperty().getValue(); + if (!DataHelper.equal(typeGroup, criteria.getFacilityTypeGroup())) { + if (typeGroup != null) { + enableFields(AefiInvestigationCriteria.FACILITY_TYPE); + FieldHelper.updateEnumData(facilityTypeField, FacilityType.getAccommodationTypes(typeGroup)); + facilityField.setValue(null); + } else { + clearAndDisableFields(facilityTypeField, facilityField); + } + } + + break; + } + case AefiInvestigationCriteria.FACILITY_TYPE: { + FacilityType facilityType = (FacilityType) event.getProperty().getValue(); + if (!DataHelper.equal(facilityType, criteria.getFacilityType())) { + if (facilityType == null) { + clearAndDisableFields(facilityField); + } else { + enableFields(facilityField); + facilityField.setValue(null); + + CommunityReferenceDto community = (CommunityReferenceDto) communityFilter.getValue(); + if (community != null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByCommunityAndType(community, facilityType, true, false)); + } else if (currentDistrict != null) { + FieldHelper.updateItems( + facilityField, + FacadeProvider.getFacilityFacade().getActiveFacilitiesByDistrictAndType(currentDistrict, facilityType, true, false)); + } + } + } + break; + } + } + } + + @Override + protected void applyDependenciesOnNewValue(AefiInvestigationCriteria criteria) { + + final UserDto user = currentUserDto(); + + UserProvider currentUserProvider = UserProvider.getCurrent(); + final JurisdictionLevel userJurisdictionLevel = currentUserProvider != null ? UserProvider.getCurrent().getJurisdictionLevel() : null; + + final ComboBox facilityTypeGroupField = getField(AefiInvestigationCriteria.FACILITY_TYPE_GROUP); + final ComboBox facilityTypeField = getField(AefiInvestigationCriteria.FACILITY_TYPE); + final ComboBox facilityField = getField(AefiInvestigationCriteria.HEALTH_FACILITY); + + // Disable all fields + clearAndDisableFields(districtFilter, communityFilter, facilityTypeGroupField, facilityTypeField, facilityField); + + // Get initial field values according to user and criteria + final RegionReferenceDto region = user.getRegion() == null ? criteria.getRegion() : user.getRegion(); + final DistrictReferenceDto district = user.getDistrict() == null ? criteria.getDistrict() : user.getDistrict(); + final CommunityReferenceDto community = user.getCommunity() == null ? criteria.getCommunity() : user.getCommunity(); + final FacilityTypeGroup facilityTypeGroup = criteria.getFacilityTypeGroup(); + final FacilityType facilityType = criteria.getFacilityType(); + + // district + if (region != null) { + enableFields(districtFilter); + districtFilter.addItems(FacadeProvider.getDistrictFacade().getAllActiveByRegion(region.getUuid())); + // community + if (district != null) { + districtFilter.setValue(district); + communityFilter.addItems(FacadeProvider.getCommunityFacade().getAllActiveByDistrict(district.getUuid())); + enableFields(communityFilter); + if (community != null) { + communityFilter.setValue(community); + } + } else { + clearAndDisableFields(communityFilter); + } + } else { + clearAndDisableFields(districtFilter, communityFilter); + } + + // facility + if (userJurisdictionLevel == JurisdictionLevel.HEALTH_FACILITY) { + facilityField.setValue(user.getHealthFacility()); + disableFields(facilityTypeGroupField, facilityTypeField, facilityField); + } else if (facilityTypeGroupField != null && district != null) { + enableFields(facilityTypeGroupField); + FieldHelper.updateEnumData(facilityTypeGroupField, FacilityTypeGroup.getAccomodationGroups()); + if (facilityTypeGroup != null) { + facilityTypeGroupField.setValue(facilityTypeGroup); + enableFields(facilityTypeField); + FieldHelper.updateEnumData(facilityTypeField, FacilityType.getAccommodationTypes(facilityTypeGroup)); + if (facilityType != null) { + facilityTypeField.setValue(facilityType); + enableFields(facilityField); + if (community != null) { + facilityField + .addItems(FacadeProvider.getFacilityFacade().getActiveFacilitiesByCommunityAndType(community, facilityType, true, false)); + } else { + facilityField + .addItems(FacadeProvider.getFacilityFacade().getActiveFacilitiesByDistrictAndType(district, facilityType, true, false)); + } + } else { + disableFields(facilityField); + } + } else { + disableFields(facilityTypeField); + } + } + + // Disable fields according to user & jurisdiction + if (userJurisdictionLevel == JurisdictionLevel.DISTRICT) { + clearAndDisableFields(districtFilter); + } else if (userJurisdictionLevel == JurisdictionLevel.COMMUNITY) { + clearAndDisableFields(districtFilter, communityFilter); + } else if (userJurisdictionLevel == JurisdictionLevel.HEALTH_FACILITY) { + clearAndDisableFields(districtFilter, communityFilter, facilityTypeGroupField, facilityTypeField, facilityField); + } + + // Date/Epi week filter + HorizontalLayout dateFilterLayout = (HorizontalLayout) getMoreFiltersContainer().getComponent(WEEK_AND_DATE_FILTER); + @SuppressWarnings("unchecked") + EpiWeekAndDateFilterComponent weekAndDateFilter = + (EpiWeekAndDateFilterComponent) dateFilterLayout.getComponent(0); + + AefiInvestigationDateType aefiInvestigationDateType = criteria.getAefiInvestigationDateType(); + weekAndDateFilter.getDateTypeSelector().setValue(aefiInvestigationDateType); + weekAndDateFilter.getDateFilterOptionFilter().setValue(criteria.getDateFilterOption()); + Date dateFrom = criteria.getFromDate(); + Date dateTo = criteria.getToDate(); + + if (DateFilterOption.EPI_WEEK.equals(criteria.getDateFilterOption())) { + weekAndDateFilter.getWeekFromFilter().setValue(dateFrom == null ? null : DateHelper.getEpiWeek(dateFrom)); + weekAndDateFilter.getWeekToFilter().setValue(dateTo == null ? null : DateHelper.getEpiWeek(dateTo)); + } else { + weekAndDateFilter.getDateFromFilter().setValue(dateFrom); + weekAndDateFilter.getDateToFilter().setValue(dateTo); + } + } + + @Override + protected Stream streamFieldsForEmptyCheck(CustomLayout layout) { + HorizontalLayout dateFilterLayout = (HorizontalLayout) getMoreFiltersContainer().getComponent(WEEK_AND_DATE_FILTER); + EpiWeekAndDateFilterComponent weekAndDateFilter = + (EpiWeekAndDateFilterComponent) dateFilterLayout.getComponent(0); + + return super.streamFieldsForEmptyCheck(layout).filter(f -> f != weekAndDateFilter.getDateFilterOptionFilter()); + } + + private HorizontalLayout buildWeekAndDateFilter() { + + EpiWeekAndDateFilterComponent weekAndDateFilter = new EpiWeekAndDateFilterComponent<>( + false, + false, + null, + AefiInvestigationDateType.values(), + I18nProperties.getString(Strings.promptAefiInvestigationDateType), + null, + this); + weekAndDateFilter.getWeekFromFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiInvestigationEpiWeekFrom)); + weekAndDateFilter.getWeekToFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiInvestigationEpiWeekTo)); + weekAndDateFilter.getDateFromFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiInvestigationDateFrom)); + weekAndDateFilter.getDateToFilter().setInputPrompt(I18nProperties.getString(Strings.promptAefiInvestigationDateTo)); + + addApplyHandler(e -> onApplyClick(weekAndDateFilter)); + + HorizontalLayout dateFilterRowLayout = new HorizontalLayout(); + dateFilterRowLayout.setSpacing(true); + dateFilterRowLayout.setSizeUndefined(); + + dateFilterRowLayout.addComponent(weekAndDateFilter); + + return dateFilterRowLayout; + } + + private void onApplyClick(EpiWeekAndDateFilterComponent weekAndDateFilter) { + AefiInvestigationCriteria criteria = getValue(); + + DateFilterOption dateFilterOption = (DateFilterOption) weekAndDateFilter.getDateFilterOptionFilter().getValue(); + Date fromDate, toDate; + if (dateFilterOption == DateFilterOption.DATE) { + Date dateFrom = weekAndDateFilter.getDateFromFilter().getValue(); + fromDate = dateFrom != null ? DateHelper.getStartOfDay(dateFrom) : null; + Date dateTo = weekAndDateFilter.getDateToFilter().getValue(); + toDate = dateFrom != null ? DateHelper.getEndOfDay(dateTo) : null; + } else { + fromDate = DateHelper.getEpiWeekStart((EpiWeek) weekAndDateFilter.getWeekFromFilter().getValue()); + toDate = DateHelper.getEpiWeekEnd((EpiWeek) weekAndDateFilter.getWeekToFilter().getValue()); + } + if ((fromDate != null && toDate != null) || (fromDate == null && toDate == null)) { + criteria.setDateFilterOption(dateFilterOption); + AefiInvestigationDateType aefiInvestigationDateType = (AefiInvestigationDateType) weekAndDateFilter.getDateTypeSelector().getValue(); + criteria.setAefiInvestigationDateType(aefiInvestigationDateType); + criteria.setFromDate(fromDate); + criteria.setToDate(toDate); + } else { + weekAndDateFilter.setNotificationsForMissingFilters(); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterFormLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterFormLayout.java new file mode 100644 index 00000000000..eddd5ee5c8b --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationFilterFormLayout.java @@ -0,0 +1,53 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import com.vaadin.ui.Button; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; + +public class AefiInvestigationFilterFormLayout extends VerticalLayout { + + private static final float PERCENTAGE_WIDTH = 100; + + private final AefiInvestigationFilterForm filterForm; + + public AefiInvestigationFilterFormLayout() { + setSpacing(false); + setMargin(false); + setWidth(PERCENTAGE_WIDTH, Unit.PERCENTAGE); + + filterForm = new AefiInvestigationFilterForm(); + addComponent(filterForm); + } + + public AefiInvestigationCriteria getValue() { + return filterForm.getValue(); + } + + public void setValue(AefiInvestigationCriteria criteria) { + filterForm.setValue(criteria); + } + + public void addResetHandler(Button.ClickListener listener) { + filterForm.addResetHandler(listener); + } + + public void addApplyHandler(Button.ClickListener listener) { + filterForm.addApplyHandler(listener); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationGrid.java new file mode 100644 index 00000000000..a0575b79a3f --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/directory/AefiInvestigationGrid.java @@ -0,0 +1,151 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.directory; + +import java.util.Date; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.ui.renderers.DateRenderer; +import com.vaadin.ui.renderers.TextRenderer; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationIndexDto; +import de.symeda.sormas.api.caze.AgeAndBirthDateDto; +import de.symeda.sormas.api.caze.Vaccine; +import de.symeda.sormas.api.feature.FeatureType; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.person.PersonHelper; +import de.symeda.sormas.ui.ControllerProvider; +import de.symeda.sormas.ui.UiUtil; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.FieldAccessColumnStyleGenerator; +import de.symeda.sormas.ui.utils.FilteredGrid; +import de.symeda.sormas.ui.utils.ShowDetailsListener; +import de.symeda.sormas.ui.utils.UuidRenderer; + +public class AefiInvestigationGrid extends FilteredGrid { + + public static final String PRIMARY_VACCINE_COLUMN = "primaryVaccineColumn"; + + public AefiInvestigationGrid(AefiInvestigationCriteria criteria) { + super(AefiInvestigationIndexDto.class); + setSizeFull(); + setLazyDataProvider(); + setCriteria(criteria); + + Column primaryVaccineColumn = addColumn(entry -> { + if (entry.getPrimaryVaccine() != null) { + if (entry.getPrimaryVaccine() != Vaccine.OTHER) { + return entry.getPrimaryVaccine().toString(); + } else { + return StringUtils.isBlank(entry.getPrimaryVaccineDetails()) + ? entry.getPrimaryVaccine().toString() + : (entry.getPrimaryVaccine() + " (" + entry.getPrimaryVaccineDetails() + ")"); + } + } else { + return "-"; + } + }); + primaryVaccineColumn.setId(PRIMARY_VACCINE_COLUMN); + primaryVaccineColumn + .setCaption(I18nProperties.getPrefixCaption(AefiInvestigationIndexDto.I18N_PREFIX, AefiInvestigationIndexDto.PRIMARY_VACCINE_NAME)); + + Column deleteColumn = addColumn(entry -> { + if (entry.getDeletionReason() != null) { + return entry.getDeletionReason() + (entry.getOtherDeletionReason() != null ? ": " + entry.getOtherDeletionReason() : ""); + } else { + return "-"; + } + }); + deleteColumn.setId(DELETE_REASON_COLUMN); + deleteColumn.setSortable(false); + deleteColumn.setCaption(I18nProperties.getCaption(Captions.deletionReason)); + + initColumns(); + + addItemClickListener( + new ShowDetailsListener<>( + AefiInvestigationIndexDto.UUID, + e -> ControllerProvider.getAefiInvestigationController().navigateToAefiInvestigation(e.getUuid()))); + addItemClickListener(new ShowDetailsListener<>(AefiInvestigationIndexDto.AEFI_REPORT_UUID, e -> { + ControllerProvider.getAefiController().navigateToAefi(e.getAefiReportUuid()); + })); + } + + public void reload() { + getDataProvider().refreshAll(); + } + + private void initColumns() { + setColumns( + AefiInvestigationIndexDto.UUID, + AefiInvestigationIndexDto.INVESTIGATION_CASE_ID, + AefiInvestigationIndexDto.AEFI_REPORT_UUID, + AefiInvestigationIndexDto.PERSON_FIRST_NAME, + AefiInvestigationIndexDto.PERSON_LAST_NAME, + AefiInvestigationIndexDto.DISEASE, + AefiInvestigationIndexDto.AGE_AND_BIRTH_DATE, + AefiInvestigationIndexDto.SEX, + AefiInvestigationIndexDto.REGION, + AefiInvestigationIndexDto.DISTRICT, + PRIMARY_VACCINE_COLUMN, + AefiInvestigationIndexDto.STATUS_ON_DATE_OF_INVESTIGATION, + AefiInvestigationIndexDto.INVESTIGATION_STATUS, + AefiInvestigationIndexDto.AEFI_CLASSIFICATION, + AefiInvestigationIndexDto.REPORT_DATE, + AefiInvestigationIndexDto.INVESTIGATION_DATE, + AefiInvestigationIndexDto.INVESTIGATION_STAGE, + DELETE_REASON_COLUMN); + + ((Column) getColumn(AefiInvestigationIndexDto.UUID)).setRenderer(new UuidRenderer()); + ((Column) getColumn(AefiInvestigationIndexDto.AEFI_REPORT_UUID)).setRenderer(new UuidRenderer()); + + ((Column) getColumn(AefiInvestigationIndexDto.AGE_AND_BIRTH_DATE)).setRenderer( + value -> value == null + ? "" + : PersonHelper.getAgeAndBirthdateString( + value.getAge(), + value.getAgeType(), + value.getDateOfBirthDD(), + value.getDateOfBirthMM(), + value.getDateOfBirthYYYY()), + new TextRenderer()); + + ((Column) getColumn(AefiInvestigationIndexDto.REPORT_DATE)) + .setRenderer(new DateRenderer(DateFormatHelper.getDateFormat())); + ((Column) getColumn(AefiInvestigationIndexDto.INVESTIGATION_DATE)) + .setRenderer(new DateRenderer(DateFormatHelper.getDateFormat())); + + for (Column column : getColumns()) { + column + .setCaption(I18nProperties.findPrefixCaptionWithDefault(column.getId(), column.getCaption(), AefiInvestigationIndexDto.I18N_PREFIX)); + column.setStyleGenerator(FieldAccessColumnStyleGenerator.getDefault(getBeanType(), column.getId())); + } + + if (UiUtil.enabled(FeatureType.HIDE_JURISDICTION_FIELDS)) { + getColumn(AefiInvestigationIndexDto.REGION).setHidden(true); + getColumn(AefiInvestigationIndexDto.DISTRICT).setHidden(true); + } + } + + private void setLazyDataProvider() { + + setLazyDataProvider(FacadeProvider.getAefiInvestigationFacade()::getIndexList, FacadeProvider.getAefiInvestigationFacade()::count); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java index 03742120e3a..8597fdac53a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -29,6 +26,7 @@ import com.vaadin.v7.ui.Table; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; import de.symeda.sormas.api.caze.Vaccine; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -41,7 +39,10 @@ public class AefiVaccinationsField extends AbstractTableField { private AefiDto aefiDto; + private AefiInvestigationDto aefiInvestigationDto; private VaccinationDto primarySuspectVaccination; + private boolean aefiReportContext; + private boolean aefiInvestigationContext; private final UiFieldAccessCheckers fieldAccessCheckers; public AefiVaccinationsField(UiFieldAccessCheckers fieldAccessCheckers) { @@ -60,7 +61,12 @@ public Class getEntryType() { protected void editEntry(VaccinationDto entry, boolean create, Consumer commitCallback) { if (create) { - ControllerProvider.getAefiController().selectPrimarySuspectVaccination(aefiDto, this::selectPrimarySuspectVaccination); + if (aefiReportContext) { + ControllerProvider.getAefiController().selectPrimarySuspectVaccination(aefiDto, this::selectPrimarySuspectVaccination); + } else if (aefiInvestigationContext) { + ControllerProvider.getAefiInvestigationController() + .selectPrimarySuspectVaccination(aefiInvestigationDto, this::selectPrimarySuspectVaccination); + } } } @@ -68,30 +74,6 @@ public void updateAddButtonCaption() { getAddButton().setCaption(I18nProperties.getCaption(Captions.actionAefiSelectPrimarySuspectVaccination)); } - /* - * @Override - * protected VaccinationDto createEntry() { - * UserDto user = UserProvider.getCurrent().getUser(); - * return VaccinationDto.build(user.toReference()); - * } - */ - - /* - * @Override - * protected Table createTable() { - * Table table = super.createTable(); - * table.addGeneratedColumn(VaccinationDto.UUID, (Table.ColumnGenerator) (source, itemId, columnId) -> { - * Label textField = new Label(DataHelper.getShortUuid(((EntityDto) itemId).getUuid())); - * return textField; - * }); - * table.addGeneratedColumn(VaccinationDto.VACCINATION_DATE, (Table.ColumnGenerator) (source, itemId, columnId) -> { - * Label textField = new Label(DateFormatHelper.formatDate(((VaccinationDto) itemId).getVaccinationDate())); - * return textField; - * }); - * return table; - * } - */ - @Override protected void updateColumns() { Table table = getTable(); @@ -175,12 +157,22 @@ public void refreshTable() { } container.removeAllItems(); - container.addAll(aefiDto.getVaccinations()); + if (aefiReportContext) { + container.addAll(aefiDto.getVaccinations()); + } else { + container.addAll(aefiInvestigationDto.getVaccinations()); + } table.refreshRowCache(); } - public void setAefiDto(AefiDto aefiDto) { + public void applyAefiReportContext(AefiDto aefiDto) { this.aefiDto = aefiDto; + aefiReportContext = true; + } + + public void applyAefiInvestigationContext(AefiInvestigationDto aefiInvestigationDto) { + this.aefiInvestigationDto = aefiInvestigationDto; + aefiInvestigationContext = true; } public void selectPrimarySuspectVaccination(VaccinationDto vaccinationDto) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java deleted file mode 100644 index 659ae920839..00000000000 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/fields/vaccines/AefiVaccinationsField_2.java +++ /dev/null @@ -1,198 +0,0 @@ -/* - * SORMAS® - Surveillance Outbreak Response Management & Analysis System - * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * - * This program is free software: you can redistribute it and/or modify - * it under the terms of the GNU General Public License as published by - * the Free Software Foundation, either version 3 of the License, or - * (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License - * along with this program. If not, see . - */ - -package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines; - -import java.util.ArrayList; -import java.util.Collection; -import java.util.List; - -import org.slf4j.Logger; -import org.slf4j.LoggerFactory; - -import com.vaadin.ui.Alignment; -import com.vaadin.ui.Button; -import com.vaadin.ui.Component; -import com.vaadin.ui.Grid; -import com.vaadin.ui.HorizontalLayout; -import com.vaadin.ui.Label; -import com.vaadin.ui.VerticalLayout; -import com.vaadin.ui.themes.ValoTheme; -import com.vaadin.v7.data.Property; -import com.vaadin.v7.data.util.converter.Converter; -import com.vaadin.v7.ui.CustomField; - -import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; -import de.symeda.sormas.api.caze.Vaccine; -import de.symeda.sormas.api.caze.VaccineManufacturer; -import de.symeda.sormas.api.i18n.Captions; -import de.symeda.sormas.api.i18n.I18nProperties; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; -import de.symeda.sormas.api.vaccination.VaccinationDto; -import de.symeda.sormas.ui.ControllerProvider; -import de.symeda.sormas.ui.utils.ButtonHelper; -import de.symeda.sormas.ui.utils.CssStyles; -import de.symeda.sormas.ui.utils.DateFormatHelper; - -@SuppressWarnings({ - "serial", - "rawtypes" }) -public class AefiVaccinationsField_2 extends CustomField { - - private final Logger logger = LoggerFactory.getLogger(getClass()); - - private VerticalLayout mainLayout; - private Label captionLabel; - private Button addButton; - private Grid vaccinationGrid; - private List value = new ArrayList<>(); - private AefiDto aefiDto; - private VaccinationDto primarySuspectVaccination; - protected UiFieldAccessCheckers fieldAccessCheckers; - - public AefiVaccinationsField_2(UiFieldAccessCheckers fieldAccessCheckers) { - this.fieldAccessCheckers = fieldAccessCheckers; - - getContent(); - //setValue(value); - } - - public void initializeGrid() { - - vaccinationGrid = new Grid<>(); - vaccinationGrid.setSizeFull(); - vaccinationGrid.setSelectionMode(Grid.SelectionMode.SINGLE); - /* vaccinationGrid.setHeightMode(HeightMode.ROW); */ - - vaccinationGrid.addColumn(vaccinationDto -> DateFormatHelper.formatDate(vaccinationDto.getVaccinationDate())) - .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINATION_DATE)); - - vaccinationGrid - .addColumn( - vaccinationDto -> Vaccine.OTHER.equals(vaccinationDto.getVaccineName()) - ? vaccinationDto.getOtherVaccineName() - : vaccinationDto.getVaccineName()) - .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_NAME)); - - vaccinationGrid - .addColumn( - vaccinationDto -> VaccineManufacturer.OTHER.equals(vaccinationDto.getVaccineManufacturer()) - ? vaccinationDto.getOtherVaccineManufacturer() - : vaccinationDto.getVaccineManufacturer()) - .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_MANUFACTURER)); - - vaccinationGrid.addColumn(VaccinationDto::getVaccineType) - .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_TYPE)); - - vaccinationGrid.addColumn(VaccinationDto::getVaccineDose) - .setCaption(I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_DOSE)); - - vaccinationGrid.setStyleGenerator(vaccinationDto -> { - if (primarySuspectVaccination != null) { - return vaccinationDto.getUuid().equals(primarySuspectVaccination.getUuid()) ? CssStyles.GRID_ROW_SELECTED : null; - } - return null; - }); - } - - @Override - protected Component initContent() { - mainLayout = new VerticalLayout(); - mainLayout.setSpacing(false); - mainLayout.setMargin(false); - - HorizontalLayout headerLayout = new HorizontalLayout(); - { - headerLayout.setWidth(100, Unit.PERCENTAGE); - - captionLabel = new Label(getCaption()); - captionLabel.setSizeUndefined(); - headerLayout.addComponent(captionLabel); - headerLayout.setComponentAlignment(captionLabel, Alignment.BOTTOM_LEFT); - headerLayout.setExpandRatio(captionLabel, 0); - - addButton = ButtonHelper.createButton(Captions.actionAefiSelectPrimarySuspectVaccination, (event) -> { - ControllerProvider.getAefiController().selectPrimarySuspectVaccination(aefiDto, this::selectPrimarySuspectVaccination); - }, ValoTheme.BUTTON_LINK); - headerLayout.addComponent(addButton); - headerLayout.setComponentAlignment(addButton, Alignment.BOTTOM_RIGHT); - headerLayout.setExpandRatio(addButton, 1); - } - mainLayout.addComponent(headerLayout); - - initializeGrid(); - mainLayout.addComponent(vaccinationGrid); - - return mainLayout; - } - - @Override - public Class getType() { - return Collection.class; - } - - @Override - public void setPropertyDataSource(Property newDataSource) { - super.setPropertyDataSource(newDataSource); - } - - /* - * @Override - * protected void setValue(Collection newFieldValue, boolean repaintIsNotNeeded, boolean ignoreReadOnly) - * throws ReadOnlyException, Converter.ConversionException, Validator.InvalidValueException { - * super.setValue(newFieldValue, repaintIsNotNeeded, ignoreReadOnly); - * value = new ArrayList<>(newFieldValue); - * vaccinationGrid.setItems(newFieldValue); - * fireValueChange(repaintIsNotNeeded); - * } - */ - - @Override - public void setValue(Collection newFieldValue) throws ReadOnlyException, Converter.ConversionException { - value = new ArrayList<>(newFieldValue); - vaccinationGrid.setItems(newFieldValue); - - super.setValue(newFieldValue); - } - - /* - * @Override - * protected void doSetValue(Collection collection) { - * value = new ArrayList<>(collection); - * vaccinationGrid.setItems(collection); - * } - */ - - @Override - public Collection getValue() { - return value; - } - - public void setAefiDto(AefiDto aefiDto) { - this.aefiDto = aefiDto; - } - - public void setPrimarySuspectVaccination(VaccinationDto primarySuspectVaccination) { - this.primarySuspectVaccination = primarySuspectVaccination; - } - - public void selectPrimarySuspectVaccination(VaccinationDto vaccinationDto) { - primarySuspectVaccination = vaccinationDto; - vaccinationGrid.select(primarySuspectVaccination); - } -} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java index 4eade7d908a..48181c91253 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AdverseEventsForm.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -44,6 +41,7 @@ import com.vaadin.ui.CustomLayout; import com.vaadin.ui.Label; +import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.ui.CheckBox; import com.vaadin.v7.ui.Field; import com.vaadin.v7.ui.TextArea; @@ -58,7 +56,9 @@ import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.FieldHelper; +import de.symeda.sormas.ui.utils.NullableOptionGroup; +@SuppressWarnings("deprecation") public class AdverseEventsForm extends AbstractEditForm { private static final long serialVersionUID = 5081846814610543073L; @@ -119,7 +119,8 @@ protected void addFields() { emptyLabel.addStyleName(H3); getContent().addComponent(emptyLabel, EMPTY_LABEL_LOC); - addField(SEIZURE_TYPE); + NullableOptionGroup seizureType = addField(SEIZURE_TYPE, NullableOptionGroup.class); + CssStyles.style(seizureType, ValoTheme.OPTIONGROUP_HORIZONTAL, CssStyles.OPTIONGROUP_CAPTION_INLINE); TextArea otherAdverseEvents = addField(OTHER_ADVERSE_EVENT_DETAILS, TextArea.class); otherAdverseEvents.setRows(6); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java index 3bbd3e2e4c0..78ee9dcb888 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -79,7 +76,8 @@ public class AefiDataForm extends AbstractEditForm { private static final String HTML_LAYOUT = divCss(CssStyles.VIEW_SECTION_MARGIN_TOP_4_MARGIN_X_4, loc(REPORTING_INFORMATION_HEADING_LOC) - + fluidRowLocs(4, AefiDto.UUID, 3, AefiDto.REPORT_DATE, 3, AefiDto.REPORTING_USER, 2, "") + + fluidRowLocs(4, AefiDto.UUID, 4, AefiDto.REPORT_DATE, 3, AefiDto.REPORTING_USER) + + fluidRowLocs(4, AefiDto.RESPONSIBLE_REGION, 4, AefiDto.RESPONSIBLE_DISTRICT, 3, AefiDto.RESPONSIBLE_COMMUNITY) + fluidRowLocs(4, AefiDto.REPORTING_ID_NUMBER, 3, ASSIGN_NEW_AEFI_ID_LOC) ) + divCss(CssStyles.VIEW_SECTION_MARGIN_X_4 + " " + CssStyles.VSPACE_TOP_3, @@ -118,6 +116,9 @@ public class AefiDataForm extends AbstractEditForm { private boolean isCreateAction; private final Consumer actionCallback; + private TextField responsibleRegion; + private TextField responsibleDistrict; + private TextField responsibleCommunity; private AefiVaccinationsField vaccinationsField; public AefiDataForm(boolean isCreateAction, boolean isPseudonymized, boolean inJurisdiction, Consumer actionCallback) { @@ -156,7 +157,14 @@ protected void addFields() { addField(AefiDto.UUID); } addField(AefiDto.REPORT_DATE, DateField.class); - addField(ImmunizationDto.REPORTING_USER, UserField.class); + addField(AefiDto.REPORTING_USER, UserField.class); + + responsibleRegion = new TextField(I18nProperties.getPrefixCaption(AefiDto.I18N_PREFIX, AefiDto.RESPONSIBLE_REGION)); + getContent().addComponent(responsibleRegion, AefiDto.RESPONSIBLE_REGION); + responsibleDistrict = new TextField(I18nProperties.getPrefixCaption(AefiDto.I18N_PREFIX, AefiDto.RESPONSIBLE_DISTRICT)); + getContent().addComponent(responsibleDistrict, AefiDto.RESPONSIBLE_DISTRICT); + responsibleCommunity = new TextField(I18nProperties.getPrefixCaption(AefiDto.I18N_PREFIX, AefiDto.RESPONSIBLE_COMMUNITY)); + getContent().addComponent(responsibleCommunity, AefiDto.RESPONSIBLE_COMMUNITY); TextField reportIdField = addField(AefiDto.REPORTING_ID_NUMBER, TextField.class); /* @@ -272,18 +280,25 @@ protected void addFields() { public void attach() { super.attach(); - vaccinationsField.setAefiDto(getValue()); - if (getValue().getPrimarySuspectVaccine() != null) { - vaccinationsField.selectPrimarySuspectVaccination(getValue().getPrimarySuspectVaccine()); + AefiDto dataFormValue = getValue(); + + ImmunizationDto immunizationDto = FacadeProvider.getImmunizationFacade().getByUuid(dataFormValue.getImmunization().getUuid()); + + responsibleRegion.setValue(immunizationDto.getResponsibleRegion().getCaption()); + responsibleDistrict.setValue(immunizationDto.getResponsibleDistrict().getCaption()); + if (immunizationDto.getResponsibleCommunity() != null) { + responsibleCommunity.setValue(immunizationDto.getResponsibleCommunity().getCaption()); } - } - /* - * @Override - * public AefiDto getValue() { - * return super.getValue(); - * } - */ + responsibleRegion.setReadOnly(true); + responsibleDistrict.setReadOnly(true); + responsibleCommunity.setReadOnly(true); + + vaccinationsField.applyAefiReportContext(dataFormValue); + if (dataFormValue.getPrimarySuspectVaccine() != null) { + vaccinationsField.selectPrimarySuspectVaccination(dataFormValue.getPrimarySuspectVaccine()); + } + } @Override public void setValue(AefiDto newFieldValue) throws ReadOnlyException, Converter.ConversionException { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordion.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordion.java new file mode 100644 index 00000000000..ce435e84443 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordion.java @@ -0,0 +1,55 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form; + +import java.util.ArrayList; +import java.util.List; + +import com.vaadin.ui.Component; +import com.vaadin.ui.VerticalLayout; + +public class FormSectionAccordion extends VerticalLayout { + + private final List formSectionAccordionPanelList = new ArrayList<>(); + + public FormSectionAccordion() { + setMargin(false); + setSpacing(false); + setWidth(100, Unit.PERCENTAGE); + } + + public void addFormSectionPanel(String sectionTitleCaption, boolean expanded, Component component) { + + FormSectionAccordionPanel formSectionAccordionPanel = new FormSectionAccordionPanel(sectionTitleCaption, expanded, FormSectionAccordion.this); + formSectionAccordionPanel.addComponentToMainLayout(component); + + formSectionAccordionPanelList.add(formSectionAccordionPanel); + + addComponent(formSectionAccordionPanel); + } + + public void toggleFormSection(FormSectionAccordionPanel formSectionAccordionPanel) { + formSectionAccordionPanel.getMainLayout().setVisible(!formSectionAccordionPanel.isExpanded()); + formSectionAccordionPanel.setExpanded(!formSectionAccordionPanel.isExpanded()); + } + + public void toggleAllFormSections(boolean expanded) { + for (FormSectionAccordionPanel formSectionAccordionPanel : formSectionAccordionPanelList) { + formSectionAccordionPanel.getMainLayout().setVisible(!expanded); + formSectionAccordionPanel.setExpanded(!expanded); + } + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordionPanel.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordionPanel.java new file mode 100644 index 00000000000..6e8cc33ef79 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/FormSectionAccordionPanel.java @@ -0,0 +1,77 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form; + +import com.vaadin.ui.Button; +import com.vaadin.ui.Component; +import com.vaadin.ui.VerticalLayout; +import com.vaadin.ui.themes.ValoTheme; + +import de.symeda.sormas.ui.utils.ButtonHelper; +import de.symeda.sormas.ui.utils.CssStyles; + +public class FormSectionAccordionPanel extends VerticalLayout { + + private final Button titleButton; + private final VerticalLayout mainLayout; + private final FormSectionAccordion formSectionAccordion; + private boolean expanded; + + public FormSectionAccordionPanel(String titleButtonCaption, boolean expanded, FormSectionAccordion formSectionAccordion) { + + this.expanded = expanded; + this.formSectionAccordion = formSectionAccordion; + + setMargin(false); + setSpacing(false); + setWidth(99, Unit.PERCENTAGE); + addStyleNames(CssStyles.VIEW_SECTION_MARGIN_X_4, CssStyles.VSPACE_3); + + titleButton = ButtonHelper.createButton(titleButtonCaption, (event) -> { + formSectionAccordion.toggleFormSection(FormSectionAccordionPanel.this); + }, ValoTheme.BUTTON_LINK, CssStyles.FORM_SECTION_ACCORDION_PANEL_TITLE_BUTTON); + addComponent(titleButton); + + mainLayout = new VerticalLayout(); + mainLayout.setMargin(false); + mainLayout.setSpacing(false); + //mainLayout.setWidth(100, Unit.PERCENTAGE); + mainLayout.setVisible(expanded); + mainLayout.addStyleNames(CssStyles.VSPACE_TOP_3); + + addComponent(mainLayout); + } + + public void addComponentToMainLayout(Component component) { + mainLayout.addComponent(component); + } + + public Button getTitleButton() { + return titleButton; + } + + public VerticalLayout getMainLayout() { + return mainLayout; + } + + public boolean isExpanded() { + return expanded; + } + + public void setExpanded(boolean expanded) { + this.expanded = expanded; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiInfo.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiInfo.java new file mode 100644 index 00000000000..43824207209 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiInfo.java @@ -0,0 +1,89 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.information; + +import java.util.function.Consumer; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.ui.Label; +import com.vaadin.ui.VerticalLayout; + +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiHelper; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiListEntryDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.utils.YesNoUnknown; +import de.symeda.sormas.api.vaccination.VaccinationDto; +import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponent; + +public class AefiInfo extends SideComponent { + + public static final String SEPARATOR = ": "; + + private AefiDto aefi; + private final VerticalLayout mainLayout; + + public AefiInfo(AefiDto aefi, Consumer actionCallback) { + super(I18nProperties.getString(Strings.entityAdverseEvent), actionCallback); + + this.aefi = aefi; + mainLayout = new VerticalLayout(); + mainLayout.setWidth(100, Unit.PERCENTAGE); + mainLayout.setMargin(false); + mainLayout.setSpacing(false); + + buildMainLayout(); + addComponent(mainLayout); + } + + public void buildMainLayout() { + + Label labelAefiType = new Label(AefiType.toString(aefi.getSerious())); + CssStyles.style(labelAefiType, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + if (aefi.getSerious() == YesNoUnknown.YES) { + CssStyles.style(labelAefiType, CssStyles.LABEL_CRITICAL); + } + mainLayout.addComponent(labelAefiType); + + Label labelVaccineName = new Label(aefi.getPrimarySuspectVaccine().getVaccineName().toString()); + CssStyles.style(labelVaccineName, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + mainLayout.addComponent(labelVaccineName); + + if (!StringUtils.isBlank(aefi.getPrimarySuspectVaccine().getVaccineDose())) { + Label labelVaccineDose = new Label( + I18nProperties.getPrefixCaption(VaccinationDto.I18N_PREFIX, VaccinationDto.VACCINE_DOSE) + + SEPARATOR + + aefi.getPrimarySuspectVaccine().getVaccineDose()); + CssStyles.style(labelVaccineDose, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + mainLayout.addComponent(labelVaccineDose); + } + + Label labelVaccineDate = new Label( + I18nProperties.getPrefixCaption(AefiListEntryDto.I18N_PREFIX, AefiListEntryDto.PRIMARY_VACCINE_VACCINATION_DATE) + + SEPARATOR + + DateFormatHelper.formatLocalDate(aefi.getPrimarySuspectVaccine().getVaccinationDate())); + mainLayout.addComponent(labelVaccineDate); + + Label labelAdverseEvents = new Label(StringUtils.abbreviate(AefiHelper.buildAdverseEventsString(aefi.getAdverseEvents()), 56)); + CssStyles.style(labelAdverseEvents, CssStyles.LABEL_BOLD, CssStyles.LABEL_UPPERCASE); + mainLayout.addComponent(labelAdverseEvents); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java index fb6a77afc5a..2945bb8f54f 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardDataProvider.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -21,7 +18,9 @@ import java.util.Map; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDashboardFilterDateType; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; import de.symeda.sormas.api.caze.Vaccine; import de.symeda.sormas.api.dashboard.AefiDashboardCriteria; @@ -33,6 +32,9 @@ public class AefiDashboardDataProvider extends AbstractDashboardDataProvider aefiCountsByType; + private int totalAefiInvestigations; + private Map> aefiInvestigationCountsByInvestigationStatus; + private Map> aefiInvestigationCountsByAefiClassification; private Map> aefiCountsByVaccine; private AefiChartData aefiByVaccineDoseChartData; private AefiChartData aefiEventsByGenderChartData; @@ -40,6 +42,16 @@ public class AefiDashboardDataProvider extends AbstractDashboardDataProvider investigationStatusValue : aefiInvestigationCountsByInvestigationStatus.values()) { + totalAefiInvestigations += Integer.parseInt(investigationStatusValue.get("total")); + } + + aefiInvestigationCountsByAefiClassification = + FacadeProvider.getAefiDashboardFacade().getAefiInvestigationCountsByAefiClassification(buildDashboardCriteriaWithDates()); aefiCountsByVaccine = FacadeProvider.getAefiDashboardFacade().getAefiCountsByVaccine(buildDashboardCriteriaWithDates()); aefiByVaccineDoseChartData = FacadeProvider.getAefiDashboardFacade().getAefiByVaccineDoseChartData(buildDashboardCriteriaWithDates()); aefiEventsByGenderChartData = FacadeProvider.getAefiDashboardFacade().getAefiEventsByGenderChartData(buildDashboardCriteriaWithDates()); @@ -67,6 +79,18 @@ public Map getAefiCountsByType() { return aefiCountsByType; } + public int getTotalAefiInvestigations() { + return totalAefiInvestigations; + } + + public Map> getAefiInvestigationCountsByInvestigationStatus() { + return aefiInvestigationCountsByInvestigationStatus; + } + + public Map> getAefiInvestigationCountsByAefiClassification() { + return aefiInvestigationCountsByAefiClassification; + } + public Map> getAefiCountsByVaccine() { return aefiCountsByVaccine; } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java index 0c4df88ccf6..c4f2cc2ff1a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardFilterLayout.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -78,7 +75,7 @@ private void createDiseaseFilter() { diseaseFilter.setWidth(200, Unit.PIXELS); diseaseFilter.setInputPrompt(I18nProperties.getString(Strings.promptDisease)); diseaseFilter.setDescription(I18nProperties.getDescription(Descriptions.aefiDashboardDiseaseFilter)); - List availableDisease = FacadeProvider.getDiseaseConfigurationFacade().getAllDiseasesWithFollowUp(true, true, true); + List availableDisease = FacadeProvider.getDiseaseConfigurationFacade().getAllDiseases(true, true, true); diseaseFilter .addItems(Stream.concat(availableDisease.stream(), Stream.of(AefiDashboardCustomDiseaseFilter.values())).collect(Collectors.toList())); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java index 15bc987c333..b114177e007 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -23,13 +20,19 @@ import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowCss; import static de.symeda.sormas.ui.utils.LayoutUtil.locCss; +import java.util.Map; + +import com.vaadin.shared.ui.MarginInfo; import com.vaadin.ui.Component; import com.vaadin.ui.CustomLayout; import com.vaadin.ui.HorizontalLayout; import com.vaadin.ui.VerticalLayout; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiClassification; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationStatus; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiType; import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.ui.dashboard.AbstractDashboardView; import de.symeda.sormas.ui.dashboard.DashboardCssStyles; import de.symeda.sormas.ui.dashboard.DashboardType; @@ -39,6 +42,7 @@ import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiReactionsByGenderChart; import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiTypeStatisticsGroupComponent; import de.symeda.sormas.ui.dashboard.components.DashboardHeadingComponent; +import de.symeda.sormas.ui.dashboard.statistics.DashboardStatisticsPercentageElement; import de.symeda.sormas.ui.utils.CssStyles; public class AefiDashboardView extends AbstractDashboardView { @@ -47,8 +51,11 @@ public class AefiDashboardView extends AbstractDashboardView { private static final int EPI_CURVE_AND_MAP_HEIGHT = 555; - private static final String HEADING_LOC = "headingLoc"; + private static final String ALL_AEFI_HEADING_LOC = "allAefiHeadingLoc"; private static final String AEFI_TYPE_LOC = "aefiTypeLoc"; + private static final String ALL_AEFI_INVESTIGATION_HEADING_LOC = "allAefiInvestigationHeadingLoc"; + private static final String INVESTIGATION_STATUS_LOC = "investigationStatusLoc"; + private static final String AEFI_CLASSIFICATION_LOC = "aefiClassificationLoc"; private static final String VACCINES_LOC = "vaccinesLoc"; private static final String VACCINE_DOSE_LOC = "vaccineDoseLoc"; private static final String REACTIONS_LOC = "reactionsLoc"; @@ -60,8 +67,14 @@ public class AefiDashboardView extends AbstractDashboardView { private final VerticalLayout epiCurveLayout; private final VerticalLayout mapLayout; - private final DashboardHeadingComponent heading; + private final DashboardHeadingComponent allAefiHeading; private final AefiCountTilesComponent aefiCountsByType; + private final DashboardHeadingComponent allAefiInvestigationHeading; + private DashboardStatisticsPercentageElement investigationStatusDone; + private DashboardStatisticsPercentageElement investigationStatusDiscarded; + private DashboardStatisticsPercentageElement aefiClassificationVaccineRelated; + private DashboardStatisticsPercentageElement aefiClassificationConincidentalAefi; + private DashboardStatisticsPercentageElement aefiClassificationUndetermined; private final AefiTypeStatisticsGroupComponent aefiCountsByVaccine; private final AefiByVaccineDoseChart vaccineDoseChart; private final AefiReactionsByGenderChart reactionsByGenderChart; @@ -89,8 +102,11 @@ public AefiDashboardView() { aefiCountsLayout.setTemplateContents( fluidRowCss( CssStyles.PADDING_X_20 + " " + CssStyles.VSPACE_TOP_2, - fluidColumn(6, 0, locCss("", HEADING_LOC)), - fluidColumn(4, 0, locCss("", AEFI_TYPE_LOC)) + fluidColumn(1, 0, locCss("", ALL_AEFI_HEADING_LOC)), + fluidColumn(3, 0, locCss("", AEFI_TYPE_LOC)), + fluidColumn(2, 0, locCss("", ALL_AEFI_INVESTIGATION_HEADING_LOC)), + fluidColumn(3, 0, locCss("", INVESTIGATION_STATUS_LOC)), + fluidColumn(3, 0, locCss("", AEFI_CLASSIFICATION_LOC)) ) + fluidRowCss( CssStyles.VSPACE_TOP_2, @@ -105,15 +121,53 @@ public AefiDashboardView() { //@formatter:on dashboardLayout.addComponent(aefiCountsLayout); - heading = new DashboardHeadingComponent(Captions.aefiDashboardAllAefi, null); - heading.setMargin(false); - aefiCountsLayout.addComponent(heading, HEADING_LOC); + allAefiHeading = new DashboardHeadingComponent(Captions.aefiDashboardAllAefi, null); + allAefiHeading.setMargin(false); + aefiCountsLayout.addComponent(allAefiHeading, ALL_AEFI_HEADING_LOC); aefiCountsByType = new AefiCountTilesComponent<>(AefiType.class, "", this::getBackgroundStyleForAefiCountByType, null); aefiCountsByType.setTitleStyleNames(CssStyles.H3, CssStyles.VSPACE_TOP_5); aefiCountsByType.setGroupLabelStyle(CssStyles.LABEL_LARGE + " " + CssStyles.LABEL_WHITE_SPACE_NORMAL); aefiCountsLayout.addComponent(aefiCountsByType, AEFI_TYPE_LOC); + allAefiInvestigationHeading = new DashboardHeadingComponent(Captions.aefiDashboardAllAefiInvestigation, null); + allAefiInvestigationHeading.setMargin(false); + aefiCountsLayout.addComponent(allAefiInvestigationHeading, ALL_AEFI_INVESTIGATION_HEADING_LOC); + + VerticalLayout investigationStatusLayout = new VerticalLayout(); + investigationStatusLayout.setMargin(new MarginInfo(false, true, false, true)); + investigationStatusLayout.setSpacing(false); + + investigationStatusDone = new DashboardStatisticsPercentageElement( + I18nProperties.getCaption(Captions.aefiDashboardAefiInvestigationDone), + CssStyles.SVG_FILL_POSITIVE); + investigationStatusDiscarded = new DashboardStatisticsPercentageElement( + I18nProperties.getCaption(Captions.aefiDashboardAefiInvestigationDiscarded), + CssStyles.SVG_FILL_NEUTRAL); + + investigationStatusLayout.addComponent(investigationStatusDone); + investigationStatusLayout.addComponent(investigationStatusDiscarded); + aefiCountsLayout.addComponent(investigationStatusLayout, INVESTIGATION_STATUS_LOC); + + VerticalLayout aefiClassificationLayout = new VerticalLayout(); + aefiClassificationLayout.setMargin(new MarginInfo(false, true, false, true)); + aefiClassificationLayout.setSpacing(false); + + aefiClassificationVaccineRelated = new DashboardStatisticsPercentageElement( + I18nProperties.getCaption(Captions.aefiDashboardAefiClassificationRelatedToVaccination), + CssStyles.SVG_FILL_POSITIVE); + aefiClassificationConincidentalAefi = new DashboardStatisticsPercentageElement( + I18nProperties.getCaption(Captions.aefiDashboardAefiClassificationCoincidentalAdverseEvent), + CssStyles.SVG_FILL_IMPORTANT); + aefiClassificationUndetermined = new DashboardStatisticsPercentageElement( + I18nProperties.getCaption(Captions.aefiDashboardAefiClassificationUndetermined), + CssStyles.SVG_FILL_NEUTRAL); + + aefiClassificationLayout.addComponent(aefiClassificationVaccineRelated); + aefiClassificationLayout.addComponent(aefiClassificationConincidentalAefi); + aefiClassificationLayout.addComponent(aefiClassificationUndetermined); + aefiCountsLayout.addComponent(aefiClassificationLayout, AEFI_CLASSIFICATION_LOC); + aefiCountsByVaccine = new AefiTypeStatisticsGroupComponent(); aefiCountsByVaccine.addStyleNames(CssStyles.PADDING_X_20); aefiCountsLayout.addComponent(aefiCountsByVaccine, VACCINES_LOC); @@ -140,9 +194,30 @@ public AefiDashboardView() { public void refreshDashboard() { dataProvider.refreshData(); - heading.updateTotalLabel(String.valueOf(dataProvider.getAefiCountsByType().values().stream().mapToLong(Long::longValue).sum())); - + allAefiHeading.updateTotalLabel(String.valueOf(dataProvider.getAefiCountsByType().values().stream().mapToLong(Long::longValue).sum())); aefiCountsByType.update(dataProvider.getAefiCountsByType()); + + allAefiInvestigationHeading.updateTotalLabel(String.valueOf(dataProvider.getTotalAefiInvestigations())); + Map> investigationStatusCountMap = + dataProvider.getAefiInvestigationCountsByInvestigationStatus(); + investigationStatusDone.updatePercentageValueWithCount( + Integer.parseInt(investigationStatusCountMap.get(AefiInvestigationStatus.DONE).get("total")), + Integer.parseInt(investigationStatusCountMap.get(AefiInvestigationStatus.DONE).get("percent"))); + investigationStatusDiscarded.updatePercentageValueWithCount( + Integer.parseInt(investigationStatusCountMap.get(AefiInvestigationStatus.DISCARDED).get("total")), + Integer.parseInt(investigationStatusCountMap.get(AefiInvestigationStatus.DISCARDED).get("percent"))); + + Map> aefiClassificationCountMap = dataProvider.getAefiInvestigationCountsByAefiClassification(); + aefiClassificationVaccineRelated.updatePercentageValueWithCount( + Integer.parseInt(aefiClassificationCountMap.get(AefiClassification.RELATED_TO_VACCINE_OR_VACCINATION).get("total")), + Integer.parseInt(aefiClassificationCountMap.get(AefiClassification.RELATED_TO_VACCINE_OR_VACCINATION).get("percent"))); + aefiClassificationConincidentalAefi.updatePercentageValueWithCount( + Integer.parseInt(aefiClassificationCountMap.get(AefiClassification.COINCIDENTAL_ADVERSE_EVENT).get("total")), + Integer.parseInt(aefiClassificationCountMap.get(AefiClassification.COINCIDENTAL_ADVERSE_EVENT).get("percent"))); + aefiClassificationUndetermined.updatePercentageValueWithCount( + Integer.parseInt(aefiClassificationCountMap.get(AefiClassification.UNDETERMINED).get("total")), + Integer.parseInt(aefiClassificationCountMap.get(AefiClassification.UNDETERMINED).get("percent"))); + aefiCountsByVaccine.update(dataProvider.getAefiCountsByVaccine()); vaccineDoseChart.update(dataProvider.getAefiByVaccineDoseChartData()); reactionsByGenderChart.update(dataProvider.getAefiEventsByGenderChartData()); @@ -206,7 +281,7 @@ private VerticalLayout createMapLayout(AefiDashboardMapComponent mapComponent) { private void setExpanded(Boolean expanded, Component componentToExpand, Component componentToRemove, int removedComponentIndex) { if (expanded) { - dashboardLayout.removeComponent(heading); + dashboardLayout.removeComponent(allAefiHeading); dashboardLayout.removeComponent(aefiCountsLayout); epiCurveAndMapLayout.removeComponent(componentToRemove); setHeight(100, Unit.PERCENTAGE); @@ -216,7 +291,7 @@ private void setExpanded(Boolean expanded, Component componentToExpand, Componen componentToExpand.setSizeFull(); dashboardLayout.setHeightFull(); } else { - dashboardLayout.addComponent(heading, 1); + dashboardLayout.addComponent(allAefiHeading, 1); dashboardLayout.addComponent(aefiCountsLayout, 2); epiCurveAndMapLayout.addComponent(componentToRemove, removedComponentIndex); mapComponent.refreshMap(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java index a3308e23824..7a08ff88730 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiCountTilesComponent.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -84,6 +81,8 @@ public AefiCountTilesComponent( titleLayout.addComponent(infoIcon); } + titleLayout.setVisible(false); + countsLayout = new HorizontalLayout(); countsLayout.setWidthFull(); addComponent(countsLayout); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java index 464872634f0..b44a29b613d 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -127,6 +124,11 @@ public void update(AefiChartData chartData) { " fontWeight: 'bold'" + " }" + " }," + + " labels: {" + + " formatter: function() {" + + " return Math.abs(this.value);" + + " }" + + " }" + " }," + " plotOptions: {" + " series: {" + @@ -135,7 +137,12 @@ public void update(AefiChartData chartData) { " }," + " credits: {" + " enabled: false" + - " },"); + " }," + + " tooltip: {" + + " formatter: function() {" + + " return this.point.category + ': ' + this.series.name + ' ' + Math.abs(this.y);" + + " }" + + " },"); //@formatter:on hcjs.append("series: ["); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java index 45e459ac760..aed7bba874e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiTypeStatisticsGroupComponent.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -39,10 +36,8 @@ public AefiTypeStatisticsGroupComponent() { public void update(Map> countsByVaccineData) { //temporary fix: re-use stored components and hide/unhide if not in new update - if (countsByVaccineData.isEmpty()) { - removeAllComponents(); - componentMap.clear(); - } + removeAllComponents(); + componentMap.clear(); AefiTypeStatisticsComponent statisticsComponent; @@ -54,7 +49,7 @@ public void update(Map> countsByVaccineData) { //statisticsComponent.getHeading().getTitleLabel().setValue(entry.getKey().toString()); statisticsComponent.hideHeading(); statisticsComponent.update(entry.getValue()); - statisticsComponent.addStyleNames(CssStyles.VIEW_SECTION, CssStyles.PADDING_X_8, CssStyles.HSPACE_RIGHT_3); + statisticsComponent.addStyleNames(CssStyles.VIEW_SECTION, CssStyles.PADDING_X_8, CssStyles.HSPACE_RIGHT_3, CssStyles.VSPACE_4); componentMap.put(entry.getKey(), statisticsComponent); addComponent(statisticsComponent); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AefiDownloadUtil.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AefiDownloadUtil.java new file mode 100644 index 00000000000..1f8174ec575 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AefiDownloadUtil.java @@ -0,0 +1,67 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.ui.utils; + +import java.util.Collection; +import java.util.Date; +import java.util.function.Supplier; + +import com.vaadin.server.StreamResource; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventsDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiCriteria; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiExportDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiIndexDto; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.importexport.ExportConfigurationDto; + +public class AefiDownloadUtil { + + private AefiDownloadUtil() { + } + + public static StreamResource createAefiExportResource( + AefiCriteria aefiCriteria, + Supplier> selectedRows, + ExportConfigurationDto exportConfiguration) { + return DownloadUtil.createCsvExportStreamResource( + AefiExportDto.class, + null, + (Integer start, Integer max) -> FacadeProvider.getAefiFacade().getExportList(aefiCriteria, selectedRows.get(), start, max), + AefiDownloadUtil::captionProvider, + ExportEntityName.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION, + exportConfiguration); + } + + public static String getPropertyCaption(String propertyId, String prefixId) { + if (prefixId != null) { + return I18nProperties.getPrefixCaption(prefixId, propertyId); + } + + return I18nProperties.findPrefixCaption(propertyId, AefiExportDto.I18N_PREFIX, AefiIndexDto.I18N_PREFIX, AdverseEventsDto.I18N_PREFIX); + } + + private static String captionProvider(String propertyId, Class type) { + String caption = getPropertyCaption(propertyId, null); + + if (Date.class.isAssignableFrom(type)) { + caption += " (" + DateFormatHelper.getDateFormatPattern() + ")"; + } + + return caption; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java index 5bfa1ac1249..ae367e005a7 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveHandlers.java @@ -32,6 +32,8 @@ import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiFacade; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationDto; +import de.symeda.sormas.api.adverseeventsfollowingimmunization.AefiInvestigationFacade; import de.symeda.sormas.api.campaign.CampaignDto; import de.symeda.sormas.api.campaign.CampaignFacade; import de.symeda.sormas.api.caze.CaseDataDto; @@ -99,6 +101,12 @@ public static CoreEntityArchiveHandler forAefi() { return new CoreEntityArchiveHandler<>(FacadeProvider.getAefiFacade(), ArchiveMessages.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION); } + public static CoreEntityArchiveHandler forAefiInvestigation() { + return new CoreEntityArchiveHandler<>( + FacadeProvider.getAefiInvestigationFacade(), + ArchiveMessages.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_INVESTIGATION); + } + public static CoreEntityArchiveHandler forTravelEntry() { return new CoreEntityArchiveHandler<>(FacadeProvider.getTravelEntryFacade(), ArchiveMessages.TRAVEL_ENTRY); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java index 079f167ae09..f7504095812 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ArchiveMessages.java @@ -106,14 +106,30 @@ public enum ArchiveMessages { null, null), - ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION(Strings.headingArchiveImmunization, - Strings.confirmationArchiveImmunization, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION(Strings.headingArchiveAdverseEvent, + Strings.confirmationArchiveAdverseEvent, null, - Strings.messageImmunizationArchived, - Strings.headingDearchiveImmunization, - Strings.confirmationDearchiveImmunization, + Strings.messageAdverseEventArchived, + Strings.headingDearchiveAdverseEvent, + Strings.confirmationDearchiveAdverseEvent, null, - Strings.messageImmunizationDearchived, + Strings.messageAdverseEventDearchived, + null, + null, + null, + null, + null, + null, + null, + null), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_INVESTIGATION(Strings.headingArchiveAdverseEventInvestigation, + Strings.confirmationArchiveAdverseEventInvestigation, + null, + Strings.messageAdverseEventInvestigationArchived, + Strings.headingDearchiveAdverseEventInvestigation, + Strings.confirmationDearchiveAdverseEventInvestigation, + null, + Strings.messageAdverseEventInvestigationDearchived, null, null, null, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java index 1c82e66f59f..a1bee68b7c0 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/CssStyles.java @@ -323,6 +323,8 @@ private CssStyles() { public static final String MARGIN_TOP_4 = "margin-top-4"; public static final String PADDING_X_8 = "padding-x-8"; public static final String PADDING_X_20 = "padding-x-20"; + public static final String FORM_SECTION_ACCORDION_PANEL_TITLE_BUTTON = "form-section-accordion-panel-title-button"; + public static final String OPTIONGROUP_CAPTION_FLEX = "optiongroup-caption-flex"; public static final String VIEW_SECTION_MARGIN_X_4 = VIEW_SECTION + " " + MARGIN_X_4; public static final String VIEW_SECTION_MARGIN_TOP_4_MARGIN_X_4 = VIEW_SECTION + " " + MARGIN_X_4 + " " + MARGIN_TOP_4; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java index fd9ad184630..ca2b6e7b088 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java @@ -43,6 +43,9 @@ public enum ExportEntityName { ENVIRONMENTS("entityEnvironments", "environments"), ENVIRONMENT_SAMPLES("entityEnvironmentSamples", "environment samples"), SELF_REPORTS("entitySelfReports", "self reports"); + ENVIRONMENT_SAMPLES("entityEnvironmentSamples", "environment samples"), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION("entityAdverseEvents", "adverse events following immunization"), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_INVESTIGATION("entityAdverseEventInvestigations", "aefi investigations"); private final String languageKey; private final String defaultName; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java index 6e0290496cd..22b80d05097 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java @@ -20,8 +20,6 @@ import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.adverseeventsfollowingimmunization.AdverseEventState; -import de.symeda.sormas.api.adverseeventsfollowingimmunization.SeizureType; -import de.symeda.sormas.api.caze.Trimester; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.symptoms.SymptomState; import de.symeda.sormas.api.utils.FieldConstraints; @@ -30,7 +28,6 @@ import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.ActivityAsCase.ActivityAsCaseField; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField; -import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField_2; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.form.AdverseEventsForm; import de.symeda.sormas.ui.clinicalcourse.HealthConditionsForm; import de.symeda.sormas.ui.exposure.ExposuresField; @@ -80,9 +77,7 @@ public T createField(Class type, Class fieldType) { if (fieldType.isAssignableFrom(Field.class) // no specific fieldType defined? && (SymptomState.class.isAssignableFrom(type) || YesNoUnknown.class.isAssignableFrom(type) - || AdverseEventState.class.isAssignableFrom(type) - || SeizureType.class.isAssignableFrom(type) - || Trimester.class.isAssignableFrom(type))) { + || AdverseEventState.class.isAssignableFrom(type))) { NullableOptionGroup field = new NullableOptionGroup(); field.setImmediate(true); populateWithEnumData(field, (Class) type); @@ -191,7 +186,7 @@ public T createField(Class type, Class fieldType) { return (T) new AefiVaccinationsField(fieldAccessCheckers); } else if (AdverseEventsForm.class.isAssignableFrom(fieldType)) { return (T) new AdverseEventsForm(fieldVisibilityCheckers, fieldAccessCheckers); - }else if (CheckBoxTree.class.isAssignableFrom(fieldType)){ + } else if (CheckBoxTree.class.isAssignableFrom(fieldType)) { return (T) new CheckBoxTree<>(); } return super.createField(type, fieldType); From 6e516b6d0f95c744f5c2b00bec917e57fe7acd76 Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Sat, 30 Mar 2024 19:56:12 +0100 Subject: [PATCH 06/56] #12976 - AEFI dashboard - Change "gender" to "sex" on the dashboard --- .../AefiDashboardView.java | 10 +++++----- ...ByGenderChart.java => AefiReactionsBySexChart.java} | 6 +++--- 2 files changed, 8 insertions(+), 8 deletions(-) rename sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/{AefiReactionsByGenderChart.java => AefiReactionsBySexChart.java} (96%) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java index b114177e007..f21df5a8aaf 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/AefiDashboardView.java @@ -39,7 +39,7 @@ import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiByVaccineDoseChart; import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiCountTilesComponent; import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiDashboardMapComponent; -import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiReactionsByGenderChart; +import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiReactionsBySexChart; import de.symeda.sormas.ui.dashboard.adverseeventsfollowingimmunization.components.AefiTypeStatisticsGroupComponent; import de.symeda.sormas.ui.dashboard.components.DashboardHeadingComponent; import de.symeda.sormas.ui.dashboard.statistics.DashboardStatisticsPercentageElement; @@ -77,7 +77,7 @@ public class AefiDashboardView extends AbstractDashboardView { private DashboardStatisticsPercentageElement aefiClassificationUndetermined; private final AefiTypeStatisticsGroupComponent aefiCountsByVaccine; private final AefiByVaccineDoseChart vaccineDoseChart; - private final AefiReactionsByGenderChart reactionsByGenderChart; + private final AefiReactionsBySexChart reactionsBySexChart; private final AefiEpiCurveComponent epiCurveComponent; private final AefiDashboardMapComponent mapComponent; @@ -175,8 +175,8 @@ public AefiDashboardView() { vaccineDoseChart = new AefiByVaccineDoseChart(); aefiCountsLayout.addComponent(vaccineDoseChart, VACCINE_DOSE_LOC); - reactionsByGenderChart = new AefiReactionsByGenderChart(); - aefiCountsLayout.addComponent(reactionsByGenderChart, REACTIONS_LOC); + reactionsBySexChart = new AefiReactionsBySexChart(); + aefiCountsLayout.addComponent(reactionsBySexChart, REACTIONS_LOC); epiCurveComponent = new AefiEpiCurveComponent(dataProvider); epiCurveLayout = createEpiCurveLayout(); @@ -220,7 +220,7 @@ public void refreshDashboard() { aefiCountsByVaccine.update(dataProvider.getAefiCountsByVaccine()); vaccineDoseChart.update(dataProvider.getAefiByVaccineDoseChartData()); - reactionsByGenderChart.update(dataProvider.getAefiEventsByGenderChartData()); + reactionsBySexChart.update(dataProvider.getAefiEventsByGenderChartData()); epiCurveComponent.clearAndFillEpiCurveChart(); mapComponent.refreshMap(); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsBySexChart.java similarity index 96% rename from sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java rename to sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsBySexChart.java index b44a29b613d..8b923778206 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsByGenderChart.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/dashboard/adverseeventsfollowingimmunization/components/AefiReactionsBySexChart.java @@ -27,11 +27,11 @@ import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.ui.highcharts.HighChart; -public class AefiReactionsByGenderChart extends VerticalLayout { +public class AefiReactionsBySexChart extends VerticalLayout { protected final HighChart chart; - public AefiReactionsByGenderChart() { + public AefiReactionsBySexChart() { setMargin(false); setSpacing(false); @@ -56,7 +56,7 @@ public void update(AefiChartData chartData) { " borderRadius: '8px'" + " }," + " title: {" + - " text: 'Proportion of AEFI reactions (events) by gender'," + + " text: 'Proportion of AEFI reactions (events) by sex'," + " align: 'left'," + " style: {" + " fontSize: '15px'," + From 6efd99da13f749a421bf4c008212accb86de75aa Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Sun, 14 Jul 2024 19:35:51 +0100 Subject: [PATCH 07/56] #12774 - aefi export entity configuration --- .../main/java/de/symeda/sormas/ui/utils/ExportEntityName.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java index ca2b6e7b088..e00f41dfe45 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/ExportEntityName.java @@ -42,8 +42,7 @@ public enum ExportEntityName { PERSONS("entityPersons", "persons"), ENVIRONMENTS("entityEnvironments", "environments"), ENVIRONMENT_SAMPLES("entityEnvironmentSamples", "environment samples"), - SELF_REPORTS("entitySelfReports", "self reports"); - ENVIRONMENT_SAMPLES("entityEnvironmentSamples", "environment samples"), + SELF_REPORTS("entitySelfReports", "self reports"), ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION("entityAdverseEvents", "adverse events following immunization"), ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_INVESTIGATION("entityAdverseEventInvestigations", "aefi investigations"); From b7ea5a05a83361a0a0c1e66314742fbf8ec832eb Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Wed, 16 Oct 2024 16:28:15 +0300 Subject: [PATCH 08/56] #13159 Automatically (Soft-)Delete Samples & Pathogen Tests with Negative Test Results for COVID-19 --- .../api/sample/PathogenTestCriteria.java | 5 + .../sormas/backend/common/CronService.java | 2 +- .../sormas/backend/sample/PathogenTest.java | 1 - .../sormas/backend/sample/SampleService.java | 45 +-- .../DeleteOldPathogenTestsAndSamplesTest.java | 264 ++++++++++++++++++ .../backend/sample/SampleServiceTest.java | 32 --- 6 files changed, 294 insertions(+), 55 deletions(-) create mode 100644 sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestCriteria.java index a9f9a1ac789..fad0fb4236f 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestCriteria.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestCriteria.java @@ -21,4 +21,9 @@ public SampleReferenceDto getSample() { public void setSample(SampleReferenceDto sample) { this.sample = sample; } + + public PathogenTestCriteria sample(SampleReferenceDto sample) { + this.sample = sample; + return this; + } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java index 6bb6d7dda83..dd443441e71 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/CronService.java @@ -303,6 +303,6 @@ public void syncUsersFromAuthenticationProvider() { @Schedule(hour = "2", minute = "40", persistent = false) public void sofDeleteOldNegativeSamples() { - sampleService.deleteOldNegativeSamples(); + sampleService.cleanupOldCovidSamples(); } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java index 52beec65f52..79aeef5cffc 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java @@ -248,7 +248,6 @@ public void setTestTypeText(String testTypeText) { } @Temporal(TemporalType.TIMESTAMP) - @Column(nullable = false) public Date getTestDateTime() { return testDateTime; } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java index 95e58b65ed8..8bb70bb1994 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java @@ -72,6 +72,7 @@ import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.sample.IsSample; +import de.symeda.sormas.api.sample.PathogenTestCriteria; import de.symeda.sormas.api.sample.PathogenTestResultType; import de.symeda.sormas.api.sample.PathogenTestType; import de.symeda.sormas.api.sample.SampleAssociationType; @@ -1284,34 +1285,36 @@ public List getAssociatedDiseaseVariants(String sampleUuid) { return em.createQuery(cq).getResultList(); } - public void deleteOldNegativeSamples() { + public void cleanupOldCovidSamples() { final Integer maxAgeDays = configFacade.getNegaiveCovidSamplesMaxAgeDays(); if (maxAgeDays == null) { return; } CriteriaBuilder cb = em.getCriteriaBuilder(); - CriteriaQuery cq = cb.createQuery(Sample.class); - Root from = cq.from(Sample.class); - SampleJoins sampleJoins = new SampleJoins(from); - - Subquery covidTestQuery = cq.subquery(PathogenTest.class); - Root covidTestRoot = covidTestQuery.from(PathogenTest.class); - covidTestQuery.where( - cb.equal(covidTestRoot.get(PathogenTest.SAMPLE), from), - cb.equal(covidTestRoot.get(PathogenTest.TESTED_DISEASE), Disease.CORONAVIRUS)); + CriteriaQuery cq = cb.createQuery(PathogenTest.class); + Root from = cq.from(PathogenTest.class); cq.where( - cb.or( - cb.equal(sampleJoins.getCaze().get(Case.DISEASE), Disease.CORONAVIRUS), - cb.equal(sampleJoins.getContact().get(Contact.DISEASE), Disease.CORONAVIRUS), - cb.equal(sampleJoins.getContactCase().get(Case.DISEASE), Disease.CORONAVIRUS), - cb.equal(sampleJoins.getEventParticipantJoins().getEvent().get(Event.DISEASE), Disease.CORONAVIRUS)), - cb.exists(covidTestQuery), - cb.equal(from.get(Sample.PATHOGEN_TEST_RESULT), PathogenTestResultType.NEGATIVE), - cb.greaterThan(from.get(Sample.SAMPLE_DATE_TIME), DateHelper.subtractDays(new Date(), maxAgeDays))); - em.createQuery(cq) - .getResultList() - .forEach(s -> delete(s, new DeletionDetails(DeletionReason.OTHER_REASON, I18nProperties.getString(Strings.entityAutomaticSoftDeletion)))); + cb.equal(from.get(PathogenTest.TESTED_DISEASE), Disease.CORONAVIRUS), + cb.equal(from.get(PathogenTest.TEST_RESULT), PathogenTestResultType.NEGATIVE), + cb.notEqual(from.get(PathogenTest.DELETED), true), + cb.lessThan( + CriteriaBuilderHelper.coalesce( + cb, + Date.class, + from.get(PathogenTest.TEST_DATE_TIME), + from.get(PathogenTest.REPORT_DATE), + from.get(PathogenTest.CREATION_DATE)), + DateHelper.subtractDays(new Date(), maxAgeDays))); + em.createQuery(cq).getResultList().stream().collect(Collectors.groupingBy(PathogenTest::getSample)).forEach((sample, tests) -> { + if (pathogenTestService.count(new PathogenTestCriteria().sample(sample.toReference())) == tests.size()) { + delete(sample, new DeletionDetails(DeletionReason.OTHER_REASON, I18nProperties.getString(Strings.entityAutomaticSoftDeletion))); + } else { + tests.forEach( + p -> pathogenTestService + .delete(p, new DeletionDetails(DeletionReason.OTHER_REASON, I18nProperties.getString(Strings.entityAutomaticSoftDeletion)))); + } + }); } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java new file mode 100644 index 00000000000..a5d70e9b207 --- /dev/null +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java @@ -0,0 +1,264 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.sample; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +import java.util.Date; +import java.util.List; + +import org.junit.jupiter.api.Test; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.caze.CaseDataDto; +import de.symeda.sormas.api.common.DeletionDetails; +import de.symeda.sormas.api.common.DeletionReason; +import de.symeda.sormas.api.sample.PathogenTestDto; +import de.symeda.sormas.api.sample.PathogenTestResultType; +import de.symeda.sormas.api.sample.PathogenTestType; +import de.symeda.sormas.api.sample.SampleDto; +import de.symeda.sormas.api.user.UserReferenceDto; +import de.symeda.sormas.api.utils.DateHelper; +import de.symeda.sormas.backend.AbstractBeanTest; +import de.symeda.sormas.backend.MockProducer; +import de.symeda.sormas.backend.TestDataCreator; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; + +public class DeleteOldPathogenTestsAndSamplesTest extends AbstractBeanTest { + + private TestDataCreator.RDCF rdcf; + private UserReferenceDto user; + private CaseDataDto caze; + + final int negativeCovidTestMaxAge = 15; + + @Override + public void init() { + super.init(); + + rdcf = creator.createRDCF(); + user = creator.createUser(rdcf).toReference(); + caze = creator.createCase(user, creator.createPerson().toReference(), rdcf); + + MockProducer.getProperties().setProperty(ConfigFacadeEjb.NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS, String.valueOf(negativeCovidTestMaxAge)); + } + + @Test + public void testDeleteOldTestsOnly() { + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + PathogenTestDto oldTest = creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + PathogenTestDto newTest = creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.RAPID_TEST); + s.setTestDateTime(new Date()); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().cleanupOldCovidSamples(); + + List sampleTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(sampleTests.size(), is(1)); + assertThat(sampleTests.get(0).getUuid(), is(newTest.getUuid())); + assertThat(getPathogenTestFacade().getByUuid(oldTest.getUuid()).isDeleted(), is(true)); + } + + @Test + public void testDeleteOnlyNegativeTests() { + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setPathogenTestResult(PathogenTestResultType.POSITIVE); + }); + PathogenTestDto positiveTest = creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.POSITIVE); + }); + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().cleanupOldCovidSamples(); + + List sampleTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(sampleTests.size(), is(1)); + assertThat(sampleTests.get(0).getUuid(), is(positiveTest.getUuid())); + } + + @Test + public void testDeleteSampleIfEmptied() { + + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setSampleDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 5)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().cleanupOldCovidSamples(); + + List sampleTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(sampleTests.size(), is(0)); + assertThat(getSampleFacade().getSampleByUuid(sample.getUuid()).isDeleted(), is(true)); + } + + @Test + public void testDeleteSampleWithOldAndDeletedTests() { + + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setSampleDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 5)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + PathogenTestDto deletedTest = creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + getPathogenTestFacade().deletePathogenTest(deletedTest.getUuid(), new DeletionDetails(DeletionReason.DELETION_REQUEST, null)); + + getSampleService().cleanupOldCovidSamples(); + + List sampleTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(sampleTests.size(), is(0)); + assertThat(getSampleFacade().getSampleByUuid(sample.getUuid()).isDeleted(), is(true)); + } + + @Test + public void testCovidTestsOnly() { + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + PathogenTestDto otherDiseaseTest = creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.DENGUE); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().cleanupOldCovidSamples(); + + List sampleTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(sampleTests.size(), is(1)); + assertThat(sampleTests.get(0).getUuid(), is(otherDiseaseTest.getUuid())); + } + + @Test + public void testDeletionReferenceDate() { + + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + // old by creation date + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(null); + s.setCreationDate(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + // old by result date + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setCreationDate(new Date()); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + // old by reporting date + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setCreationDate(new Date()); + s.setTestDateTime(null); + s.setReportDate(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().cleanupOldCovidSamples(); + + List pathogenTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(getSampleFacade().getSampleByUuid(sample.getUuid()).isDeleted(), is(true)); + } + + @Test + public void testNotConfigured() { + MockProducer.getProperties().remove(ConfigFacadeEjb.NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS); + + SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { + s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); + }); + creator.createPathogenTest(sample.toReference(), user, s -> { + s.setLab(rdcf.facility); + s.setTestedDisease(Disease.CORONAVIRUS); + s.setTestType(PathogenTestType.PCR_RT_PCR); + s.setTestDateTime(DateHelper.subtractDays(new Date(), negativeCovidTestMaxAge + 1)); + s.setTestResult(PathogenTestResultType.NEGATIVE); + }); + + getSampleService().cleanupOldCovidSamples(); + + assertThat(getSampleFacade().getSampleByUuid(sample.getUuid()).isDeleted(), is(false)); + List pathogenTests = getPathogenTestFacade().getAllBySample(sample.toReference()); + assertThat(pathogenTests.size(), is(1)); + } +} diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java index 942585a5c7a..27cba075c46 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/SampleServiceTest.java @@ -1,13 +1,10 @@ package de.symeda.sormas.backend.sample; -import static org.hamcrest.MatcherAssert.assertThat; -import static org.hamcrest.Matchers.is; import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertNotNull; import static org.junit.jupiter.api.Assertions.assertNull; import static org.junit.jupiter.api.Assertions.assertTrue; -import java.util.Date; import java.util.List; import org.junit.jupiter.api.Test; @@ -16,16 +13,10 @@ import de.symeda.sormas.api.common.DeletionDetails; import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.externalmessage.ExternalMessageDto; -import de.symeda.sormas.api.i18n.I18nProperties; -import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.person.PersonDto; -import de.symeda.sormas.api.sample.PathogenTestResultType; -import de.symeda.sormas.api.sample.SampleCriteria; import de.symeda.sormas.api.sample.SampleDto; import de.symeda.sormas.api.user.DefaultUserRole; import de.symeda.sormas.api.user.UserDto; -import de.symeda.sormas.api.user.UserReferenceDto; -import de.symeda.sormas.api.utils.DateHelper; import de.symeda.sormas.backend.AbstractBeanTest; import de.symeda.sormas.backend.TestDataCreator; @@ -73,27 +64,4 @@ public void testSamplePermanentDeletion() { assertNull(getSampleService().getByUuid(referralSample.getUuid()).getReferredTo()); assertNull(getSampleReportService().getByUuid(labMessage.getSampleReports().get(0).getUuid()).getSample()); } - - @Test - public void testDeleteOldNegativeSamples() { - TestDataCreator.RDCF rdcf = creator.createRDCF(); - UserReferenceDto user = creator.createUser(rdcf).toReference(); - CaseDataDto caze = creator.createCase(user, creator.createPerson().toReference(), rdcf); - SampleDto newSample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { - s.setSampleDateTime(new Date()); - s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); - }); - SampleDto oldSample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { - s.setSampleDateTime(DateHelper.subtractDays(new Date(), 14)); - s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); - }); - - getSampleService().deleteOldNegativeSamples(); - - assertThat(getSampleFacade().count(new SampleCriteria()), is(1L)); - SampleDto deletedSample = getSampleFacade().getSampleByUuid(newSample.getUuid()); - assertThat(deletedSample.isDeleted(), is(true)); - assertThat(deletedSample.getDeletionReason(), is(DeletionReason.OTHER_REASON)); - assertThat(deletedSample.getOtherDeletionReason(), is(I18nProperties.getString(Strings.entityAutomaticSoftDeletion))); - } } From fb42dc583af3733a68d8fc4e84385a2d2cfd5544 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Mon, 21 Oct 2024 12:40:28 +0300 Subject: [PATCH 09/56] #13160 Add "Disease" Attribute to Document Templates for Filtering - WIP --- .../docgeneneration/DocumentTemplateDto.java | 67 ++++++++++ .../DocumentTemplateFacade.java | 21 +-- .../DocumentTemplateReferenceDto.java | 25 ++++ .../docgeneneration/EventDocumentFacade.java | 10 +- .../QuarantineOrderDocumentOptionsDto.java | 17 ++- .../QuarantineOrderFacade.java | 20 ++- .../externalemail/ExternalEmailFacade.java | 3 +- .../ExternalEmailOptionsDto.java | 11 +- .../sormas/backend/action/ActionService.java | 3 +- .../sormas/backend/caze/CaseService.java | 3 +- .../sormas/backend/common/AdoService.java | 5 +- .../sormas/backend/common/BaseAdoService.java | 3 +- .../backend/contact/ContactService.java | 3 +- .../docgeneration/DocGenerationHelper.java | 5 +- .../docgeneration/DocumentTemplate.java | 66 ++++++++++ .../DocumentTemplateFacadeEjb.java | 121 +++++++++++------- .../DocumentTemplateService.java | 66 ++++++++++ .../docgeneration/EventDocumentFacadeEjb.java | 18 +-- .../QuarantineOrderFacadeEjb.java | 35 ++--- .../event/EventParticipantService.java | 3 +- .../sormas/backend/event/EventService.java | 3 +- .../externalemail/ExternalEmailFacadeEjb.java | 19 ++- .../ExternalMessageService.java | 3 +- .../labmessage/SampleReportService.java | 10 +- .../immunization/ImmunizationService.java | 7 +- .../sormas/backend/person/PersonService.java | 3 +- .../sormas/backend/sample/SampleService.java | 3 +- .../services/TravelEntryService.java | 3 +- .../sormas/backend/user/UserRoleService.java | 3 +- .../DocumentTemplateFacadeEjbTest.java | 2 +- .../EventDocumentFacadeEjbTest.java | 2 +- .../QuarantineOrderFacadeEjbTest.java | 15 +-- .../ExternalEmailFacadeEjbTest.java | 18 +-- .../docgeneration/DocumentTemplatesGrid.java | 39 +++--- .../AbstractDocgenerationLayout.java | 20 +-- .../DocGenerationController.java | 40 +++--- .../ui/docgeneration/EventDocumentLayout.java | 7 +- .../docgeneration/QuarantineOrderLayout.java | 14 +- 38 files changed, 494 insertions(+), 222 deletions(-) create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateReferenceDto.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java new file mode 100644 index 00000000000..cc6d26988d8 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java @@ -0,0 +1,67 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.docgeneneration; + +import javax.validation.constraints.NotNull; +import javax.validation.constraints.Size; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.EntityDto; +import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.utils.FieldConstraints; + +public class DocumentTemplateDto extends EntityDto { + + private static final long serialVersionUID = 8591649635400893169L; + + public static final String I18N_PREFIX = "DocumentTemplate"; + public static final String DISEASE = "disease"; + public static final String FILE_NAME = "fileName"; + + @NotNull + private DocumentWorkflow workflow; + private Disease disease; + @Size(max = FieldConstraints.CHARACTER_LIMIT_BIG, message = Validations.textTooLong) + private String fileName; + + public DocumentWorkflow getWorkflow() { + return workflow; + } + + public void setWorkflow(DocumentWorkflow workflow) { + this.workflow = workflow; + } + + public Disease getDisease() { + return disease; + } + + public void setDisease(Disease disease) { + this.disease = disease; + } + + public String getFileName() { + return fileName; + } + + public void setFileName(String fileName) { + this.fileName = fileName; + } + + public DocumentTemplateReferenceDto toReference() { + return new DocumentTemplateReferenceDto(getUuid(), fileName); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java index ed431525eb3..7af68b34888 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java @@ -5,32 +5,33 @@ import javax.ejb.Remote; +import de.symeda.sormas.api.Disease; + @Remote public interface DocumentTemplateFacade { byte[] generateDocumentDocxFromEntities( - DocumentWorkflow documentWorkflow, - String templateName, + DocumentTemplateReferenceDto templateReference, DocumentTemplateEntities entities, Properties extraProperties) throws DocumentTemplateException; String generateDocumentTxtFromEntities( - DocumentWorkflow documentWorkflow, - String templateName, + DocumentTemplateReferenceDto templateReference, DocumentTemplateEntities entities, Properties extraProperties) throws DocumentTemplateException; - List getAvailableTemplates(DocumentWorkflow documentWorkflow); + List getAvailableTemplates(DocumentWorkflow documentWorkflow, Disease disease); - DocumentVariables getDocumentVariables(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException; + DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; - boolean isExistingTemplate(DocumentWorkflow documentWorkflow, String templateName); + boolean isExistingTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease); - void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName, byte[] document) throws DocumentTemplateException; + void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease, byte[] document) + throws DocumentTemplateException; - boolean deleteDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException; + boolean deleteDocumentTemplate(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; - byte[] getDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException; + byte[] getDocumentTemplateContent(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateReferenceDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateReferenceDto.java new file mode 100644 index 00000000000..f3ae8bdbe52 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateReferenceDto.java @@ -0,0 +1,25 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.api.docgeneneration; + +import de.symeda.sormas.api.ReferenceDto; + +public class DocumentTemplateReferenceDto extends ReferenceDto { + + public DocumentTemplateReferenceDto(String uuid, String caption) { + super(uuid, caption); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java index 3d323c795f8..4418aa69b45 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java @@ -27,17 +27,21 @@ @Remote public interface EventDocumentFacade { - String getGeneratedDocument(String templateName, EventReferenceDto eventReference, Properties extraProperties, Boolean shouldUploadGeneratedDoc) + String getGeneratedDocument( + DocumentTemplateReferenceDto templateReferenceDto, + EventReferenceDto eventReference, + Properties extraProperties, + Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; Map getGeneratedDocuments( - String templateName, + DocumentTemplateReferenceDto templateReference, List eventReferences, Properties extraProperties, Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; - List getAvailableTemplates(); + List getAvailableTemplates(); DocumentVariables getDocumentVariables(String templateName) throws DocumentTemplateException; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderDocumentOptionsDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderDocumentOptionsDto.java index dd7e406c78a..6e17f81dbcf 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderDocumentOptionsDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderDocumentOptionsDto.java @@ -1,16 +1,15 @@ package de.symeda.sormas.api.docgeneneration; +import java.io.Serializable; +import java.util.Properties; + import de.symeda.sormas.api.sample.PathogenTestReferenceDto; import de.symeda.sormas.api.sample.SampleReferenceDto; import de.symeda.sormas.api.vaccination.VaccinationReferenceDto; -import javax.validation.constraints.NotNull; -import java.io.Serializable; -import java.util.Properties; - public class QuarantineOrderDocumentOptionsDto implements Serializable { - private String templateFile; + private DocumentTemplateReferenceDto template; private SampleReferenceDto sample; private PathogenTestReferenceDto pathogenTest; private VaccinationReferenceDto vaccinationReference; @@ -19,12 +18,12 @@ public class QuarantineOrderDocumentOptionsDto implements Serializable { private DocumentWorkflow documentWorkflow; - public String getTemplateFile() { - return templateFile; + public DocumentTemplateReferenceDto getTemplate() { + return template; } - public void setTemplateFile(String templateFile) { - this.templateFile = templateFile; + public void setTemplate(DocumentTemplateReferenceDto template) { + this.template = template; } public SampleReferenceDto getSample() { diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java index dfda902b11d..0e24f8fb88b 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java @@ -32,8 +32,7 @@ public interface QuarantineOrderFacade { byte[] getGeneratedDocument( - String templateName, - DocumentWorkflow workflow, + DocumentTemplateReferenceDto templateReference, RootEntityType rootEntityType, ReferenceDto rootEntityReference, SampleReferenceDto sampleReference, @@ -44,22 +43,21 @@ byte[] getGeneratedDocument( throws DocumentTemplateException; Map getGeneratedDocuments( - String templateName, - DocumentWorkflow workflow, + DocumentTemplateReferenceDto templateReference, List rootEntityReferences, Properties extraProperties, Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; Map getGeneratedDocumentsForEventParticipants( - String templateName, - List rootEntityReferences, - Disease eventDisease, - Properties extraProperties, - Boolean shouldUploadGeneratedDoc) - throws DocumentTemplateException; + DocumentTemplateReferenceDto templateReference, + List rootEntityReferences, + Disease eventDisease, + Properties extraProperties, + Boolean shouldUploadGeneratedDoc) + throws DocumentTemplateException; - List getAvailableTemplates(DocumentWorkflow workflow); + List getAvailableTemplates(DocumentWorkflow workflow); DocumentVariables getDocumentVariables(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java index 127e707212d..725235c028f 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java @@ -24,6 +24,7 @@ import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.common.progress.ProcessedEntity; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.document.DocumentReferenceDto; @@ -33,7 +34,7 @@ @Remote public interface ExternalEmailFacade { - List getTemplateNames(DocumentWorkflow documentWorkflow); + List getTemplateNames(DocumentWorkflow documentWorkflow); List getAttachableDocuments(DocumentWorkflow documentWorkflow, String relatedEntityUuid); diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java index 64d40170261..de1179f2569 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java @@ -23,6 +23,7 @@ import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.audit.AuditedClass; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.QuarantineOrderDocumentOptionsDto; import de.symeda.sormas.api.docgeneneration.RootEntityType; @@ -48,7 +49,7 @@ public class ExternalEmailOptionsDto implements Serializable { private ReferenceDto rootEntityReference; @NotNull(message = Validations.requiredField) @Size(min = 1, message = Validations.requiredField) - private String templateName; + private DocumentTemplateReferenceDto template; @NotNull(message = Validations.requiredField) @Size(min = 1, message = Validations.requiredField) private String recipientEmail; @@ -78,12 +79,12 @@ public void setRootEntityReference(ReferenceDto rootEntityReference) { this.rootEntityReference = rootEntityReference; } - public String getTemplateName() { - return templateName; + public DocumentTemplateReferenceDto getTemplate() { + return template; } - public void setTemplateName(String templateName) { - this.templateName = templateName; + public void setTemplate(DocumentTemplateReferenceDto template) { + this.template = template; } public String getRecipientEmail() { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/action/ActionService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/action/ActionService.java index 12698b51144..d0deeea0f54 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/action/ActionService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/action/ActionService.java @@ -612,11 +612,12 @@ public long countActions(ActionCriteria actionCriteria) { } @Override - public void deletePermanent(Action action) { + public boolean deletePermanent(Action action) { documentService.getRelatedToEntity(DocumentRelatedEntityType.ACTION, action.getUuid()).forEach(d -> documentService.markAsDeleted(d)); super.deletePermanent(action); + return false; } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java index 4d4d35ab8aa..fd359454347 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java @@ -1033,7 +1033,7 @@ public Predicate createDefaultFilter(CriteriaBuilder cb, From root) { } @Override - public void deletePermanent(Case caze) { + public boolean deletePermanent(Case caze) { // Delete all tasks associated with this case Optional.ofNullable(caze.getTasks()).ifPresent(tl -> tl.forEach(t -> taskService.deletePermanent(t))); @@ -1111,6 +1111,7 @@ public void deletePermanent(Case caze) { deleteCaseLinks(caze); super.deletePermanent(caze); + return false; } private void initVisitSubqueryForDeletion(CriteriaBuilder cb, Root visitRoot, Subquery contactVisitsSubquery) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AdoService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AdoService.java index 111cd6abb80..a625f731622 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AdoService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AdoService.java @@ -34,10 +34,11 @@ public interface AdoService { /** * DELETES an entity from the database! - * + * * @param ado + * @return */ - void deletePermanent(ADO ado); + boolean deletePermanent(ADO ado); /** * Speichert ein neues Objekt in der Datenbank. diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/BaseAdoService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/BaseAdoService.java index f1fb15356d4..1b022714d67 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/BaseAdoService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/BaseAdoService.java @@ -517,9 +517,10 @@ public void persist(ADO persistme) { } @Override - public void deletePermanent(ADO ado) { + public boolean deletePermanent(ADO ado) { em.remove(em.contains(ado) ? ado : em.merge(ado)); // todo: investigate why the entity might be detached (example: AdditionalTest) em.flush(); + return false; } @TransactionAttribute(TransactionAttributeType.REQUIRES_NEW) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java index 3c8f1cd2b22..07b51eb8efa 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java @@ -1722,7 +1722,7 @@ public void restore(Contact contact) { } @Override - public void deletePermanent(Contact contact) { + public boolean deletePermanent(Contact contact) { // Delete all tasks associated with this case Optional.ofNullable(contact.getTasks()).ifPresent(tl -> tl.forEach(t -> taskService.deletePermanent(t))); @@ -1763,6 +1763,7 @@ public void deletePermanent(Contact contact) { deleteContactLinks(contact); super.deletePermanent(contact); + return false; } private void deleteContactLinks(Contact contact) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocGenerationHelper.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocGenerationHelper.java index bd1ba299331..94b7b993860 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocGenerationHelper.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocGenerationHelper.java @@ -13,6 +13,7 @@ import de.symeda.sormas.api.action.ActionReferenceDto; import de.symeda.sormas.api.caze.CaseReferenceDto; import de.symeda.sormas.api.contact.ContactReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.document.DocumentDto; import de.symeda.sormas.api.document.DocumentRelatedEntityDto; import de.symeda.sormas.api.document.DocumentRelatedEntityType; @@ -45,13 +46,13 @@ public DocumentRelatedEntityType getDocumentRelatedEntityType(ReferenceDto rootE } } - public String getDocumentFileName(ReferenceDto rootEntityReference, String templateFileName) { + public String getDocumentFileName(ReferenceDto rootEntityReference, DocumentTemplateReferenceDto templateReference) { List docs = documentFacade.getDocumentsRelatedToEntity(getDocumentRelatedEntityType(rootEntityReference), rootEntityReference.getUuid()); return generateNewFileName( docs.stream().map(DocumentDto::getName).collect(Collectors.toList()), DataHelper.getShortUuid(rootEntityReference), - '-' + templateFileName); + '-' + templateReference.getCaption()); } private String generateNewFileName(List docs, String shortUuid, String templateFileName) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java new file mode 100644 index 00000000000..8aa96440e36 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java @@ -0,0 +1,66 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.docgeneration; + +import javax.persistence.Column; +import javax.persistence.Entity; +import javax.persistence.EnumType; +import javax.persistence.Enumerated; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; +import de.symeda.sormas.api.utils.FieldConstraints; +import de.symeda.sormas.backend.common.AbstractDomainObject; + +@Entity +public class DocumentTemplate extends AbstractDomainObject { + + public static final String WORKFLOW = "workflow"; + public static final String DISEASE = "disease"; + public static final String DOCUMENT_PATH = "documentPath"; + + private DocumentWorkflow workflow; + private Disease disease; + private String fileName; + + @Enumerated(EnumType.STRING) + @Column(nullable = false) + public DocumentWorkflow getWorkflow() { + return workflow; + } + + public void setWorkflow(DocumentWorkflow workflow) { + this.workflow = workflow; + } + + @Enumerated(EnumType.STRING) + public Disease getDisease() { + return disease; + } + + public void setDisease(Disease disease) { + this.disease = disease; + } + + @Column(length = FieldConstraints.CHARACTER_LIMIT_BIG) + public String getFileName() { + return fileName; + } + + public void setFileName(String fileName) { + this.fileName = fileName; + } +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java index ae62fc5870c..4cc5bb6df32 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java @@ -23,8 +23,6 @@ import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; -import java.util.Arrays; -import java.util.Collections; import java.util.List; import java.util.Properties; import java.util.Set; @@ -32,6 +30,7 @@ import java.util.regex.Pattern; import java.util.stream.Collectors; +import javax.annotation.Nullable; import javax.annotation.security.PermitAll; import javax.ejb.EJB; import javax.ejb.LocalBean; @@ -41,13 +40,16 @@ import org.apache.commons.io.FilenameUtils; import org.apache.commons.lang3.StringUtils; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.EntityDtoAccessHelper; import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.caze.CaseReferenceDto; import de.symeda.sormas.api.contact.ContactReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateEntities; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentTemplateFacade; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.DocumentWorkflowType; @@ -85,6 +87,7 @@ import de.symeda.sormas.backend.travelentry.TravelEntryFacadeEjb.TravelEntryFacadeEjbLocal; import de.symeda.sormas.backend.user.UserFacadeEjb.UserFacadeEjbLocal; import de.symeda.sormas.backend.user.UserService; +import de.symeda.sormas.backend.util.DtoHelper; import de.symeda.sormas.backend.util.RightsAllowed; @Stateless(name = "DocumentTemplateFacade") @@ -94,6 +97,9 @@ public class DocumentTemplateFacadeEjb implements DocumentTemplateFacade { public static final int EMAIL_SUBJECT_MAX_LENGTH = 50; public static final String EMAIL_TEMPLATE_SUBJECT_PREFIX = "#"; + @EJB + private DocumentTemplateService documentTemplateService; + @EJB private ConfigFacadeEjbLocal configFacade; @@ -144,18 +150,20 @@ public class DocumentTemplateFacadeEjb implements DocumentTemplateFacade { @Override @PermitAll public byte[] generateDocumentDocxFromEntities( - DocumentWorkflow documentWorkflow, - String templateName, + DocumentTemplateReferenceDto templateReference, DocumentTemplateEntities entities, Properties extraProperties) throws DocumentTemplateException { + DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); + DocumentWorkflow documentWorkflow = template.getWorkflow(); + if (!documentWorkflow.isDocx()) { throw new DocumentTemplateException( String.format(I18nProperties.getString(Strings.messageWrongTemplateFileType), documentWorkflow, documentWorkflow.getFileExtension())); } // 1. Read template from custom directory - File templateFile = getTemplateFile(documentWorkflow, templateName); + File templateFile = getTemplateFile(template); // 2. Extract document variables DocumentVariables documentVariables = getTemplateVariablesDocx(templateFile); @@ -170,24 +178,27 @@ public byte[] generateDocumentDocxFromEntities( @Override @PermitAll public String generateDocumentTxtFromEntities( - DocumentWorkflow documentWorkflow, - String templateName, + DocumentTemplateReferenceDto templateReference, DocumentTemplateEntities entities, Properties extraProperties) throws DocumentTemplateException { - if (documentWorkflow.isDocx()) { + DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); + if (template.getWorkflow().isDocx()) { throw new DocumentTemplateException( - String.format(I18nProperties.getString(Strings.messageWrongTemplateFileType), documentWorkflow, documentWorkflow.getFileExtension())); + String.format( + I18nProperties.getString(Strings.messageWrongTemplateFileType), + template.getWorkflow(), + template.getWorkflow().getFileExtension())); } // 1. Read template from custom directory - File templateFile = getTemplateFile(documentWorkflow, templateName); + File templateFile = getTemplateFile(template); // 2. Extract document variables DocumentVariables documentVariables = getTemplateVariablesTxt(templateFile); // 3. prepare properties - Properties properties = prepareProperties(documentWorkflow, entities, extraProperties, documentVariables); + Properties properties = prepareProperties(template.getWorkflow(), entities, extraProperties, documentVariables); // 4. generate document return generateDocumentTxt(templateFile, properties); @@ -283,34 +294,33 @@ private String generateDocumentTxt(File templateFile, Properties properties) { @Override @PermitAll - public List getAvailableTemplates(DocumentWorkflow documentWorkflow) { - File workflowTemplateDir = new File(getWorkflowTemplateDirPath(documentWorkflow).toUri()); - if (!workflowTemplateDir.exists() || !workflowTemplateDir.isDirectory()) { - return Collections.emptyList(); - } - File[] availableTemplates = - workflowTemplateDir.listFiles((d, name) -> name.toLowerCase().endsWith("." + documentWorkflow.getFileExtension())); - if (availableTemplates == null) { - return Collections.emptyList(); - } - return Arrays.stream(availableTemplates).map(File::getName).sorted(String::compareTo).collect(Collectors.toList()); + public List getAvailableTemplates(DocumentWorkflow documentWorkflow, @Nullable Disease disease) { + List templates = documentTemplateService.getByPredicate( + (cb, root, cq) -> cb.and( + cb.equal(root.get(DocumentTemplate.WORKFLOW), documentWorkflow), + cb.or(cb.isNull(root.get(DocumentTemplate.DISEASE)), cb.equal(root.get(DocumentTemplate.DISEASE), disease)))); + + return templates.stream().map(DocumentTemplateFacadeEjb::toDto).collect(Collectors.toList()); } @Override @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public boolean isExistingTemplate(DocumentWorkflow documentWorkflow, String templateName) { + public boolean isExistingTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease) { assertRequredUserRight(documentWorkflow); - File templateFile = new File(getWorkflowTemplateDirPath(documentWorkflow).resolve(templateName).toUri()); + File templateFile = new File(getWorkflowTemplateDirPath(documentWorkflow, disease).resolve(templateName).toUri()); return templateFile.exists(); } @Override @PermitAll - public DocumentVariables getDocumentVariables(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException { - File templateFile = getTemplateFile(documentWorkflow, templateName); + public DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); + DocumentWorkflow documentWorkflow = template.getWorkflow(); + + File templateFile = getTemplateFile(template); DocumentVariables documentVariables = documentWorkflow.isDocx() ? getTemplateVariablesDocx(templateFile) : getTemplateVariablesTxt(templateFile); Set propertyKeys = documentVariables.getVariables(); @@ -326,7 +336,8 @@ public DocumentVariables getDocumentVariables(DocumentWorkflow documentWorkflow, @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName, byte[] document) throws DocumentTemplateException { + public void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease, byte[] document) + throws DocumentTemplateException { assertRequredUserRight(documentWorkflow); if (!documentWorkflow.getFileExtension().equalsIgnoreCase(FilenameUtils.getExtension(templateName))) { @@ -349,7 +360,7 @@ public void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String temp validateEmailTemplate(document); } - Path workflowTemplateDirPath = getWorkflowTemplateDirPath(documentWorkflow); + Path workflowTemplateDirPath = getWorkflowTemplateDirPath(documentWorkflow, disease); try { Files.createDirectories(workflowTemplateDirPath); } catch (IOException e) { @@ -367,36 +378,35 @@ public void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String temp @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public boolean deleteDocumentTemplate(DocumentWorkflow documentWorkflow, String fileName) throws DocumentTemplateException { - assertRequredUserRight(documentWorkflow); + public boolean deleteDocumentTemplate(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); - File templateFile = new File(getWorkflowTemplateDirPath(documentWorkflow).resolve(fileName).toUri()); - if (templateFile.exists() && templateFile.isFile()) { - return templateFile.delete(); - } else { - throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), fileName)); - } + assertRequredUserRight(template.getWorkflow()); + + documentTemplateService.deletePermanent(template); } @Override @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public byte[] getDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException { - assertRequredUserRight(documentWorkflow); + public byte[] getDocumentTemplateContent(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); + assertRequredUserRight(template.getWorkflow()); try { - return FileUtils.readFileToByteArray(getTemplateFile(documentWorkflow, templateName)); + return FileUtils.readFileToByteArray(getTemplateFile(template)); } catch (IOException e) { - throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorReadingTemplate), templateName)); + throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorReadingTemplate), template.getFileName())); } } - private File getTemplateFile(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException { - File templateFile = new File(getWorkflowTemplateDirPath(documentWorkflow).resolve(templateName).toString()); + private File getTemplateFile(DocumentTemplate template) throws DocumentTemplateException { + File templateFile = + new File(getWorkflowTemplateDirPath(template.getWorkflow(), template.getDisease()).resolve(template.getFileName()).toString()); if (!templateFile.exists()) { - throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), templateName)); + throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), template.getFileName())); } return templateFile; } @@ -423,8 +433,14 @@ private String getVariableBaseName(String propertyKey) { return matcher.matches() ? matcher.group(1) : ""; } - private Path getWorkflowTemplateDirPath(DocumentWorkflow documentWorkflow) { - return Paths.get(configFacade.getCustomFilesPath()).resolve("docgeneration").resolve(documentWorkflow.getTemplateDirectory()); + private Path getWorkflowTemplateDirPath(DocumentWorkflow documentWorkflow, Disease disease) { + Path path = Paths.get(configFacade.getCustomFilesPath()).resolve("docgeneration").resolve(documentWorkflow.getTemplateDirectory()); + + if (disease != null) { + path = path.resolve(disease.name()); + } + + return path; } private EntityDtoAccessHelper.IReferenceDtoResolver getReferenceDtoResolver() { @@ -482,6 +498,21 @@ private static void validateEmailTemplate(byte[] document) { } } + public static DocumentTemplateDto toDto(DocumentTemplate source) { + if (source == null) { + return null; + } + + DocumentTemplateDto target = new DocumentTemplateDto(); + DtoHelper.fillDto(target, source); + + target.setWorkflow(source.getWorkflow()); + target.setDisease(source.getDisease()); + target.setFileName(source.getFileName()); + + return target; + } + public static EmailTemplateTexts splitTemplateContent(String content) { return splitTemplateContent(content, true); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java new file mode 100644 index 00000000000..14bd08f89dc --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java @@ -0,0 +1,66 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.docgeneration; + +import java.io.File; +import java.nio.file.Path; +import java.nio.file.Paths; + +import javax.ejb.EJB; +import javax.ejb.LocalBean; +import javax.ejb.Stateless; + +import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.backend.common.BaseAdoService; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; + +@Stateless +@LocalBean +public class DocumentTemplateService extends BaseAdoService { + + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; + + protected DocumentTemplateService() { + super(DocumentTemplate.class); + } + + @Override + public boolean deletePermanent(DocumentTemplate documentTemplate) throws DocumentTemplateException { + String fileName = documentTemplate.getFileName(); + File templateFile = new File(getWorkflowTemplateDirPath(documentTemplate).resolve(fileName).toUri()); + if (templateFile.exists() && templateFile.isFile()) { + return templateFile.delete(); + } else { + throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), fileName)); + } + super.deletePermanent(documentTemplate); + } + + private Path getWorkflowTemplateDirPath(DocumentTemplate documentTemplate) { + Path path = + Paths.get(configFacade.getCustomFilesPath()).resolve("docgeneration").resolve(documentTemplate.getWorkflow().getTemplateDirectory()); + + if (documentTemplate.getDisease() != null) { + path = path.resolve(documentTemplate.getDisease().name()); + } + + return path; + } + +} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java index 02f6ab5a25f..b6cfc009cde 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java @@ -15,8 +15,10 @@ import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.action.ActionCriteria; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateEntities; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EventDocumentFacade; @@ -50,7 +52,7 @@ public class EventDocumentFacadeEjb implements EventDocumentFacade { @Override public String getGeneratedDocument( - String templateName, + DocumentTemplateReferenceDto templateReference, EventReferenceDto eventReference, Properties extraProperties, Boolean shouldUploadGeneratedDoc) @@ -64,13 +66,13 @@ public String getGeneratedDocument( entities .addEntity(RootEntityType.ROOT_EVENT_PARTICIPANTS, eventParticipantFacade.getAllActiveEventParticipantsByEvent(eventReference.getUuid())); - String body = documentTemplateFacade.generateDocumentTxtFromEntities(DOCUMENT_WORKFLOW, templateName, entities, extraProperties); - String styledHtml = createStyledHtml(templateName, body); + String body = documentTemplateFacade.generateDocumentTxtFromEntities(templateReference, entities, extraProperties); + String styledHtml = createStyledHtml(templateReference.getCaption(), body); if (shouldUploadGeneratedDoc) { byte[] documentToSave = styledHtml.getBytes(StandardCharsets.UTF_8);//mandatory UTF_8 try { helper.saveDocument( - helper.getDocumentFileName(eventReference, templateName), + helper.getDocumentFileName(eventReference, templateReference.getCaption()), DocumentDto.MIME_TYPE_DEFAULT, documentToSave.length, helper.getDocumentRelatedEntityType(eventReference), @@ -85,7 +87,7 @@ public String getGeneratedDocument( @Override public Map getGeneratedDocuments( - String templateName, + DocumentTemplateReferenceDto templateReference, List eventReferences, Properties extraProperties, Boolean shouldUploadGeneratedDoc) @@ -93,7 +95,7 @@ public Map getGeneratedDocuments( Map documents = new HashMap<>(eventReferences.size()); for (EventReferenceDto referenceDto : eventReferences) { - String documentContent = getGeneratedDocument(templateName, referenceDto, extraProperties, shouldUploadGeneratedDoc); + String documentContent = getGeneratedDocument(templateReference, referenceDto, extraProperties, shouldUploadGeneratedDoc); documents.put(referenceDto, documentContent.getBytes(StandardCharsets.UTF_8)); } @@ -101,8 +103,8 @@ public Map getGeneratedDocuments( } @Override - public List getAvailableTemplates() { - return documentTemplateFacade.getAvailableTemplates(DOCUMENT_WORKFLOW); + public List getAvailableTemplates() { + return documentTemplateFacade.getAvailableTemplates(DOCUMENT_WORKFLOW, null); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java index 8933054aeb8..05cae3d6464 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java @@ -26,8 +26,10 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.ReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateEntities; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.QuarantineOrderFacade; @@ -59,8 +61,7 @@ public class QuarantineOrderFacadeEjb implements QuarantineOrderFacade { @Override public byte[] getGeneratedDocument( - String templateName, - DocumentWorkflow workflow, + DocumentTemplateReferenceDto templateReference, RootEntityType rootEntityType, ReferenceDto rootEntityReference, SampleReferenceDto sampleReference, @@ -72,9 +73,9 @@ public byte[] getGeneratedDocument( DocumentTemplateEntities entities = entitiesBuilder .getQuarantineOrderEntities(rootEntityType, rootEntityReference, sampleReference, pathogenTestReference, vaccinationReference); - byte[] documentToSave = documentTemplateFacade.generateDocumentDocxFromEntities(workflow, templateName, entities, extraProperties); + byte[] documentToSave = documentTemplateFacade.generateDocumentDocxFromEntities(templateReference, entities, extraProperties); if (shouldUploadGeneratedDoc) { - uploadDocument(helper.getDocumentFileName(rootEntityReference, templateName), rootEntityReference, documentToSave); + uploadDocument(helper.getDocumentFileName(rootEntityReference, templateReference), rootEntityReference, documentToSave); } return documentToSave; } @@ -105,8 +106,7 @@ private boolean isFileSizeLimitExceeded(int length) { @Override public Map getGeneratedDocuments( - String templateName, - DocumentWorkflow workflow, + DocumentTemplateReferenceDto templateReference, List rootEntityReferences, Properties extraProperties, Boolean shouldUploadGeneratedDoc) @@ -115,12 +115,12 @@ public Map getGeneratedDocuments( Map quarantineOrderEntities = entitiesBuilder.getQuarantineOrderEntities(workflow, rootEntityReferences); - return getGeneratedDocuments(templateName, workflow, quarantineOrderEntities, extraProperties, shouldUploadGeneratedDoc); + return getGeneratedDocuments(templateReference, quarantineOrderEntities, extraProperties, shouldUploadGeneratedDoc); } @Override public Map getGeneratedDocumentsForEventParticipants( - String templateName, + DocumentTemplateReferenceDto templateReference, List rootEntityReferences, Disease eventDisease, Properties extraProperties, @@ -130,17 +130,11 @@ public Map getGeneratedDocumentsForEventParticipants( Map quarantineOrderEntities = entitiesBuilder.getEventParticipantQuarantineOrderEntities(rootEntityReferences, eventDisease); - return getGeneratedDocuments( - templateName, - DocumentWorkflow.QUARANTINE_ORDER_EVENT_PARTICIPANT, - quarantineOrderEntities, - extraProperties, - shouldUploadGeneratedDoc); + return getGeneratedDocuments(templateReference, quarantineOrderEntities, extraProperties, shouldUploadGeneratedDoc); } private Map getGeneratedDocuments( - String templateName, - DocumentWorkflow workflow, + DocumentTemplateReferenceDto templateReference, Map quarantineOrderEntities, Properties extraProperties, Boolean shouldUploadGeneratedDoc) @@ -149,10 +143,9 @@ private Map getGeneratedDocuments( Map documents = new HashMap<>(quarantineOrderEntities.size()); for (Map.Entry entities : quarantineOrderEntities.entrySet()) { - byte[] documentContent = - documentTemplateFacade.generateDocumentDocxFromEntities(workflow, templateName, entities.getValue(), extraProperties); + byte[] documentContent = documentTemplateFacade.generateDocumentDocxFromEntities(templateReference, entities.getValue(), extraProperties); if (shouldUploadGeneratedDoc) { - uploadDocument(helper.getDocumentFileName(entities.getKey(), templateName), entities.getKey(), documentContent); + uploadDocument(helper.getDocumentFileName(entities.getKey(), templateReference), entities.getKey(), documentContent); } documents.put(entities.getKey(), documentContent); } @@ -161,8 +154,8 @@ private Map getGeneratedDocuments( } @Override - public List getAvailableTemplates(DocumentWorkflow workflow) { - return documentTemplateFacade.getAvailableTemplates(workflow); + public List getAvailableTemplates(DocumentWorkflow workflow) { + return documentTemplateFacade.getAvailableTemplates(workflow, null); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantService.java index a9941a14f7e..781b2fb686a 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantService.java @@ -333,7 +333,7 @@ public void restore(EventParticipant eventParticipant) { } @Override - public void deletePermanent(EventParticipant eventParticipant) { + public boolean deletePermanent(EventParticipant eventParticipant) { if (eventParticipant.getSamples() != null) { for (Sample sample : eventParticipant.getSamples()) { if (sample.getAssociatedCase() == null && sample.getAssociatedContact() == null) { @@ -354,6 +354,7 @@ public void deletePermanent(EventParticipant eventParticipant) { }); super.deletePermanent(eventParticipant); + return false; } public List getAllUuidsByEventUuids(List eventUuids) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java index 7814c312f48..89f07b85306 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java @@ -638,7 +638,7 @@ public void restore(Event event) { } @Override - public void deletePermanent(Event event) { + public boolean deletePermanent(Event event) { // Delete all tasks associated with this event List tasks = taskService.findBy(new TaskCriteria().event(new EventReferenceDto(event.getUuid())), true); @@ -677,6 +677,7 @@ public void deletePermanent(Event event) { removeFromSubordinateEvents(event); super.deletePermanent(event); + return false; } private void removeFromSubordinateEvents(Event event) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java index c61d503ac4f..f9681ee9ccb 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java @@ -51,6 +51,7 @@ import de.symeda.sormas.api.common.progress.ProcessedEntity; import de.symeda.sormas.api.common.progress.ProcessedEntityStatus; import de.symeda.sormas.api.contact.ContactReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateEntities; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; @@ -157,8 +158,8 @@ public class ExternalEmailFacadeEjb implements ExternalEmailFacade { private DocGenerationHelper docGenerationHelper; @Override - public List getTemplateNames(DocumentWorkflow documentWorkflow) { - return documentTemplateFacade.getAvailableTemplates(documentWorkflow); + public List getTemplateNames(DocumentWorkflow documentWorkflow) { + return documentTemplateFacade.getAvailableTemplates(documentWorkflow, null); } @Override @@ -200,7 +201,7 @@ private void sendEmail(@Valid ExternalEmailOptionsDto options, boolean onlyLinke String personalizedFileName = null; Map filesToBeEncryped = new HashMap<>(); - if (options.getQuarantineOrderDocumentOptions() != null && options.getQuarantineOrderDocumentOptions().getTemplateFile() != null) { + if (options.getQuarantineOrderDocumentOptions() != null && options.getQuarantineOrderDocumentOptions().getTemplate() != null) { Map.Entry fileFromTemplate = createFileFromTemplate(options); personalizedFile = fileFromTemplate.getKey(); personalizedFileName = fileFromTemplate.getValue(); @@ -232,8 +233,7 @@ private void sendEmail(@Valid ExternalEmailOptionsDto options, boolean onlyLinke emailAttachments = attachmentService.createEncryptedPdfs(filesToBeEncryped, password); } - String generatedText = - documentTemplateFacade.generateDocumentTxtFromEntities(options.getDocumentWorkflow(), options.getTemplateName(), documentEntities, null); + String generatedText = documentTemplateFacade.generateDocumentTxtFromEntities(options.getTemplate(), documentEntities, null); EmailTemplateTexts emailTexts = splitTemplateContent(generatedText); try { @@ -281,8 +281,7 @@ private void sendEmail(@Valid ExternalEmailOptionsDto options, boolean onlyLinke private Map.Entry createFileFromTemplate(ExternalEmailOptionsDto emailOptions) throws DocumentTemplateException, IOException { QuarantineOrderDocumentOptionsDto documentOptions = emailOptions.getQuarantineOrderDocumentOptions(); byte[] generatedDocument = quarantineOrderFacade.getGeneratedDocument( - documentOptions.getTemplateFile(), - documentOptions.getDocumentWorkflow(), + documentOptions.getTemplate(), emailOptions.getRootEntityType(), emailOptions.getRootEntityReference(), null, @@ -291,7 +290,7 @@ private Map.Entry createFileFromTemplate(ExternalEmailOptionsDto e documentOptions.getExtraProperties(), // document uploading will be handled in another place false); - String fileName = docGenerationHelper.getDocumentFileName(emailOptions.getRootEntityReference(), documentOptions.getTemplateFile()); + String fileName = docGenerationHelper.getDocumentFileName(emailOptions.getRootEntityReference(), documentOptions.getTemplate()); File tempTemplateFile = Paths.get(configFacade.getTempFilesPath()).resolve(fileName).toFile(); BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(Files.newOutputStream(tempTemplateFile.toPath())); bufferedOutputStream.write(generatedDocument); @@ -341,7 +340,7 @@ public List sendBulkEmail(@Valid ExternalEmailOptionsWithAttach try { ExternalEmailOptionsDto emailOptions = new ExternalEmailOptionsDto(options.getDocumentWorkflow(), options.getRootEntityType(), entityRef); - emailOptions.setTemplateName(options.getTemplateName()); + emailOptions.setTemplate(options.getTemplateName()); emailOptions.setAttachedDocuments(attachedDocuments); emailOptions.setRootEntityReference(entityRef); @@ -481,7 +480,7 @@ private ManualMessageLog createMessageLog( log.setRecipientPerson(personService.getByReferenceDto(personRef)); log.setSentDate(new Date()); log.setEmailAddress(options.getRecipientEmail()); - log.setUsedTemplate(options.getTemplateName()); + log.setUsedTemplate(options.getTemplate().getCaption()); log.setAttachedDocuments(attachedDocumentsName); // `*Service::getByReferenceDto` does a null check, so we don't need to do it here diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java index f4bd04ba457..d163cb10330 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java @@ -50,9 +50,10 @@ public ExternalMessageService() { } @Override - public void deletePermanent(ExternalMessage externalMessage) { + public boolean deletePermanent(ExternalMessage externalMessage) { super.deletePermanent(externalMessage); + return false; } @TransactionAttribute(TransactionAttributeType.REQUIRES_NEW) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/SampleReportService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/SampleReportService.java index 8153c30db63..542a4ca6fd0 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/SampleReportService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/SampleReportService.java @@ -15,8 +15,7 @@ package de.symeda.sormas.backend.externalmessage.labmessage; -import de.symeda.sormas.backend.common.BaseAdoService; -import de.symeda.sormas.backend.common.DeletableAdo; +import java.util.Optional; import javax.ejb.EJB; import javax.ejb.LocalBean; @@ -25,7 +24,9 @@ import javax.persistence.criteria.JoinType; import javax.persistence.criteria.Predicate; import javax.persistence.criteria.Root; -import java.util.Optional; + +import de.symeda.sormas.backend.common.BaseAdoService; +import de.symeda.sormas.backend.common.DeletableAdo; @Stateless @LocalBean @@ -47,10 +48,11 @@ public Predicate createDefaultFilter(CriteriaBuilder cb, Root root) } @Override - public void deletePermanent(SampleReport sampleReport) { + public boolean deletePermanent(SampleReport sampleReport) { Optional.ofNullable(sampleReport.getTestReports()).ifPresent(testReports -> testReports.forEach(t -> testReportService.deletePermanent(t))); super.deletePermanent(sampleReport); + return false; } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java index 237cdad9507..2a9dca086ff 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationService.java @@ -22,7 +22,6 @@ import java.util.HashSet; import java.util.LinkedList; import java.util.List; -import java.util.stream.Collectors; import javax.ejb.EJB; import javax.ejb.LocalBean; @@ -53,7 +52,6 @@ import de.symeda.sormas.api.immunization.ImmunizationStatus; import de.symeda.sormas.api.immunization.MeansOfImmunization; import de.symeda.sormas.api.person.PersonAssociation; -import de.symeda.sormas.api.person.PersonCriteria; import de.symeda.sormas.api.sormastosormas.share.incoming.ShareRequestStatus; import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.common.AbstractCoreAdoService; @@ -70,7 +68,6 @@ import de.symeda.sormas.backend.person.Person; import de.symeda.sormas.backend.person.PersonJoins; import de.symeda.sormas.backend.person.PersonJurisdictionPredicateValidator; -import de.symeda.sormas.backend.person.PersonQueryContext; import de.symeda.sormas.backend.person.PersonService; import de.symeda.sormas.backend.sormastosormas.origin.SormasToSormasOriginInfo; import de.symeda.sormas.backend.sormastosormas.share.outgoing.ShareRequestInfo; @@ -87,7 +84,6 @@ import de.symeda.sormas.backend.util.QueryHelper; import de.symeda.sormas.backend.vaccination.LastVaccinationDate; import de.symeda.sormas.backend.vaccination.Vaccination; -import org.hibernate.query.Query; @Stateless @LocalBean @@ -109,7 +105,7 @@ public ImmunizationService() { } @Override - public void deletePermanent(Immunization immunization) { + public boolean deletePermanent(Immunization immunization) { // Remove the immunization from any S2S share info referencing it sormasToSormasShareInfoService.getByAssociatedEntity(SormasToSormasShareInfo.IMMUNIZATION, immunization.getUuid()).forEach(s -> { @@ -131,6 +127,7 @@ public void deletePermanent(Immunization immunization) { } super.deletePermanent(immunization); + return false; } public List getEntriesList(Long personId, Disease disease, Integer first, Integer max) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java index b681af27ae7..8ea0fd1cd9a 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java @@ -1054,13 +1054,14 @@ private Subquery createSubquery( } @Override - public void deletePermanent(Person person) { + public boolean deletePermanent(Person person) { manualMessageLogService.getByPersonUuid(person.getUuid()) .forEach(manualMessageLog -> manualMessageLogService.deletePermanent(manualMessageLog)); visitService.deletePersonVisits(Collections.singletonList(person.getUuid())); super.deletePermanent(person); + return false; } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java index 1295c31c7e8..6169bf89272 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java @@ -1027,7 +1027,7 @@ private Predicate allAssignedEntitiesAreArchived(CriteriaBuilder cb, SampleJoins } @Override - public void deletePermanent(Sample sample) { + public boolean deletePermanent(Sample sample) { // Delete all pathogen tests of this sample for (PathogenTest pathogenTest : sample.getPathogenTests()) { @@ -1052,6 +1052,7 @@ public void deletePermanent(Sample sample) { deleteSampleLinks(sample); super.deletePermanent(sample); + return false; } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/services/TravelEntryService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/services/TravelEntryService.java index 7b006c17a25..775b1d97b91 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/services/TravelEntryService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/services/TravelEntryService.java @@ -277,7 +277,7 @@ public EditPermissionType getEditPermissionType(TravelEntry travelEntry) { } @Override - public void deletePermanent(TravelEntry travelEntry) { + public boolean deletePermanent(TravelEntry travelEntry) { // Delete all tasks associated with this travel entry List tasks = taskService.findBy( new TaskCriteria().travelEntry( @@ -295,6 +295,7 @@ public void deletePermanent(TravelEntry travelEntry) { .forEach(document -> documentService.markAsDeleted(document)); super.deletePermanent(travelEntry); + return false; } public TravelEntry getLastTravelEntry() { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/user/UserRoleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/user/UserRoleService.java index 2ba7576570b..a435131a022 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/user/UserRoleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/user/UserRoleService.java @@ -175,7 +175,7 @@ public boolean isCaptionUnique(String excludedUuid, String caption) { } @Override - public void deletePermanent(UserRole userRole) { + public boolean deletePermanent(UserRole userRole) { List usersWithRole = userService.getAllWithRole(userRole); for (User u : usersWithRole) { @@ -191,5 +191,6 @@ public void deletePermanent(UserRole userRole) { } super.deletePermanent(userRole); + return false; } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java index 8f410e4df3a..7f0bdccdcf3 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java @@ -132,7 +132,7 @@ public void testEmailTemplateValidation() throws DocumentTemplateException { @Test public void readTemplateTest() throws DocumentTemplateException { - byte[] template = documentTemplateFacade.getDocumentTemplate(QUARANTINE_ORDER_CASE, "Quarantine.docx"); + byte[] template = documentTemplateFacade.getDocumentTemplateContent(QUARANTINE_ORDER_CASE, "Quarantine.docx"); assertEquals(12731, template.length); } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java index 441e8356e8d..2cb9c486600 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java @@ -133,7 +133,7 @@ public void generateEventHandoutTest() throws IOException, URISyntaxException, D String testcaseBasename = FilenameUtils.getBaseName(testCaseHtml.getName()); String htmlText = - eventDocumentFacade.getGeneratedDocument(testcaseBasename + ".html", eventDto.toReference(), new Properties(), Boolean.FALSE); + eventDocumentFacade.getGeneratedDocument(, testcaseBasename + ".html", eventDto.toReference(), new Properties(), Boolean.FALSE); String actual = cleanLineSeparators( htmlText.replaceAll("

Event-ID: [A-Z0-9-]*

", "

Event-ID: STN3WX-5JTGYV-IU2LRM-4UHCSOEE

")); diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java index 5dac8076044..9a062b80f32 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java @@ -254,7 +254,7 @@ public void testBulkCaseDocumentCreation() throws DocumentTemplateException, IOE DocumentWorkflow workflow = DocumentWorkflow.QUARANTINE_ORDER_CASE; Map documentContents = quarantineOrderFacadeEjb - .getGeneratedDocuments("Quarantine.docx", workflow, Collections.singletonList(rootEntityReference), properties, false); + .getGeneratedDocuments(, "Quarantine.docx", Collections.singletonList(rootEntityReference), properties, false); verifyGeneratedDocument(rootEntityReference, workflow, "QuarantineCase.cmp", documentContents.get(rootEntityReference)); } @@ -268,7 +268,7 @@ public void testBulkContactDocumentCreation() throws DocumentTemplateException, DocumentWorkflow workflow = DocumentWorkflow.QUARANTINE_ORDER_CONTACT; Map documentContents = quarantineOrderFacadeEjb - .getGeneratedDocuments("Quarantine.docx", workflow, Collections.singletonList(rootEntityReference), properties, false); + .getGeneratedDocuments(, "Quarantine.docx", Collections.singletonList(rootEntityReference), properties, false); verifyGeneratedDocument(rootEntityReference, workflow, "QuarantineContact.cmp", documentContents.get(rootEntityReference)); } @@ -282,8 +282,8 @@ public void testBulkEventParticipantDocumentCreation() throws DocumentTemplateEx properties.setProperty("extra.remark.no3", "the third remark"); DocumentWorkflow workflow = DocumentWorkflow.QUARANTINE_ORDER_EVENT_PARTICIPANT; - Map documentContents = quarantineOrderFacadeEjb.getGeneratedDocumentsForEventParticipants( - "Quarantine.docx", + Map documentContents = quarantineOrderFacadeEjb.getGeneratedDocumentsForEventParticipants(, + "Quarantine.docx", Collections.singletonList(rootEntityReference), eventDto.getDisease(), properties, @@ -310,10 +310,9 @@ private void generateQuarantineOrderTest( rootEntityReference, documentWorkflow, comparisonFile, - quarantineOrderFacadeEjb.getGeneratedDocument( - "Quarantine.docx", - documentWorkflow, - rootEntityType, + quarantineOrderFacadeEjb.getGeneratedDocument(, + "Quarantine.docx", + rootEntityType, rootEntityReference, sampleReference, pathogenTest, diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java index a5a811d255c..95ed274840a 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java @@ -297,7 +297,7 @@ public void testSendEmailToCasePerson() }).when(emailService).sendEmail(any(), any(), any(), any()); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate("CaseEmail.txt"); options.setRecipientEmail("test@mail.com"); getExternalEmailFacade().sendEmail(options); @@ -336,7 +336,7 @@ public void testSendEmailToContactPerson() ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CONTACT_EMAIL, RootEntityType.ROOT_CONTACT, contact.toReference()); - options.setTemplateName("ContactEmail.txt"); + options.setTemplate("ContactEmail.txt"); options.setRecipientEmail("test@mail.com"); getExternalEmailFacade().sendEmail(options); @@ -397,7 +397,7 @@ public void testSendEmailWithAttachments() }).when(emailService).sendEmail(any(), any(), any(), any()); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate("CaseEmail.txt"); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(documents.stream().map(DocumentDto::toReference).collect(Collectors.toSet())); @@ -476,7 +476,7 @@ public void testEncryptAttachmentsWithRandomPassword() }).when(smsService).sendSms(any(), any()); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate("CaseEmail.txt"); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(documents.stream().map(DocumentDto::toReference).collect(Collectors.toSet())); @@ -497,7 +497,7 @@ public void testSendEmailWithUnsupportedAttachment() throws MessagingException, createDocument("SomeDocument.txt", DocumentRelatedEntityType.CASE, caze.getUuid(), "Some content".getBytes(StandardCharsets.UTF_8)); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate("CaseEmail.txt"); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(Collections.singleton(document.toReference())); @@ -518,7 +518,7 @@ public void testSendAttachmentWithUnavailablePassword() throws MessagingExceptio createDocument("SomeDocument.txt", DocumentRelatedEntityType.CASE, caze.getUuid(), "Some content".getBytes(StandardCharsets.UTF_8)); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate("CaseEmail.txt"); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(Collections.singleton(document.toReference())); @@ -541,7 +541,7 @@ public void testSendAttachmentNotRelatedToEntity() throws MessagingException, IO createDocument("SomeDocument.txt", DocumentRelatedEntityType.CONTACT, "mock-uuid", "Some content".getBytes(StandardCharsets.UTF_8)); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate("CaseEmail.txt"); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(Collections.singleton(document.toReference())); @@ -705,7 +705,7 @@ public void testSendBulkEmailToCasePersonAttachmentsAndTemplateDocument() throws options.setTemplateName("CaseEmail.txt"); QuarantineOrderDocumentOptionsDto quarantineOrderDocumentOptions = new QuarantineOrderDocumentOptionsDto(); - quarantineOrderDocumentOptions.setTemplateFile("Quarantine.docx"); + quarantineOrderDocumentOptions.setTemplate("Quarantine.docx"); quarantineOrderDocumentOptions.setExtraProperties(new Properties()); quarantineOrderDocumentOptions.setShouldUploadGeneratedDoc(false); quarantineOrderDocumentOptions.setDocumentWorkflow(DocumentWorkflow.QUARANTINE_ORDER_CASE); @@ -801,7 +801,7 @@ public void testSendBulkEmailToCasePersonTemplateDocument() throws MessagingExce new ExternalEmailOptionsWithAttachmentsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE); options.setTemplateName("CaseEmail.txt"); QuarantineOrderDocumentOptionsDto quarantineOrderDocumentOptions = new QuarantineOrderDocumentOptionsDto(); - quarantineOrderDocumentOptions.setTemplateFile("Quarantine.docx"); + quarantineOrderDocumentOptions.setTemplate("Quarantine.docx"); quarantineOrderDocumentOptions.setExtraProperties(new Properties()); quarantineOrderDocumentOptions.setShouldUploadGeneratedDoc(true); quarantineOrderDocumentOptions.setDocumentWorkflow(DocumentWorkflow.QUARANTINE_ORDER_CASE); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java index 24978bdfb57..ef5e3f6182b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java @@ -31,6 +31,7 @@ import com.vaadin.ui.Notification; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.i18n.Captions; @@ -39,23 +40,27 @@ import de.symeda.sormas.ui.utils.ButtonHelper; import de.symeda.sormas.ui.utils.VaadinUiUtil; -public class DocumentTemplatesGrid extends Grid { +public class DocumentTemplatesGrid extends Grid { private static final long serialVersionUID = 2589713987152595369L; private final DocumentWorkflow documentWorkflow; public DocumentTemplatesGrid(DocumentWorkflow documentWorkflow) { - super(String.class); + super(DocumentTemplateDto.class); this.documentWorkflow = documentWorkflow; setSizeFull(); - List availableTemplates = FacadeProvider.getDocumentTemplateFacade().getAvailableTemplates(documentWorkflow); - ListDataProvider dataProvider = DataProvider.fromStream(availableTemplates.stream()); + List availableTemplates = FacadeProvider.getDocumentTemplateFacade().getAvailableTemplates(documentWorkflow, null); + ListDataProvider dataProvider = DataProvider.fromStream(availableTemplates.stream()); setDataProvider(dataProvider); - removeAllColumns(); - addColumn(String::toString).setCaption(I18nProperties.getString(Strings.fileName)).setExpandRatio(1); + setColumns(DocumentTemplateDto.FILE_NAME, DocumentTemplateDto.DISEASE); + + for (Column column : getColumns()) { + column.setCaption(I18nProperties.getPrefixCaption(DocumentTemplateDto.I18N_PREFIX, column.getId(), column.getCaption())); + } + addComponentColumn(this::buildActionButtons).setCaption(I18nProperties.getCaption(Captions.eventActionsView)) .setWidth(100) .setStyleGenerator(item -> "v-align-center"); @@ -67,20 +72,20 @@ public DocumentTemplatesGrid(DocumentWorkflow documentWorkflow) { public void reload() { // This is bad practice but it works (unlike refreshAll), and in this case its sufficient - List availableTemplates = FacadeProvider.getDocumentTemplateFacade().getAvailableTemplates(documentWorkflow); + List availableTemplates = FacadeProvider.getDocumentTemplateFacade().getAvailableTemplates(documentWorkflow, null); setItems(availableTemplates); getDataProvider().refreshAll(); setHeightByRows(Math.max(1, availableTemplates.size())); } - private Button buildDeleteButton(String templateFileName) { + private Button buildDeleteButton(DocumentTemplateDto template) { return ButtonHelper.createIconButton( "", VaadinIcons.TRASH, e -> VaadinUiUtil - .showDeleteConfirmationWindow(String.format(I18nProperties.getString(Strings.confirmationDeleteFile), templateFileName), () -> { + .showDeleteConfirmationWindow(String.format(I18nProperties.getString(Strings.confirmationDeleteFile), template.getFileName()), () -> { try { - FacadeProvider.getDocumentTemplateFacade().deleteDocumentTemplate(documentWorkflow, templateFileName); + FacadeProvider.getDocumentTemplateFacade().deleteDocumentTemplate(template.toReference()); } catch (DocumentTemplateException ex) { new Notification( I18nProperties.getString(Strings.errorDeletingDocumentTemplate), @@ -92,21 +97,21 @@ private Button buildDeleteButton(String templateFileName) { })); } - private Button buildViewDocumentButton(String templateFileName) { + private Button buildViewDocumentButton(DocumentTemplateDto template) { Button viewButton = ButtonHelper.createIconButton(VaadinIcons.DOWNLOAD); StreamResource streamResource = new StreamResource((StreamResource.StreamSource) () -> { try { - return new ByteArrayInputStream(FacadeProvider.getDocumentTemplateFacade().getDocumentTemplate(documentWorkflow, templateFileName)); + return new ByteArrayInputStream(FacadeProvider.getDocumentTemplateFacade().getDocumentTemplateContent(template.toReference())); } catch (DocumentTemplateException e) { new Notification( - String.format(I18nProperties.getString(Strings.errorReadingTemplate), templateFileName), + String.format(I18nProperties.getString(Strings.errorReadingTemplate), template.getFileName()), e.getMessage(), Notification.Type.ERROR_MESSAGE, false).show(Page.getCurrent()); return null; } - }, templateFileName); + }, template.getFileName()); FileDownloader fileDownloader = new FileDownloader(streamResource); fileDownloader.extend(viewButton); fileDownloader.setFileDownloadResource(streamResource); @@ -114,11 +119,11 @@ private Button buildViewDocumentButton(String templateFileName) { return viewButton; } - private HorizontalLayout buildActionButtons(String s) { + private HorizontalLayout buildActionButtons(DocumentTemplateDto template) { HorizontalLayout horizontalLayout = new HorizontalLayout(); - horizontalLayout.addComponent(buildViewDocumentButton(s)); - horizontalLayout.addComponent(buildDeleteButton(s)); + horizontalLayout.addComponent(buildViewDocumentButton(template)); + horizontalLayout.addComponent(buildDeleteButton(template)); horizontalLayout.setSpacing(false); horizontalLayout.setMargin(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java index 7e67258e814..7ed3718a08b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java @@ -20,7 +20,6 @@ import java.util.Properties; import java.util.function.Function; -import org.apache.commons.lang3.StringUtils; import org.slf4j.LoggerFactory; import com.vaadin.icons.VaadinIcons; @@ -41,6 +40,7 @@ import com.vaadin.ui.Window; import com.vaadin.ui.themes.ValoTheme; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.i18n.Captions; @@ -56,7 +56,7 @@ public abstract class AbstractDocgenerationLayout extends VerticalLayout { protected final Button createButton; protected final Button cancelButton; - public ComboBox templateSelector; + public ComboBox templateSelector; public final VerticalLayout additionalVariablesComponent; public final VerticalLayout additionalParametersComponent; public FileDownloader fileDownloader; @@ -129,15 +129,15 @@ private void addTemplateSelector(String captionTemplateSelector, Function(captionTemplateSelector); templateSelector.setWidth(100F, Unit.PERCENTAGE); templateSelector.addValueChangeListener(e -> { - String templateFile = e.getValue(); - boolean isValidTemplateFile = StringUtils.isNotBlank(templateFile); + DocumentTemplateDto templateFile = e.getValue(); + boolean isValidTemplateFile = templateFile != null; createButton.setEnabled(isValidTemplateFile); additionalVariablesComponent.removeAllComponents(); hideTextfields(); documentVariables = null; if (isValidTemplateFile) { try { - documentVariables = getDocumentVariables(templateFile); + documentVariables = getDocumentVariables(templateFile.getUuid()); List additionalVariables = documentVariables.getAdditionalVariables(); if (additionalVariables != null && !additionalVariables.isEmpty()) { for (String variable : additionalVariables) { @@ -149,7 +149,7 @@ private void addTemplateSelector(String captionTemplateSelector, Function function) { } } - private void setStreamResource(String templateFile, String fileName) { - StreamResource streamResource = createStreamResource(templateFile, fileName); + private void setStreamResource(DocumentTemplateDto template, String fileName) { + StreamResource streamResource = createStreamResource(template, fileName); if (fileDownloader == null) { fileDownloader = new FileDownloader(streamResource); fileDownloader.extend(createButton); @@ -235,11 +235,11 @@ protected boolean shouldUploadGeneratedDocument() { return checkBoxUploadGeneratedDoc != null && Boolean.TRUE.equals(checkBoxUploadGeneratedDoc.getValue()); } - protected abstract List getAvailableTemplates(); + protected abstract List getAvailableTemplates(); protected abstract DocumentVariables getDocumentVariables(String templateFile) throws IOException, DocumentTemplateException; - protected abstract StreamResource createStreamResource(String templateFile, String filename); + protected abstract StreamResource createStreamResource(DocumentTemplateDto template, String filename); protected abstract String getWindowCaption(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java index 95086c7a8ef..29fa05d0b70 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java @@ -39,6 +39,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.ReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EventDocumentFacade; import de.symeda.sormas.api.docgeneneration.QuarantineOrderFacade; @@ -74,13 +75,12 @@ public void showQuarantineOrderDocumentDialog( sampleCriteria, vaccinationCriteria, documentListComponent, - (templateFile, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { + (templateReference, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { QuarantineOrderFacade quarantineOrderFacade = FacadeProvider.getQuarantineOrderFacade(); return new ByteArrayInputStream( - quarantineOrderFacade.getGeneratedDocument( - templateFile, - workflow, + quarantineOrderFacade.getGeneratedDocument(, + templateReference, rootEntityType, referenceDto, sample, @@ -89,7 +89,7 @@ public void showQuarantineOrderDocumentDialog( extraProperties, shouldUploadGeneratedDoc)); }, - (templateFile) -> getDocumentFileName(referenceDto, templateFile))); + (templateFile) -> getDocumentFileName(referenceDto, templateReference))); } public void showBulkQuarantineOrderDocumentDialog(List referenceDtos, DocumentWorkflow workflow) { @@ -100,13 +100,13 @@ public void showBulkQuarantineOrderDocumentDialog(List referenceDt null, null, null, - (templateFile, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { + (templateReference, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { QuarantineOrderFacade quarantineOrderFacade = FacadeProvider.getQuarantineOrderFacade(); Map generatedDocumentContents = - quarantineOrderFacade.getGeneratedDocuments(templateFile, workflow, referenceDtos, extraProperties, shouldUploadGeneratedDoc); + quarantineOrderFacade.getGeneratedDocuments(templateReference, referenceDtos, extraProperties, shouldUploadGeneratedDoc); - return generateZip(templateFile, shouldUploadGeneratedDoc, generatedDocumentContents); + return generateZip(templateReference, shouldUploadGeneratedDoc, generatedDocumentContents); }, (templateFile) -> filename)); @@ -120,24 +120,24 @@ public void showBulkEventParticipantQuarantineOrderDocumentDialog(List { + (templateReference, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { QuarantineOrderFacade quarantineOrderFacade = FacadeProvider.getQuarantineOrderFacade(); Map generatedDocumentContents = quarantineOrderFacade.getGeneratedDocumentsForEventParticipants( - templateFile, + templateReference, referenceDtos, eventDisease, extraProperties, shouldUploadGeneratedDoc); - return generateZip(templateFile, shouldUploadGeneratedDoc, generatedDocumentContents); + return generateZip(templateReference, shouldUploadGeneratedDoc, generatedDocumentContents); }, (templateFile) -> filename)); } private ByteArrayInputStream generateZip( - String templateFile, + DocumentTemplateReferenceDto templateReference, Boolean shouldUploadGeneratedDoc, Map generatedDocumentContents) { long fileSizeLimitMB = FacadeProvider.getConfigFacade().getDocumentUploadSizeLimitMb(); @@ -149,7 +149,7 @@ private ByteArrayInputStream generateZip( for (Map.Entry referenceDocumentContent : generatedDocumentContents.entrySet()) { ReferenceDto referenceDto = referenceDocumentContent.getKey(); - ZipEntry entry = new ZipEntry(getDocumentFileName(referenceDto, templateFile)); + ZipEntry entry = new ZipEntry(getDocumentFileName(referenceDto, templateReference)); zos.putNextEntry(entry); byte[] document = referenceDocumentContent.getValue(); @@ -211,11 +211,11 @@ public void showEventDocumentDialog(EventReferenceDto eventReferenceDto, Documen new EventDocumentLayout( documentListComponent, (templateFileName) -> getDocumentFileName(eventReferenceDto, templateFileName), - (templateFile, properties, shouldUploadGeneratedDoc) -> { + (template, properties, shouldUploadGeneratedDoc) -> { EventDocumentFacade eventDocumentFacade = FacadeProvider.getEventDocumentFacade(); return new ByteArrayInputStream( - eventDocumentFacade.getGeneratedDocument(templateFile, eventReferenceDto, properties, shouldUploadGeneratedDoc) + eventDocumentFacade.getGeneratedDocument(, template.getUuid(), eventReferenceDto, properties, shouldUploadGeneratedDoc) .getBytes(StandardCharsets.UTF_8)); })); } @@ -223,13 +223,13 @@ public void showEventDocumentDialog(EventReferenceDto eventReferenceDto, Documen public void showEventDocumentDialog(List referenceDtos) { String filename = DownloadUtil.createFileNameWithCurrentDate(ExportEntityName.EVENTS, ".zip"); - showDialog(new EventDocumentLayout(null, (templateFile) -> filename, (templateFile, properties, shouldUploadGeneratedDoc) -> { + showDialog(new EventDocumentLayout(null, (templateFile) -> filename, (template, properties, shouldUploadGeneratedDoc) -> { EventDocumentFacade eventDocumentFacade = FacadeProvider.getEventDocumentFacade(); Map generatedDocumentContents = - eventDocumentFacade.getGeneratedDocuments(templateFile, referenceDtos, properties, shouldUploadGeneratedDoc); + eventDocumentFacade.getGeneratedDocuments(, template, referenceDtos, properties, shouldUploadGeneratedDoc); - return generateZip(templateFile, shouldUploadGeneratedDoc, generatedDocumentContents); + return generateZip(template, shouldUploadGeneratedDoc, generatedDocumentContents); })); } @@ -240,7 +240,7 @@ private void showDialog(AbstractDocgenerationLayout docgenerationLayout) { window.setCaption(I18nProperties.getCaption(docgenerationLayout.getWindowCaption())); } - private String getDocumentFileName(ReferenceDto eventReferenceDto, String templateFileName) { - return DataHelper.getShortUuid(eventReferenceDto) + '-' + templateFileName; + private String getDocumentFileName(ReferenceDto eventReferenceDto, DocumentTemplateReferenceDto templateReference) { + return DataHelper.getShortUuid(eventReferenceDto) + '-' + templateReference.getCaption(); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java index 5ed024ae1fc..68d8f09e853 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java @@ -28,6 +28,7 @@ import com.vaadin.ui.Notification; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.i18n.Captions; @@ -53,7 +54,7 @@ public EventDocumentLayout( } @Override - protected List getAvailableTemplates() { + protected List getAvailableTemplates() { return FacadeProvider.getEventDocumentFacade().getAvailableTemplates(); } @@ -63,7 +64,7 @@ protected DocumentVariables getDocumentVariables(String templateFile) throws Doc } @Override - protected StreamResource createStreamResource(String templateFile, String filename) { + protected StreamResource createStreamResource(DocumentTemplateDto template, String filename) { return new StreamResource((StreamResource.StreamSource) () -> { try { return documentInputStreamSupplier.get(templateFile, readAdditionalVariables(), shouldUploadGeneratedDocument()); @@ -86,6 +87,6 @@ protected String getWindowCaption() { interface DocumentInputStreamSupplier { - InputStream get(String templateFile, Properties properties, Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; + InputStream get(DocumentTemplateDto template, Properties properties, Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java index b31f8320381..9dc7126c603 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java @@ -26,7 +26,6 @@ import javax.annotation.Nullable; -import de.symeda.sormas.ui.utils.CssStyles; import org.slf4j.LoggerFactory; import com.vaadin.server.Page; @@ -36,7 +35,9 @@ import com.vaadin.ui.Notification; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.QuarantineOrderDocumentOptionsDto; @@ -56,6 +57,7 @@ import de.symeda.sormas.api.vaccination.VaccinationListEntryDto; import de.symeda.sormas.api.vaccination.VaccinationReferenceDto; import de.symeda.sormas.ui.document.DocumentListComponent; +import de.symeda.sormas.ui.utils.CssStyles; public class QuarantineOrderLayout extends AbstractDocgenerationLayout { @@ -164,7 +166,7 @@ protected void createVaccinationSelector(VaccinationCriteria vaccinationCriteria } @Override - protected List getAvailableTemplates() { + protected List getAvailableTemplates() { try { return FacadeProvider.getQuarantineOrderFacade().getAvailableTemplates(workflow); } catch (Exception e) { @@ -180,7 +182,7 @@ protected DocumentVariables getDocumentVariables(String templateFile) throws Doc } @Override - protected StreamResource createStreamResource(String templateFile, String filename) { + protected StreamResource createStreamResource(DocumentTemplateDto template, String filename) { return new StreamResource((StreamSource) () -> { SampleReferenceDto sampleReference = sampleSelector != null && sampleSelector.getValue() != null ? sampleSelector.getValue().toReference() : null; @@ -192,7 +194,7 @@ protected StreamResource createStreamResource(String templateFile, String filena try { InputStream stream = documentStreamSupplier.getStream( - templateFile, + template.toReference(), sampleReference, pathogenTestReference, vaccinationReference, @@ -260,7 +262,7 @@ public QuarantineOrderDocumentOptionsDto getFieldValues() { options.setPathogenTest(new PathogenTestReferenceDto(pathogenTestSelector.getValue().getUuid())); } if (templateSelector != null) { - options.setTemplateFile(templateSelector.getValue()); + options.setTemplate(templateSelector.getValue().toReference()); } options.setShouldUploadGeneratedDoc(shouldUploadGeneratedDocument()); @@ -275,7 +277,7 @@ public QuarantineOrderDocumentOptionsDto getFieldValues() { public interface DocumentStreamSupplier { InputStream getStream( - String templateFile, + DocumentTemplateReferenceDto templateReference, SampleReferenceDto sample, PathogenTestReferenceDto pathogenTest, VaccinationReferenceDto vaccinationReference, From 45dbe8b012339f86804ec6455d7b576f0c0c66a8 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Mon, 21 Oct 2024 13:47:46 +0300 Subject: [PATCH 10/56] #13168 Normalization of Automatic Processing for Name, Birthdate, and Address Matching --- .../AutomaticLabMessageProcessor.java | 16 +++++++++++++++ .../AutomaticLabMessageProcessorTest.java | 20 +++++++++++++++++++ 2 files changed, 36 insertions(+) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java index beaa4e0eb02..74f141f2b50 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java @@ -15,6 +15,7 @@ package de.symeda.sormas.backend.externalmessage.labmessage; +import java.text.Collator; import java.util.List; import java.util.Set; import java.util.concurrent.CompletableFuture; @@ -316,5 +317,20 @@ private boolean unsetOrMatches(T personValue, T messageValue) { return personValue.equals(messageValue); } + + private boolean unsetOrMatches(String personValue, String messageValue) { + if (personValue == null || messageValue == null) { + return true; + } + + Collator collator = Collator.getInstance(); + collator.setStrength(Collator.PRIMARY); + + return collator.compare(normalizatString(personValue), normalizatString(messageValue)) == 0; + } + + private String normalizatString(String string) { + return string.replaceAll("[\\s-,;:]", ""); + } } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java index 0a0ba4d7f90..dd55f0c32cf 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java @@ -310,6 +310,26 @@ public void testProcessWithExistingPersonWithSameNationalHealthIdButDifferentDet assertThat(getPathogenTestFacade().getAllActiveUuids(), hasSize(0)); } + @Test + public void testProcessWithExistingPersonWithSameNationalHealthIdAndPersonDetailsNormalizedCheck() + throws ExecutionException, InterruptedException { + ExternalMessageDto externalMessage = createExternalMessage(m -> { + m.setPersonCity("person city"); + m.setPersonStreet("PERSON STREET, 12a"); + }); + + creator.createPerson("john", "DOÉ", Sex.MALE, p -> { + p.setNationalHealthId(externalMessage.getPersonNationalHealthId()); + p.getAddress().setCity(" PERSON city \n"); + p.getAddress().setStreet(" person STREET 12A"); + }); + + ProcessingResult result = runFlow(externalMessage); + assertThat(result.getStatus(), is(DONE)); + assertThat(externalMessage.getStatus(), is(ExternalMessageStatus.PROCESSED)); + assertThat(getExternalMessageFacade().getByUuid(externalMessage.getUuid()).getStatus(), is(ExternalMessageStatus.PROCESSED)); + } + @Test public void testProcessWithExistingSimilarPerson() throws ExecutionException, InterruptedException { ExternalMessageDto externalMessage = createExternalMessage(m -> m.setPersonNationalHealthId(null)); From f22a66bb2f1a96aef09318ab6f02c7b300699dec Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Mon, 21 Oct 2024 13:55:53 +0300 Subject: [PATCH 11/56] #13159 Automatically (Soft-)Delete Samples & Pathogen Tests with Negative Test Results for COVID-19 --- sormas-base/setup/sormas.properties | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/sormas-base/setup/sormas.properties b/sormas-base/setup/sormas.properties index d5f57df3d3e..287bf8b66f5 100644 --- a/sormas-base/setup/sormas.properties +++ b/sormas-base/setup/sormas.properties @@ -140,6 +140,10 @@ app.url= # Number of days after which system events are deleted from the database. An example for a system event is the last date at which data from an external service was pulled. # default: 90 #daysAfterSystemEventGetsDeleted=90 +# Number of days after which negative CORONAVIRUS pathogen tests and their samples are soft deleted +# default: not set, meaning no delete +# possible values: any integer representing the number of days +# negativeCovidSamplesMaxAgeDays # The similarity threshold after which two names are identified as similar enough to consider them for duplicate detection. # The default value should work for most servers. If you need to change it, please change it carefully as slightly higher or lower values already lead to significant differences. From 8bb7730a15134a85c8ae7f0693c694a1ce040bf3 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Wed, 23 Oct 2024 10:48:15 +0300 Subject: [PATCH 12/56] #13168 Normalization of Automatic Processing for Name, Birthdate, and Address Matching - extended test --- .../AutomaticLabMessageProcessorTest.java | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java index dd55f0c32cf..e3e22f325c7 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java @@ -40,6 +40,7 @@ import de.symeda.sormas.api.externalmessage.processing.ExternalMessageProcessingResult; import de.symeda.sormas.api.infrastructure.facility.FacilityDto; import de.symeda.sormas.api.infrastructure.facility.FacilityType; +import de.symeda.sormas.api.person.PersonCriteria; import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.person.Sex; import de.symeda.sormas.api.sample.PathogenTestDto; @@ -314,20 +315,23 @@ public void testProcessWithExistingPersonWithSameNationalHealthIdButDifferentDet public void testProcessWithExistingPersonWithSameNationalHealthIdAndPersonDetailsNormalizedCheck() throws ExecutionException, InterruptedException { ExternalMessageDto externalMessage = createExternalMessage(m -> { - m.setPersonCity("person city"); - m.setPersonStreet("PERSON STREET, 12a"); + m.setPersonFirstName("john vander"); + m.setPersonLastName("DOÉ"); + m.setPersonCity(" PERSON city \n"); + m.setPersonStreet(" person STREET 12A"); }); - creator.createPerson("john", "DOÉ", Sex.MALE, p -> { + creator.createPerson("John Van Der", "Doe", Sex.MALE, p -> { p.setNationalHealthId(externalMessage.getPersonNationalHealthId()); - p.getAddress().setCity(" PERSON city \n"); - p.getAddress().setStreet(" person STREET 12A"); + p.getAddress().setCity("person city"); + p.getAddress().setStreet("PERSON STREET, 12a"); }); ProcessingResult result = runFlow(externalMessage); assertThat(result.getStatus(), is(DONE)); assertThat(externalMessage.getStatus(), is(ExternalMessageStatus.PROCESSED)); assertThat(getExternalMessageFacade().getByUuid(externalMessage.getUuid()).getStatus(), is(ExternalMessageStatus.PROCESSED)); + assertThat(getPersonFacade().count(new PersonCriteria()), is(1L)); } @Test From 322d5f708b4320943456c6021d3b187ff5457164 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Wed, 23 Oct 2024 16:08:07 +0300 Subject: [PATCH 13/56] #13093 - Update Data Protection for certain Data Fields --- .../symeda/sormas/api/caze/CaseDataDto.java | 3 + .../de/symeda/sormas/api/event/EventDto.java | 4 + .../de/symeda/sormas/api/i18n/Strings.java | 1 + .../sormas/api/location/LocationDto.java | 6 +- .../symeda/sormas/api/utils/PersonalData.java | 2 + .../sormas/api/utils/SensitiveData.java | 9 +-- .../AnnotationBasedFieldAccessChecker.java | 2 +- .../PersonalDataFieldAccessChecker.java | 42 ++++++++-- .../SensitiveDataFieldAccessChecker.java | 45 +++++++++-- .../src/main/resources/strings.properties | 1 + .../sormas/backend/caze/CaseFacadeEjb.java | 7 +- .../PortHealthInfoFacadeEjb.java | 5 +- .../SurveillanceReportFacadeEjb.java | 5 +- .../ClinicalVisitFacadeEjb.java | 17 ++-- .../backend/common/AbstractBaseEjb.java | 14 ++-- .../backend/contact/ContactFacadeEjb.java | 5 +- .../backend/document/DocumentFacadeEjb.java | 9 ++- .../event/EventParticipantFacadeEjb.java | 5 +- .../immunization/ImmunizationFacadeEjb.java | 7 +- .../sormas/backend/info/EntityColumn.java | 25 +++++- .../ManualMessageLogFacadeEjb.java | 6 +- .../backend/person/PersonFacadeEjb.java | 5 +- .../sample/AdditionalTestFacadeEjb.java | 5 +- .../backend/sample/PathogenTestFacadeEjb.java | 5 +- .../sormas/backend/sample/SampleService.java | 12 ++- .../share/ExternalShareInfoFacadeEjb.java | 6 +- .../sormas/backend/task/TaskFacadeEjb.java | 6 +- .../therapy/PrescriptionFacadeEjb.java | 15 ++-- .../backend/therapy/TreatmentFacadeEjb.java | 15 ++-- .../travelentry/TravelEntryFacadeEjb.java | 5 +- .../sormas/backend/util/Pseudonymizer.java | 65 +++++++++------ .../vaccination/VaccinationFacadeEjb.java | 5 +- .../sormas/backend/visit/VisitFacadeEjb.java | 8 +- .../sormas/ui/action/ActionEditForm.java | 7 +- .../symeda/sormas/ui/caze/CaseDataForm.java | 4 +- .../clinicalcourse/HealthConditionsForm.java | 79 +++++++++++-------- .../sormas/ui/events/EventDataForm.java | 2 +- .../symeda/sormas/ui/task/TaskEditForm.java | 7 +- .../sormas/ui/utils/AbstractEditForm.java | 4 + .../sormas/ui/utils/NullableOptionGroup.java | 10 +++ .../sormas/ui/utils/RichTextAreaCustom.java | 24 ++++++ .../utils/SormasFieldGroupFieldFactory.java | 2 + .../symeda/sormas/ui/utils/VaadinUiUtil.java | 65 +++++++++++++++ 43 files changed, 439 insertions(+), 137 deletions(-) create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java index 8500490afa3..224afe0be89 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java @@ -346,6 +346,7 @@ public class CaseDataDto extends SormasToSormasShareableDto implements IsCase { @Valid @EmbeddedPersonalData @EmbeddedSensitiveData + @SensitiveData private HealthConditionsDto healthConditions; private YesNoUnknown pregnant; @@ -509,7 +510,9 @@ public class CaseDataDto extends SormasToSormasShareableDto implements IsCase { COUNTRY_CODE_GERMANY, COUNTRY_CODE_SWITZERLAND }) private Date quarantineOfficialOrderSentDate; + @SensitiveData private YesNoUnknown postpartum; + @SensitiveData private Trimester trimester; private FollowUpStatus followUpStatus; @SensitiveData diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventDto.java index 6db5fa85872..648098f13be 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventDto.java @@ -167,12 +167,16 @@ public class EventDto extends SormasToSormasShareableDto { private InstitutionalPartnerType srcInstitutionalPartnerType; @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String srcInstitutionalPartnerTypeDetails; + @SensitiveData @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String srcFirstName; + @SensitiveData @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String srcLastName; + @SensitiveData @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String srcTelNo; + @SensitiveData @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String srcEmail; @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index ed87fae2ee3..e77b185d269 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -1247,6 +1247,7 @@ public interface Strings { String messageCountImmunizationsNotRestored = "messageCountImmunizationsNotRestored"; String messageCountriesArchived = "messageCountriesArchived"; String messageCountriesDearchived = "messageCountriesDearchived"; + String messageCountriesExcludedFromDataProtection = "messageCountriesExcludedFromDataProtection"; String messageCountryArchived = "messageCountryArchived"; String messageCountryDearchived = "messageCountryDearchived"; String messageCountryDearchivingNotPossible = "messageCountryDearchivingNotPossible"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/location/LocationDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/location/LocationDto.java index 576e6085873..5f146a7bf22 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/location/LocationDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/location/LocationDto.java @@ -101,8 +101,10 @@ public class LocationDto extends PseudonymizableDto { CountryHelper.COUNTRY_CODE_GERMANY, CountryHelper.COUNTRY_CODE_FRANCE }) private String details; - @PersonalData - @SensitiveData + @PersonalData(excludeForCountries = { + CountryHelper.COUNTRY_CODE_LUXEMBOURG }) + @SensitiveData(excludeForCountries = { + CountryHelper.COUNTRY_CODE_LUXEMBOURG }) @Size(max = FieldConstraints.CHARACTER_LIMIT_DEFAULT, message = Validations.textTooLong) private String city; @PersonalData diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/PersonalData.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/PersonalData.java index c96ed3e9295..4d81471a021 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/PersonalData.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/PersonalData.java @@ -27,4 +27,6 @@ public @interface PersonalData { boolean mandatoryField() default false; + + String[] excludeForCountries() default {}; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/SensitiveData.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/SensitiveData.java index 6ce06fd20d8..9602b676142 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/SensitiveData.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/SensitiveData.java @@ -1,19 +1,16 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2020 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License - * along with this program. If not, see . + * along with this program. If not, see . */ package de.symeda.sormas.api.utils; @@ -28,4 +25,6 @@ public @interface SensitiveData { boolean mandatoryField() default false; + + String[] excludeForCountries() default {}; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/AnnotationBasedFieldAccessChecker.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/AnnotationBasedFieldAccessChecker.java index 402ac5219da..d87b4038db4 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/AnnotationBasedFieldAccessChecker.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/AnnotationBasedFieldAccessChecker.java @@ -22,7 +22,7 @@ public abstract class AnnotationBasedFieldAccessChecker implements FieldAccessChecker { - private final Class fieldAnnotation; + protected final Class fieldAnnotation; private final Class embeddedAnnotation; private final boolean hasRight; private final SpecialAccessCheck specialAccessCheck; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PersonalDataFieldAccessChecker.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PersonalDataFieldAccessChecker.java index be838b8a0eb..cc38ed1c6f5 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PersonalDataFieldAccessChecker.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PersonalDataFieldAccessChecker.java @@ -18,6 +18,7 @@ package de.symeda.sormas.api.utils.fieldaccess.checkers; import java.lang.reflect.Field; +import java.util.Arrays; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.EmbeddedPersonalData; @@ -25,20 +26,32 @@ public final class PersonalDataFieldAccessChecker extends AnnotationBasedFieldAccessChecker { - private PersonalDataFieldAccessChecker(final boolean hasRight, SpecialAccessCheck specialAccessCheck) { + private final String serverCountry; + + private PersonalDataFieldAccessChecker(final boolean hasRight, SpecialAccessCheck specialAccessCheck, String serverCountry) { super(PersonalData.class, EmbeddedPersonalData.class, hasRight, specialAccessCheck); + this.serverCountry = serverCountry; } - public static PersonalDataFieldAccessChecker inJurisdiction(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck) { - return new PersonalDataFieldAccessChecker<>(rightCheck.check(UserRight.SEE_PERSONAL_DATA_IN_JURISDICTION), specialAccessCheck); + public static PersonalDataFieldAccessChecker inJurisdiction( + RightCheck rightCheck, + SpecialAccessCheck specialAccessCheck, + String serverCountry) { + return new PersonalDataFieldAccessChecker<>(rightCheck.check(UserRight.SEE_PERSONAL_DATA_IN_JURISDICTION), specialAccessCheck, serverCountry); } - public static PersonalDataFieldAccessChecker outsideJurisdiction(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck) { - return new PersonalDataFieldAccessChecker<>(rightCheck.check(UserRight.SEE_PERSONAL_DATA_OUTSIDE_JURISDICTION), specialAccessCheck); + public static PersonalDataFieldAccessChecker outsideJurisdiction( + RightCheck rightCheck, + SpecialAccessCheck specialAccessCheck, + String serverCountry) { + return new PersonalDataFieldAccessChecker<>( + rightCheck.check(UserRight.SEE_PERSONAL_DATA_OUTSIDE_JURISDICTION), + specialAccessCheck, + serverCountry); } public static PersonalDataFieldAccessChecker forcedNoAccess() { - return new PersonalDataFieldAccessChecker<>(false, t -> false); + return new PersonalDataFieldAccessChecker<>(false, t -> false, null); } @Override @@ -46,6 +59,23 @@ protected boolean isAnnotatedFieldMandatory(Field annotatedField) { return annotatedField.getAnnotation(PersonalData.class).mandatoryField(); } + @Override + public boolean isConfiguredForCheck(Field field, boolean withMandatory) { + boolean annotationPresent = field.isAnnotationPresent(fieldAnnotation); + + if (annotationPresent) { + String[] excludeForCountries = field.getAnnotation(PersonalData.class).excludeForCountries(); + if (Arrays.asList(excludeForCountries).contains(serverCountry)) { + return false; + } + } + + if (!annotationPresent || withMandatory) { + return annotationPresent; + } + return !isAnnotatedFieldMandatory(field); + } + public interface RightCheck { boolean check(UserRight userRight); diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/SensitiveDataFieldAccessChecker.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/SensitiveDataFieldAccessChecker.java index fb0c1f68005..d5b716c26a3 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/SensitiveDataFieldAccessChecker.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/SensitiveDataFieldAccessChecker.java @@ -16,6 +16,7 @@ package de.symeda.sormas.api.utils.fieldaccess.checkers; import java.lang.reflect.Field; +import java.util.Arrays; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.EmbeddedSensitiveData; @@ -23,20 +24,35 @@ public final class SensitiveDataFieldAccessChecker extends AnnotationBasedFieldAccessChecker { - private SensitiveDataFieldAccessChecker(final boolean hasRight, SpecialAccessCheck specialAccessCheck) { + private final String serverCountry; + + private SensitiveDataFieldAccessChecker(final boolean hasRight, SpecialAccessCheck specialAccessCheck, String serverCountry) { super(SensitiveData.class, EmbeddedSensitiveData.class, hasRight, specialAccessCheck); + this.serverCountry = serverCountry; } - public static SensitiveDataFieldAccessChecker inJurisdiction(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck) { - return new SensitiveDataFieldAccessChecker<>(rightCheck.check(UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION), specialAccessCheck); + public static SensitiveDataFieldAccessChecker inJurisdiction( + RightCheck rightCheck, + SpecialAccessCheck specialAccessCheck, + String serverCountry) { + return new SensitiveDataFieldAccessChecker<>( + rightCheck.check(UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION), + specialAccessCheck, + serverCountry); } - public static SensitiveDataFieldAccessChecker outsideJurisdiction(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck) { - return new SensitiveDataFieldAccessChecker<>(rightCheck.check(UserRight.SEE_SENSITIVE_DATA_OUTSIDE_JURISDICTION), specialAccessCheck); + public static SensitiveDataFieldAccessChecker outsideJurisdiction( + RightCheck rightCheck, + SpecialAccessCheck specialAccessCheck, + String serverCountry) { + return new SensitiveDataFieldAccessChecker<>( + rightCheck.check(UserRight.SEE_SENSITIVE_DATA_OUTSIDE_JURISDICTION), + specialAccessCheck, + serverCountry); } public static SensitiveDataFieldAccessChecker forcedNoAccess() { - return new SensitiveDataFieldAccessChecker<>(false, t -> false); + return new SensitiveDataFieldAccessChecker<>(false, t -> false, null); } @Override @@ -44,6 +60,23 @@ protected boolean isAnnotatedFieldMandatory(Field annotatedField) { return annotatedField.getAnnotation(SensitiveData.class).mandatoryField(); } + @Override + public boolean isConfiguredForCheck(Field field, boolean withMandatory) { + boolean annotationPresent = field.isAnnotationPresent(fieldAnnotation); + + if (annotationPresent) { + String[] excludeForCountries = field.getAnnotation(SensitiveData.class).excludeForCountries(); + if (Arrays.asList(excludeForCountries).contains(serverCountry)) { + return false; + } + } + + if (!annotationPresent || withMandatory) { + return annotationPresent; + } + return !isAnnotatedFieldMandatory(field); + } + public interface RightCheck { boolean check(UserRight userRight); diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index fc1854df695..2a02c9a7230 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -1522,6 +1522,7 @@ messageExternalEmailAttachmentPassword=Please use this password to open the docu messageExternalEmailAttachmentNotAvailableInfo=Attaching documents is disabled because encryption would not be possible. To encrypt documents, the person needs to have either a national health ID specified, or a primary mobile phone number set with SMS sending set up on this system. messagePersonNationalHealthIdInvalid=The entered national health ID does not seem to be correct messageSyncUsersFromAuthProviderConfigurationError=Syncing users from authentication provider is not possible because the configuration is incorrect. Please contact an admin and inform them about this issue. +messageCountriesExcludedFromDataProtection=Countries excluded from data protection for this field: # Notifications notificationCaseClassificationChanged = The classification of case %s has changed to %s. diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java index d72209821be..e0030adea8e 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java @@ -4168,7 +4168,8 @@ public List getCaseFollowUpList( private Pseudonymizer createPseudonymizerForDtoWithClinician( @Nullable String pseudonymizedValue, Collection cases) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, createSpecialAccessChecker(cases), pseudonymizedValue); + Pseudonymizer pseudonymizer = + Pseudonymizer.getDefault(userService, createSpecialAccessChecker(cases), pseudonymizedValue, configFacade.getCountryCode()); UserRightFieldAccessChecker clinicianViewRightChecker = new UserRightFieldAccessChecker<>(UserRight.CASE_CLINICIAN_VIEW, userService.hasRight(UserRight.CASE_CLINICIAN_VIEW)); @@ -4179,12 +4180,12 @@ private Pseudonymizer createPseudonymizerForDtoWithClinici @PermitAll public Pseudonymizer createSimplePseudonymizer(Collection cases) { - return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(cases)); + return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(cases), configFacade.getCountryCode()); } @PermitAll public Pseudonymizer createSimplePlaceholderPseudonymizer(Collection cases) { - return Pseudonymizer.getDefaultWithPlaceHolder(userService, createSpecialAccessChecker(cases)); + return Pseudonymizer.getDefaultWithPlaceHolder(userService, createSpecialAccessChecker(cases), configFacade.getCountryCode()); } private AnnotationBasedFieldAccessChecker.SpecialAccessCheck createSpecialAccessChecker( diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/porthealthinfo/PortHealthInfoFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/porthealthinfo/PortHealthInfoFacadeEjb.java index 941f1d44268..e42bb80b028 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/porthealthinfo/PortHealthInfoFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/porthealthinfo/PortHealthInfoFacadeEjb.java @@ -7,6 +7,7 @@ import de.symeda.sormas.api.caze.porthealthinfo.PortHealthInfoDto; import de.symeda.sormas.api.caze.porthealthinfo.PortHealthInfoFacade; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.user.UserService; import de.symeda.sormas.backend.util.DtoHelper; import de.symeda.sormas.backend.util.Pseudonymizer; @@ -18,6 +19,8 @@ public class PortHealthInfoFacadeEjb implements PortHealthInfoFacade { private PortHealthInfoService portHealthInfoService; @EJB private UserService userService; + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; public static PortHealthInfoDto toDto(PortHealthInfo source) { if (source == null) { @@ -89,7 +92,7 @@ public PortHealthInfoDto getByCaseUuid(String caseUuid) { } private PortHealthInfoDto toPseudonymizedDto(PortHealthInfo portHealthInfo) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); PortHealthInfoDto portHealthInfoDto = toDto(portHealthInfo); pseudonymizer.pseudonymizeDto(PortHealthInfoDto.class, portHealthInfoDto, portHealthInfoService.inJurisdictionOrOwned(portHealthInfo), null); return portHealthInfoDto; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/surveillancereport/SurveillanceReportFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/surveillancereport/SurveillanceReportFacadeEjb.java index 53d7addf02e..4d9bc572eb7 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/surveillancereport/SurveillanceReportFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/surveillancereport/SurveillanceReportFacadeEjb.java @@ -51,6 +51,7 @@ import de.symeda.sormas.backend.caze.CaseFacadeEjb; import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractBaseEjb; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.externalmessage.ExternalMessageFacadeEjb.ExternalMessageFacadeEjbLocal; import de.symeda.sormas.backend.infrastructure.district.DistrictFacadeEjb; import de.symeda.sormas.backend.infrastructure.district.DistrictService; @@ -102,6 +103,8 @@ public class SurveillanceReportFacadeEjb private SormasToSormasCaseFacadeEjbLocal sormasToSormasCaseFacade; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; public SurveillanceReportFacadeEjb() { super(); @@ -198,7 +201,7 @@ protected SurveillanceReportReferenceDto toRefDto(SurveillanceReport surveillanc @Override protected Pseudonymizer createPseudonymizer(List surveillanceReports) { List withSpecialAccess = specialCaseAccessService.getSurveillanceReportUuidsWithSpecialAccess(surveillanceReports); - return Pseudonymizer.getDefault(userService, r -> withSpecialAccess.contains(r.getUuid())); + return Pseudonymizer.getDefault(userService, r -> withSpecialAccess.contains(r.getUuid()), configFacade.getCountryCode()); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/clinicalcourse/ClinicalVisitFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/clinicalcourse/ClinicalVisitFacadeEjb.java index 4133237fad0..d830f08488d 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/clinicalcourse/ClinicalVisitFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/clinicalcourse/ClinicalVisitFacadeEjb.java @@ -42,6 +42,7 @@ import de.symeda.sormas.backend.caze.CaseFacadeEjb.CaseFacadeEjbLocal; import de.symeda.sormas.backend.caze.CaseQueryContext; import de.symeda.sormas.backend.caze.CaseService; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.person.Person; import de.symeda.sormas.backend.symptoms.Symptoms; @@ -78,6 +79,8 @@ public class ClinicalVisitFacadeEjb implements ClinicalVisitFacade { private CaseService caseService; @EJB private SymptomsService symptomsService; + @EJB + private ConfigFacadeEjbLocal configFacade; // private String countPositiveSymptomsQuery; @@ -159,7 +162,7 @@ public List getIndexList(ClinicalVisitCriteria criteria, List results = QueryHelper.getResultList(em, cq, first, max); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); pseudonymizer.pseudonymizeDtoCollection(ClinicalVisitIndexDto.class, results, ClinicalVisitIndexDto::getInJurisdiction, null); // Build the query to count positive symptoms @@ -209,7 +212,7 @@ public List getIndexList(ClinicalVisitCriteria criteria, @Override public ClinicalVisitDto getClinicalVisitByUuid(String uuid) { - return convertToDto(service.getByUuid(uuid), Pseudonymizer.getDefault(userService)); + return convertToDto(service.getByUuid(uuid), Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } @Override @@ -244,7 +247,7 @@ private ClinicalVisitDto saveClinicalVisit(ClinicalVisitDto clinicalVisit, Strin caseFacade.save(caze); } - return convertToDto(entity, Pseudonymizer.getDefault(userService)); + return convertToDto(entity, Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } /** @@ -286,7 +289,7 @@ public List getAllActiveClinicalVisitsAfter(Date date, Integer private List toPseudonymizedDtos(List entities) { List inJurisdictionIds = service.getInJurisdictionIds(entities); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); List dtos = entities.stream().map(p -> convertToDto(p, pseudonymizer, inJurisdictionIds.contains(p.getId()))).collect(Collectors.toList()); return dtos; @@ -294,7 +297,7 @@ private List toPseudonymizedDtos(List entities) @Override public List getByUuids(List uuids) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); return service.getByUuids(uuids).stream().map(t -> convertToDto(t, pseudonymizer)).collect(Collectors.toList()); } @@ -340,7 +343,7 @@ public List getExportList(CaseCriteria criteria, Collect List resultList = QueryHelper.getResultList(em, cq, first, max); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); for (ClinicalVisitExportDto exportDto : resultList) { exportDto.setSymptoms(SymptomsFacadeEjb.toSymptomsDto(symptomsService.getById(exportDto.getSymptomsId()))); @@ -382,7 +385,7 @@ private void pseudonymizeDto(ClinicalVisit source, ClinicalVisitDto dto, Pseudon private void restorePseudonymizedDto(ClinicalVisitDto clinicalVisit, ClinicalVisit existingClinicalVisit) { if (existingClinicalVisit != null) { boolean inJurisdiction = caseService.inJurisdictionOrOwned(existingClinicalVisit.getClinicalCourse().getCaze()); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); ClinicalVisitDto existingDto = toDto(existingClinicalVisit); pseudonymizer.restorePseudonymizedValues(ClinicalVisitDto.class, clinicalVisit, existingDto, inJurisdiction); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AbstractBaseEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AbstractBaseEjb.java index 07b968e4974..c6bdfd09d65 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AbstractBaseEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/AbstractBaseEjb.java @@ -35,6 +35,8 @@ public abstract class AbstractBaseEjb adoClass; protected Class dtoClass; + @Inject + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; protected AbstractBaseEjb() { } @@ -141,23 +143,25 @@ protected Pseudonymizer createPseudonymizer(ADO ado) { } protected Pseudonymizer createPseudonymizer(List adoList) { - return Pseudonymizer.getDefault(userService); + return Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); } protected Pseudonymizer createGenericPseudonymizer() { - return Pseudonymizer.getDefault(userService); + return Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); } protected Pseudonymizer createGenericPseudonymizer(SpecialAccessCheck specialAccessCheck) { - return Pseudonymizer.getDefault(userService, specialAccessCheck); + return Pseudonymizer.getDefault(userService, specialAccessCheck, configFacade.getCountryCode()); } protected Pseudonymizer createGenericPlaceholderPseudonymizer() { - return Pseudonymizer.getDefault(userService, I18nProperties.getCaption(Captions.inaccessibleValue)); + return Pseudonymizer.getDefault(userService, I18nProperties.getCaption(Captions.inaccessibleValue), configFacade.getCountryCode()); } protected Pseudonymizer createGenericPlaceholderPseudonymizer(SpecialAccessCheck specialAccessCheck) { - return Pseudonymizer.getDefault(userService, specialAccessCheck, I18nProperties.getCaption(Captions.inaccessibleValue)); + Pseudonymizer aDefault = Pseudonymizer + .getDefault(userService, specialAccessCheck, I18nProperties.getCaption(Captions.inaccessibleValue), configFacade.getCountryCode()); + return aDefault; } protected abstract ADO fillOrBuildEntity(@NotNull DTO source, ADO target, boolean checkChangeDate); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java index e09a585e276..a37c6e0de46 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java @@ -161,6 +161,7 @@ import de.symeda.sormas.backend.clinicalcourse.HealthConditionsMapper; import de.symeda.sormas.backend.common.AbstractCoreFacadeEjb; import de.symeda.sormas.backend.common.AbstractDomainObject; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.common.TaskCreationException; import de.symeda.sormas.backend.disease.DiseaseConfigurationFacadeEjb.DiseaseConfigurationFacadeEjbLocal; @@ -291,6 +292,8 @@ public class ContactFacadeEjb private UserRoleFacadeEjb.UserRoleFacadeEjbLocal userRoleFacadeEjb; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; @Resource private ManagedScheduledExecutorService executorService; @@ -1725,7 +1728,7 @@ protected List toPseudonymizedDtos(List adoList) { @Override protected Pseudonymizer createPseudonymizer(List contacts) { - return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(contacts)); + return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(contacts), configFacade.getCountryCode()); } private SpecialAccessCheck createSpecialAccessChecker(Collection contacts) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/document/DocumentFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/document/DocumentFacadeEjb.java index ae27dda00f1..7f5fbb42f81 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/document/DocumentFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/document/DocumentFacadeEjb.java @@ -112,7 +112,7 @@ public class DocumentFacadeEjb implements DocumentFacade { @Override public DocumentDto getDocumentByUuid(String uuid) { - return convertToDto(documentService.getByUuid(uuid), Pseudonymizer.getDefault(userService)); + return convertToDto(documentService.getByUuid(uuid), Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } @Override @@ -157,7 +157,7 @@ public DocumentDto saveDocument(@Valid DocumentDto dto, byte[] content, @Nonnull document.setRelatedEntities(documentRelatedEntitySet); documentService.ensurePersisted(document); - return convertToDto(document, Pseudonymizer.getDefault(userService)); + return convertToDto(document, Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } catch (Throwable t) { try { documentStorageService.delete(storageReference); @@ -209,9 +209,10 @@ public void deleteDocument(String documentUuid, String relatedEntityUuid, Docume documentService.markAsDeleted(document); } } + @Override public List getDocumentsRelatedToEntity(DocumentRelatedEntityType type, String uuid) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); return documentService.getRelatedToEntity(type, uuid).stream().map(d -> convertToDto(d, pseudonymizer)).collect(Collectors.toList()); } @@ -222,7 +223,7 @@ public List getReferencesRelatedToEntity(DocumentRelatedEn @Override public Map> getDocumentsRelatedToEntities(DocumentCriteria criteria, List sortProperties) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); List allDocuments = documentService.getRelatedToEntities(criteria.getDocumentRelatedEntityType(), criteria.getEntityUuids(), sortProperties); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantFacadeEjb.java index 39f4e5766a2..f5526b3588e 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventParticipantFacadeEjb.java @@ -104,6 +104,7 @@ import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractCoreFacadeEjb; import de.symeda.sormas.backend.common.AbstractDomainObject; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.common.NotificationService; import de.symeda.sormas.backend.common.messaging.MessageContents; @@ -180,6 +181,8 @@ public class EventParticipantFacadeEjb private VaccinationService vaccinationService; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; public EventParticipantFacadeEjb() { } @@ -1157,7 +1160,7 @@ public EventParticipant fillOrBuildEntity(@NotNull EventParticipantDto source, E @Override protected Pseudonymizer createPseudonymizer(List eventParticipants) { - return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(eventParticipants)); + return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(eventParticipants), configFacade.getCountryCode()); } private SpecialAccessCheck createSpecialAccessChecker( diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationFacadeEjb.java index f3132958c5b..10265699d6b 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/immunization/ImmunizationFacadeEjb.java @@ -41,6 +41,9 @@ import javax.validation.Valid; import javax.validation.constraints.NotNull; +import de.symeda.sormas.api.ConfigFacade; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -170,6 +173,8 @@ public class ImmunizationFacadeEjb private VaccinationService vaccinationService; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; public ImmunizationFacadeEjb() { } @@ -440,7 +445,7 @@ public ImmunizationDto save(@Valid @NotNull ImmunizationDto dto, boolean checkCh @Override protected Pseudonymizer createPseudonymizer(List immunizations) { - return Pseudonymizer.getDefault(userService, getSpecialAccessChecker(immunizations)); + return Pseudonymizer.getDefault(userService, getSpecialAccessChecker(immunizations), configFacade.getCountryCode()); } private SpecialAccessCheck getSpecialAccessChecker(Collection immunizations) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java index 72d898f1a5f..7661514f8fb 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java @@ -18,6 +18,7 @@ import java.lang.reflect.Field; import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; +import java.util.Arrays; import java.util.Collection; import java.util.Date; import java.util.Map; @@ -171,7 +172,29 @@ private static String getCaption(FieldData fieldData) { } private static String getDescription(FieldData fieldData) { - return I18nProperties.getPrefixDescription(fieldData.getI18NPrefix(), fieldData.getField().getName(), ""); + String prefixDescription = I18nProperties.getPrefixDescription(fieldData.getI18NPrefix(), fieldData.getField().getName(), ""); + + String[] excludedForCountries = null; + + Field field = fieldData.getField(); + if (field.getAnnotation(PersonalData.class) != null) { + excludedForCountries = field.getAnnotation(PersonalData.class).excludeForCountries(); + + } else { + if (field.getAnnotation(SensitiveData.class) != null) { + excludedForCountries = field.getAnnotation(SensitiveData.class).excludeForCountries(); + } + } + + String description; + if (excludedForCountries != null && excludedForCountries.length > 0) { + description = prefixDescription + I18nProperties.getString(Strings.messageCountriesExcludedFromDataProtection) + " " + + Arrays.toString(excludedForCountries); + } else { + description = prefixDescription; + } + + return description; } private static String getNotNull(FieldData fieldData) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/manualmessagelog/ManualMessageLogFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/manualmessagelog/ManualMessageLogFacadeEjb.java index 68546300739..983a2b33d54 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/manualmessagelog/ManualMessageLogFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/manualmessagelog/ManualMessageLogFacadeEjb.java @@ -35,6 +35,7 @@ import de.symeda.sormas.api.manualmessagelog.ManualMessageLogIndexDto; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.fieldaccess.checkers.AnnotationBasedFieldAccessChecker.SpecialAccessCheck; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccessService; import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.user.UserService; @@ -55,6 +56,8 @@ public class ManualMessageLogFacadeEjb implements ManualMessageLogFacade { private UserService userService; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; @RightsAllowed({ UserRight._SEND_MANUAL_EXTERNAL_MESSAGES, @@ -104,7 +107,7 @@ public List getIndexList(ManualMessageLogCriteria crit @NotNull private Pseudonymizer createPseudonymizerWithPlaceholder(Collection manualMessageLogs) { - return Pseudonymizer.getDefaultWithPlaceHolder(userService, getSpecialAccessChecker(manualMessageLogs)); + return Pseudonymizer.getDefaultWithPlaceHolder(userService, getSpecialAccessChecker(manualMessageLogs), configFacade.getCountryCode()); } private SpecialAccessCheck getSpecialAccessChecker(Collection manualMessageLogs) { @@ -113,7 +116,6 @@ private SpecialAccessCheck getSpecialAccessChecker(Col return i -> specialAccessUuids.contains(i.getUuid()); } - @LocalBean @Stateless public static class ManualMessageLogFacadeEjbLocal extends ManualMessageLogFacadeEjb { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java index 0c9d9f9f17f..2266e8b4f51 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java @@ -134,6 +134,7 @@ import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractBaseEjb; import de.symeda.sormas.backend.common.AbstractDomainObject; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.contact.Contact; import de.symeda.sormas.backend.contact.ContactFacadeEjb; @@ -258,6 +259,8 @@ public class PersonFacadeEjb extends AbstractBaseEjb getIndexPage(PersonCriteria personCriteria, Integer @Override protected Pseudonymizer createPseudonymizer(List persons) { - return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(persons)); + return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(persons), configFacade.getCountryCode()); } private SpecialAccessCheck createSpecialAccessChecker(Collection persons) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/AdditionalTestFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/AdditionalTestFacadeEjb.java index 6e553c5bd93..394de84a91b 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/AdditionalTestFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/AdditionalTestFacadeEjb.java @@ -20,6 +20,7 @@ import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.SortProperty; import de.symeda.sormas.backend.FacadeHelper; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.user.UserService; import de.symeda.sormas.backend.util.DtoHelper; @@ -38,6 +39,8 @@ public class AdditionalTestFacadeEjb implements AdditionalTestFacade { private SampleService sampleService; @EJB private UserService userService; + @EJB + private ConfigFacadeEjbLocal configFacade; @Override public AdditionalTestDto getByUuid(String uuid) { @@ -118,7 +121,7 @@ public List getAllActiveAdditionalTestsAfter(Date date, Integ private List toPseudonymizedDtos(List entities) { List inJurisdictionIds = service.getInJurisdictionIds(entities); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); return entities.stream().map(p -> convertToDto(p, pseudonymizer, inJurisdictionIds.contains(p.getId()))).collect(Collectors.toList()); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTestFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTestFacadeEjb.java index 4dc4a8cbe50..46a80c89851 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTestFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTestFacadeEjb.java @@ -60,6 +60,7 @@ import de.symeda.sormas.backend.FacadeHelper; import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.caze.CaseFacadeEjb.CaseFacadeEjbLocal; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CoreAdo; import de.symeda.sormas.backend.common.NotificationService; import de.symeda.sormas.backend.common.messaging.MessageContents; @@ -120,6 +121,8 @@ public class PathogenTestFacadeEjb implements PathogenTestFacade { private CountryService countryService; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; @Override public List getAllActiveUuids() { @@ -485,7 +488,7 @@ private static void pseudonymizeDto( } private Pseudonymizer createPseudonymizer(Collection tests) { - return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(tests)); + return Pseudonymizer.getDefault(userService, createSpecialAccessChecker(tests), configFacade.getCountryCode()); } private SpecialAccessCheck createSpecialAccessChecker(Collection tests) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java index 1295c31c7e8..1c7caa51825 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java @@ -87,6 +87,7 @@ import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractDeletableAdoService; import de.symeda.sormas.backend.common.AbstractDomainObject; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.common.DeletableAdo; import de.symeda.sormas.backend.common.JurisdictionFlagsService; @@ -145,6 +146,8 @@ public class SampleService extends AbstractDeletableAdoService protected FeatureConfigurationFacadeEjbLocal featureConfigurationFacade; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjbLocal configFacade; public SampleService() { super(Sample.class, DeletableEntityType.SAMPLE); @@ -324,12 +327,15 @@ public SamplePseudonymizer createPseudonymizer(boolean w AnnotationBasedFieldAccessChecker.SpecialAccessCheck specialAccessChecker = createSpecialAccessChecker(samples); Pseudonymizer rootPseudonymizer = withPlaceHolder - ? Pseudonymizer.getDefaultWithPlaceHolder(userService, specialAccessChecker) - : Pseudonymizer.getDefault(userService, specialAccessChecker); + ? Pseudonymizer.getDefaultWithPlaceHolder(userService, specialAccessChecker, configFacade.getCountryCode()) + : Pseudonymizer.getDefault(userService, specialAccessChecker, configFacade.getCountryCode()); Collection cases = samples.stream().map(IsSample::getAssociatedCase).filter(Objects::nonNull).collect(Collectors.toList()); - return new SamplePseudonymizer<>(rootPseudonymizer, caseFacade.createSimplePseudonymizer(cases), Pseudonymizer.getDefault(userService)); + return new SamplePseudonymizer<>( + rootPseudonymizer, + caseFacade.createSimplePseudonymizer(cases), + Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } private AnnotationBasedFieldAccessChecker.SpecialAccessCheck createSpecialAccessChecker( diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/share/ExternalShareInfoFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/share/ExternalShareInfoFacadeEjb.java index 099709e5e4f..056b9582832 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/share/ExternalShareInfoFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/share/ExternalShareInfoFacadeEjb.java @@ -36,6 +36,7 @@ import de.symeda.sormas.api.share.ExternalShareInfoFacade; import de.symeda.sormas.api.share.ExternalShareStatus; import de.symeda.sormas.backend.caze.Case; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.event.Event; import de.symeda.sormas.backend.user.UserFacadeEjb; @@ -57,6 +58,9 @@ public class ExternalShareInfoFacadeEjb implements ExternalShareInfoFacade { @EJB private UserService userService; + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; + @Override public List getIndexList(ExternalShareInfoCriteria criteria, Integer first, Integer max) { final CriteriaBuilder cb = em.getCriteriaBuilder(); @@ -78,7 +82,7 @@ public List getIndexList(ExternalShareInfoCriteria criteri List shareInfoList = QueryHelper.getResultList(em, cq, first, max); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); return shareInfoList.stream().map(i -> convertToDto(i, pseudonymizer)).collect(Collectors.toList()); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/task/TaskFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/task/TaskFacadeEjb.java index 8aa6c858e34..6067f8e63f3 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/task/TaskFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/task/TaskFacadeEjb.java @@ -358,10 +358,10 @@ private TaskPseudonymizer createPseudonymizer(boolean with return new TaskPseudonymizer<>( withPlaceHolder - ? Pseudonymizer.getDefaultWithPlaceHolder(userService, specialAccessCheck) - : Pseudonymizer.getDefault(userService, specialAccessCheck), + ? Pseudonymizer.getDefaultWithPlaceHolder(userService, specialAccessCheck, configFacade.getCountryCode()) + : Pseudonymizer.getDefault(userService, specialAccessCheck, configFacade.getCountryCode()), caseFacade.createSimplePseudonymizer(associatedCases), - Pseudonymizer.getDefault(userService)); + Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/PrescriptionFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/PrescriptionFacadeEjb.java index f10a766ad9f..188e78bec4f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/PrescriptionFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/PrescriptionFacadeEjb.java @@ -30,6 +30,7 @@ import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.caze.CaseQueryContext; import de.symeda.sormas.backend.caze.CaseService; +import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.person.Person; import de.symeda.sormas.backend.user.User; @@ -56,6 +57,8 @@ public class PrescriptionFacadeEjb implements PrescriptionFacade { private TherapyService therapyService; @EJB private CaseService caseService; + @EJB + private ConfigFacadeEjbLocal configFacade; @Override public List getIndexList(PrescriptionCriteria criteria) { @@ -87,7 +90,7 @@ public List getIndexList(PrescriptionCriteria criteria) { List indexList = em.createQuery(cq).getResultList(); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); pseudonymizer.pseudonymizeDtoCollection(PrescriptionIndexDto.class, indexList, PrescriptionIndexDto::getInJurisdiction, null); return indexList; @@ -95,7 +98,7 @@ public List getIndexList(PrescriptionCriteria criteria) { @Override public PrescriptionDto getPrescriptionByUuid(String uuid) { - return convertToDto(service.getByUuid(uuid), Pseudonymizer.getDefault(userService)); + return convertToDto(service.getByUuid(uuid), Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } @Override @@ -114,7 +117,7 @@ public PrescriptionDto savePrescription(@Valid PrescriptionDto prescription) { service.ensurePersisted(entity); - return convertToDto(entity, Pseudonymizer.getDefault(userService)); + return convertToDto(entity, Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } @Override @@ -143,7 +146,7 @@ public List getAllActivePrescriptionsAfter(Date date, Integer b private List toPseudonymizedDtos(List entities) { List inJurisdictionIds = service.getInJurisdictionIds(entities); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); return entities.stream().map(p -> convertToDto(p, pseudonymizer, inJurisdictionIds.contains(p.getId()))).collect(Collectors.toList()); } @@ -202,7 +205,7 @@ public List getExportList(CaseCriteria criteria, Collecti List exportList = QueryHelper.getResultList(em, cq, first, max); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); pseudonymizer.pseudonymizeDtoCollection(PrescriptionExportDto.class, exportList, PrescriptionExportDto::getInJurisdiction, null); return exportList; @@ -233,7 +236,7 @@ private void pseudonymizeDto(Prescription source, PrescriptionDto dto, Pseudonym private void restorePseudonymizedDto(PrescriptionDto prescription, Prescription existingPrescription, PrescriptionDto existingPrescriptionDto) { if (existingPrescription != null) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); pseudonymizer.restorePseudonymizedValues( PrescriptionDto.class, prescription, diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/TreatmentFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/TreatmentFacadeEjb.java index e71060be315..85cf01f177a 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/TreatmentFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/therapy/TreatmentFacadeEjb.java @@ -29,6 +29,7 @@ import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.caze.CaseQueryContext; import de.symeda.sormas.backend.caze.CaseService; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.person.Person; import de.symeda.sormas.backend.user.User; @@ -57,6 +58,8 @@ public class TreatmentFacadeEjb implements TreatmentFacade { private PrescriptionService prescriptionService; @EJB private CaseService caseService; + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; @Override public List getIndexList(TreatmentCriteria criteria) { @@ -86,7 +89,7 @@ public List getIndexList(TreatmentCriteria criteria) { List indexList = em.createQuery(cq).getResultList(); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); pseudonymizer.pseudonymizeDtoCollection(TreatmentIndexDto.class, indexList, TreatmentIndexDto::getInJurisdiction, null); return indexList; @@ -122,7 +125,7 @@ public List getTreatmentForPrescription(List prescrip List treatmentIndexDtos = em.createQuery(cq).getResultList(); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); pseudonymizer.pseudonymizeDtoCollection(TreatmentIndexDto.class, treatmentIndexDtos, TreatmentIndexDto::getInJurisdiction, null); return treatmentIndexDtos; @@ -130,7 +133,7 @@ public List getTreatmentForPrescription(List prescrip @Override public TreatmentDto getTreatmentByUuid(String uuid) { - return convertToDto(service.getByUuid(uuid), Pseudonymizer.getDefault(userService)); + return convertToDto(service.getByUuid(uuid), Pseudonymizer.getDefault(userService, configFacade.getCountryCode())); } @Override @@ -195,7 +198,7 @@ public List getByUuids(List uuids) { private List toPseudonymizedDtos(List entities) { List inJurisdictionIds = service.getInJurisdictionIds(entities); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); return entities.stream().map(p -> convertToDto(p, pseudonymizer, inJurisdictionIds.contains(p.getId()))).collect(Collectors.toList()); } @@ -245,7 +248,7 @@ public List getExportList(CaseCriteria criteria, Collection< List exportList = QueryHelper.getResultList(em, cq, first, max); - Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefaultWithPlaceHolder(userService, configFacade.getCountryCode()); pseudonymizer.pseudonymizeDtoCollection(TreatmentExportDto.class, exportList, TreatmentExportDto::getInJurisdiction, null); return exportList; @@ -276,7 +279,7 @@ private void pseudonymizeDto(Treatment source, TreatmentDto dto, Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService); + Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, configFacade.getCountryCode()); pseudonymizer.restorePseudonymizedValues(TreatmentDto.class, source, existingDto, service.inJurisdictionOrOwned(existingTreatment)); } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntryFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntryFacadeEjb.java index e311abde4f4..accd4e2dc1f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntryFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntryFacadeEjb.java @@ -50,6 +50,7 @@ import de.symeda.sormas.backend.caze.CaseFacadeEjb; import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractCoreFacadeEjb; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.infrastructure.community.CommunityFacadeEjb; import de.symeda.sormas.backend.infrastructure.community.CommunityService; import de.symeda.sormas.backend.infrastructure.district.DistrictFacadeEjb; @@ -96,6 +97,8 @@ public class TravelEntryFacadeEjb private TravelEntryService travelEntryService; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; public TravelEntryFacadeEjb() { } @@ -223,7 +226,7 @@ public long count(TravelEntryCriteria criteria, boolean ignoreUserFilter) { @Override protected Pseudonymizer createPseudonymizer(List travelEntries) { - return Pseudonymizer.getDefault(userService, getSpecialAccessChecker(travelEntries)); + return Pseudonymizer.getDefault(userService, getSpecialAccessChecker(travelEntries), configFacade.getCountryCode()); } private SpecialAccessCheck getSpecialAccessChecker(Collection entries) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/util/Pseudonymizer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/util/Pseudonymizer.java index 48cc1772c09..211cf1729da 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/util/Pseudonymizer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/util/Pseudonymizer.java @@ -33,48 +33,62 @@ public final class Pseudonymizer extends DtoPseudonymizer { - public static Pseudonymizer getDefault(UserService userService) { - return getDefault(userService::hasRight, noopSpecialAccessCheck()); + public static Pseudonymizer getDefault(UserService userService, String serverCountry) { + return getDefault(userService::hasRight, noopSpecialAccessCheck(), serverCountry); } - public static Pseudonymizer getDefault(UserService userService, SpecialAccessCheck specialAccessCheck) { - return getDefault(userService::hasRight, specialAccessCheck); + public static Pseudonymizer getDefault(UserService userService, SpecialAccessCheck specialAccessCheck, String serverCountry) { + return getDefault(userService::hasRight, specialAccessCheck, serverCountry); } - public static Pseudonymizer getDefault(UserService userService, SpecialAccessCheck specialAccessCheck, String stringValuePlaceholder) { - return getDefault(userService::hasRight, specialAccessCheck, stringValuePlaceholder); + public static Pseudonymizer getDefault( + UserService userService, + SpecialAccessCheck specialAccessCheck, + String stringValuePlaceholder, + String serverCountry) { + return getDefault(userService::hasRight, specialAccessCheck, stringValuePlaceholder, serverCountry); } - public static Pseudonymizer getDefault(UserService userService, String stringValuePlaceholder) { - return getDefault(userService::hasRight, noopSpecialAccessCheck(), stringValuePlaceholder); + public static Pseudonymizer getDefault(UserService userService, String stringValuePlaceholder, String serverCountry) { + return getDefault(userService::hasRight, noopSpecialAccessCheck(), stringValuePlaceholder, serverCountry); } - private static Pseudonymizer getDefault(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck) { + private static Pseudonymizer getDefault(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck, String serverCountry) { return new Pseudonymizer<>( - createDefaultFieldAccessCheckers(true, rightCheck, specialAccessCheck), - createDefaultFieldAccessCheckers(false, rightCheck, specialAccessCheck), + createDefaultFieldAccessCheckers(true, rightCheck, specialAccessCheck, serverCountry), + createDefaultFieldAccessCheckers(false, rightCheck, specialAccessCheck, serverCountry), "", true); } - private static Pseudonymizer getDefault(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck, String stringValuePlaceholder) { + private static Pseudonymizer getDefault( + RightCheck rightCheck, + SpecialAccessCheck specialAccessCheck, + String stringValuePlaceholder, + String serverCountry) { return new Pseudonymizer<>( - createDefaultFieldAccessCheckers(true, rightCheck, specialAccessCheck), - createDefaultFieldAccessCheckers(false, rightCheck, specialAccessCheck), + createDefaultFieldAccessCheckers(true, rightCheck, specialAccessCheck, serverCountry), + createDefaultFieldAccessCheckers(false, rightCheck, specialAccessCheck, serverCountry), stringValuePlaceholder, true); } - public static Pseudonymizer getDefaultWithPlaceHolder(UserService userService) { - return getDefaultWithPlaceHolder(userService, noopSpecialAccessCheck()); + public static Pseudonymizer getDefaultWithPlaceHolder(UserService userService, String serverCountry) { + return getDefaultWithPlaceHolder(userService, noopSpecialAccessCheck(), serverCountry); } - public static Pseudonymizer getDefaultWithPlaceHolder(UserService userService, SpecialAccessCheck specialAccessCheck) { - return getDefaultWithPlaceHolder(userService::hasRight, specialAccessCheck); + public static Pseudonymizer getDefaultWithPlaceHolder( + UserService userService, + SpecialAccessCheck specialAccessCheck, + String serverCountry) { + return getDefaultWithPlaceHolder(userService::hasRight, specialAccessCheck, serverCountry); } - private static Pseudonymizer getDefaultWithPlaceHolder(RightCheck rightCheck, SpecialAccessCheck specialAccessCheck) { - return getDefault(rightCheck, specialAccessCheck, I18nProperties.getCaption(Captions.inaccessibleValue)); + private static Pseudonymizer getDefaultWithPlaceHolder( + RightCheck rightCheck, + SpecialAccessCheck specialAccessCheck, + String serverCountry) { + return getDefault(rightCheck, specialAccessCheck, I18nProperties.getCaption(Captions.inaccessibleValue), serverCountry); } public static Pseudonymizer getDefaultNoCheckers(boolean pseudonymizeMandatoryFields) { @@ -161,13 +175,14 @@ private static boolean isUserInJurisdiction(User user, User currentUser) { private static FieldAccessCheckers createDefaultFieldAccessCheckers( boolean inJurisdiction, final RightCheck rightCheck, - SpecialAccessCheck specialAccessCheck) { + SpecialAccessCheck specialAccessCheck, + String serverCountry) { PersonalDataFieldAccessChecker personalFieldAccessChecker = inJurisdiction - ? PersonalDataFieldAccessChecker.inJurisdiction(rightCheck::hasRight, specialAccessCheck) - : PersonalDataFieldAccessChecker.outsideJurisdiction(rightCheck::hasRight, specialAccessCheck); + ? PersonalDataFieldAccessChecker.inJurisdiction(rightCheck::hasRight, specialAccessCheck, serverCountry) + : PersonalDataFieldAccessChecker.outsideJurisdiction(rightCheck::hasRight, specialAccessCheck, serverCountry); SensitiveDataFieldAccessChecker sensitiveFieldAccessChecker = inJurisdiction - ? SensitiveDataFieldAccessChecker.inJurisdiction(rightCheck::hasRight, specialAccessCheck) - : SensitiveDataFieldAccessChecker.outsideJurisdiction(rightCheck::hasRight, specialAccessCheck); + ? SensitiveDataFieldAccessChecker.inJurisdiction(rightCheck::hasRight, specialAccessCheck, serverCountry) + : SensitiveDataFieldAccessChecker.outsideJurisdiction(rightCheck::hasRight, specialAccessCheck, serverCountry); return FieldAccessCheckers.withCheckers(Arrays.asList(personalFieldAccessChecker, sensitiveFieldAccessChecker)); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/vaccination/VaccinationFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/vaccination/VaccinationFacadeEjb.java index d1dee05a51f..4bbc8281817 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/vaccination/VaccinationFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/vaccination/VaccinationFacadeEjb.java @@ -71,6 +71,7 @@ import de.symeda.sormas.backend.clinicalcourse.HealthConditions; import de.symeda.sormas.backend.clinicalcourse.HealthConditionsMapper; import de.symeda.sormas.backend.common.AbstractBaseEjb; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.contact.Contact; import de.symeda.sormas.backend.contact.ContactService; import de.symeda.sormas.backend.event.Event; @@ -113,6 +114,8 @@ public class VaccinationFacadeEjb private HealthConditionsMapper healthConditionsMapper; @EJB private SpecialCaseAccessService specialCaseAccessService; + @EJB + private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; public VaccinationFacadeEjb() { } @@ -397,7 +400,7 @@ protected Pseudonymizer createPseudonymizer(List va SpecialAccessCheck specialAccessCheck = t -> specialAccessUuids.contains(t.getUuid()); - return Pseudonymizer.getDefault(userService, specialAccessCheck); + return Pseudonymizer.getDefault(userService, specialAccessCheck, configFacade.getCountryCode()); } @Override diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/visit/VisitFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/visit/VisitFacadeEjb.java index ea579696a92..8c1eba2a289 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/visit/VisitFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/visit/VisitFacadeEjb.java @@ -87,6 +87,7 @@ import de.symeda.sormas.backend.caze.CaseFacadeEjb.CaseFacadeEjbLocal; import de.symeda.sormas.backend.caze.CaseService; import de.symeda.sormas.backend.common.AbstractBaseEjb; +import de.symeda.sormas.backend.common.ConfigFacadeEjb; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.common.NotificationService; import de.symeda.sormas.backend.common.messaging.MessageContents; @@ -137,6 +138,8 @@ public class VisitFacadeEjb extends AbstractBaseEjb getVisitsExportList( if (!resultList.isEmpty()) { - Pseudonymizer pseudonymizer = Pseudonymizer.getDefault(userService, getSpecialAccessChecker(resultList)); + Pseudonymizer pseudonymizer = + Pseudonymizer.getDefault(userService, getSpecialAccessChecker(resultList), configFacade.getCountryCode()); Set userIds = resultList.stream().map(VisitExportDto::getVisitUserId).filter(Objects::nonNull).collect(Collectors.toSet()); Map visitUsers = userIds.isEmpty() ? null @@ -587,7 +591,7 @@ public Visit fillOrBuildEntity(@NotNull VisitDto source, Visit target, boolean c @Override protected Pseudonymizer createPseudonymizer(List visits) { - return Pseudonymizer.getDefault(userService, getSpecialAccessChecker(visits)); + return Pseudonymizer.getDefault(userService, getSpecialAccessChecker(visits), configFacade.getCountryCode()); } private SpecialAccessCheck getSpecialAccessChecker(Collection visits) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java index e1b6811807d..018be86c216 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java @@ -41,6 +41,8 @@ import de.symeda.sormas.ui.utils.DateTimeField; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; +import de.symeda.sormas.ui.utils.RichTextAreaCustom; +import de.symeda.sormas.ui.utils.VaadinUiUtil; public class ActionEditForm extends AbstractEditForm { @@ -99,9 +101,12 @@ protected void addFields() { addField(ActionDto.ACTION_MEASURE, TextField.class); TextField title = addField(ActionDto.TITLE, TextField.class); title.addStyleName(SOFT_REQUIRED); - RichTextArea description = addField(ActionDto.DESCRIPTION, RichTextArea.class); + RichTextAreaCustom description = addField(ActionDto.DESCRIPTION, RichTextAreaCustom.class); + description.setNullRepresentation(""); description.setImmediate(true); + VaadinUiUtil.addGdprMessageOnClick(description); + RichTextArea reply = addField(ActionDto.REPLY, RichTextArea.class); reply.setNullRepresentation(""); reply.setImmediate(true); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java index 2f9385b057a..62ff0097a06 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java @@ -949,8 +949,10 @@ protected void addFields() { CssStyles.style(additionalDetails, CssStyles.CAPTION_HIDDEN); addField(CaseDataDto.PREGNANT, NullableOptionGroup.class); + addField(CaseDataDto.POSTPARTUM, NullableOptionGroup.class); addField(CaseDataDto.TRIMESTER, NullableOptionGroup.class); + FieldHelper.setVisibleWhen(getFieldGroup(), CaseDataDto.TRIMESTER, CaseDataDto.PREGNANT, Arrays.asList(YesNoUnknown.YES), true); addField(CaseDataDto.VACCINATION_STATUS); addFields(CaseDataDto.SMALLPOX_VACCINATION_SCAR, CaseDataDto.SMALLPOX_VACCINATION_RECEIVED); @@ -1082,8 +1084,6 @@ protected void addFields() { differentPlaceOfStayJurisdiction.setVisible(false); } - FieldHelper.setVisibleWhen(getFieldGroup(), CaseDataDto.TRIMESTER, CaseDataDto.PREGNANT, Arrays.asList(YesNoUnknown.YES), true); - diseaseField.addValueChangeListener((ValueChangeListener) valueChangeEvent -> { Disease disease = (Disease) valueChangeEvent.getProperty().getValue(); List diseaseVariants = diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java index c377cb9281f..cad1c6f6615 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java @@ -38,13 +38,17 @@ import com.vaadin.v7.ui.TextArea; import de.symeda.sormas.api.clinicalcourse.HealthConditionsDto; +import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.Descriptions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; +import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.FieldHelper; public class HealthConditionsForm extends AbstractEditForm { @@ -52,6 +56,7 @@ public class HealthConditionsForm extends AbstractEditForm private static final long serialVersionUID = 1L; private static final String HEALTH_CONDITIONS_HEADINGS_LOC = "healthConditionsHeadingLoc"; + private static final String CONFIDENTIAL_LABEL = "confidentialLabel"; //@formatter:off private static final String HTML_LAYOUT = @@ -66,7 +71,7 @@ public class HealthConditionsForm extends AbstractEditForm CHRONIC_NEUROLOGIC_CONDITION, CARDIOVASCULAR_DISEASE_INCLUDING_HYPERTENSION, OBESITY, CURRENT_SMOKER, FORMER_SMOKER, ASTHMA, SICKLE_CELL_DISEASE)) ) + - loc(OTHER_CONDITIONS); + loc(OTHER_CONDITIONS) + loc(CONFIDENTIAL_LABEL); //@formatter:on public HealthConditionsForm(FieldVisibilityCheckers fieldVisibilityCheckers, UiFieldAccessCheckers fieldAccessCheckers) { @@ -80,39 +85,45 @@ protected void addFields() { healthConditionsHeadingLabel.addStyleName(H3); getContent().addComponent(healthConditionsHeadingLabel, HEALTH_CONDITIONS_HEADINGS_LOC); - addFields( - TUBERCULOSIS, - ASPLENIA, - HEPATITIS, - DIABETES, - HIV, - HIV_ART, - CHRONIC_LIVER_DISEASE, - MALIGNANCY_CHEMOTHERAPY, - CHRONIC_HEART_FAILURE, - CHRONIC_PULMONARY_DISEASE, - CHRONIC_KIDNEY_DISEASE, - CHRONIC_NEUROLOGIC_CONDITION, - DOWN_SYNDROME, - CONGENITAL_SYPHILIS, - IMMUNODEFICIENCY_OTHER_THAN_HIV, - CARDIOVASCULAR_DISEASE_INCLUDING_HYPERTENSION, - OBESITY, - CURRENT_SMOKER, - FORMER_SMOKER, - ASTHMA, - SICKLE_CELL_DISEASE, - IMMUNODEFICIENCY_INCLUDING_HIV); - TextArea otherConditions = addField(OTHER_CONDITIONS, TextArea.class); - otherConditions.setRows(6); - otherConditions.setDescription( - I18nProperties.getPrefixDescription(HealthConditionsDto.I18N_PREFIX, OTHER_CONDITIONS, "") + "\n" - + I18nProperties.getDescription(Descriptions.descGdpr)); - - initializeVisibilitiesAndAllowedVisibilities(); - initializeAccessAndAllowedAccesses(); - - FieldHelper.setVisibleWhen(getFieldGroup(), HIV_ART, HIV, Arrays.asList(YesNoUnknown.YES), true); + if (UiUtil.permitted(UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION)) { + addFields( + TUBERCULOSIS, + ASPLENIA, + HEPATITIS, + DIABETES, + HIV, + HIV_ART, + CHRONIC_LIVER_DISEASE, + MALIGNANCY_CHEMOTHERAPY, + CHRONIC_HEART_FAILURE, + CHRONIC_PULMONARY_DISEASE, + CHRONIC_KIDNEY_DISEASE, + CHRONIC_NEUROLOGIC_CONDITION, + DOWN_SYNDROME, + CONGENITAL_SYPHILIS, + IMMUNODEFICIENCY_OTHER_THAN_HIV, + CARDIOVASCULAR_DISEASE_INCLUDING_HYPERTENSION, + OBESITY, + CURRENT_SMOKER, + FORMER_SMOKER, + ASTHMA, + SICKLE_CELL_DISEASE, + IMMUNODEFICIENCY_INCLUDING_HIV); + TextArea otherConditions = addField(OTHER_CONDITIONS, TextArea.class); + otherConditions.setRows(6); + otherConditions.setDescription( + I18nProperties.getPrefixDescription(HealthConditionsDto.I18N_PREFIX, OTHER_CONDITIONS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + initializeVisibilitiesAndAllowedVisibilities(); + initializeAccessAndAllowedAccesses(); + + FieldHelper.setVisibleWhen(getFieldGroup(), HIV_ART, HIV, Arrays.asList(YesNoUnknown.YES), true); + } else { + Label confidentialLabel = new Label(I18nProperties.getCaption(Captions.inaccessibleValue)); + confidentialLabel.addStyleName(CssStyles.INACCESSIBLE_LABEL); + getContent().addComponent(confidentialLabel, CONFIDENTIAL_LABEL); + } } @Override diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java index c704207bec8..7d3a92e2013 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java @@ -454,7 +454,7 @@ protected void addFields() { diseaseVariantDetailsField.setVisible(diseaseVariant != null && diseaseVariant.matchPropertyValue(DiseaseVariant.HAS_DETAILS, true)); }); - setRequired(true, EventDto.EVENT_STATUS, EventDto.UUID, EventDto.EVENT_TITLE, EventDto.REPORT_DATE_TIME, EventDto.REPORTING_USER); + setRequired(true, EventDto.EVENT_STATUS, EventDto.UUID, EventDto.EVENT_TITLE, EventDto.REPORT_DATE_TIME); reportDate.addValidator( new DateComparisonValidator( diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskEditForm.java index 6b28f33007c..63d02222c71 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskEditForm.java @@ -63,6 +63,7 @@ import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.TaskStatusValidator; +import de.symeda.sormas.ui.utils.VaadinUiUtil; import de.symeda.sormas.ui.utils.components.MultiSelect; @SuppressWarnings("deprecation") @@ -161,7 +162,11 @@ protected void addFields() { TextArea creatorComment = addField(TaskDto.CREATOR_COMMENT, TextArea.class); creatorComment.setRows(2); creatorComment.setImmediate(true); - addField(TaskDto.ASSIGNEE_REPLY, TextArea.class).setRows(4); + VaadinUiUtil.addGdprMessageOnClick(creatorComment); + + TextArea assigneeReply = addField(TaskDto.ASSIGNEE_REPLY, TextArea.class); + assigneeReply.setRows(4); + VaadinUiUtil.addGdprMessageOnClick(assigneeReply); MultiSelect observerUsers = addField(TaskDto.OBSERVER_USERS, MultiSelect.class); observerUsers.addValueChangeListener(e -> { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java index 5e91c4022b7..1162f807383 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java @@ -564,6 +564,10 @@ protected void initializeAccessAndAllowedAccesses() { if (field instanceof ComboBoxWithPlaceholder) { FieldHelper.setComboInaccessible((ComboBoxWithPlaceholder) field); } + + if (field instanceof NullableOptionGroup) { + ((NullableOptionGroup) field).setInaccessible(); + } } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/NullableOptionGroup.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/NullableOptionGroup.java index 91dd75ae380..69a6fcfb312 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/NullableOptionGroup.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/NullableOptionGroup.java @@ -23,9 +23,13 @@ import org.apache.commons.lang3.ObjectUtils; import com.vaadin.v7.data.Container; +import com.vaadin.v7.data.Item; import com.vaadin.v7.data.util.converter.Converter; import com.vaadin.v7.ui.OptionGroup; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; + public class NullableOptionGroup extends OptionGroup { public NullableOptionGroup() { @@ -79,4 +83,10 @@ public void setRequired(boolean required) { private Object getFirstValue(Set value) { return value.stream().findFirst().orElse(null); } + + public void setInaccessible() { + this.removeAllItems(); + Item item = this.addItem(1); + item.getItemProperty(item.getItemPropertyIds().stream().findFirst().get()).setValue(I18nProperties.getCaption(Captions.inaccessibleValue)); + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java new file mode 100644 index 00000000000..8f26e2c24b9 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java @@ -0,0 +1,24 @@ +package de.symeda.sormas.ui.utils; + +import com.vaadin.event.FieldEvents; +import com.vaadin.v7.ui.RichTextArea; + +public class RichTextAreaCustom extends RichTextArea { + + public void addFocusListener(FieldEvents.FocusListener listener) { + addListener(FieldEvents.FocusEvent.EVENT_ID, FieldEvents.FocusEvent.class, listener, FieldEvents.FocusListener.focusMethod); + } + + public void removeFocusListener(FieldEvents.FocusListener listener) { + removeListener(FieldEvents.FocusEvent.EVENT_ID, FieldEvents.FocusEvent.class, listener); + + } + + public void addBlurListener(FieldEvents.BlurListener listener) { + addListener(FieldEvents.BlurEvent.EVENT_ID, FieldEvents.BlurEvent.class, listener, FieldEvents.BlurListener.blurMethod); + } + + public void removeBlurListener(FieldEvents.BlurListener listener) { + removeListener(FieldEvents.BlurEvent.EVENT_ID, FieldEvents.BlurEvent.class, listener); + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java index ad062f3b4b8..d8109192d5e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java @@ -179,6 +179,8 @@ public T createField(Class type, Class fieldType) { return (T) new UserField(); } else if (CheckBoxTree.class.isAssignableFrom(fieldType)) { return (T) new CheckBoxTree<>(); + } else if (RichTextAreaCustom.class.isAssignableFrom(fieldType)) { + return (T) new RichTextAreaCustom(); } return super.createField(type, fieldType); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java index a4de6114cdf..84509ae6295 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java @@ -21,6 +21,7 @@ import java.math.RoundingMode; import java.util.concurrent.CompletableFuture; import java.util.concurrent.CompletionStage; +import java.util.concurrent.atomic.AtomicBoolean; import java.util.function.Consumer; import java.util.function.Function; import java.util.function.UnaryOperator; @@ -43,12 +44,15 @@ import com.vaadin.v7.data.Item; import com.vaadin.v7.data.util.GeneratedPropertyContainer; import com.vaadin.v7.data.util.PropertyValueGenerator; +import com.vaadin.v7.ui.AbstractField; import com.vaadin.v7.ui.Grid; +import com.vaadin.v7.ui.TextArea; import com.vaadin.v7.ui.renderers.HtmlRenderer; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.user.UserDto; public final class VaadinUiUtil { @@ -632,4 +636,65 @@ public static void showWarningPopup(String message) { popupWindow.addCloseListener(e -> popupWindow.close()); popupWindow.setWidth(400, Unit.PIXELS); } + + public static void addGdprMessageOnClick(TextArea textArea) { + AtomicBoolean gdprMessageTriggered = new AtomicBoolean(false); + Window subWindowGdpR = VaadinUiUtil.createPopupWindow(); + + textArea.addFocusListener(i -> { + showGdprWindow(textArea, gdprMessageTriggered, subWindowGdpR); + }); + + textArea.addBlurListener(blurEvent -> { + if (gdprMessageTriggered.get() && subWindowGdpR.getParent() == null) { + gdprMessageTriggered.set(false); + } + }); + } + + public static void addGdprMessageOnClick(RichTextAreaCustom richTextArea) { + AtomicBoolean gdprMessageTriggered = new AtomicBoolean(false); + Window subWindowGdpR = VaadinUiUtil.createPopupWindow(); + + richTextArea.addValueChangeListener(event -> { + if (!gdprMessageTriggered.get()) { + showGdprWindow(richTextArea, gdprMessageTriggered, subWindowGdpR); + subWindowGdpR.focus(); + gdprMessageTriggered.set(true); + } + }); + } + + private static void showGdprWindow(AbstractField textArea, AtomicBoolean gdprMessageTriggered, Window subWindowGdpR) { + if (!gdprMessageTriggered.get()) { + gdprMessageTriggered.set(true); + subWindowGdpR.setCaption(I18nProperties.getPrefixCaption(UserDto.I18N_PREFIX, UserDto.HAS_CONSENTED_TO_GDPR)); + VerticalLayout subContentGdpr = new VerticalLayout(); + subWindowGdpR.setContent(subContentGdpr); + subWindowGdpR.center(); + subWindowGdpR.setWidth("40%"); + subWindowGdpR.setModal(true); + subWindowGdpR.setClosable(true); + + Label textGdpr = new Label(); + textGdpr.setWidth("80%"); + textGdpr.setSizeFull(); + textGdpr.setValue(I18nProperties.getString(Strings.messageGdpr)); + subContentGdpr.addComponent(textGdpr); + + HorizontalLayout buttonLayout = new HorizontalLayout(); + buttonLayout.setMargin(false); + buttonLayout.setSpacing(true); + buttonLayout.setWidth(100, Unit.PERCENTAGE); + Button buttonGdpr = ButtonHelper.createButton(I18nProperties.getCaption(Captions.actionConfirm), event -> { + subWindowGdpR.close(); + textArea.focus(); + }, ValoTheme.BUTTON_PRIMARY); + buttonLayout.addComponent(buttonGdpr); + subContentGdpr.addComponent(buttonLayout); + buttonLayout.setComponentAlignment(buttonGdpr, Alignment.BOTTOM_RIGHT); + buttonLayout.setExpandRatio(buttonGdpr, 0); + UI.getCurrent().addWindow(subWindowGdpR); + } + } } From 946919913eb78e58fda6c63f3d7b29e0db8fe9c5 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Fri, 25 Oct 2024 14:30:58 +0300 Subject: [PATCH 14/56] #13093 - Update Data Protection for certain Data Fields --- .../fieldaccess/UiFieldAccessCheckers.java | 25 +++++----- .../PseudonymizedFieldAccessChecker.java | 48 +++++++++++++++---- .../ActivityAsCase/ActivityAsCaseField.java | 5 +- .../symeda/sormas/ui/caze/CaseController.java | 9 ++-- .../symeda/sormas/ui/caze/CaseDataForm.java | 4 +- .../symeda/sormas/ui/caze/CaseInfoLayout.java | 7 +-- .../maternalhistory/MaternalHistoryForm.java | 5 +- .../porthealthinfo/PortHealthInfoForm.java | 5 +- .../SurveillanceReportForm.java | 10 +--- .../SurveillanceReportList.java | 4 +- .../ui/clinicalcourse/ClinicalCourseForm.java | 5 +- .../ui/clinicalcourse/ClinicalVisitForm.java | 6 ++- .../ui/clinicalcourse/ClinicalVisitGrid.java | 5 +- .../sormas/ui/contact/ContactDataForm.java | 4 +- .../ui/environment/EnvironmentDataForm.java | 5 +- .../symeda/sormas/ui/epidata/EpiDataForm.java | 5 +- .../sormas/ui/events/EventDataForm.java | 4 +- .../ui/events/EventParticipantEditForm.java | 4 +- .../SuperordinateEventComponent.java | 8 +--- .../sormas/ui/exposure/ExposuresField.java | 5 +- ...ciansReportCaseImmunizationsComponent.java | 18 +++---- .../hospitalization/HospitalizationForm.java | 4 +- .../components/form/ImmunizationDataForm.java | 4 +- .../ui/person/PersonContactDetailsField.java | 6 ++- .../sormas/ui/person/PersonCreateForm.java | 2 +- .../sormas/ui/person/PersonEditForm.java | 9 ++-- .../sormas/ui/samples/PathogenTestForm.java | 7 +-- .../EnvironmentSampleEditForm.java | 2 +- .../samples/humansample/SampleDataView.java | 11 ++--- .../samples/humansample/SampleEditForm.java | 8 +--- .../ui/selfreport/SelfReportDataForm.java | 5 +- .../de/symeda/sormas/ui/task/TaskGrid.java | 2 +- .../sormas/ui/therapy/PrescriptionForm.java | 5 +- .../sormas/ui/therapy/PrescriptionGrid.java | 5 +- .../sormas/ui/therapy/TreatmentForm.java | 5 +- .../sormas/ui/therapy/TreatmentGrid.java | 5 +- .../ui/travelentry/TravelEntryDataForm.java | 3 +- .../FieldAccessColumnStyleGenerator.java | 11 ++++- .../sormas/ui/utils/FieldAccessHelper.java | 19 ++++++++ .../ui/vaccination/list/VaccinationList.java | 6 +-- .../symeda/sormas/ui/visit/VisitEditForm.java | 5 +- 41 files changed, 173 insertions(+), 142 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/UiFieldAccessCheckers.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/UiFieldAccessCheckers.java index 26c1b2e5c9f..555e1913608 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/UiFieldAccessCheckers.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/UiFieldAccessCheckers.java @@ -48,41 +48,44 @@ public static UiFieldAccessCheckers getNoop() { return new UiFieldAccessCheckers<>(); } - public static UiFieldAccessCheckers getDefault(boolean isPseudonymized) { + public static UiFieldAccessCheckers getDefault(boolean isPseudonymized, String serverCountry) { UiFieldAccessCheckers fieldAccessCheckers = new UiFieldAccessCheckers<>(); - fieldAccessCheckers.add(PseudonymizedFieldAccessChecker.forPersonalData(isPseudonymized)) - .add(PseudonymizedFieldAccessChecker.forSensitiveData(isPseudonymized)); + fieldAccessCheckers.add(PseudonymizedFieldAccessChecker.forPersonalData(isPseudonymized, serverCountry)) + .add(PseudonymizedFieldAccessChecker.forSensitiveData(isPseudonymized, serverCountry)); return fieldAccessCheckers; } - public static UiFieldAccessCheckers forPersonalData(boolean isPseudonymized) { + public static UiFieldAccessCheckers forPersonalData(boolean isPseudonymized, String serverCountry) { UiFieldAccessCheckers fieldAccessCheckers = new UiFieldAccessCheckers<>(); - fieldAccessCheckers.add(PseudonymizedFieldAccessChecker.forPersonalData(isPseudonymized)); + fieldAccessCheckers.add(PseudonymizedFieldAccessChecker.forPersonalData(isPseudonymized, serverCountry)); return fieldAccessCheckers; } - public static UiFieldAccessCheckers forSensitiveData(boolean isPseudonymized) { + public static UiFieldAccessCheckers forSensitiveData(boolean isPseudonymized, String serverCountry) { UiFieldAccessCheckers fieldAccessCheckers = new UiFieldAccessCheckers<>(); - fieldAccessCheckers.add(PseudonymizedFieldAccessChecker.forSensitiveData(isPseudonymized)); + fieldAccessCheckers.add(PseudonymizedFieldAccessChecker.forSensitiveData(isPseudonymized, serverCountry)); return fieldAccessCheckers; } - public static UiFieldAccessCheckers forDataAccessLevel(PseudonymizableDataAccessLevel accessLevel, boolean isPseudonymized) { + public static UiFieldAccessCheckers forDataAccessLevel( + PseudonymizableDataAccessLevel accessLevel, + boolean isPseudonymized, + String serverCountry) { switch (accessLevel) { case ALL: case NONE: - return getDefault(isPseudonymized); + return getDefault(isPseudonymized, serverCountry); case PERSONAL: - return forSensitiveData(isPseudonymized); + return forSensitiveData(isPseudonymized, serverCountry); case SENSITIVE: - return forPersonalData(isPseudonymized); + return forPersonalData(isPseudonymized, serverCountry); default: throw new IllegalArgumentException(accessLevel.name()); } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PseudonymizedFieldAccessChecker.java b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PseudonymizedFieldAccessChecker.java index 28605b139fa..c88550f853c 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PseudonymizedFieldAccessChecker.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/utils/fieldaccess/checkers/PseudonymizedFieldAccessChecker.java @@ -17,6 +17,7 @@ import java.lang.annotation.Annotation; import java.lang.reflect.Field; +import java.util.Arrays; import de.symeda.sormas.api.utils.EmbeddedPersonalData; import de.symeda.sormas.api.utils.EmbeddedSensitiveData; @@ -24,15 +25,16 @@ import de.symeda.sormas.api.utils.SensitiveData; import de.symeda.sormas.api.utils.fieldaccess.FieldAccessChecker; -public final class PseudonymizedFieldAccessChecker implements FieldAccessChecker { +public abstract class PseudonymizedFieldAccessChecker implements FieldAccessChecker { private final WrappedFieldAccessChecker wrapped; - public PseudonymizedFieldAccessChecker( + private PseudonymizedFieldAccessChecker( Class annotation, Class embeddedAnnotation, - boolean isPseudonymized) { - this.wrapped = new WrappedFieldAccessChecker(annotation, embeddedAnnotation, isPseudonymized); + boolean isPseudonymized, + String serverCountry) { + this.wrapped = new WrappedFieldAccessChecker(annotation, embeddedAnnotation, isPseudonymized, serverCountry); } @Override @@ -56,24 +58,52 @@ public boolean hasRight(T object) { private final class WrappedFieldAccessChecker extends AnnotationBasedFieldAccessChecker { + private final String serverCountry; + private WrappedFieldAccessChecker( Class annotation, Class embeddedAnnotation, - boolean isPseudonymized) { + boolean isPseudonymized, + String serverCountry) { super(annotation, embeddedAnnotation, !isPseudonymized, t -> false); + this.serverCountry = serverCountry; } @Override protected boolean isAnnotatedFieldMandatory(Field annotatedField) { return false; } + + @Override + public boolean isConfiguredForCheck(Field field, boolean withMandatory) { + if (isExcludedForCountry(field, serverCountry)) { + return false; + } + return super.isConfiguredForCheck(field, withMandatory); + } } - public static PseudonymizedFieldAccessChecker forPersonalData(boolean isPseudonymized) { - return new PseudonymizedFieldAccessChecker<>(PersonalData.class, EmbeddedPersonalData.class, isPseudonymized); + protected abstract boolean isExcludedForCountry(Field field, String serverCountry); + + public static PseudonymizedFieldAccessChecker forPersonalData(boolean isPseudonymized, String serverCountry) { + return new PseudonymizedFieldAccessChecker<>(PersonalData.class, EmbeddedPersonalData.class, isPseudonymized, serverCountry) { + + @Override + protected boolean isExcludedForCountry(Field field, String serverCountry) { + return field.getAnnotation(PersonalData.class) != null + && Arrays.asList(field.getAnnotation(PersonalData.class).excludeForCountries()).contains(serverCountry); + } + }; } - public static PseudonymizedFieldAccessChecker forSensitiveData(boolean isPseudonymized) { - return new PseudonymizedFieldAccessChecker<>(SensitiveData.class, EmbeddedSensitiveData.class, isPseudonymized); + public static PseudonymizedFieldAccessChecker forSensitiveData(boolean isPseudonymized, String serverCountry) { + return new PseudonymizedFieldAccessChecker<>(SensitiveData.class, EmbeddedSensitiveData.class, isPseudonymized, serverCountry) { + + @Override + protected boolean isExcludedForCountry(Field field, String serverCountry) { + return field.getAnnotation(SensitiveData.class) != null + && Arrays.asList(field.getAnnotation(SensitiveData.class).excludeForCountries()).contains(serverCountry); + } + }; } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/ActivityAsCase/ActivityAsCaseField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/ActivityAsCase/ActivityAsCaseField.java index 95f37cd7415..fbfc08b132e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/ActivityAsCase/ActivityAsCaseField.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/ActivityAsCase/ActivityAsCaseField.java @@ -31,6 +31,7 @@ import com.vaadin.v7.ui.Label; import com.vaadin.v7.ui.Table; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.activityascase.ActivityAsCaseDto; import de.symeda.sormas.api.activityascase.ActivityAsCaseType; import de.symeda.sormas.api.event.TypeOfPlace; @@ -80,7 +81,9 @@ protected void updateColumns() { .setVisibleColumns(ACTION_COLUMN_ID, COLUMN_ACTIVITY_AS_CASE_TYPE, COLUMN_TYPE_OF_PLACE, COLUMN_DATE, COLUMN_ADDRESS, COLUMN_DESCRIPTION); table.setCellStyleGenerator( - FieldAccessCellStyleGenerator.withFieldAccessCheckers(ActivityAsCaseDto.class, UiFieldAccessCheckers.forSensitiveData(isPseudonymized))); + FieldAccessCellStyleGenerator.withFieldAccessCheckers( + ActivityAsCaseDto.class, + UiFieldAccessCheckers.forSensitiveData(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale()))); for (Object columnId : table.getVisibleColumns()) { if (!columnId.equals(ACTION_COLUMN_ID)) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java index f6f1a2cb389..92d01dd0a56 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java @@ -110,7 +110,6 @@ import de.symeda.sormas.api.utils.DateHelper; import de.symeda.sormas.api.utils.ValidationRuntimeException; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.uuid.HasUuid; import de.symeda.sormas.api.visit.VisitDto; import de.symeda.sormas.ui.ControllerProvider; @@ -141,6 +140,7 @@ import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DeleteRestoreHandlers; import de.symeda.sormas.ui.utils.DetailSubComponentWrapper; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.VaadinUiUtil; import de.symeda.sormas.ui.utils.ViewMode; @@ -1298,8 +1298,7 @@ public CommitDiscardWrapperComponent getSymptomsEditComponent(fina person, SymptomsContext.CASE, viewMode, - UiFieldAccessCheckers - .forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(caseDataDto.isInJurisdiction()), caseDataDto.isPseudonymized())); + FieldAccessHelper.getFieldAccessCheckers(caseDataDto)); symptomsForm.setValue(caseDataDto.getSymptoms()); CommitDiscardWrapperComponent editView = @@ -1381,9 +1380,7 @@ public DetailSubComponentWrapper getExternalDataComponent(final String caseUuid, CaseDataDto caseDataDto = findCase(caseUuid); - CaseExternalDataForm caseExternalDataForm = new CaseExternalDataForm( - UiFieldAccessCheckers - .forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(caseDataDto.isInJurisdiction()), caseDataDto.isPseudonymized())); + CaseExternalDataForm caseExternalDataForm = new CaseExternalDataForm(FieldAccessHelper.getFieldAccessCheckers(caseDataDto)); caseExternalDataForm.setValue(caseDataDto); DetailSubComponentWrapper wrapper = new DetailSubComponentWrapper(() -> null); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java index b9e57040952..b2f181ee080 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java @@ -118,7 +118,6 @@ import de.symeda.sormas.api.utils.DateHelper; import de.symeda.sormas.api.utils.ExtendedReduced; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.api.utils.fieldvisibility.checkers.CountryFieldVisibilityChecker; import de.symeda.sormas.api.utils.fieldvisibility.checkers.FeatureTypeFieldVisibilityChecker; @@ -135,6 +134,7 @@ import de.symeda.sormas.ui.utils.ConfirmationComponent; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.InfrastructureFieldsHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -327,7 +327,7 @@ public CaseDataForm( .add(new CountryFieldVisibilityChecker(FacadeProvider.getConfigFacade().getCountryLocale())) .add(new UserRightFieldVisibilityChecker(UiUtil::permitted)) .add(new FeatureTypeFieldVisibilityChecker(FacadeProvider.getFeatureConfigurationFacade().getActiveServerFeatureConfigurations())), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.caseUuid = caseUuid; this.person = person; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseInfoLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseInfoLayout.java index fd7a689f335..529827e3de6 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseInfoLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseInfoLayout.java @@ -32,11 +32,11 @@ import de.symeda.sormas.api.symptoms.SymptomsDto; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.DataHelper; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.ui.AbstractInfoLayout; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.FieldAccessHelper; @SuppressWarnings("serial") public class CaseInfoLayout extends AbstractInfoLayout { @@ -49,10 +49,7 @@ public CaseInfoLayout(CaseDataDto caseDto) { } public CaseInfoLayout(CaseDataDto caseDto, boolean isTravelEntry) { - super( - CaseDataDto.class, - UiFieldAccessCheckers - .forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(caseDto.isInJurisdiction()), caseDto.isPseudonymized())); + super(CaseDataDto.class, FieldAccessHelper.getFieldAccessCheckers(caseDto)); this.caseDto = caseDto; this.isTravelEntry = isTravelEntry; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/maternalhistory/MaternalHistoryForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/maternalhistory/MaternalHistoryForm.java index 9bfc0a029ed..75a24f1bb71 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/maternalhistory/MaternalHistoryForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/maternalhistory/MaternalHistoryForm.java @@ -18,10 +18,9 @@ import de.symeda.sormas.api.infrastructure.district.DistrictReferenceDto; import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.ViewMode; @@ -55,7 +54,7 @@ public MaternalHistoryForm(ViewMode viewMode, boolean isPseudonymized, boolean i MaternalHistoryDto.I18N_PREFIX, true, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.viewMode = viewMode; } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/porthealthinfo/PortHealthInfoForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/porthealthinfo/PortHealthInfoForm.java index 52abe79dec9..2fecf46b4c4 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/porthealthinfo/PortHealthInfoForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/porthealthinfo/PortHealthInfoForm.java @@ -24,13 +24,12 @@ import de.symeda.sormas.api.infrastructure.pointofentry.PointOfEntryReferenceDto; import de.symeda.sormas.api.utils.DataHelper; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -83,7 +82,7 @@ public PortHealthInfoForm(PointOfEntryDto pointOfEntry, String pointOfEntryDetai PortHealthInfoDto.I18N_PREFIX, false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), pseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, pseudonymized)); this.pointOfEntry = pointOfEntry; this.pointOfEntryDetails = pointOfEntryDetails; this.pseudonymized = pseudonymized; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/surveillancereport/SurveillanceReportForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/surveillancereport/SurveillanceReportForm.java index dfd24c9a6b2..880ee52d8d6 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/surveillancereport/SurveillanceReportForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/surveillancereport/SurveillanceReportForm.java @@ -24,10 +24,9 @@ import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.infrastructure.facility.FacilityTypeGroup; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.ComboBoxHelper; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.InfrastructureFieldsHelper; public class SurveillanceReportForm extends AbstractEditForm { @@ -46,12 +45,7 @@ public class SurveillanceReportForm extends AbstractEditForm { @@ -19,7 +18,7 @@ public ClinicalCourseForm(boolean isPseudonymized, boolean inJurisdiction) { ClinicalCourseDto.I18N_PREFIX, true, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); } @Override diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitForm.java index 36b9e4926a0..ce80ece4dbe 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitForm.java @@ -6,6 +6,7 @@ import com.vaadin.v7.ui.TextField; import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.clinicalcourse.ClinicalVisitDto; import de.symeda.sormas.api.i18n.Descriptions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -36,7 +37,10 @@ public ClinicalVisitForm(boolean create, Disease disease, PersonDto person, bool ClinicalVisitDto.I18N_PREFIX, false, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + UiFieldAccessCheckers.forDataAccessLevel( + UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), + isPseudonymized, + FacadeProvider.getConfigFacade().getCountryLocale())); if (create) { hideValidationUntilNextCommit(); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitGrid.java index 1318e7b3e78..fbebeec4aa3 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/ClinicalVisitGrid.java @@ -66,8 +66,9 @@ public ClinicalVisitGrid(CaseReferenceDto caseRef, boolean isPseudonymized, bool } setCellStyleGenerator( - FieldAccessCellStyleGenerator - .withFieldAccessCheckers(ClinicalVisitIndexDto.class, UiFieldAccessCheckers.forSensitiveData(isPseudonymized))); + FieldAccessCellStyleGenerator.withFieldAccessCheckers( + ClinicalVisitIndexDto.class, + UiFieldAccessCheckers.forSensitiveData(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale()))); addItemClickListener(e -> { if (ACTION_BTN_ID.equals(e.getPropertyId()) || e.isDoubleClick()) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java index 5b0fe16c52c..e76a5f9ad9b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java @@ -81,7 +81,6 @@ import de.symeda.sormas.api.utils.Diseases.DiseasesConfiguration; import de.symeda.sormas.api.utils.ExtendedReduced; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.UiUtil; @@ -90,6 +89,7 @@ import de.symeda.sormas.ui.utils.ButtonHelper; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.LayoutUtil; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -204,7 +204,7 @@ public ContactDataForm(Disease disease, ViewMode viewMode, boolean isPseudonymiz FieldVisibilityCheckers.withDisease(disease) .andWithCountry(FacadeProvider.getConfigFacade().getCountryLocale()) .andWithFeatureType(FacadeProvider.getFeatureConfigurationFacade().getActiveServerFeatureConfigurations()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.viewMode = viewMode; this.disease = disease; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/environment/EnvironmentDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/environment/EnvironmentDataForm.java index 154a0bedbd4..96fed8d8d73 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/environment/EnvironmentDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/environment/EnvironmentDataForm.java @@ -30,13 +30,12 @@ import de.symeda.sormas.api.user.JurisdictionLevel; import de.symeda.sormas.api.user.UserReferenceDto; import de.symeda.sormas.api.user.UserRight; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.location.LocationEditForm; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CheckBoxTree; import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.ResizableTextAreaWrapper; import de.symeda.sormas.ui.utils.UserField; @@ -71,7 +70,7 @@ public EnvironmentDataForm(boolean isPseudonymized, boolean inJurisdiction, bool EnvironmentDto.I18N_PREFIX, false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized), + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized), isEditAllowed); addFields(); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/epidata/EpiDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/epidata/EpiDataForm.java index 21f396a0e1f..81fa1fe4cd6 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/epidata/EpiDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/epidata/EpiDataForm.java @@ -44,12 +44,11 @@ import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.ActivityAsCase.ActivityAsCaseField; import de.symeda.sormas.ui.exposure.ExposuresField; import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.components.MultilineLabel; @@ -98,7 +97,7 @@ public EpiDataForm( EpiDataDto.I18N_PREFIX, false, FieldVisibilityCheckers.withDisease(disease).andWithCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized), + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized), isEditAllowed); this.disease = disease; this.parentClass = parentClass; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java index 7d3a92e2013..6fdfd864d4b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java @@ -79,13 +79,13 @@ import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.location.LocationEditForm; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CheckBoxTree; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.PhoneNumberValidator; @@ -197,7 +197,7 @@ private static UiFieldAccessCheckers createFieldAccessCheckers( boolean withPersonalAndSensitive) { if (withPersonalAndSensitive) { - return UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized); + return FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized); } return UiFieldAccessCheckers.getNoop(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantEditForm.java index 0ceb8e3094d..4b4a439cd83 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantEditForm.java @@ -32,10 +32,10 @@ import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.infrastructure.region.RegionReferenceDto; import de.symeda.sormas.api.location.LocationDto; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.UserField; @@ -63,7 +63,7 @@ public EventParticipantEditForm(EventDto event, boolean isPseudonymized, boolean EventParticipantDto.I18N_PREFIX, false, FieldVisibilityCheckers.withDisease(event.getDisease()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.event = event; addFields(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/eventLink/SuperordinateEventComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/eventLink/SuperordinateEventComponent.java index 6c51364f2ad..71cdf0688e1 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/eventLink/SuperordinateEventComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/eventLink/SuperordinateEventComponent.java @@ -33,13 +33,13 @@ import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.DataHelper; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.ui.AbstractInfoLayout; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.ButtonHelper; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateFormatHelper; +import de.symeda.sormas.ui.utils.FieldAccessHelper; public class SuperordinateEventComponent extends VerticalLayout { @@ -129,11 +129,7 @@ private class SuperordinateEventInfoLayout extends AbstractInfoLayout private final EventDto superordinateEvent; public SuperordinateEventInfoLayout(EventDto superordinateEvent) { - super( - EventDto.class, - UiFieldAccessCheckers.forDataAccessLevel( - UiUtil.getPseudonymizableDataAccessLevel(superordinateEvent.isInJurisdiction()), - superordinateEvent.isPseudonymized())); + super(EventDto.class, FieldAccessHelper.getFieldAccessCheckers(superordinateEvent)); this.superordinateEvent = superordinateEvent; setSpacing(true); setMargin(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/exposure/ExposuresField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/exposure/ExposuresField.java index 3bdde88a1e1..4b02af90252 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/exposure/ExposuresField.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/exposure/ExposuresField.java @@ -32,6 +32,7 @@ import com.vaadin.v7.ui.Table; import de.symeda.sormas.api.EntityDto; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.caze.CaseDataDto; import de.symeda.sormas.api.contact.ContactDto; import de.symeda.sormas.api.contact.ContactReferenceDto; @@ -112,7 +113,9 @@ protected void updateColumns() { COLUMN_DESCRIPTION); } table.setCellStyleGenerator( - FieldAccessCellStyleGenerator.withFieldAccessCheckers(ExposureDto.class, UiFieldAccessCheckers.forSensitiveData(isPseudonymized))); + FieldAccessCellStyleGenerator.withFieldAccessCheckers( + ExposureDto.class, + UiFieldAccessCheckers.forSensitiveData(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale()))); for (Object columnId : table.getVisibleColumns()) { if (columnId.equals(ACTION_COLUMN_ID)) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/externalmessage/physiciansreport/PhysiciansReportCaseImmunizationsComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/externalmessage/physiciansreport/PhysiciansReportCaseImmunizationsComponent.java index ee1f9f9e4bf..bc2c86116a7 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/externalmessage/physiciansreport/PhysiciansReportCaseImmunizationsComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/externalmessage/physiciansreport/PhysiciansReportCaseImmunizationsComponent.java @@ -57,6 +57,7 @@ import de.symeda.sormas.ui.utils.DateFormatHelper; import de.symeda.sormas.ui.utils.DeletableUtils; import de.symeda.sormas.ui.utils.DirtyCheckPopup; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.vaccination.VaccinationEditForm; public class PhysiciansReportCaseImmunizationsComponent extends CommitDiscardWrapperComponent { @@ -210,18 +211,11 @@ private void handleEditVaccination(VaccinationDto vaccination, CaseDataDto caze) VaccinationListItem collapsedComponent = vaccinationList.getItemByVaccination(vaccination); currentVaccinationEditComponent = ControllerProvider.getVaccinationController() - .getVaccinationEditComponent( - vaccination, - caze.getDisease(), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(caze.isInJurisdiction()), caze.isPseudonymized()), - false, - (v) -> { - if (!vaccinationsToCreate.contains(v)) { - vaccinationsToUpdate.add(v); - } - }, - true, - UiUtil.permitted(UserRight.IMMUNIZATION_DELETE)); + .getVaccinationEditComponent(vaccination, caze.getDisease(), FieldAccessHelper.getFieldAccessCheckers(caze), false, (v) -> { + if (!vaccinationsToCreate.contains(v)) { + vaccinationsToUpdate.add(v); + } + }, true, UiUtil.permitted(UserRight.IMMUNIZATION_DELETE)); currentVaccinationEditComponent.getDiscardButton().setCaption(I18nProperties.getCaption(Captions.actionCancel)); currentVaccinationEditComponent.getButtonsPanel() diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/hospitalization/HospitalizationForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/hospitalization/HospitalizationForm.java index 716ba30ac49..7f00d8bdc11 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/hospitalization/HospitalizationForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/hospitalization/HospitalizationForm.java @@ -50,12 +50,12 @@ import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.DateComparator; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.OutbreakFieldVisibilityChecker; @@ -100,7 +100,7 @@ public HospitalizationForm(CaseDataDto caze, ViewMode viewMode, boolean isPseudo false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()) .add(new OutbreakFieldVisibilityChecker(viewMode)), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized), + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized), isEditAllowed); this.caze = caze; this.viewMode = viewMode; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java index af1e2f96a19..fbe86c47094 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java @@ -66,7 +66,6 @@ import de.symeda.sormas.api.infrastructure.facility.FacilityType; import de.symeda.sormas.api.infrastructure.facility.FacilityTypeGroup; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.SearchSpecificLayout; @@ -78,6 +77,7 @@ import de.symeda.sormas.ui.utils.ComboBoxWithPlaceholder; import de.symeda.sormas.ui.utils.ConfirmationComponent; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.InfrastructureFieldsHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -139,7 +139,7 @@ public ImmunizationDataForm(boolean isPseudonymized, boolean inJurisdiction, Cas ImmunizationDto.I18N_PREFIX, false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.relatedCase = relatedCase; this.actionCallback = actionCallback; addFields(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonContactDetailsField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonContactDetailsField.java index 66c471fd9da..491cd82be6a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonContactDetailsField.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonContactDetailsField.java @@ -11,6 +11,7 @@ import com.vaadin.v7.data.Property; import com.vaadin.v7.ui.Table; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; @@ -161,8 +162,9 @@ protected void updateColumns() { table.setColumnExpandRatio(ACTION_COLUMN_ID, 0); table.setCellStyleGenerator( - FieldAccessCellStyleGenerator - .withFieldAccessCheckers(PersonContactDetailDto.class, UiFieldAccessCheckers.forSensitiveData(isPseudonymized))); + FieldAccessCellStyleGenerator.withFieldAccessCheckers( + PersonContactDetailDto.class, + UiFieldAccessCheckers.forSensitiveData(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale()))); table.setColumnExpandRatio(COLUMN_PRIMARY, 0); table.setColumnExpandRatio(COLUMN_OWNER, 0); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java index 075fc55ed00..9f6cab9751c 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java @@ -111,7 +111,7 @@ public PersonCreateForm( PersonDto.I18N_PREFIX, false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.getDefault(false)); + UiFieldAccessCheckers.getDefault(false, FacadeProvider.getConfigFacade().getCountryLocale())); this.showHomeAddressForm = showHomeAddressForm; this.showPresentCondition = showPresentCondition; this.showSymptomsOnsetDate = showSymptomsOnsetDate; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index d3af772d3bc..4367a571630 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -78,17 +78,16 @@ import de.symeda.sormas.api.person.Salutation; import de.symeda.sormas.api.utils.DataHelper.Pair; import de.symeda.sormas.api.utils.DateHelper; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.api.utils.fieldvisibility.checkers.CountryFieldVisibilityChecker; import de.symeda.sormas.api.utils.luxembourg.LuxembourgNationalHealthIdValidator; import de.symeda.sormas.ui.ControllerProvider; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.location.LocationEditForm; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.ApproximateAgeValidator; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.OutbreakFieldVisibilityChecker; import de.symeda.sormas.ui.utils.ResizableTextAreaWrapper; @@ -202,7 +201,7 @@ public PersonEditForm( FieldVisibilityCheckers.withDisease(disease) .add(new OutbreakFieldVisibilityChecker(viewMode)) .add(new CountryFieldVisibilityChecker(FacadeProvider.getConfigFacade().getCountryLocale())), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized), + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized), isEditAllowed); this.personContext = personContext; @@ -233,7 +232,7 @@ public PersonEditForm( FieldVisibilityCheckers.withDisease(disease) .add(new OutbreakFieldVisibilityChecker(viewMode)) .add(new CountryFieldVisibilityChecker(FacadeProvider.getConfigFacade().getCountryLocale())), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.personContext = personContext; this.disease = disease; @@ -256,7 +255,7 @@ public PersonEditForm(boolean isEditAllowed, boolean isPseudonymized, boolean in false, new FieldVisibilityCheckers().add(new OutbreakFieldVisibilityChecker(ViewMode.NORMAL)) .add(new CountryFieldVisibilityChecker(FacadeProvider.getConfigFacade().getCountryLocale())), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized), + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized), isEditAllowed); CssStyles.style(CssStyles.H3, occupationHeader, addressHeader, addressesHeader, contactInformationHeader); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java index d2b201c05e5..281313c0b50 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java @@ -58,14 +58,13 @@ import de.symeda.sormas.api.sample.PathogenTestType; import de.symeda.sormas.api.sample.SampleDto; import de.symeda.sormas.api.sample.SamplePurpose; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; import de.symeda.sormas.ui.utils.DateFormatHelper; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldConfiguration; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -158,9 +157,7 @@ public PathogenTestForm(boolean create, int caseSampleCount, boolean isPseudonym PathogenTestDto.I18N_PREFIX, false, FieldVisibilityCheckers.withDisease(null).andWithCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel( - UiUtil.getPseudonymizableDataAccessLevel(create || inJurisdiction), // Jurisdiction doesn't matter for creation forms - !create && isPseudonymized)); // Pseudonymization doesn't matter for creation forms + FieldAccessHelper.getFieldAccessCheckers(create || inJurisdiction, !create && isPseudonymized));// Jurisdiction doesn't matter for creation forms // Pseudonymization doesn't matter for creation forms this.caseSampleCount = caseSampleCount; this.create = create; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/environmentsample/EnvironmentSampleEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/environmentsample/EnvironmentSampleEditForm.java index 530e5d69aea..799b3380fb1 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/environmentsample/EnvironmentSampleEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/environmentsample/EnvironmentSampleEditForm.java @@ -117,7 +117,7 @@ public EnvironmentSampleEditForm(boolean isPseudonymized, boolean isCreate) { EnvironmentSampleDto.I18N_PREFIX, false, FieldVisibilityCheckers.getNoop(), - UiFieldAccessCheckers.getDefault(isPseudonymized)); + UiFieldAccessCheckers.getDefault(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale())); this.isCreate = isCreate; addFields(); addValueChangeListener(e -> defaultValueChangeListener((EnvironmentSampleDto) e.getProperty().getValue())); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleDataView.java index cca12f46ed6..0544db8e6c4 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleDataView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleDataView.java @@ -37,7 +37,6 @@ import de.symeda.sormas.api.sample.SampleDto; import de.symeda.sormas.api.sample.SampleReferenceDto; import de.symeda.sormas.api.user.UserRight; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.ui.AbstractInfoLayout; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.UiUtil; @@ -52,6 +51,7 @@ import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DetailSubComponentWrapper; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.LayoutWithSidePanel; import de.symeda.sormas.ui.utils.components.sidecomponent.SideComponentLayout; @@ -201,8 +201,7 @@ private AbstractInfoLayout getDependentSideComponent(SampleDto sample disease = contactDto.getDisease(); - final ContactInfoLayout contactInfoLayout = - new ContactInfoLayout(contactDto, UiFieldAccessCheckers.getDefault(contactDto.isPseudonymized())); + final ContactInfoLayout contactInfoLayout = new ContactInfoLayout(contactDto, FieldAccessHelper.getFieldAccessCheckers(contactDto)); contactInfoLayout.addStyleName(CssStyles.SIDE_COMPONENT); return (AbstractInfoLayout) contactInfoLayout; @@ -216,10 +215,8 @@ private AbstractInfoLayout getDependentSideComponent(SampleDto sample disease = eventDto.getDisease(); - final EventParticipantInfoLayout eventParticipantInfoLayout = new EventParticipantInfoLayout( - eventParticipantDto, - eventDto, - UiFieldAccessCheckers.getDefault(eventParticipantDto.isPseudonymized())); + final EventParticipantInfoLayout eventParticipantInfoLayout = + new EventParticipantInfoLayout(eventParticipantDto, eventDto, FieldAccessHelper.getFieldAccessCheckers(eventParticipantDto)); eventParticipantInfoLayout.addStyleName(CssStyles.SIDE_COMPONENT); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleEditForm.java index f2f6cabaa40..8e5073026a7 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/humansample/SampleEditForm.java @@ -41,12 +41,12 @@ import de.symeda.sormas.api.sample.SampleDto; import de.symeda.sormas.api.user.UserReferenceDto; import de.symeda.sormas.api.user.UserRight; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.samples.AbstractSampleForm; import de.symeda.sormas.ui.utils.DateComparisonValidator; import de.symeda.sormas.ui.utils.DateFormatHelper; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; public class SampleEditForm extends AbstractSampleForm { @@ -62,11 +62,7 @@ public class SampleEditForm extends AbstractSampleForm { private Label laboratorySampleHeadingLabel; public SampleEditForm(boolean isPseudonymized, boolean inJurisdiction, Disease disease) { - super( - SampleDto.class, - SampleDto.I18N_PREFIX, - disease, - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + super(SampleDto.class, SampleDto.I18N_PREFIX, disease, FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); testsToBeRemovedOnCommit = new ArrayList(); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java index fc9861b1581..870f42355e0 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java @@ -49,14 +49,13 @@ import de.symeda.sormas.api.user.JurisdictionLevel; import de.symeda.sormas.api.user.UserReferenceDto; import de.symeda.sormas.api.user.UserRight; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.api.utils.fieldvisibility.checkers.CountryFieldVisibilityChecker; import de.symeda.sormas.api.utils.luxembourg.LuxembourgNationalHealthIdValidator; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.location.LocationEditForm; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.PhoneNumberValidator; import de.symeda.sormas.ui.utils.ValidationUtils; @@ -95,7 +94,7 @@ public SelfReportDataForm(Disease disease, boolean inJurisdiction, boolean isPse SelfReportDto.I18N_PREFIX, true, FieldVisibilityCheckers.withDisease(disease).add(new CountryFieldVisibilityChecker(FacadeProvider.getConfigFacade().getCountryLocale())), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); } @Override diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskGrid.java index c723f40551b..a20d3560303 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/task/TaskGrid.java @@ -166,7 +166,7 @@ public TaskGrid(TaskCriteria criteria) { getColumn(TaskIndexDto.CONTEXT_REFERENCE).setStyleGenerator(new FieldAccessColumnStyleGenerator<>(task -> { boolean isInJurisdiction = task.getTaskJurisdictionFlagsDto().getInJurisdiction(); - return UiFieldAccessCheckers.getDefault(!isInJurisdiction).hasRight(); + return UiFieldAccessCheckers.getDefault(!isInJurisdiction, FacadeProvider.getConfigFacade().getCountryLocale()).hasRight(); })); addItemClickListener(new ShowDetailsListener<>(TaskIndexDto.CONTEXT_REFERENCE, false, this::navigateToData)); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionForm.java index ce174d85b2e..80bba42de13 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionForm.java @@ -17,11 +17,10 @@ import de.symeda.sormas.api.therapy.PrescriptionDto; import de.symeda.sormas.api.therapy.TreatmentRoute; import de.symeda.sormas.api.therapy.TreatmentType; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -43,7 +42,7 @@ public PrescriptionForm(boolean create, boolean readOnly, boolean isPseudonymize PrescriptionDto.I18N_PREFIX, false, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); addFields(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionGrid.java index f4c18f9d98b..fb60dd7c797 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/PrescriptionGrid.java @@ -96,8 +96,9 @@ public Class getType() { I18nProperties.getPrefixCaption(PrescriptionIndexDto.I18N_PREFIX, column.getPropertyId().toString(), column.getHeaderCaption())); setCellStyleGenerator( - FieldAccessCellStyleGenerator - .withFieldAccessCheckers(PrescriptionIndexDto.class, UiFieldAccessCheckers.forSensitiveData(isPseudonymized))); + FieldAccessCellStyleGenerator.withFieldAccessCheckers( + PrescriptionIndexDto.class, + UiFieldAccessCheckers.forSensitiveData(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale()))); } addItemClickListener(e -> { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentForm.java index e91756f128b..ecf5bd9d10e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentForm.java @@ -13,11 +13,10 @@ import de.symeda.sormas.api.therapy.TreatmentDto; import de.symeda.sormas.api.therapy.TreatmentRoute; import de.symeda.sormas.api.therapy.TreatmentType; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -38,7 +37,7 @@ public TreatmentForm(boolean create, boolean isPseudonymized, boolean inJurisdic TreatmentDto.I18N_PREFIX, true, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); setWidth(680, Unit.PIXELS); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentGrid.java index 6172746f547..13d534355bd 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/therapy/TreatmentGrid.java @@ -67,8 +67,9 @@ public TreatmentGrid( I18nProperties.getPrefixCaption(TreatmentIndexDto.I18N_PREFIX, column.getPropertyId().toString(), column.getHeaderCaption())); setCellStyleGenerator( - FieldAccessCellStyleGenerator - .withFieldAccessCheckers(TreatmentIndexDto.class, UiFieldAccessCheckers.forSensitiveData(isPseudonymized))); + FieldAccessCellStyleGenerator.withFieldAccessCheckers( + TreatmentIndexDto.class, + UiFieldAccessCheckers.forSensitiveData(isPseudonymized, FacadeProvider.getConfigFacade().getCountryLocale()))); } addItemClickListener(e -> { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java index a5b7c5979ee..3c0f0d024d9 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java @@ -52,6 +52,7 @@ import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateComparisonValidator; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.InfrastructureFieldsHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -439,7 +440,7 @@ private static UiFieldAccessCheckers createFieldAccessCheckers( boolean inJurisdiction, boolean withPersonalAndSensitive) { if (withPersonalAndSensitive) { - return UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized); + return FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized); } return UiFieldAccessCheckers.getNoop(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessColumnStyleGenerator.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessColumnStyleGenerator.java index 7ab5f9338ac..d602b942a3a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessColumnStyleGenerator.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessColumnStyleGenerator.java @@ -19,6 +19,7 @@ import com.vaadin.ui.StyleGenerator; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.pseudonymization.Pseudonymizable; @@ -28,12 +29,18 @@ public class FieldAccessColumnStyleGenerator implements StyleGenerator { public static FieldAccessColumnStyleGenerator getDefault(Class beanType, String columnId) { - return forFieldAccessCheckers(beanType, columnId, UiFieldAccessCheckers.getDefault(true)); + return forFieldAccessCheckers( + beanType, + columnId, + UiFieldAccessCheckers.getDefault(true, FacadeProvider.getConfigFacade().getCountryLocale())); } public static FieldAccessColumnStyleGenerator forSensitiveData(Class beanType, String columnId) { - return forFieldAccessCheckers(beanType, columnId, UiFieldAccessCheckers.forSensitiveData(true)); + return forFieldAccessCheckers( + beanType, + columnId, + UiFieldAccessCheckers.forSensitiveData(true, FacadeProvider.getConfigFacade().getCountryLocale())); } private static FieldAccessColumnStyleGenerator forFieldAccessCheckers( diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessHelper.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessHelper.java index 0c4ea358aa3..fe9a895990a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessHelper.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/FieldAccessHelper.java @@ -17,12 +17,31 @@ import java.util.stream.Stream; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; +import de.symeda.sormas.api.utils.pseudonymization.PseudonymizableDto; +import de.symeda.sormas.ui.UiUtil; public class FieldAccessHelper { public static boolean isAllInaccessible(String... values) { return Stream.of(values).allMatch(v -> I18nProperties.getCaption(Captions.inaccessibleValue).equals(v)); } + + public static UiFieldAccessCheckers getFieldAccessCheckers(boolean inJurisdiction, boolean isPseudonymized) { + return UiFieldAccessCheckers.forDataAccessLevel( + UiUtil.getPseudonymizableDataAccessLevel(inJurisdiction), + isPseudonymized, + FacadeProvider.getConfigFacade().getCountryLocale()); + } + + public static UiFieldAccessCheckers getFieldAccessCheckers(T dto) { + return UiFieldAccessCheckers.forDataAccessLevel( + UiUtil.getPseudonymizableDataAccessLevel(dto.isInJurisdiction()), + dto.isPseudonymized(), + FacadeProvider.getConfigFacade().getCountryLocale()); + } + } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/vaccination/list/VaccinationList.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/vaccination/list/VaccinationList.java index 81f86a48f2e..4dc5127e504 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/vaccination/list/VaccinationList.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/vaccination/list/VaccinationList.java @@ -26,11 +26,11 @@ import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.user.UserRight; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.vaccination.VaccinationListEntryDto; import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.SormasUI; import de.symeda.sormas.ui.UiUtil; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.PaginationList; public class VaccinationList extends PaginationList { @@ -88,9 +88,7 @@ protected void drawDisplayedEntries() { .edit( FacadeProvider.getVaccinationFacade().getByUuid(listEntry.getVaccination().getUuid()), listEntry.getVaccination().getDisease(), - UiFieldAccessCheckers.forDataAccessLevel( - UiUtil.getPseudonymizableDataAccessLevel(vaccination.isInJurisdiction()), - vaccination.isPseudonymized()), + FieldAccessHelper.getFieldAccessCheckers(vaccination.isInJurisdiction(), vaccination.isPseudonymized()), true, v -> SormasUI.refreshView(), deleteCallback(), diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/visit/VisitEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/visit/VisitEditForm.java index 6f4d7259a62..f8db4394dac 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/visit/VisitEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/visit/VisitEditForm.java @@ -42,13 +42,12 @@ import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.symptoms.SymptomsContext; import de.symeda.sormas.api.utils.DateHelper; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.visit.VisitDto; import de.symeda.sormas.api.visit.VisitStatus; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.symptoms.SymptomsForm; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; @@ -82,7 +81,7 @@ public VisitEditForm( VisitDto.I18N_PREFIX, false, null, - UiFieldAccessCheckers.forDataAccessLevel(UiUtil.getPseudonymizableDataAccessLevel(create || inJurisdiction), !create && isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(create || inJurisdiction, !create && isPseudonymized)); if (create) { hideValidationUntilNextCommit(); } From 8a9119a3e771a6cc8fd06b6f45a5b0a0b61a1ab3 Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Wed, 16 Oct 2024 14:40:16 +0100 Subject: [PATCH 15/56] update sormas_schema.sql for aefi module, fix tests, update user rights in web descriptors --- .../AefiChartData.java | 6 +- .../api/importexport/DatabaseTable.java | 8 + .../sormas/api/user/DefaultUserRole.java | 14 + .../entity/Aefi.java | 4 +- .../entity/AefiInvestigation.java | 4 +- .../resources/META-INF/glassfish-ejb-jar.xml | 35 ++ .../src/main/resources/sql/sormas_schema.sql | 411 ++++++++++++++++++ .../backend/common/HistoryTablesTest.java | 7 + .../DatabaseExportServiceTest.java | 12 +- .../src/test/resources/checkHistoryTables.sql | 4 +- .../src/main/webapp/WEB-INF/glassfish-web.xml | 35 ++ sormas-rest/src/main/webapp/WEB-INF/web.xml | 28 ++ .../AbstractAefiView.java | 2 + .../AefiInvestigationView.java | 4 +- .../AefiView.java | 4 +- .../AefiInvestigationList.java | 4 +- .../aefilink/AefiList.java | 11 +- .../webapp/VAADIN/themes/sormas/global.scss | 8 + .../src/main/webapp/WEB-INF/glassfish-web.xml | 35 ++ sormas-ui/src/main/webapp/WEB-INF/web.xml | 28 ++ 20 files changed, 637 insertions(+), 27 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java index 0ba1d2093bc..b6190530a39 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/dashboard/adverseeventsfollowingimmunization/AefiChartData.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -22,6 +19,9 @@ import java.util.ArrayList; import java.util.List; +import de.symeda.sormas.api.audit.AuditedClass; + +@AuditedClass public class AefiChartData implements Serializable { private static final long serialVersionUID = 3538219674050390425L; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/importexport/DatabaseTable.java b/sormas-api/src/main/java/de/symeda/sormas/api/importexport/DatabaseTable.java index 63f15937443..171ad6e8303 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/importexport/DatabaseTable.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/importexport/DatabaseTable.java @@ -68,6 +68,14 @@ public enum DatabaseTable { IMMUNIZATIONS(DatabaseTableType.SORMAS, "immunizations", dependingOnFeature(FeatureType.IMMUNIZATION_MANAGEMENT)), VACCINATIONS(DatabaseTableType.SORMAS, IMMUNIZATIONS, "vaccinations"), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATIONS(DatabaseTableType.SORMAS, + "adverse_events_following_immunizations", + dependingOnFeature(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT)), + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_INVESTIGATIONS(DatabaseTableType.SORMAS, + "adverse_events_following_immunization_investigations", + dependingOnFeature(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT)), + ADVERSE_EVENTS(DatabaseTableType.SORMAS, "adverse_events", dependingOnFeature(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT)), + SAMPLES(DatabaseTableType.SORMAS, "samples", dependingOnFeature(FeatureType.SAMPLES_LAB)), PATHOGEN_TESTS(DatabaseTableType.SORMAS, SAMPLES, "pathogen_tests"), ADDITIONAL_TESTS(DatabaseTableType.SORMAS, SAMPLES, "additional_tests", dependingOnFeature(FeatureType.ADDITIONAL_TESTS)), diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/user/DefaultUserRole.java b/sormas-api/src/main/java/de/symeda/sormas/api/user/DefaultUserRole.java index 5e7a2fc06ab..b91d28b515b 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/user/DefaultUserRole.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/user/DefaultUserRole.java @@ -304,6 +304,13 @@ public Set getDefaultUserRights() { IMMUNIZATION_DELETE, IMMUNIZATION_ARCHIVE, IMMUNIZATION_VIEW_ARCHIVED, + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT, PERSON_VIEW, PERSON_EDIT, PERSON_DELETE, @@ -1315,6 +1322,13 @@ public Set getDefaultUserRights() { IMMUNIZATION_CREATE, IMMUNIZATION_EDIT, IMMUNIZATION_DELETE, + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE, + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT, PERSON_VIEW, PERSON_EDIT, PERSON_DELETE, diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java index 92b65406c2b..63df6410c97 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/Aefi.java @@ -32,7 +32,6 @@ import javax.persistence.ManyToMany; import javax.persistence.ManyToOne; import javax.persistence.OneToOne; -import javax.persistence.Table; import javax.persistence.Temporal; import javax.persistence.TemporalType; @@ -53,8 +52,7 @@ import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.vaccination.Vaccination; -@Entity -@Table(name = "adverseeventsfollowingimmunization") +@Entity(name = "adverseeventsfollowingimmunization") public class Aefi extends CoreAdo { private static final long serialVersionUID = -7845660472641846292L; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java index e4157a4c6c0..be0e4de6af1 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/adverseeventsfollowingimmunization/entity/AefiInvestigation.java @@ -32,7 +32,6 @@ import javax.persistence.ManyToMany; import javax.persistence.ManyToOne; import javax.persistence.OneToOne; -import javax.persistence.Table; import javax.persistence.Temporal; import javax.persistence.TemporalType; import javax.persistence.Transient; @@ -66,8 +65,7 @@ import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.vaccination.Vaccination; -@Entity -@Table(name = "adverseeventsfollowingimmunizationinvestigation") +@Entity(name = "adverseeventsfollowingimmunizationinvestigation") public class AefiInvestigation extends CoreAdo { private static final long serialVersionUID = 6128204752074963848L; diff --git a/sormas-backend/src/main/resources/META-INF/glassfish-ejb-jar.xml b/sormas-backend/src/main/resources/META-INF/glassfish-ejb-jar.xml index 584b2fa6fd6..c907e3669f4 100644 --- a/sormas-backend/src/main/resources/META-INF/glassfish-ejb-jar.xml +++ b/sormas-backend/src/main/resources/META-INF/glassfish-ejb-jar.xml @@ -122,6 +122,36 @@ IMMUNIZATION_ARCHIVE + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + + PERSON_VIEW PERSON_VIEW @@ -607,6 +637,11 @@ DASHBOARD_SAMPLES_VIEW + + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + CASE_CLINICIAN_VIEW CASE_CLINICIAN_VIEW diff --git a/sormas-backend/src/main/resources/sql/sormas_schema.sql b/sormas-backend/src/main/resources/sql/sormas_schema.sql index f08ecaec095..fb32127ab77 100644 --- a/sormas-backend/src/main/resources/sql/sormas_schema.sql +++ b/sormas-backend/src/main/resources/sql/sormas_schema.sql @@ -13257,4 +13257,415 @@ ALTER TABLE externalmessage_history ADD COLUMN personadditionaldetails text; INSERT INTO schema_version (version_number, comment) VALUES (551, '#13147 Phone Number Validation for E-Santé Reports – Remove and Store Non-Numeric Text'); + +-- 2024-10-16 Adverse Events Following Immunization (AEFI) - Entities #12634 +-- adverse events +create table adverseevents +( + id bigint not null, + uuid varchar(36) not null, + changedate timestamp(3) not null, + creationdate timestamp(3) not null, + severelocalreaction varchar(255), + severelocalreactionmorethanthreedays boolean, + severelocalreactionbeyondnearestjoint boolean, + seizures varchar(255), + seizuretype varchar(255), + abscess varchar(255), + sepsis varchar(255), + encephalopathy varchar(255), + toxicshocksyndrome varchar(255), + thrombocytopenia varchar(255), + anaphylaxis varchar(255), + feverishfeeling varchar(255), + otheradverseeventdetails varchar(255), + sys_period tstzrange not null, + change_user_id bigint +); + +alter table adverseevents owner to sormas_user; + +alter table adverseevents add primary key (id); +alter table adverseevents add unique (uuid); +alter table adverseevents add constraint fk_adverseevents_change_user_id foreign key (change_user_id) references users; + +-- adverse events history +CREATE TABLE adverseevents_history (LIKE adverseevents); +DROP TRIGGER IF EXISTS versioning_trigger ON adverseevents; +CREATE TRIGGER versioning_trigger + BEFORE INSERT OR UPDATE ON adverseevents + FOR EACH ROW EXECUTE PROCEDURE versioning('sys_period', 'adverseevents_history', true); +DROP TRIGGER IF EXISTS delete_history_trigger ON adverseevents; +CREATE TRIGGER delete_history_trigger + AFTER DELETE ON adverseevents + FOR EACH ROW EXECUTE PROCEDURE delete_history_trigger('adverseevents_history', 'id'); +ALTER TABLE adverseevents_history OWNER TO sormas_user; + +-- adverse events following immunization (AEFI) +create table adverseeventsfollowingimmunization +( + id bigint not null, + uuid varchar(36) not null, + changedate timestamp(3) not null, + creationdate timestamp(3) not null, + immunization_id bigint not null, + address_id bigint, + primarysuspectvaccine_id bigint not null, + adverseevents_id bigint not null, + person_id bigint, + reportdate timestamp not null, + reportinguser_id bigint not null, + externalid varchar(512), + responsibleregion_id bigint, + responsibledistrict_id bigint, + responsiblecommunity_id bigint, + country_id bigint, + reportingidnumber varchar(512) not null, + firstname varchar(255), + lastname varchar(255), + phonenumber varchar(255), + pregnant varchar(255), + trimester varchar(255), + lactating varchar(255), + onsetageyears integer, + onsetagemonths integer, + onsetagedays integer, + agegroup varchar(255), + healthfacility_id bigint, + healthfacilitydetails varchar(512), + reportingofficername varchar(255), + reportingofficerfacility_id bigint, + reportingofficerfacilitydetails varchar(512), + reportingofficerdesignation varchar(255), + reportingofficerdepartment varchar(255), + reportingofficeraddress_id bigint, + reportingofficerphonenumber varchar(255), + reportingofficeremail varchar(255), + healthsystemnotifieddate timestamp, + todaysdate timestamp, + startdatetime timestamp, + aefidescription text, + serious varchar(255) not null, + seriousreason varchar(255), + seriousreasondetails text, + outcome varchar(255) not null, + deathdate timestamp, + autopsydone varchar(255), + pastmedicalhistory text, + investigationneeded varchar(255), + investigationplanneddate timestamp, + receivedatnationalleveldate timestamp, + worldwideid varchar(255), + nationallevelcomment text, + archived boolean default false, + endofprocessingdate timestamp, + archiveundonereason varchar(255), + deleted boolean default false, + deletionreason varchar(255), + otherdeletionreason text, + sys_period tstzrange not null, + change_user_id bigint +); + +alter table adverseeventsfollowingimmunization owner to sormas_user; + +alter table adverseeventsfollowingimmunization add primary key (id); +alter table adverseeventsfollowingimmunization add unique (uuid); +alter table adverseeventsfollowingimmunization add constraint fk_aefi_address_id foreign key (address_id) references location; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_adverseevents_id foreign key (adverseevents_id) references adverseevents; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_change_user_id foreign key (change_user_id) references users; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_country_id foreign key (country_id) references country; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_healthfacility_id foreign key (healthfacility_id) references facility; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_immunization_id foreign key (immunization_id) references immunization; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_person_id foreign key (person_id) references person; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_primarysuspectvaccine_id foreign key (primarysuspectvaccine_id) references vaccination; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_reporting_user_id foreign key (reportinguser_id) references users; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_reportingofficeraddress_id foreign key (reportingofficeraddress_id) references location; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_reportingofficerfacility_id foreign key (reportingofficerfacility_id) references facility; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_responsiblecommunity_id foreign key (responsiblecommunity_id) references community; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_responsibledistrict_id foreign key (responsibledistrict_id) references district; +alter table adverseeventsfollowingimmunization add constraint fk_aefi_responsibleregion_id foreign key (responsibleregion_id) references region; + +-- AEFI history +CREATE TABLE adverseeventsfollowingimmunization_history (LIKE adverseeventsfollowingimmunization); +DROP TRIGGER IF EXISTS versioning_trigger ON adverseeventsfollowingimmunization; +CREATE TRIGGER versioning_trigger + BEFORE INSERT OR UPDATE ON adverseeventsfollowingimmunization + FOR EACH ROW EXECUTE PROCEDURE versioning('sys_period', 'adverseeventsfollowingimmunization_history', true); +DROP TRIGGER IF EXISTS delete_history_trigger ON adverseeventsfollowingimmunization; +CREATE TRIGGER delete_history_trigger + AFTER DELETE ON adverseeventsfollowingimmunization + FOR EACH ROW EXECUTE PROCEDURE delete_history_trigger('adverseeventsfollowingimmunization_history', 'id'); +ALTER TABLE adverseeventsfollowingimmunization_history OWNER TO sormas_user; + +-- AEFI vaccination +create table adverseeventsfollowingimmunization_vaccinations +( + adverseeventsfollowingimmunization_id bigint not null, + vaccination_id bigint not null, + sys_period tstzrange not null +); + +alter table adverseeventsfollowingimmunization_vaccinations owner to sormas_user; + +alter table adverseeventsfollowingimmunization_vaccinations add constraint aefi_vaccinations_pkey primary key (adverseeventsfollowingimmunization_id, vaccination_id); +alter table adverseeventsfollowingimmunization_vaccinations add constraint fk_aefi_vaccinations_aefi_id foreign key (adverseeventsfollowingimmunization_id) references adverseeventsfollowingimmunization; +alter table adverseeventsfollowingimmunization_vaccinations add constraint fk_aefi_vaccinations_vaccination_id foreign key (vaccination_id) references vaccination; + +-- AEFI vaccination history +CREATE TABLE adverseeventsfollowingimmunization_vaccinations_history (LIKE adverseeventsfollowingimmunization_vaccinations); +DROP TRIGGER IF EXISTS versioning_trigger ON adverseeventsfollowingimmunization_vaccinations; +CREATE TRIGGER versioning_trigger + BEFORE INSERT OR UPDATE ON adverseeventsfollowingimmunization_vaccinations + FOR EACH ROW EXECUTE PROCEDURE versioning('sys_period', 'adverseeventsfollowingimmunization_vaccinations_history', true); +DROP TRIGGER IF EXISTS delete_history_trigger ON adverseeventsfollowingimmunization_vaccinations; +CREATE TRIGGER delete_history_trigger + AFTER DELETE ON adverseeventsfollowingimmunization_vaccinations + FOR EACH ROW EXECUTE PROCEDURE delete_history_trigger('adverseeventsfollowingimmunization_vaccinations_history', 'id'); +ALTER TABLE adverseeventsfollowingimmunization_vaccinations_history OWNER TO sormas_user; + +-- AEFI investigation +create table adverseeventsfollowingimmunizationinvestigation +( + id bigint not null, + uuid varchar(36) not null, + changedate timestamp(3) not null, + creationdate timestamp(3) not null, + adverseeventsfollowingimmunization_id bigint not null, + address_id bigint, + primarysuspectvaccine_id bigint, + reportdate timestamp not null, + reportinguser_id bigint not null, + externalid varchar(512), + responsibleregion_id bigint, + responsibledistrict_id bigint, + responsiblecommunity_id bigint, + country_id bigint, + investigationcaseid varchar(255), + placeofvaccination varchar(255) not null, + placeofvaccinationdetails varchar(512), + vaccinationactivity varchar(255) not null, + vaccinationactivitydetails varchar(512), + vaccinationfacility_id bigint, + vaccinationfacilitydetails varchar(512), + reportingofficername varchar(255), + reportingofficerfacility_id bigint, + reportingofficerfacilitydetails varchar(512), + reportingofficerdesignation varchar(255), + reportingofficerdepartment varchar(255), + reportingofficeraddress_id bigint, + reportingofficerlandlinephonenumber varchar(255), + reportingofficermobilephonenumber varchar(255), + reportingofficeremail varchar(255), + investigationdate timestamp not null, + formcompletiondate timestamp not null, + investigationstage varchar(255), + onsetageyears integer, + onsetagemonths integer, + onsetagedays integer, + agegroup varchar(255), + typeofsite varchar(255) not null, + typeofsitedetails varchar(255), + keysymptomdatetime timestamp, + hospitalizationdate timestamp, + reportedtohealthauthoritydate timestamp, + statusondateofinvestigation varchar(255), + deathdatetime timestamp, + autopsydone varchar(255), + autopsydate timestamp, + autopsyplanneddatetime timestamp, + autopsyreportname varchar(255), + pasthistoryofsimilarevent varchar(255), + pasthistoryofsimilareventdetails varchar(512), + adverseeventafterpreviousvaccinations varchar(255), + adverseeventafterpreviousvaccinationsdetails varchar(512), + historyofallergytovaccinedrugorfood varchar(255), + historyofallergytovaccinedrugorfooddetails varchar(512), + preexistingillnessthirtydaysorcongenitaldisorder varchar(255), + preexistingillnessthirtydaysorcongenitaldisorderdetails varchar(512), + historyofhospitalizationinlastthirtydayswithcause varchar(255), + historyofhospitalizationinlastthirtydayswithcausedetails varchar(512), + currentlyonconcomitantmedication varchar(255), + currentlyonconcomitantmedicationdetails varchar(255), + familyhistoryofdiseaseorallergy varchar(255), + familyhistoryofdiseaseorallergydetails varchar(512), + pregnant varchar(255), + numberofweekspregnant integer, + breastfeeding varchar(255), + birthterm varchar(255), + birthweight real, + deliveryprocedure varchar(255), + deliveryproceduredetails varchar(512), + seriousaefiinfosourcestring varchar(512), + seriousaefiinfosourcedetails varchar(512), + seriousaefiverbalautopsyinfosourcedetails varchar(512), + firstcaregiversname varchar(255), + othercaregiversnames varchar(512), + othersourceswhoprovidedinfo varchar(512), + signsandsymptomsfromtimeofvaccination text, + clinicaldetailsofficername varchar(255), + clinicaldetailsofficerphonenumber varchar(255), + clinicaldetailsofficeremail varchar(255), + clinicaldetailsofficerdesignation varchar(255), + clinicaldetailsdatetime timestamp, + patientreceivedmedicalcare varchar(255), + patientreceivedmedicalcaredetails text, + provisionalorfinaldiagnosis text, + patientimmunizedperiod varchar(255), + patientimmunizedperioddetails varchar(512), + vaccinegivenperiod varchar(255), + vaccinegivenperioddetails varchar(512), + errorprescribingvaccine varchar(255), + errorprescribingvaccinedetails varchar(512), + vaccinecouldhavebeenunsterile varchar(255), + vaccinecouldhavebeenunsteriledetails varchar(512), + vaccinephysicalconditionabnormal varchar(255), + vaccinephysicalconditionabnormaldetails varchar(255), + errorinvaccinereconstitution varchar(255), + errorinvaccinereconstitutiondetails varchar(512), + errorinvaccinehandling varchar(255), + errorinvaccinehandlingdetails varchar(512), + vaccineadministeredincorrectly varchar(255), + vaccineadministeredincorrectlydetails varchar(255), + numberimmunizedfromconcernedvaccinevial integer, + numberimmunizedwithconcernedvaccineinsamesession integer, + numberimmunizedconcernedvaccinesamebatchnumberotherlocations integer, + numberimmunizedconcernedvaccinesamebatchnumberlocationdetails varchar(512), + vaccinehasqualitydefect varchar(255), + vaccinehasqualitydefectdetails varchar(512), + eventisastressresponserelatedtoimmunization varchar(255), + eventisastressresponserelatedtoimmunizationdetails varchar(255), + caseispartofacluster varchar(255), + caseispartofaclusterdetails varchar(255), + numberofcasesdetectedincluster integer, + allcasesinclusterreceivedvaccinefromsamevial varchar(255), + allcasesinclusterreceivedvaccinefromsamevialdetails varchar(512), + numberofvialsusedincluster integer, + numberofvialsusedinclusterdetails varchar(255), + adsyringesusedforimmunization varchar(255), + typeofsyringesused varchar(255), + typeofsyringesuseddetails varchar(512), + syringesusedadditionaldetails text, + samereconstitutionsyringeusedformultiplevialsofsamevaccine varchar(255), + samereconstitutionsyringeusedforreconstitutingdifferentvaccines varchar(255), + samereconstitutionsyringeforeachvaccinevial varchar(255), + samereconstitutionsyringeforeachvaccination varchar(255), + vaccinesanddiluentsusedrecommendedbymanufacturer varchar(255), + reconstitutionadditionaldetails text, + correctdoseorroute varchar(255), + timeofreconstitutionmentionedonthevial varchar(255), + nontouchtechniquefollowed varchar(255), + contraindicationscreenedpriortovaccination varchar(255), + numberofaefireportedfromvaccinedistributioncenterlastthirtydays integer, + trainingreceivedbyvaccinator varchar(255), + lasttrainingreceivedbyvaccinatordate timestamp, + injectiontechniqueadditionaldetails text, + vaccinestoragerefrigeratortemperaturemonitored varchar(255), + anystoragetemperaturedeviationoutsidetwotoeightdegrees varchar(255), + storagetemperaturemonitoringadditionaldetails text, + correctprocedureforstoragefollowed varchar(255), + anyotheriteminrefrigerator varchar(255), + partiallyusedreconstitutedvaccinesinrefrigerator varchar(255), + unusablevaccinesinrefrigerator varchar(255), + unusablediluentsinstore varchar(255), + vaccinestoragepointadditionaldetails text, + vaccinecarriertype varchar(255), + vaccinecarriertypedetails varchar(512), + vaccinecarriersenttositeonsamedateasvaccination varchar(255), + vaccinecarrierreturnedfromsiteonsamedateasvaccination varchar(255), + conditionedicepackused varchar(255), + vaccinetransportationadditionaldetails text, + similareventsreportedsameperiodandlocality varchar(255), + similareventsreportedsameperiodandlocalitydetails varchar(512), + numberofsimilareventsreportedsameperiodandlocality integer, + numberofthoseaffectedvaccinated integer, + numberofthoseaffectednotvaccinated integer, + numberofthoseaffectedvaccinatedunknown integer, + communityinvestigationadditionaldetails text, + otherinvestigationfindings text, + investigationstatus varchar(255), + investigationstatusdetails text, + adverseeventfollowingimmunizationclassification varchar(255), + adverseeventfollowingimmunizationclassificationsubtype varchar(255), + adverseeventfollowingimmunizationclassificationdetails varchar(512), + causality varchar(255), + causalitydetails text, + investigationcompletiondate timestamp, + archived boolean, + endofprocessingdate timestamp, + archiveundonereason varchar(255), + deleted boolean, + deletionreason varchar(255), + otherdeletionreason text, + sys_period tstzrange not null, + change_user_id bigint +); + +alter table adverseeventsfollowingimmunizationinvestigation owner to sormas_user; + +alter table adverseeventsfollowingimmunizationinvestigation add primary key (id); +alter table adverseeventsfollowingimmunizationinvestigation add unique (uuid); +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_address_id foreign key (address_id) references location; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_aefi_id foreign key (adverseeventsfollowingimmunization_id) references adverseeventsfollowingimmunization; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_change_user_id foreign key (change_user_id) references users; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_country_id foreign key (country_id) references country; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_primarysuspectvaccine_id foreign key (primarysuspectvaccine_id) references vaccination; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_reportingofficeraddress_id foreign key (reportingofficeraddress_id) references location; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_reportingofficerfacility_id foreign key (reportingofficerfacility_id) references facility; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_reportinguser_id foreign key (reportinguser_id) references users; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_responsiblecommunity_id foreign key (responsiblecommunity_id) references community; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_responsibledistrict_id foreign key (responsibledistrict_id) references district; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_responsibleregion_id foreign key (responsibleregion_id) references region; +alter table adverseeventsfollowingimmunizationinvestigation add constraint fk_aefiinvestigation_vaccinationfacility_id foreign key (vaccinationfacility_id) references facility; + +-- AEFI investigation history +CREATE TABLE adverseeventsfollowingimmunizationinvestigation_history (LIKE adverseeventsfollowingimmunizationinvestigation); + +DROP TRIGGER IF EXISTS versioning_trigger ON adverseeventsfollowingimmunizationinvestigation; +CREATE TRIGGER versioning_trigger + BEFORE INSERT OR UPDATE ON adverseeventsfollowingimmunizationinvestigation + FOR EACH ROW EXECUTE PROCEDURE versioning('sys_period', 'adverseeventsfollowingimmunizationinvestigation_history', true); +DROP TRIGGER IF EXISTS delete_history_trigger ON adverseeventsfollowingimmunizationinvestigation; +CREATE TRIGGER delete_history_trigger + AFTER DELETE ON adverseeventsfollowingimmunizationinvestigation + FOR EACH ROW EXECUTE PROCEDURE delete_history_trigger('adverseeventsfollowingimmunizationinvestigation_history', 'id'); + +ALTER TABLE adverseeventsfollowingimmunizationinvestigation_history OWNER TO sormas_user; + +-- AEFI investigation vaccination +create table adverseeventsfollowingimmunizationinvestigation_vaccinations +( + adverseeventsfollowingimmunizationinvestigation_id bigint not null, + vaccination_id bigint not null, + sys_period tstzrange not null +); + +alter table adverseeventsfollowingimmunizationinvestigation_vaccinations owner to sormas_user; + +alter table adverseeventsfollowingimmunizationinvestigation_vaccinations add constraint aefi_investigation_vaccinations_pkey primary key (adverseeventsfollowingimmunizationinvestigation_id, vaccination_id); +alter table adverseeventsfollowingimmunizationinvestigation_vaccinations add constraint fk_aefiinvestigation_vaccinations_aefiinvestigation_id foreign key (adverseeventsfollowingimmunizationinvestigation_id) references adverseeventsfollowingimmunizationinvestigation; +alter table adverseeventsfollowingimmunizationinvestigation_vaccinations add constraint fk_aefiinvestigation_vaccinations_vaccination_id foreign key (vaccination_id) references vaccination; + +-- AEFI investigation vaccination history +CREATE TABLE aefiinvestigation_vaccinations_history (LIKE adverseeventsfollowingimmunizationinvestigation_vaccinations); +DROP TRIGGER IF EXISTS versioning_trigger ON adverseeventsfollowingimmunizationinvestigation_vaccinations; +CREATE TRIGGER versioning_trigger + BEFORE INSERT OR UPDATE ON adverseeventsfollowingimmunizationinvestigation_vaccinations + FOR EACH ROW EXECUTE PROCEDURE versioning('sys_period', 'aefiinvestigation_vaccinations_history', true); +DROP TRIGGER IF EXISTS delete_history_trigger ON adverseeventsfollowingimmunizationinvestigation_vaccinations; +CREATE TRIGGER delete_history_trigger + AFTER DELETE ON adverseeventsfollowingimmunizationinvestigation_vaccinations + FOR EACH ROW EXECUTE PROCEDURE delete_history_trigger('aefiinvestigation_vaccinations_history', 'id'); +ALTER TABLE aefiinvestigation_vaccinations_history OWNER TO sormas_user; + +-- Assign AEFI user rights to default admin and national_user user roles +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); +INSERT INTO userroles_userrights (userrole_id, userright) SELECT id, 'ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT' FROM public.userroles WHERE userroles.linkeddefaultuserrole in ('ADMIN','NATIONAL_USER'); + +INSERT INTO schema_version (version_number, comment) VALUES (552, 'Adverse Events Following Immunization (AEFI) - Entities #12634'); -- *** Insert new sql commands BEFORE this line. Remember to always consider _history tables. *** diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/common/HistoryTablesTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/common/HistoryTablesTest.java index ecaba04ef41..725e266a9ed 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/common/HistoryTablesTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/common/HistoryTablesTest.java @@ -27,13 +27,19 @@ import org.testcontainers.containers.JdbcDatabaseContainer; import org.testcontainers.containers.wait.strategy.LogMessageWaitStrategy; import org.testcontainers.images.builder.ImageFromDockerfile; +import org.testcontainers.shaded.org.apache.commons.lang3.StringUtils; import de.hilling.junit.cdi.CdiTestJunitExtension; import de.hilling.junit.cdi.annotations.BypassTestInterceptor; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; @ExtendWith(CdiTestJunitExtension.class) public class HistoryTablesTest { + private static final List NO_HISTORY_REQUIRED_TABLES = + Arrays.asList(Aefi.AEFI_VACCINATIONS_TABLE_NAME, AefiInvestigation.AEFI_INVESTIGATION_VACCINATIONS_TABLE_NAME); + /** * Test that the *_history tables have the same columns as the corresponding production tables * @@ -71,6 +77,7 @@ public void testHistoryTablesMatch() throws IOException, URISyntaxException { Files.readAllBytes(Paths.get(Objects.requireNonNull(getClass().getClassLoader().getResource("checkHistoryTables.sql")).toURI()))); @SuppressWarnings("unchecked") List results = (List) em.createNativeQuery(checkHistoryTablesSql).getResultList(); + results.removeIf(o -> StringUtils.containsAny(o[1].toString(), NO_HISTORY_REQUIRED_TABLES.toArray(String[]::new))); StringBuilder result = new StringBuilder(); results.forEach(objects -> { result.append("\n"); diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java index afcc4faf9bf..e6b42ec468c 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java @@ -23,6 +23,9 @@ import com.tngtech.archunit.core.importer.ClassFileImporter; import de.symeda.sormas.api.importexport.DatabaseTable; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; +import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; import de.symeda.sormas.backend.environment.Environment; import de.symeda.sormas.backend.environment.environmentsample.EnvironmentSample; import de.symeda.sormas.backend.immunization.entity.DirectoryImmunization; @@ -64,7 +67,10 @@ public void testGetConfigFullyDefined() { FirstVaccinationDate.class, Environment.class, EnvironmentSample.class, - SelfReport.class); + SelfReport.class, + Aefi.class, + AdverseEvents.class, + AefiInvestigation.class); @Test public void test_all_entities_have_export_configuration() { @@ -107,6 +113,10 @@ public void test_all_entities_have_export_configuration() { // remove not exported entities from the list of missing ones NOT_EXPORTED_ENTITIES.forEach(e -> missingEntities.remove(e.getSimpleName())); + //remove aefi join tables + missingJoinTables.remove(Aefi.AEFI_VACCINATIONS_TABLE_NAME); + missingJoinTables.remove(AefiInvestigation.AEFI_INVESTIGATION_VACCINATIONS_TABLE_NAME); + assertThat("Missing export configuration for entities [" + String.join(", ", missingEntities) + "]", missingEntities, hasSize(0)); assertThat( "Export configuration not wanted for entities [" + String.join(", ", exportedButNotWantedEntity) + "]", diff --git a/sormas-backend/src/test/resources/checkHistoryTables.sql b/sormas-backend/src/test/resources/checkHistoryTables.sql index 7d0a5860b3a..9f8056274ef 100644 --- a/sormas-backend/src/test/resources/checkHistoryTables.sql +++ b/sormas-backend/src/test/resources/checkHistoryTables.sql @@ -23,11 +23,11 @@ AND c_hist.column_name IS NULL AND c.table_name NOT IN (SELECT t.table_name FROM information_schema."tables" t WHERE t.table_schema = 'public' AND t.table_name NOT LIKE '%_history' - AND (SELECT COUNT(t_hist.table_name) FROM information_schema."tables" t_hist WHERE concat(t.table_name,'_history') = t_hist .table_name) = 0) + AND (SELECT COUNT(t_hist.table_name) FROM information_schema."tables" t_hist WHERE concat(t.table_name,'_history') = t_hist.table_name) = 0) UNION SELECT 'no history table' as remark, t.table_name, null as column_name, null as data_type FROM information_schema."tables" t WHERE t.table_schema = 'public' AND t.table_name NOT LIKE '%_history' AND t.table_name NOT IN ('schema_version', 'systemevent') AND t.table_name NOT like 'pg_%' -AND (SELECT COUNT(t_hist.table_name) FROM information_schema."tables" t_hist WHERE concat(t.table_name,'_history') = t_hist .table_name) = 0 +AND (SELECT COUNT(t_hist.table_name) FROM information_schema."tables" t_hist WHERE concat(t.table_name,'_history') = t_hist.table_name) = 0 UNION SELECT 'missing delete history trigger' as remark, t.table_name, null as column_name, null as data_type FROM information_schema."tables" t WHERE t.table_schema = 'public' diff --git a/sormas-rest/src/main/webapp/WEB-INF/glassfish-web.xml b/sormas-rest/src/main/webapp/WEB-INF/glassfish-web.xml index c78da1af03b..d9f6a362c55 100644 --- a/sormas-rest/src/main/webapp/WEB-INF/glassfish-web.xml +++ b/sormas-rest/src/main/webapp/WEB-INF/glassfish-web.xml @@ -115,6 +115,36 @@ IMMUNIZATION_ARCHIVE + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + + PERSON_VIEW PERSON_VIEW @@ -590,6 +620,11 @@ DASHBOARD_SAMPLES_VIEW + + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + CASE_CLINICIAN_VIEW CASE_CLINICIAN_VIEW diff --git a/sormas-rest/src/main/webapp/WEB-INF/web.xml b/sormas-rest/src/main/webapp/WEB-INF/web.xml index 3decccae6a9..dc2a67e6b83 100644 --- a/sormas-rest/src/main/webapp/WEB-INF/web.xml +++ b/sormas-rest/src/main/webapp/WEB-INF/web.xml @@ -100,6 +100,30 @@ IMMUNIZATION_ARCHIVE + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + + PERSON_VIEW @@ -480,6 +504,10 @@ DASHBOARD_SAMPLES_VIEW + + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + CASE_CLINICIAN_VIEW diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java index 4499d4a7278..66cff5e28c8 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AbstractAefiView.java @@ -24,6 +24,8 @@ public class AbstractAefiView extends AbstractSubNavigationView { + public static final String ROOT_VIEW_NAME = "adverseevents"; + protected AbstractAefiView(String viewName) { super(viewName); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java index ebc4814cfd8..5dccd5e14cb 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationView.java @@ -57,7 +57,7 @@ public class AefiInvestigationView extends AbstractAefiView { - public static final String VIEW_NAME = "adverseeventinvestigations"; + public static final String VIEW_NAME = ROOT_VIEW_NAME + "/investigations"; private final AefiInvestigationCriteria criteria; @@ -111,7 +111,7 @@ public AefiInvestigationView() { gridLayout.setMargin(false); gridLayout.setSpacing(false); gridLayout.setSizeFull(); - CssStyles.style(gridLayout, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + //CssStyles.style(gridLayout, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); gridLayout.addComponent(createStatusFilterBar()); gridLayout.addComponent(dataLayout); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java index ef58c7bc968..c68ab2828be 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiView.java @@ -61,7 +61,7 @@ public class AefiView extends AbstractAefiView { - public static final String VIEW_NAME = "adverseevents"; + public static final String VIEW_NAME = ROOT_VIEW_NAME; private final AefiCriteria criteria; @@ -125,7 +125,7 @@ public AefiView() { gridLayout.setMargin(false); gridLayout.setSpacing(false); gridLayout.setSizeFull(); - CssStyles.style(gridLayout, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); + //CssStyles.style(gridLayout, CssStyles.VIEW_SECTION, CssStyles.VSPACE_TOP_3); gridLayout.addComponent(createStatusFilterBar()); gridLayout.addComponent(dataLayout); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java index 7a78c37b890..b063ad4f542 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefiinvestigationlink/AefiInvestigationList.java @@ -18,8 +18,6 @@ import java.util.List; import java.util.function.Consumer; -import org.apache.commons.collections.CollectionUtils; - import com.vaadin.ui.Button; import com.vaadin.ui.Label; @@ -59,7 +57,7 @@ public void reload() { FacadeProvider.getAefiInvestigationFacade().getEntriesList(listCriteria, 0, maxDisplayedEntries * 20); setEntries(listEntries); - if (CollectionUtils.isNotEmpty(listEntries)) { + if (!listEntries.isEmpty()) { showPage(1); } else { listLayout.removeAllComponents(); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java index 1c88db1072b..d4c8a1dd8ef 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/aefilink/AefiList.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -21,8 +18,6 @@ import java.util.List; import java.util.function.Consumer; -import org.apache.commons.collections.CollectionUtils; - import com.vaadin.ui.Button; import com.vaadin.ui.Label; @@ -58,10 +53,10 @@ public AefiList(AefiListCriteria aefiListCriteria, Consumer actionCall @Override public void reload() { - List aefiListEntries = FacadeProvider.getAefiFacade().getEntriesList(aefiListCriteria, 0, maxDisplayedEntries * 20); + List listEntries = FacadeProvider.getAefiFacade().getEntriesList(aefiListCriteria, 0, maxDisplayedEntries * 20); - setEntries(aefiListEntries); - if (CollectionUtils.isNotEmpty(aefiListEntries)) { + setEntries(listEntries); + if (!listEntries.isEmpty()) { showPage(1); } else { listLayout.removeAllComponents(); diff --git a/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss b/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss index a84710ea9e3..aaa7a9eea7e 100644 --- a/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss +++ b/sormas-ui/src/main/webapp/VAADIN/themes/sormas/global.scss @@ -410,6 +410,14 @@ border-bottom-right-radius: 4px; } + .form-section-accordion-panel-title-button{ + font-size: 16px !important; + font-weight: 600; + color: #6591C4; + text-decoration: none !important; + padding-left: 0; + } + .spacing-small { .v-spacing { width: 4px; diff --git a/sormas-ui/src/main/webapp/WEB-INF/glassfish-web.xml b/sormas-ui/src/main/webapp/WEB-INF/glassfish-web.xml index 99b99fa4b15..cc6f9e6ccf8 100644 --- a/sormas-ui/src/main/webapp/WEB-INF/glassfish-web.xml +++ b/sormas-ui/src/main/webapp/WEB-INF/glassfish-web.xml @@ -115,6 +115,36 @@ IMMUNIZATION_ARCHIVE + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + + PERSON_VIEW PERSON_VIEW @@ -590,6 +620,11 @@ DASHBOARD_SAMPLES_VIEW + + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + CASE_CLINICIAN_VIEW CASE_CLINICIAN_VIEW diff --git a/sormas-ui/src/main/webapp/WEB-INF/web.xml b/sormas-ui/src/main/webapp/WEB-INF/web.xml index 998d7088943..b62eb6a327a 100644 --- a/sormas-ui/src/main/webapp/WEB-INF/web.xml +++ b/sormas-ui/src/main/webapp/WEB-INF/web.xml @@ -105,6 +105,30 @@ IMMUNIZATION_ARCHIVE + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_CREATE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EDIT + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_ARCHIVE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_DELETE + + + + ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_EXPORT + + PERSON_VIEW @@ -485,6 +509,10 @@ DASHBOARD_SAMPLES_VIEW + + DASHBOARD_ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_VIEW + + CASE_CLINICIAN_VIEW From c7d9561b943c3c033e5c5d733f4b1e2917349360 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Tue, 29 Oct 2024 12:20:22 +0200 Subject: [PATCH 16/56] #13160 Add "Disease" Attribute to Document Templates for Filtering --- .../docgeneneration/DocumentTemplateDto.java | 17 +++ .../DocumentTemplateFacade.java | 6 +- .../docgeneneration/EventDocumentFacade.java | 5 +- .../QuarantineOrderFacade.java | 4 +- .../externalemail/ExternalEmailFacade.java | 2 +- .../ExternalEmailOptionsDto.java | 3 +- ...xternalEmailOptionsWithAttachmentsDto.java | 15 +-- .../de/symeda/sormas/api/i18n/Strings.java | 1 + .../src/main/resources/captions.properties | 2 + .../src/main/resources/strings.properties | 1 + .../common/StartupShutdownService.java | 25 ++++ .../docgeneration/DocumentTemplate.java | 4 +- .../DocumentTemplateFacadeEjb.java | 110 +++++++++--------- .../DocumentTemplateService.java | 101 +++++++++++++--- .../docgeneration/EventDocumentFacadeEjb.java | 11 +- .../QuarantineOrderFacadeEjb.java | 12 +- .../backend/docgeneration/TemplateEngine.java | 13 ++- .../externalemail/ExternalEmailFacadeEjb.java | 4 +- .../main/resources/META-INF/persistence.xml | 1 + .../src/main/resources/sql/sormas_schema.sql | 30 +++++ .../sormas/backend/AbstractBeanTest.java | 5 + .../AbstractDocGenerationTest.java | 18 +++ .../DocumentTemplateFacadeEjbTest.java | 95 ++++++++++----- .../EventDocumentFacadeEjbTest.java | 13 ++- .../QuarantineOrderFacadeEjbTest.java | 77 ++++++------ .../ExternalEmailFacadeEjbTest.java | 57 +++++---- .../test/resources/META-INF/persistence.xml | 1 + .../java/de/symeda/sormas/ui/MainScreen.java | 2 +- .../DocumentTemplateSection.java | 10 +- .../DocumentTemplateUploadLayout.java | 29 ++++- .../docgeneration/DocumentTemplatesGrid.java | 7 +- .../docgeneration/DocumentTemplatesView.java | 10 +- .../emailtemplate/EmailTemplatesView.java | 8 +- .../sormas/ui/contact/ContactDataView.java | 8 +- .../AbstractDocgenerationLayout.java | 67 +++++++---- .../DocGenerationController.java | 46 ++++---- .../ui/docgeneration/EventDocumentLayout.java | 22 ++-- .../EventDocumentsComponent.java | 6 +- .../QuarantineOrderDocumentsComponent.java | 28 ++++- .../docgeneration/QuarantineOrderLayout.java | 19 +-- .../email/ExternalBulkEmailOptionsForm.java | 3 +- .../ui/email/ExternalEmailOptionsForm.java | 91 ++++++++------- .../sormas/ui/events/EventDataView.java | 2 +- .../ui/events/EventParticipantDataView.java | 1 + .../ui/importer/DocumentTemplateReceiver.java | 14 ++- .../test/resources/META-INF/persistence.xml | 1 + 46 files changed, 678 insertions(+), 329 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java index cc6d26988d8..9f2e999960a 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateDto.java @@ -21,6 +21,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.EntityDto; import de.symeda.sormas.api.i18n.Validations; +import de.symeda.sormas.api.utils.DataHelper; import de.symeda.sormas.api.utils.FieldConstraints; public class DocumentTemplateDto extends EntityDto { @@ -37,6 +38,22 @@ public class DocumentTemplateDto extends EntityDto { @Size(max = FieldConstraints.CHARACTER_LIMIT_BIG, message = Validations.textTooLong) private String fileName; + public static DocumentTemplateDto build(DocumentWorkflow documentWorkflow, String fileName) { + DocumentTemplateDto dto = new DocumentTemplateDto(); + dto.setUuid(DataHelper.createUuid()); + dto.setWorkflow(documentWorkflow); + dto.setFileName(fileName); + + return dto; + } + + public static DocumentTemplateDto build(DocumentWorkflow documentWorkflow, String fileName, Disease disease) { + DocumentTemplateDto dto = build(documentWorkflow, fileName); + dto.setDisease(disease); + + return dto; + } + public DocumentWorkflow getWorkflow() { return workflow; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java index 7af68b34888..fff4aca3ad9 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/DocumentTemplateFacade.java @@ -24,11 +24,11 @@ String generateDocumentTxtFromEntities( List getAvailableTemplates(DocumentWorkflow documentWorkflow, Disease disease); - DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; + boolean isExistingTemplateFile(DocumentWorkflow documentWorkflow, Disease disease, String templateName); - boolean isExistingTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease); + DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; - void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease, byte[] document) + DocumentTemplateDto saveDocumentTemplate(DocumentTemplateDto template, byte[] document) throws DocumentTemplateException; boolean deleteDocumentTemplate(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java index 4418aa69b45..13971a70757 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/EventDocumentFacade.java @@ -21,6 +21,7 @@ import javax.ejb.Remote; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.event.EventReferenceDto; @@ -41,7 +42,7 @@ Map getGeneratedDocuments( Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; - List getAvailableTemplates(); + List getAvailableTemplates(Disease disease); - DocumentVariables getDocumentVariables(String templateName) throws DocumentTemplateException; + DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java index 0e24f8fb88b..66ee00927d3 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/docgeneneration/QuarantineOrderFacade.java @@ -57,7 +57,7 @@ Map getGeneratedDocumentsForEventParticipants( Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException; - List getAvailableTemplates(DocumentWorkflow workflow); + List getAvailableTemplates(DocumentWorkflow workflow, Disease disease); - DocumentVariables getDocumentVariables(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException; + DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java index 725235c028f..c45aa09a2c4 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailFacade.java @@ -34,7 +34,7 @@ @Remote public interface ExternalEmailFacade { - List getTemplateNames(DocumentWorkflow documentWorkflow); + List getTemplates(DocumentWorkflow documentWorkflow); List getAttachableDocuments(DocumentWorkflow documentWorkflow, String relatedEntityUuid); diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java index de1179f2569..48599792a4a 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsDto.java @@ -37,7 +37,7 @@ public class ExternalEmailOptionsDto implements Serializable { public static final String I18N_PREFIX = "ExternalEmailOptions"; - public static final String TEMPLATE_NAME = "templateName"; + public static final String TEMPLATE = "template"; public static final String RECIPIENT_EMAIL = "recipientEmail"; public static final String ATTACHED_DOCUMENTS = "attachedDocuments"; @@ -48,7 +48,6 @@ public class ExternalEmailOptionsDto implements Serializable { @NotNull(message = Validations.requiredField) private ReferenceDto rootEntityReference; @NotNull(message = Validations.requiredField) - @Size(min = 1, message = Validations.requiredField) private DocumentTemplateReferenceDto template; @NotNull(message = Validations.requiredField) @Size(min = 1, message = Validations.requiredField) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsWithAttachmentsDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsWithAttachmentsDto.java index 366a1e86b8a..1e2e3cee2eb 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsWithAttachmentsDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalemail/ExternalEmailOptionsWithAttachmentsDto.java @@ -21,14 +21,12 @@ package de.symeda.sormas.api.externalemail; import java.io.Serializable; -import java.util.Properties; import java.util.Set; -import javax.validation.Valid; import javax.validation.constraints.NotNull; -import javax.validation.constraints.Size; import de.symeda.sormas.api.audit.AuditedClass; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EmailAttachementDto; import de.symeda.sormas.api.docgeneneration.QuarantineOrderDocumentOptionsDto; @@ -51,8 +49,7 @@ public class ExternalEmailOptionsWithAttachmentsDto implements Serializable { @NotNull(message = Validations.requiredField) private final RootEntityType rootEntityType; @NotNull(message = Validations.requiredField) - @Size(min = 1, message = Validations.requiredField) - private String templateName; + private DocumentTemplateReferenceDto template; private Set attachedDocuments; private QuarantineOrderDocumentOptionsDto quarantineOrderDocumentOptionsDto; @@ -70,12 +67,12 @@ public RootEntityType getRootEntityType() { return rootEntityType; } - public String getTemplateName() { - return templateName; + public DocumentTemplateReferenceDto getTemplate() { + return template; } - public void setTemplateName(String templateName) { - this.templateName = templateName; + public void setTemplate(DocumentTemplateReferenceDto template) { + this.template = template; } public Set getAttachedDocuments() { diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index ed87fae2ee3..dec6f7891cb 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -315,6 +315,7 @@ public interface Strings { String errorDeletingDocumentTemplate = "errorDeletingDocumentTemplate"; String errorDocumentGeneration = "errorDocumentGeneration"; String errorDocumentGenerationMultipleDiseasses = "errorDocumentGenerationMultipleDiseasses"; + String errorDocumentTemplateWorkflowChangeNotAllowed = "errorDocumentTemplateWorkflowChangeNotAllowed"; String errorEntityNotEditable = "errorEntityNotEditable"; String errorEntityOutdated = "errorEntityOutdated"; String errorEnvironmentSampleNoDispatchRight = "errorEnvironmentSampleNoDispatchRight"; diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index a2fdb57909f..5cf062ece88 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -1109,6 +1109,8 @@ documentNoDocuments=There are no documents for this %s bulkActionCreatDocuments=Create quarantine order documents # DocumentTemplate DocumentTemplate=Document Template +DocumentTemplate.fileName=File name +DocumentTemplate.disease=Disease DocumentTemplate.buttonUploadTemplate=Upload Template DocumentTemplate.documentTemplateGuide=Document Template Guide DocumentTemplate.plural=Document Templates diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index fc1854df695..5bf15e53dc2 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -427,6 +427,7 @@ errorEnvironmentSampleNoReceivalRight = You do not have the necessary user right errorSendingExternalEmail = Email could not be sent. Please contact an admin and notify them about this problem. errorExternalEmailAttachmentCannotEncrypt=Can't send email with attachments. The person has no national health id or primary phone number to send the password to or the SMS service is not set up in the system. errorExternalEmailMissingPersonEmailAddress=This person does not have an email address +errorDocumentTemplateWorkflowChangeNotAllowed=The workflow of this document template cannot be changed. # headings headingAccessDenied = Access denied diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java index d81b66d14cb..b7f4794b7f9 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java @@ -14,6 +14,7 @@ */ package de.symeda.sormas.backend.common; +import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.UncheckedIOException; @@ -26,6 +27,7 @@ import java.util.Date; import java.util.HashSet; import java.util.List; +import java.util.Map; import java.util.NoSuchElementException; import java.util.Objects; import java.util.Optional; @@ -64,6 +66,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.Language; import de.symeda.sormas.api.customizableenum.CustomizableEnumType; +import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.externaljournal.PatientDiaryConfig; import de.symeda.sormas.api.externaljournal.SymptomJournalConfig; import de.symeda.sormas.api.externaljournal.UserConfig; @@ -88,6 +91,8 @@ import de.symeda.sormas.backend.deletionconfiguration.DeletionConfigurationService; import de.symeda.sormas.backend.disease.DiseaseConfiguration; import de.symeda.sormas.backend.disease.DiseaseConfigurationService; +import de.symeda.sormas.backend.docgeneration.DocumentTemplate; +import de.symeda.sormas.backend.docgeneration.DocumentTemplateService; import de.symeda.sormas.backend.feature.FeatureConfigurationFacadeEjb; import de.symeda.sormas.backend.feature.FeatureConfigurationService; import de.symeda.sormas.backend.importexport.ImportFacadeEjb.ImportFacadeEjbLocal; @@ -186,6 +191,8 @@ public class StartupShutdownService { private CustomizableEnumFacadeEjb.CustomizableEnumFacadeEjbLocal customizableEnumFacade; @EJB private CustomizableEnumValueService customizableEnumValueService; + @EJB + private DocumentTemplateService documentTemplateService; @Inject private Event passwordResetEvent; @@ -868,6 +875,9 @@ private void upgrade() { case 516: fillDefaultUserRole(DefaultUserRole.ENVIRONMENTAL_SURVEILLANCE_USER); break; + case 552: + createEntitiesForDocumentTemplates(); + break; default: throw new NoSuchElementException(DataHelper.toStringNullable(versionNeedingUpgrade)); } @@ -1019,6 +1029,21 @@ private void createMissingDiseaseConfigurations() { }); } + private void createEntitiesForDocumentTemplates() { + Map> templateFiles = documentTemplateService.getAllTemplateFiles(); + + templateFiles.forEach((workflow, files) -> { + for (File file : files) { + DocumentTemplate documentTemplate = new DocumentTemplate(); + documentTemplate.setUuid(DataHelper.createUuid()); + documentTemplate.setWorkflow(workflow); + documentTemplate.setFileName(file.getName()); + + documentTemplateService.ensurePersisted(documentTemplate); + } + }); + } + @PreDestroy public void shutdown() { auditLogger.logApplicationStop(); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java index 8aa96440e36..1babf8c57ac 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplate.java @@ -25,9 +25,11 @@ import de.symeda.sormas.api.utils.FieldConstraints; import de.symeda.sormas.backend.common.AbstractDomainObject; -@Entity +@Entity(name = "documenttemplates") public class DocumentTemplate extends AbstractDomainObject { + private static final long serialVersionUID = -8191658284208086022L; + public static final String WORKFLOW = "workflow"; public static final String DISEASE = "disease"; public static final String DOCUMENT_PATH = "documentPath"; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java index 4cc5bb6df32..8b592fb5d46 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java @@ -17,12 +17,8 @@ import java.io.ByteArrayInputStream; import java.io.File; -import java.io.FileOutputStream; import java.io.IOException; import java.nio.charset.StandardCharsets; -import java.nio.file.Files; -import java.nio.file.Path; -import java.nio.file.Paths; import java.util.List; import java.util.Properties; import java.util.Set; @@ -35,6 +31,7 @@ import javax.ejb.EJB; import javax.ejb.LocalBean; import javax.ejb.Stateless; +import javax.persistence.criteria.Predicate; import org.apache.commons.io.FileUtils; import org.apache.commons.io.FilenameUtils; @@ -74,6 +71,7 @@ import de.symeda.sormas.api.uuid.HasUuid; import de.symeda.sormas.backend.caze.CaseFacadeEjb.CaseFacadeEjbLocal; import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; +import de.symeda.sormas.backend.common.CriteriaBuilderHelper; import de.symeda.sormas.backend.contact.ContactFacadeEjb.ContactFacadeEjbLocal; import de.symeda.sormas.backend.event.EventFacadeEjb.EventFacadeEjbLocal; import de.symeda.sormas.backend.event.EventParticipantFacadeEjb.EventParticipantFacadeEjbLocal; @@ -163,7 +161,7 @@ public byte[] generateDocumentDocxFromEntities( } // 1. Read template from custom directory - File templateFile = getTemplateFile(template); + File templateFile = documentTemplateService.getTemplateFile(template); // 2. Extract document variables DocumentVariables documentVariables = getTemplateVariablesDocx(templateFile); @@ -192,7 +190,7 @@ public String generateDocumentTxtFromEntities( } // 1. Read template from custom directory - File templateFile = getTemplateFile(template); + File templateFile = documentTemplateService.getTemplateFile(template); // 2. Extract document variables DocumentVariables documentVariables = getTemplateVariablesTxt(templateFile); @@ -295,10 +293,15 @@ private String generateDocumentTxt(File templateFile, Properties properties) { @Override @PermitAll public List getAvailableTemplates(DocumentWorkflow documentWorkflow, @Nullable Disease disease) { - List templates = documentTemplateService.getByPredicate( - (cb, root, cq) -> cb.and( - cb.equal(root.get(DocumentTemplate.WORKFLOW), documentWorkflow), - cb.or(cb.isNull(root.get(DocumentTemplate.DISEASE)), cb.equal(root.get(DocumentTemplate.DISEASE), disease)))); + List templates = documentTemplateService.getByPredicate((cb, root, cq) -> { + Predicate diseasePredicate = null; + + if (disease != null) { + diseasePredicate = cb.or(cb.isNull(root.get(DocumentTemplate.DISEASE)), cb.equal(root.get(DocumentTemplate.DISEASE), disease)); + } + + return CriteriaBuilderHelper.and(cb, cb.equal(root.get(DocumentTemplate.WORKFLOW), documentWorkflow), diseasePredicate); + }); return templates.stream().map(DocumentTemplateFacadeEjb::toDto).collect(Collectors.toList()); } @@ -307,11 +310,11 @@ public List getAvailableTemplates(DocumentWorkflow document @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public boolean isExistingTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease) { + public boolean isExistingTemplateFile(DocumentWorkflow documentWorkflow, Disease disease, String templateName) { assertRequredUserRight(documentWorkflow); - File templateFile = new File(getWorkflowTemplateDirPath(documentWorkflow, disease).resolve(templateName).toUri()); - return templateFile.exists(); + return documentTemplateService.existsFile(documentWorkflow, disease, templateName); + } @Override @@ -320,7 +323,7 @@ public DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templ DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); DocumentWorkflow documentWorkflow = template.getWorkflow(); - File templateFile = getTemplateFile(template); + File templateFile = documentTemplateService.getTemplateFile(template); DocumentVariables documentVariables = documentWorkflow.isDocx() ? getTemplateVariablesDocx(templateFile) : getTemplateVariablesTxt(templateFile); Set propertyKeys = documentVariables.getVariables(); @@ -336,17 +339,18 @@ public DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templ @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String templateName, Disease disease, byte[] document) - throws DocumentTemplateException { + public DocumentTemplateDto saveDocumentTemplate(DocumentTemplateDto template, byte[] document) throws DocumentTemplateException { + DocumentWorkflow documentWorkflow = template.getWorkflow(); assertRequredUserRight(documentWorkflow); - if (!documentWorkflow.getFileExtension().equalsIgnoreCase(FilenameUtils.getExtension(templateName))) { - throw new DocumentTemplateException(I18nProperties.getString(Strings.headingWrongFileType)); + String fileName = template.getFileName(); + if (!documentWorkflow.getFileExtension().equalsIgnoreCase(FilenameUtils.getExtension(fileName))) { + throw new ValidationRuntimeException(I18nProperties.getString(Strings.headingWrongFileType)); } - String path = FilenameUtils.getPath(templateName); + String path = FilenameUtils.getPath(fileName); if (StringUtils.isNotBlank(path)) { - throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorIllegalFilename), templateName)); + throw new ValidationRuntimeException(String.format(I18nProperties.getString(Strings.errorIllegalFilename), fileName)); } ByteArrayInputStream templateInputStream = new ByteArrayInputStream(document); @@ -360,30 +364,32 @@ public void writeDocumentTemplate(DocumentWorkflow documentWorkflow, String temp validateEmailTemplate(document); } - Path workflowTemplateDirPath = getWorkflowTemplateDirPath(documentWorkflow, disease); - try { - Files.createDirectories(workflowTemplateDirPath); - } catch (IOException e) { - throw new DocumentTemplateException(I18nProperties.getString(Strings.errorCreatingTemplateDirectory)); - } - try (FileOutputStream fileOutputStream = - new FileOutputStream(new File(workflowTemplateDirPath.resolve(FilenameUtils.getName(templateName)).toUri()))) { - fileOutputStream.write(document); - } catch (IOException e) { - throw new DocumentTemplateException(I18nProperties.getString(Strings.errorWritingTemplate)); + DocumentTemplate existingTemplate = documentTemplateService.getByReferenceDto(template.toReference()); + if (existingTemplate != null) { + if (existingTemplate.getWorkflow() != template.getWorkflow()) { + throw new ValidationRuntimeException(I18nProperties.getString(Strings.errorDocumentTemplateWorkflowChangeNotAllowed)); + } + if (existingTemplate.getDisease() != template.getDisease() || existingTemplate.getFileName() != template.getFileName()) { + documentTemplateService.deleteTemplateFile(existingTemplate); + } } + + DocumentTemplate documentTemplate = fillOrBuildEntity(template, existingTemplate); + documentTemplateService.ensurePersisted(documentTemplate, document); + + return toDto(documentTemplate); } @Override @RightsAllowed({ UserRight._DOCUMENT_TEMPLATE_MANAGEMENT, UserRight._EMAIL_TEMPLATE_MANAGEMENT }) - public boolean deleteDocumentTemplate(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + public boolean deleteDocumentTemplate(DocumentTemplateReferenceDto templateReference) { DocumentTemplate template = documentTemplateService.getByReferenceDto(templateReference); assertRequredUserRight(template.getWorkflow()); - documentTemplateService.deletePermanent(template); + return documentTemplateService.deletePermanent(template); } @Override @@ -395,22 +401,12 @@ public byte[] getDocumentTemplateContent(DocumentTemplateReferenceDto templateRe assertRequredUserRight(template.getWorkflow()); try { - return FileUtils.readFileToByteArray(getTemplateFile(template)); + return FileUtils.readFileToByteArray(documentTemplateService.getTemplateFile(template)); } catch (IOException e) { throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorReadingTemplate), template.getFileName())); } } - private File getTemplateFile(DocumentTemplate template) throws DocumentTemplateException { - File templateFile = - new File(getWorkflowTemplateDirPath(template.getWorkflow(), template.getDisease()).resolve(template.getFileName()).toString()); - - if (!templateFile.exists()) { - throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), template.getFileName())); - } - return templateFile; - } - private DocumentVariables getTemplateVariablesDocx(File templateFile) throws DocumentTemplateException { return templateEngine.extractTemplateVariablesDocx(templateFile); } @@ -433,16 +429,6 @@ private String getVariableBaseName(String propertyKey) { return matcher.matches() ? matcher.group(1) : ""; } - private Path getWorkflowTemplateDirPath(DocumentWorkflow documentWorkflow, Disease disease) { - Path path = Paths.get(configFacade.getCustomFilesPath()).resolve("docgeneration").resolve(documentWorkflow.getTemplateDirectory()); - - if (disease != null) { - path = path.resolve(disease.name()); - } - - return path; - } - private EntityDtoAccessHelper.IReferenceDtoResolver getReferenceDtoResolver() { EntityDtoAccessHelper.IReferenceDtoResolver referenceDtoResolver = referenceDto -> { if (referenceDto != null) { @@ -513,6 +499,16 @@ public static DocumentTemplateDto toDto(DocumentTemplate source) { return target; } + private DocumentTemplate fillOrBuildEntity(DocumentTemplateDto source, DocumentTemplate target) { + target = DtoHelper.fillOrBuildEntity(source, target, DocumentTemplate::new, true); + + target.setWorkflow(source.getWorkflow()); + target.setDisease(source.getDisease()); + target.setFileName(source.getFileName()); + + return target; + } + public static EmailTemplateTexts splitTemplateContent(String content) { return splitTemplateContent(content, true); } @@ -530,6 +526,10 @@ private static EmailTemplateTexts splitTemplateContent(String templateString, bo return new EmailTemplateTexts(cleanupSubject ? subjectLine.substring(1).trim() : subjectLine, content); } + public DocumentTemplateDto getByUuid(String uuid) { + return toDto(documentTemplateService.getByUuid(uuid)); + } + public static final class EmailTemplateTexts { private final String subject; @@ -549,7 +549,7 @@ public String getContent() { } } - public class ObjectFormatter { + public static class ObjectFormatter { public Object format(Object value) { return EntityDtoAccessHelper.formatObject(value); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java index 14bd08f89dc..7aa7cba0c79 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateService.java @@ -16,14 +16,26 @@ package de.symeda.sormas.backend.docgeneration; import java.io.File; +import java.io.FileOutputStream; +import java.io.IOException; +import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.stream.Collectors; +import java.util.stream.Stream; import javax.ejb.EJB; import javax.ejb.LocalBean; import javax.ejb.Stateless; +import org.apache.commons.io.FilenameUtils; + +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.backend.common.BaseAdoService; @@ -36,31 +48,94 @@ public class DocumentTemplateService extends BaseAdoService { @EJB private ConfigFacadeEjb.ConfigFacadeEjbLocal configFacade; - protected DocumentTemplateService() { + public DocumentTemplateService() { super(DocumentTemplate.class); } @Override - public boolean deletePermanent(DocumentTemplate documentTemplate) throws DocumentTemplateException { + public boolean deletePermanent(DocumentTemplate documentTemplate) { + boolean fileDeleted = deleteTemplateFile(documentTemplate); + if (fileDeleted) { + super.deletePermanent(documentTemplate); + } + + return fileDeleted; + } + + public boolean deleteTemplateFile(DocumentTemplate documentTemplate) { String fileName = documentTemplate.getFileName(); - File templateFile = new File(getWorkflowTemplateDirPath(documentTemplate).resolve(fileName).toUri()); - if (templateFile.exists() && templateFile.isFile()) { - return templateFile.delete(); - } else { - throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), fileName)); + File templateFile = new File(getTemplateDirPath(documentTemplate).resolve(fileName).toUri()); + + boolean deleted = false; + boolean exists = templateFile.exists(); + if (!exists) { + return true; + } + + if (templateFile.isFile()) { + deleted = templateFile.delete(); } - super.deletePermanent(documentTemplate); + + return deleted; + } + + private Path getTemplateDirPath(DocumentTemplate documentTemplate) { + return getTemplateDirPath(documentTemplate.getWorkflow(), documentTemplate.getDisease()); } - private Path getWorkflowTemplateDirPath(DocumentTemplate documentTemplate) { - Path path = - Paths.get(configFacade.getCustomFilesPath()).resolve("docgeneration").resolve(documentTemplate.getWorkflow().getTemplateDirectory()); + private Path getTemplateDirPath(DocumentWorkflow documentWorkflow, Disease disease) { + Path path = Paths.get(configFacade.getCustomFilesPath()).resolve("docgeneration").resolve(documentWorkflow.getTemplateDirectory()); - if (documentTemplate.getDisease() != null) { - path = path.resolve(documentTemplate.getDisease().name()); + if (disease != null) { + path = path.resolve(disease.name()); } return path; } + public boolean existsFile(DocumentWorkflow documentWorkflow, Disease disease, String templateName) { + File templateFile = new File(this.getTemplateDirPath(documentWorkflow, disease).resolve(templateName).toUri()); + return templateFile.exists(); + } + + public void ensurePersisted(DocumentTemplate documentTemplate, byte[] document) throws DocumentTemplateException { + Path workflowTemplateDirPath = getTemplateDirPath(documentTemplate.getWorkflow(), documentTemplate.getDisease()); + try { + Files.createDirectories(workflowTemplateDirPath); + } catch (IOException e) { + throw new DocumentTemplateException(I18nProperties.getString(Strings.errorCreatingTemplateDirectory)); + } + + try (FileOutputStream fileOutputStream = + new FileOutputStream(new File(workflowTemplateDirPath.resolve(FilenameUtils.getName(documentTemplate.getFileName())).toUri()))) { + fileOutputStream.write(document); + ensurePersisted(documentTemplate); + } catch (IOException e) { + throw new DocumentTemplateException(I18nProperties.getString(Strings.errorWritingTemplate)); + } + } + + public File getTemplateFile(DocumentTemplate template) throws DocumentTemplateException { + File templateFile = new File(getTemplateDirPath(template).resolve(template.getFileName()).toString()); + + if (!templateFile.exists()) { + throw new DocumentTemplateException(String.format(I18nProperties.getString(Strings.errorFileNotFound), template.getFileName())); + } + return templateFile; + } + + public Map> getAllTemplateFiles() { + Map> templateFiles = new HashMap<>(); + + for (DocumentWorkflow workflow : DocumentWorkflow.values()) { + Path templateDirPath = getTemplateDirPath(workflow, null); + File templateDir = templateDirPath.toFile(); + + if (templateDir.exists() && templateDir.isDirectory()) { + templateFiles.put(workflow, Stream.of(templateDir.listFiles()).filter(File::isFile).collect(Collectors.toList())); + } + } + + return templateFiles; + } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java index b6cfc009cde..d30496f7bef 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjb.java @@ -13,6 +13,7 @@ import org.apache.commons.io.IOUtils; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.action.ActionCriteria; import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; @@ -72,7 +73,7 @@ public String getGeneratedDocument( byte[] documentToSave = styledHtml.getBytes(StandardCharsets.UTF_8);//mandatory UTF_8 try { helper.saveDocument( - helper.getDocumentFileName(eventReference, templateReference.getCaption()), + helper.getDocumentFileName(eventReference, templateReference), DocumentDto.MIME_TYPE_DEFAULT, documentToSave.length, helper.getDocumentRelatedEntityType(eventReference), @@ -103,13 +104,13 @@ public Map getGeneratedDocuments( } @Override - public List getAvailableTemplates() { - return documentTemplateFacade.getAvailableTemplates(DOCUMENT_WORKFLOW, null); + public List getAvailableTemplates(Disease disease) { + return documentTemplateFacade.getAvailableTemplates(DOCUMENT_WORKFLOW, disease); } @Override - public DocumentVariables getDocumentVariables(String templateName) throws DocumentTemplateException { - return documentTemplateFacade.getDocumentVariables(DOCUMENT_WORKFLOW, templateName); + public DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + return documentTemplateFacade.getDocumentVariables(templateReference); } private String createStyledHtml(String title, String body) throws DocumentTemplateException { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java index 05cae3d6464..c92a09cb6e7 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjb.java @@ -112,8 +112,10 @@ public Map getGeneratedDocuments( Boolean shouldUploadGeneratedDoc) throws DocumentTemplateException { + DocumentTemplateDto template = documentTemplateFacade.getByUuid(templateReference.getUuid()); + Map quarantineOrderEntities = - entitiesBuilder.getQuarantineOrderEntities(workflow, rootEntityReferences); + entitiesBuilder.getQuarantineOrderEntities(template.getWorkflow(), rootEntityReferences); return getGeneratedDocuments(templateReference, quarantineOrderEntities, extraProperties, shouldUploadGeneratedDoc); } @@ -154,13 +156,13 @@ private Map getGeneratedDocuments( } @Override - public List getAvailableTemplates(DocumentWorkflow workflow) { - return documentTemplateFacade.getAvailableTemplates(workflow, null); + public List getAvailableTemplates(DocumentWorkflow workflow, Disease disease) { + return documentTemplateFacade.getAvailableTemplates(workflow, disease); } @Override - public DocumentVariables getDocumentVariables(DocumentWorkflow documentWorkflow, String templateName) throws DocumentTemplateException { - return documentTemplateFacade.getDocumentVariables(documentWorkflow, templateName); + public DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + return documentTemplateFacade.getDocumentVariables(templateReference); } @LocalBean diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/TemplateEngine.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/TemplateEngine.java index 3f7da3c0d99..f0b40263530 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/TemplateEngine.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/TemplateEngine.java @@ -55,6 +55,7 @@ import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.utils.HtmlHelper; +import de.symeda.sormas.api.utils.ValidationRuntimeException; import fr.opensagres.xdocreport.core.XDocReportException; import fr.opensagres.xdocreport.document.IXDocReport; import fr.opensagres.xdocreport.document.registry.XDocReportRegistry; @@ -181,13 +182,15 @@ public String generateDocumentTxt(Properties properties, File templateFile) { return Jsoup.clean(stringWriter.toString(), "", HTML_TEMPLATE_WHITELIST, outputSettings); } - public void validateTemplateDocx(InputStream templateInputStream) throws DocumentTemplateException { + public void validateTemplateDocx(InputStream templateInputStream) { try { IXDocReport report = readXDocReport(templateInputStream); FieldsExtractor extractor = FieldsExtractor.create(); report.extractFields(extractor); } catch (XDocReportException | IOException e) { - throw new DocumentTemplateException(I18nProperties.getString(Strings.errorProcessingTemplate)); + throw new ValidationRuntimeException(I18nProperties.getString(Strings.errorProcessingTemplate)); + } catch (DocumentTemplateException e) { + throw new ValidationRuntimeException(e.getMessage()); } } @@ -214,11 +217,11 @@ protected IXDocReport readXDocReport(InputStream templateInputStream) throws Doc } } - public void validateTemplateTxt(InputStream templateInputStream) throws DocumentTemplateException { + public void validateTemplateTxt(InputStream templateInputStream) { getFieldExtractorTxt(new InputStreamReader(templateInputStream), "validate"); } - private FieldsExtractor getFieldExtractorTxt(Reader templateFileReader, String templateName) throws DocumentTemplateException { + private FieldsExtractor getFieldExtractorTxt(Reader templateFileReader, String templateName) { FieldsExtractor extractor = FieldsExtractor.create(); ExtractVariablesVelocityVisitor visitor = new ExtractVariablesVelocityVisitor(extractor); try { @@ -228,7 +231,7 @@ private FieldsExtractor getFieldExtractorTxt(Reader templateFile document.jjtAccept(visitor, null); return extractor; } catch (ParseException e) { - throw new DocumentTemplateException(I18nProperties.getString(Strings.errorProcessingTemplate)); + throw new ValidationRuntimeException(I18nProperties.getString(Strings.errorProcessingTemplate)); } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java index f9681ee9ccb..aac294570ad 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjb.java @@ -158,7 +158,7 @@ public class ExternalEmailFacadeEjb implements ExternalEmailFacade { private DocGenerationHelper docGenerationHelper; @Override - public List getTemplateNames(DocumentWorkflow documentWorkflow) { + public List getTemplates(DocumentWorkflow documentWorkflow) { return documentTemplateFacade.getAvailableTemplates(documentWorkflow, null); } @@ -340,7 +340,7 @@ public List sendBulkEmail(@Valid ExternalEmailOptionsWithAttach try { ExternalEmailOptionsDto emailOptions = new ExternalEmailOptionsDto(options.getDocumentWorkflow(), options.getRootEntityType(), entityRef); - emailOptions.setTemplate(options.getTemplateName()); + emailOptions.setTemplate(options.getTemplate()); emailOptions.setAttachedDocuments(attachedDocuments); emailOptions.setRootEntityReference(entityRef); diff --git a/sormas-backend/src/main/resources/META-INF/persistence.xml b/sormas-backend/src/main/resources/META-INF/persistence.xml index b792ff39147..9b0aa30b9dd 100644 --- a/sormas-backend/src/main/resources/META-INF/persistence.xml +++ b/sormas-backend/src/main/resources/META-INF/persistence.xml @@ -88,6 +88,7 @@ de.symeda.sormas.backend.externalmessage.labmessage.SampleReport de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess de.symeda.sormas.backend.selfreport.SelfReport + de.symeda.sormas.backend.docgeneration.DocumentTemplate true diff --git a/sormas-backend/src/main/resources/sql/sormas_schema.sql b/sormas-backend/src/main/resources/sql/sormas_schema.sql index f08ecaec095..914026fd5fc 100644 --- a/sormas-backend/src/main/resources/sql/sormas_schema.sql +++ b/sormas-backend/src/main/resources/sql/sormas_schema.sql @@ -13257,4 +13257,34 @@ ALTER TABLE externalmessage_history ADD COLUMN personadditionaldetails text; INSERT INTO schema_version (version_number, comment) VALUES (551, '#13147 Phone Number Validation for E-Santé Reports – Remove and Store Non-Numeric Text'); + + +-- 2024-10-23 Add "Disease" Attribute to Document Templates for Filtering #13160 +CREATE TABLE documenttemplates ( + id bigint not null, + uuid varchar(36) not null unique, + changedate timestamp not null, + creationdate timestamp not null, + change_user_id bigint, + + workflow varchar(255) not null, + disease varchar(255), + fileName text, + + sys_period tstzrange not null, + primary key(id) +); + +ALTER TABLE documenttemplates OWNER TO sormas_user; +ALTER TABLE documenttemplates ADD CONSTRAINT fk_change_user_id FOREIGN KEY (change_user_id) REFERENCES users (id); + +CREATE TABLE documenttemplates_history (LIKE documenttemplates); +CREATE TRIGGER versioning_trigger BEFORE INSERT OR UPDATE ON documenttemplates + FOR EACH ROW EXECUTE PROCEDURE versioning('sys_period', 'documenttemplates_history', true); +CREATE TRIGGER delete_history_trigger + AFTER DELETE ON documenttemplates + FOR EACH ROW EXECUTE PROCEDURE delete_history_trigger('documenttemplates_history', 'id'); +ALTER TABLE documenttemplates_history OWNER TO sormas_user; + +INSERT INTO schema_version (version_number, comment, upgradeneeded) VALUES (552, 'Add "Disease" Attribute to Document Templates for Filtering #13160', true); -- *** Insert new sql commands BEFORE this line. Remember to always consider _history tables. *** diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/AbstractBeanTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/AbstractBeanTest.java index a8dcfb20fbf..4a940ef86a5 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/AbstractBeanTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/AbstractBeanTest.java @@ -155,6 +155,7 @@ import de.symeda.sormas.backend.disease.DiseaseConfigurationFacadeEjb.DiseaseConfigurationFacadeEjbLocal; import de.symeda.sormas.backend.disease.DiseaseConfigurationService; import de.symeda.sormas.backend.docgeneration.DocumentTemplateFacadeEjb.DocumentTemplateFacadeEjbLocal; +import de.symeda.sormas.backend.docgeneration.DocumentTemplateService; import de.symeda.sormas.backend.docgeneration.EventDocumentFacadeEjb; import de.symeda.sormas.backend.docgeneration.QuarantineOrderFacadeEjb; import de.symeda.sormas.backend.document.DocumentFacadeEjb; @@ -899,6 +900,10 @@ public DocumentTemplateFacade getDocumentTemplateFacade() { return getBean(DocumentTemplateFacadeEjbLocal.class); } + public DocumentTemplateService getDocumentTemplateService() { + return getBean(DocumentTemplateService.class); + } + public QuarantineOrderFacade getQuarantineOrderFacade() { return getBean(QuarantineOrderFacadeEjb.QuarantineOrderFacadeEjbLocal.class); } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/AbstractDocGenerationTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/AbstractDocGenerationTest.java index 913be35319d..187a888afb2 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/AbstractDocGenerationTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/AbstractDocGenerationTest.java @@ -6,6 +6,9 @@ import org.junit.jupiter.api.AfterEach; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; +import de.symeda.sormas.api.utils.DataHelper; import de.symeda.sormas.backend.AbstractBeanTest; import de.symeda.sormas.backend.MockProducer; import de.symeda.sormas.backend.common.ConfigFacadeEjb; @@ -36,4 +39,19 @@ protected void resetDefaultNullReplacement() { public void teardown() throws URISyntaxException { reset(); } + + protected DocumentTemplate createDocumentTemplate(DocumentWorkflow workflow, String templateFileName) { + DocumentTemplate template = new DocumentTemplate(); + template.setUuid(DataHelper.createUuid()); + template.setFileName(templateFileName); + template.setWorkflow(workflow); + + getDocumentTemplateService().persist(template); + + return template; + } + + protected static DocumentTemplateReferenceDto toReference(DocumentTemplate documentTemplate) { + return new DocumentTemplateReferenceDto(documentTemplate.getUuid(), documentTemplate.getFileName()); + } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java index 7f0bdccdcf3..69cec0fe7ce 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjbTest.java @@ -27,14 +27,18 @@ import java.io.IOException; import java.net.URISyntaxException; import java.nio.charset.StandardCharsets; +import java.util.List; import org.apache.commons.io.FileUtils; import org.apache.commons.io.IOUtils; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentTemplateFacade; +import de.symeda.sormas.api.utils.DataHelper; import de.symeda.sormas.api.utils.ValidationRuntimeException; import de.symeda.sormas.backend.MockProducer; import de.symeda.sormas.backend.common.ConfigFacadeEjb; @@ -54,46 +58,74 @@ public void writeAndDeleteTemplateTest() throws IOException, URISyntaxException, String testDirectory = "target" + File.separator + "doctest"; byte[] document = IOUtils.toByteArray(getClass().getResourceAsStream("/docgeneration/quarantine/Quarantine.docx")); MockProducer.getProperties().setProperty(ConfigFacadeEjb.CUSTOM_FILES_PATH, testDirectory); - documentTemplateFacade.writeDocumentTemplate(QUARANTINE_ORDER_CASE, "TemplateFileToBeDeleted.docx", document); - assertTrue(documentTemplateFacade.getAvailableTemplates(QUARANTINE_ORDER_CASE).contains("TemplateFileToBeDeleted.docx")); - assertTrue(documentTemplateFacade.deleteDocumentTemplate(QUARANTINE_ORDER_CASE, "TemplateFileToBeDeleted.docx")); - assertFalse(documentTemplateFacade.getAvailableTemplates(QUARANTINE_ORDER_CASE).contains("TemplateFileToBeDeleted.docx")); + + DocumentTemplateDto templateDto = DocumentTemplateDto.build(QUARANTINE_ORDER_CASE, "TemplateFileToBeDeleted.docx"); + documentTemplateFacade.saveDocumentTemplate(templateDto, document); + + assertTrue(documentTemplateFacade.getAvailableTemplates(QUARANTINE_ORDER_CASE, null).contains(templateDto)); + + DocumentTemplateDto templateForCovid = DocumentTemplateDto.build(QUARANTINE_ORDER_CASE, "TemplateFileToBeDeletedCovid.docx"); + templateForCovid.setDisease(Disease.CORONAVIRUS); + documentTemplateFacade.saveDocumentTemplate(templateForCovid, document); + + List templatesForCovid = documentTemplateFacade.getAvailableTemplates(QUARANTINE_ORDER_CASE, Disease.CORONAVIRUS); + assertTrue(templatesForCovid.contains(templateDto)); + assertTrue(templatesForCovid.contains(templateForCovid)); + + List templatesForLassa = documentTemplateFacade.getAvailableTemplates(QUARANTINE_ORDER_CASE, Disease.LASSA); + assertTrue(templatesForLassa.contains(templateDto)); + assertFalse(templatesForLassa.contains(templateForCovid)); + + assertTrue(documentTemplateFacade.deleteDocumentTemplate(templateDto.toReference())); + assertFalse(documentTemplateFacade.getAvailableTemplates(QUARANTINE_ORDER_CASE, null).contains(templateDto)); + FileUtils.deleteDirectory(new File(testDirectory)); resetCustomPath(); } @Test - public void isExistingTemplateTest() { - assertTrue(documentTemplateFacade.isExistingTemplate(QUARANTINE_ORDER_CASE, "Quarantine.docx")); - assertFalse(documentTemplateFacade.isExistingTemplate(QUARANTINE_ORDER_CASE, "ThisTemplateDoesNotExist.docx")); + public void isExistingTemplateFileTest() { + assertTrue(documentTemplateFacade.isExistingTemplateFile(QUARANTINE_ORDER_CASE, null, "Quarantine.docx")); + assertFalse(documentTemplateFacade.isExistingTemplateFile(QUARANTINE_ORDER_CASE, null, "ThisTemplateDoesNotExist.docx")); } @Test public void validateTemplateTest() throws IOException { try { - documentTemplateFacade.writeDocumentTemplate(QUARANTINE_ORDER_CASE, "TemplateFileToBeValidated.txt", new byte[0]); + documentTemplateFacade + .saveDocumentTemplate(DocumentTemplateDto.build(QUARANTINE_ORDER_CASE, "TemplateFileToBeValidated.txt"), new byte[0]); fail("Invalid file extension not recognized."); - } catch (DocumentTemplateException e) { + } catch (ValidationRuntimeException e) { assertEquals("Wrong file type", e.getMessage()); + } catch (DocumentTemplateException e) { + fail("Invalid file extension not recognized.", e); } try { - documentTemplateFacade.writeDocumentTemplate(QUARANTINE_ORDER_CASE, "../TemplateFileToBeValidated.docx", new byte[0]); + documentTemplateFacade + .saveDocumentTemplate(DocumentTemplateDto.build(QUARANTINE_ORDER_CASE, "../TemplateFileToBeValidated.docx"), new byte[0]); fail("Invalid file extension not recognized."); - } catch (DocumentTemplateException e) { + } catch (ValidationRuntimeException e) { assertEquals("Illegal file name: ../TemplateFileToBeValidated.docx", e.getMessage()); + } catch (DocumentTemplateException e) { + fail("Invalid file extension not recognized.", e); } try { - documentTemplateFacade.writeDocumentTemplate(QUARANTINE_ORDER_CASE, "TemplateFileToBeValidated.docx", new byte[0]); + documentTemplateFacade + .saveDocumentTemplate(DocumentTemplateDto.build(QUARANTINE_ORDER_CASE, "TemplateFileToBeValidated.docx"), new byte[0]); fail("Invalid docx file not recognized."); - } catch (DocumentTemplateException e) { + } catch (ValidationRuntimeException e) { assertEquals("The template file is corrupt.", e.getMessage()); + } catch (DocumentTemplateException e) { + fail("Invalid docx file not recognized", e); } try { byte[] document = IOUtils.toByteArray(getClass().getResourceAsStream("/docgeneration/quarantine/FaultyTemplate.docx")); - documentTemplateFacade.writeDocumentTemplate(QUARANTINE_ORDER_CASE, "TemplateFileToBeValidated.docx", document); + documentTemplateFacade.saveDocumentTemplate(DocumentTemplateDto.build(QUARANTINE_ORDER_CASE, "TemplateFileToBeValidated.docx"), document); fail("Syntax error not recognized."); - } catch (DocumentTemplateException e) { + } catch (ValidationRuntimeException e) { assertEquals("Error processing template.", e.getMessage()); + } catch (DocumentTemplateException e) { + fail("Syntax error not recognized.", e); } } @@ -101,38 +133,41 @@ public void validateTemplateTest() throws IOException { public void testEmailTemplateValidation() throws DocumentTemplateException { assertThrows( ValidationRuntimeException.class, - () -> documentTemplateFacade - .writeDocumentTemplate(CASE_EMAIL, "CaseEmailTemplate.txt", "Email template without subject".getBytes(StandardCharsets.UTF_8))); + () -> documentTemplateFacade.saveDocumentTemplate( + DocumentTemplateDto.build(CASE_EMAIL, "CaseEmailTemplate.txt"), + "Email template without subject".getBytes(StandardCharsets.UTF_8))); assertThrows( ValidationRuntimeException.class, - () -> documentTemplateFacade.writeDocumentTemplate( - CASE_EMAIL, - "CaseEmailTemplate.txt", + () -> documentTemplateFacade.saveDocumentTemplate( + DocumentTemplateDto.build(CASE_EMAIL, "CaseEmailTemplate.txt"), "Email template without subject\nSecond line".getBytes(StandardCharsets.UTF_8))); assertThrows( ValidationRuntimeException.class, - () -> documentTemplateFacade.writeDocumentTemplate( - CASE_EMAIL, - "CaseEmailTemplate.txt", + () -> documentTemplateFacade.saveDocumentTemplate( + DocumentTemplateDto.build(CASE_EMAIL, "CaseEmailTemplate.txt"), "#\nEmail template without subject\nSecond line".getBytes(StandardCharsets.UTF_8))); assertThrows( ValidationRuntimeException.class, - () -> documentTemplateFacade.writeDocumentTemplate( - CASE_EMAIL, - "CaseEmailTemplate.txt", + () -> documentTemplateFacade.saveDocumentTemplate( + DocumentTemplateDto.build(CASE_EMAIL, "CaseEmailTemplate.txt"), "# \nEmail template without subject\nSecond line".getBytes(StandardCharsets.UTF_8))); assertThrows( ValidationRuntimeException.class, - () -> documentTemplateFacade.writeDocumentTemplate( - CASE_EMAIL, - "CaseEmailTemplate.txt", + () -> documentTemplateFacade.saveDocumentTemplate( + DocumentTemplateDto.build(CASE_EMAIL, "CaseEmailTemplate.txt"), "*Subject\nEmail template without subject\nSecond line".getBytes(StandardCharsets.UTF_8))); } @Test public void readTemplateTest() throws DocumentTemplateException { - byte[] template = documentTemplateFacade.getDocumentTemplateContent(QUARANTINE_ORDER_CASE, "Quarantine.docx"); + DocumentTemplate templateEntity = new DocumentTemplate(); + templateEntity.setUuid(DataHelper.createUuid()); + templateEntity.setWorkflow(QUARANTINE_ORDER_CASE); + templateEntity.setFileName("Quarantine.docx"); + + getDocumentTemplateService().ensurePersisted(templateEntity); + byte[] template = documentTemplateFacade.getDocumentTemplateContent(toReference(templateEntity)); assertEquals(12731, template.length); } } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java index 2cb9c486600..222f107e9a6 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/EventDocumentFacadeEjbTest.java @@ -9,6 +9,8 @@ import java.io.StringWriter; import java.net.URISyntaxException; import java.text.ParseException; +import java.util.HashMap; +import java.util.Map; import java.util.Properties; import org.apache.commons.io.FilenameUtils; @@ -20,6 +22,7 @@ import de.symeda.sormas.api.action.ActionContext; import de.symeda.sormas.api.action.ActionDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EventDocumentFacade; import de.symeda.sormas.api.event.EventDto; import de.symeda.sormas.api.event.EventInvestigationStatus; @@ -129,11 +132,19 @@ public void generateEventHandoutTest() throws IOException, URISyntaxException, D File[] testCasesHtml = testCasesDir.listFiles((d, name) -> name.endsWith(".html")); assertNotNull(testCasesHtml); + Map documentTemplates = new HashMap<>(testCasesHtml.length); for (File testCaseHtml : testCasesHtml) { String testcaseBasename = FilenameUtils.getBaseName(testCaseHtml.getName()); + DocumentTemplate documentTemplate = createDocumentTemplate(DocumentWorkflow.EVENT_HANDOUT, testcaseBasename + ".html"); + documentTemplates.put(testcaseBasename, documentTemplate); + } + + for (File testCaseHtml : testCasesHtml) { + String testcaseBasename = FilenameUtils.getBaseName(testCaseHtml.getName()); + DocumentTemplate documentTemplate = documentTemplates.get(testcaseBasename); String htmlText = - eventDocumentFacade.getGeneratedDocument(, testcaseBasename + ".html", eventDto.toReference(), new Properties(), Boolean.FALSE); + eventDocumentFacade.getGeneratedDocument(toReference(documentTemplate), eventDto.toReference(), new Properties(), Boolean.FALSE); String actual = cleanLineSeparators( htmlText.replaceAll("

Event-ID: [A-Z0-9-]*

", "

Event-ID: STN3WX-5JTGYV-IU2LRM-4UHCSOEE

")); diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java index 9a062b80f32..a885d1ecf9c 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/docgeneration/QuarantineOrderFacadeEjbTest.java @@ -44,6 +44,7 @@ import de.symeda.sormas.api.caze.CaseReferenceDto; import de.symeda.sormas.api.contact.ContactDto; import de.symeda.sormas.api.contact.ContactReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; @@ -69,9 +70,7 @@ import de.symeda.sormas.api.travelentry.TravelEntryReferenceDto; import de.symeda.sormas.api.user.DefaultUserRole; import de.symeda.sormas.api.user.UserDto; -import de.symeda.sormas.backend.MockProducer; import de.symeda.sormas.backend.TestDataCreator; -import de.symeda.sormas.backend.common.ConfigFacadeEjb; public class QuarantineOrderFacadeEjbTest extends AbstractDocGenerationTest { @@ -85,6 +84,11 @@ public class QuarantineOrderFacadeEjbTest extends AbstractDocGenerationTest { private SampleDto sampleDto; private PathogenTestDto pathogenTestDto; + private DocumentTemplate caseDocumentTemplate; + private DocumentTemplate contactDocumentTemplate; + private DocumentTemplate eventParticipantDocumentTemplate; + private DocumentTemplate travelEntryDocumentTemplate; + @BeforeEach public void setup() throws URISyntaxException { TestDataCreator.RDCF rdcf = creator.createRDCF("Region", "District", "Community", "Facility", "PointOfEntry"); @@ -167,6 +171,12 @@ public void setup() throws URISyntaxException { travelEntryDto.setResponsibleRegion(rdcf.region); travelEntryDto.setResponsibleDistrict(rdcf.district); travelEntryDto = getTravelEntryFacade().save(travelEntryDto); + + caseDocumentTemplate = createDocumentTemplate(DocumentWorkflow.QUARANTINE_ORDER_CASE, "Quarantine.docx"); + createDocumentTemplate(DocumentWorkflow.QUARANTINE_ORDER_CASE, "FaultyTemplate.docx"); + contactDocumentTemplate = createDocumentTemplate(DocumentWorkflow.QUARANTINE_ORDER_CONTACT, "Quarantine.docx"); + eventParticipantDocumentTemplate = createDocumentTemplate(DocumentWorkflow.QUARANTINE_ORDER_EVENT_PARTICIPANT, "Quarantine.docx"); + travelEntryDocumentTemplate = createDocumentTemplate(DocumentWorkflow.QUARANTINE_ORDER_TRAVEL_ENTRY, "Quarantine.docx"); } private Date parseDate(String dateString) { @@ -182,7 +192,7 @@ public void generateQuarantineOrderCaseTest() throws IOException, DocumentTempla generateQuarantineOrderTest( RootEntityType.ROOT_CASE, caseDataDto.toReference(), - DocumentWorkflow.QUARANTINE_ORDER_CASE, + caseDocumentTemplate, sampleDto.toReference(), pathogenTestDto.toReference(), "QuarantineCase.cmp"); @@ -193,7 +203,7 @@ public void generateQuarantineOrderContactTest() throws IOException, DocumentTem generateQuarantineOrderTest( RootEntityType.ROOT_CONTACT, contactDto.toReference(), - DocumentWorkflow.QUARANTINE_ORDER_CONTACT, + contactDocumentTemplate, null, null, "QuarantineContact.cmp"); @@ -204,7 +214,7 @@ public void generateQuarantineOrderEventParticipantTest() throws IOException, Do generateQuarantineOrderTest( RootEntityType.ROOT_EVENT_PARTICIPANT, eventParticipantDto.toReference(), - DocumentWorkflow.QUARANTINE_ORDER_EVENT_PARTICIPANT, + eventParticipantDocumentTemplate, null, null, "QuarantineEvent.cmp"); @@ -215,7 +225,7 @@ public void generateQuarantineOrderTravelEntryTest() throws IOException, Documen generateQuarantineOrderTest( RootEntityType.ROOT_TRAVEL_ENTRY, travelEntryDto.toReference(), - DocumentWorkflow.QUARANTINE_ORDER_TRAVEL_ENTRY, + travelEntryDocumentTemplate, null, null, "QuarantineTravelEntry.cmp"); @@ -229,7 +239,7 @@ public void generateQuarantineOrderCustomNullReplacementTest() throws IOExceptio generateQuarantineOrderTest( RootEntityType.ROOT_CASE, rootEntityReference, - DocumentWorkflow.QUARANTINE_ORDER_CASE, + caseDocumentTemplate, null, null, "QuarantineCaseEmptyNullReplacement.cmp"); @@ -238,7 +248,7 @@ public void generateQuarantineOrderCustomNullReplacementTest() throws IOExceptio generateQuarantineOrderTest( RootEntityType.ROOT_CASE, rootEntityReference, - DocumentWorkflow.QUARANTINE_ORDER_CASE, + caseDocumentTemplate, null, null, "QuarantineCaseCustomNullReplacement.cmp"); @@ -252,11 +262,10 @@ public void testBulkCaseDocumentCreation() throws DocumentTemplateException, IOE properties.setProperty("extraremark1", "the first remark"); properties.setProperty("extra.remark.no3", "the third remark"); - DocumentWorkflow workflow = DocumentWorkflow.QUARANTINE_ORDER_CASE; Map documentContents = quarantineOrderFacadeEjb - .getGeneratedDocuments(, "Quarantine.docx", Collections.singletonList(rootEntityReference), properties, false); + .getGeneratedDocuments(toReference(caseDocumentTemplate), Collections.singletonList(rootEntityReference), properties, false); - verifyGeneratedDocument(rootEntityReference, workflow, "QuarantineCase.cmp", documentContents.get(rootEntityReference)); + verifyGeneratedDocument(rootEntityReference, caseDocumentTemplate, "QuarantineCase.cmp", documentContents.get(rootEntityReference)); } @Test @@ -266,11 +275,10 @@ public void testBulkContactDocumentCreation() throws DocumentTemplateException, properties.setProperty("extraremark1", "the first remark"); properties.setProperty("extra.remark.no3", "the third remark"); - DocumentWorkflow workflow = DocumentWorkflow.QUARANTINE_ORDER_CONTACT; Map documentContents = quarantineOrderFacadeEjb - .getGeneratedDocuments(, "Quarantine.docx", Collections.singletonList(rootEntityReference), properties, false); + .getGeneratedDocuments(toReference(contactDocumentTemplate), Collections.singletonList(rootEntityReference), properties, false); - verifyGeneratedDocument(rootEntityReference, workflow, "QuarantineContact.cmp", documentContents.get(rootEntityReference)); + verifyGeneratedDocument(rootEntityReference, contactDocumentTemplate, "QuarantineContact.cmp", documentContents.get(rootEntityReference)); } @Test @@ -281,21 +289,24 @@ public void testBulkEventParticipantDocumentCreation() throws DocumentTemplateEx properties.setProperty("extraremark1", "the first remark"); properties.setProperty("extra.remark.no3", "the third remark"); - DocumentWorkflow workflow = DocumentWorkflow.QUARANTINE_ORDER_EVENT_PARTICIPANT; - Map documentContents = quarantineOrderFacadeEjb.getGeneratedDocumentsForEventParticipants(, - "Quarantine.docx", + Map documentContents = quarantineOrderFacadeEjb.getGeneratedDocumentsForEventParticipants( + toReference(eventParticipantDocumentTemplate), Collections.singletonList(rootEntityReference), eventDto.getDisease(), properties, false); - verifyGeneratedDocument(rootEntityReference, workflow, "QuarantineEvent.cmp", documentContents.get(rootEntityReference)); + verifyGeneratedDocument( + rootEntityReference, + eventParticipantDocumentTemplate, + "QuarantineEvent.cmp", + documentContents.get(rootEntityReference)); } private void generateQuarantineOrderTest( RootEntityType rootEntityType, ReferenceDto rootEntityReference, - DocumentWorkflow documentWorkflow, + DocumentTemplate template, SampleReferenceDto sampleReference, PathogenTestReferenceDto pathogenTest, String comparisonFile) @@ -308,11 +319,11 @@ private void generateQuarantineOrderTest( verifyGeneratedDocument( rootEntityReference, - documentWorkflow, + template, comparisonFile, - quarantineOrderFacadeEjb.getGeneratedDocument(, - "Quarantine.docx", - rootEntityType, + quarantineOrderFacadeEjb.getGeneratedDocument( + toReference(template), + rootEntityType, rootEntityReference, sampleReference, pathogenTest, @@ -323,19 +334,19 @@ private void generateQuarantineOrderTest( private void verifyGeneratedDocument( ReferenceDto rootEntityReference, - DocumentWorkflow documentWorkflow, + DocumentTemplate docuembtTemplate, String comparisonFile, byte[] documentContent) throws IOException, DocumentTemplateException { - DocumentVariables documentVariables = quarantineOrderFacadeEjb.getDocumentVariables(documentWorkflow, "Quarantine.docx"); + DocumentVariables documentVariables = quarantineOrderFacadeEjb.getDocumentVariables(toReference(docuembtTemplate)); List additionalVariables = documentVariables.getAdditionalVariables(); String rootEntityName; List expectedVariables = Arrays.asList("extraremark1", "extra.remark2", "extra.remark.no3"); List expectedUsedEntities = new ArrayList<>(Arrays.asList("person", "user", "sample", "pathogenTest")); - switch (documentWorkflow) { + switch (docuembtTemplate.getWorkflow()) { case QUARANTINE_ORDER_CASE: rootEntityName = "case"; break; @@ -390,17 +401,11 @@ private void verifyGeneratedDocument( } @Test - public void getAvailableTemplatesTest() throws URISyntaxException { - List availableTemplates = quarantineOrderFacadeEjb.getAvailableTemplates(DocumentWorkflow.QUARANTINE_ORDER_CASE); + public void getAvailableTemplatesTest() { + List availableTemplates = quarantineOrderFacadeEjb.getAvailableTemplates(DocumentWorkflow.QUARANTINE_ORDER_CASE, null); assertEquals(2, availableTemplates.size()); - assertTrue(availableTemplates.contains("Quarantine.docx")); - assertTrue(availableTemplates.contains("FaultyTemplate.docx")); - - MockProducer.getProperties().setProperty(ConfigFacadeEjb.CUSTOM_FILES_PATH, "thisDirectoryDoesNotExist"); - - assertTrue(quarantineOrderFacadeEjb.getAvailableTemplates(DocumentWorkflow.QUARANTINE_ORDER_CASE).isEmpty()); - - resetCustomPath(); + assertTrue(availableTemplates.stream().anyMatch(dt -> dt.getFileName().equals("Quarantine.docx"))); + assertTrue(availableTemplates.stream().anyMatch(dt -> dt.getFileName().equals("FaultyTemplate.docx"))); } private DocumentWorkflow getDocumentWorkflow(ReferenceDto reference) { diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java index 95ed274840a..efe9c0f1122 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalemail/ExternalEmailFacadeEjbTest.java @@ -62,6 +62,7 @@ import de.symeda.sormas.api.caze.CaseReferenceDto; import de.symeda.sormas.api.contact.ContactDto; import de.symeda.sormas.api.contact.ContactStatus; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EmailAttachementDto; @@ -96,6 +97,7 @@ import de.symeda.sormas.backend.common.messaging.InvalidPhoneNumberException; import de.symeda.sormas.backend.common.messaging.SmsService; import de.symeda.sormas.backend.docgeneration.AbstractDocGenerationTest; +import de.symeda.sormas.backend.docgeneration.DocumentTemplate; import de.symeda.sormas.backend.document.Document; import de.symeda.sormas.backend.document.DocumentRelatedEntity; @@ -107,6 +109,10 @@ public class ExternalEmailFacadeEjbTest extends AbstractDocGenerationTest { private PersonDto personDto; private LocationDto locationDto2; + private DocumentTemplate caseEmailTemplate; + private DocumentTemplate quarantineOrderTemplate; + private DocumentTemplate contactEmailTemplate; + @Mock private EmailService emailService; @Mock @@ -154,6 +160,10 @@ public void init() { personDto.setEmailAddress("testEmail@email.com"); getPersonFacade().save(personDto); + + caseEmailTemplate = createDocumentTemplate(DocumentWorkflow.CASE_EMAIL, "CaseEmail.txt"); + quarantineOrderTemplate = createDocumentTemplate(DocumentWorkflow.QUARANTINE_ORDER_CASE, "Quarantine.docx"); + contactEmailTemplate = createDocumentTemplate(DocumentWorkflow.CONTACT_EMAIL, "ContactEmail.txt"); } @BeforeEach @@ -165,21 +175,28 @@ public void setup() throws URISyntaxException { public void testGetAvailableTemplates() throws DocumentTemplateException { loginWith(admin); - getDocumentTemplateFacade() - .writeDocumentTemplate(DocumentWorkflow.CASE_EMAIL, "CaseEmailMock.txt", "#Subject\nContent".getBytes(StandardCharsets.UTF_8)); - getDocumentTemplateFacade() - .writeDocumentTemplate(DocumentWorkflow.CONTACT_EMAIL, "ContactEmailMock.txt", "#Subject\nContent".getBytes(StandardCharsets.UTF_8)); + getDocumentTemplateFacade().saveDocumentTemplate( + DocumentTemplateDto.build(DocumentWorkflow.CASE_EMAIL, "CaseEmailMock.txt"), + "#Subject\nContent".getBytes(StandardCharsets.UTF_8)); + getDocumentTemplateFacade().saveDocumentTemplate( + DocumentTemplateDto.build(DocumentWorkflow.CONTACT_EMAIL, "ContactEmailMock.txt"), + "#Subject\nContent".getBytes(StandardCharsets.UTF_8)); loginWith(userDto); - List templateNames = getExternalEmailFacade().getTemplateNames(DocumentWorkflow.CASE_EMAIL); + List templates = getExternalEmailFacade().getTemplates(DocumentWorkflow.CASE_EMAIL); - assertThat(templateNames.size(), is(2)); + assertThat(templates.size(), is(2)); // should return predefined test template "CaseEmail.txt" and the one just created by the test - assertThat(templateNames, containsInAnyOrder("CaseEmail.txt", "CaseEmailMock.txt")); + assertThat( + templates.stream().map(DocumentTemplateDto::getFileName).collect(Collectors.toList()), + containsInAnyOrder("CaseEmail.txt", "CaseEmailMock.txt")); assertThat( - getExternalEmailFacade().getTemplateNames(DocumentWorkflow.CONTACT_EMAIL), + getExternalEmailFacade().getTemplates(DocumentWorkflow.CONTACT_EMAIL) + .stream() + .map(DocumentTemplateDto::getFileName) + .collect(Collectors.toList()), containsInAnyOrder("ContactEmail.txt", "ContactEmailMock.txt")); } @@ -297,7 +314,7 @@ public void testSendEmailToCasePerson() }).when(emailService).sendEmail(any(), any(), any(), any()); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplate("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); options.setRecipientEmail("test@mail.com"); getExternalEmailFacade().sendEmail(options); @@ -336,7 +353,7 @@ public void testSendEmailToContactPerson() ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CONTACT_EMAIL, RootEntityType.ROOT_CONTACT, contact.toReference()); - options.setTemplate("ContactEmail.txt"); + options.setTemplate(toReference(contactEmailTemplate)); options.setRecipientEmail("test@mail.com"); getExternalEmailFacade().sendEmail(options); @@ -397,7 +414,7 @@ public void testSendEmailWithAttachments() }).when(emailService).sendEmail(any(), any(), any(), any()); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplate("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(documents.stream().map(DocumentDto::toReference).collect(Collectors.toSet())); @@ -476,7 +493,7 @@ public void testEncryptAttachmentsWithRandomPassword() }).when(smsService).sendSms(any(), any()); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplate("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(documents.stream().map(DocumentDto::toReference).collect(Collectors.toSet())); @@ -497,7 +514,7 @@ public void testSendEmailWithUnsupportedAttachment() throws MessagingException, createDocument("SomeDocument.txt", DocumentRelatedEntityType.CASE, caze.getUuid(), "Some content".getBytes(StandardCharsets.UTF_8)); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplate("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(Collections.singleton(document.toReference())); @@ -518,7 +535,7 @@ public void testSendAttachmentWithUnavailablePassword() throws MessagingExceptio createDocument("SomeDocument.txt", DocumentRelatedEntityType.CASE, caze.getUuid(), "Some content".getBytes(StandardCharsets.UTF_8)); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplate("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(Collections.singleton(document.toReference())); @@ -541,7 +558,7 @@ public void testSendAttachmentNotRelatedToEntity() throws MessagingException, IO createDocument("SomeDocument.txt", DocumentRelatedEntityType.CONTACT, "mock-uuid", "Some content".getBytes(StandardCharsets.UTF_8)); ExternalEmailOptionsDto options = new ExternalEmailOptionsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE, caze.toReference()); - options.setTemplate("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); options.setRecipientEmail("test@mail.com"); options.setAttachedDocuments(Collections.singleton(document.toReference())); @@ -613,7 +630,7 @@ public void testSendBulkEmailToCasePerson() throws MessagingException, IOExcepti ExternalEmailOptionsWithAttachmentsDto options = new ExternalEmailOptionsWithAttachmentsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); DocumentDto documentDto = DocumentDto.build(); documentDto.setUploadingUser(admin.toReference()); @@ -702,10 +719,10 @@ public void testSendBulkEmailToCasePersonAttachmentsAndTemplateDocument() throws ExternalEmailOptionsWithAttachmentsDto options = new ExternalEmailOptionsWithAttachmentsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); QuarantineOrderDocumentOptionsDto quarantineOrderDocumentOptions = new QuarantineOrderDocumentOptionsDto(); - quarantineOrderDocumentOptions.setTemplate("Quarantine.docx"); + quarantineOrderDocumentOptions.setTemplate(toReference(quarantineOrderTemplate)); quarantineOrderDocumentOptions.setExtraProperties(new Properties()); quarantineOrderDocumentOptions.setShouldUploadGeneratedDoc(false); quarantineOrderDocumentOptions.setDocumentWorkflow(DocumentWorkflow.QUARANTINE_ORDER_CASE); @@ -799,9 +816,9 @@ public void testSendBulkEmailToCasePersonTemplateDocument() throws MessagingExce ExternalEmailOptionsWithAttachmentsDto options = new ExternalEmailOptionsWithAttachmentsDto(DocumentWorkflow.CASE_EMAIL, RootEntityType.ROOT_CASE); - options.setTemplateName("CaseEmail.txt"); + options.setTemplate(toReference(caseEmailTemplate)); QuarantineOrderDocumentOptionsDto quarantineOrderDocumentOptions = new QuarantineOrderDocumentOptionsDto(); - quarantineOrderDocumentOptions.setTemplate("Quarantine.docx"); + quarantineOrderDocumentOptions.setTemplate(toReference(quarantineOrderTemplate)); quarantineOrderDocumentOptions.setExtraProperties(new Properties()); quarantineOrderDocumentOptions.setShouldUploadGeneratedDoc(true); quarantineOrderDocumentOptions.setDocumentWorkflow(DocumentWorkflow.QUARANTINE_ORDER_CASE); diff --git a/sormas-backend/src/test/resources/META-INF/persistence.xml b/sormas-backend/src/test/resources/META-INF/persistence.xml index 3d878a502f2..5182f469a14 100644 --- a/sormas-backend/src/test/resources/META-INF/persistence.xml +++ b/sormas-backend/src/test/resources/META-INF/persistence.xml @@ -89,6 +89,7 @@ de.symeda.sormas.backend.externalmessage.labmessage.SampleReport de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess de.symeda.sormas.backend.selfreport.SelfReport + de.symeda.sormas.backend.docgeneration.DocumentTemplate true diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java index 890097e8b3b..1e3ddbddcc2 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java @@ -300,7 +300,7 @@ public View getView(String viewName) { } if (permitted(FeatureType.TRAVEL_ENTRIES, UserRight.TRAVEL_ENTRY_MANAGEMENT_ACCESS) - && FacadeProvider.getConfigFacade().isConfiguredCountry(CountryHelper.COUNTRY_CODE_GERMANY)) { + && FacadeProvider.getConfigFacade().isConfiguredCountry(CountryHelper.COUNTRY_CODE_LUXEMBOURG)) { ControllerProvider.getTravelEntryController().registerViews(navigator); menu.addView( TravelEntriesView.class, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateSection.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateSection.java index 4b6509ca069..f141b16e31c 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateSection.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateSection.java @@ -39,15 +39,15 @@ public class DocumentTemplateSection extends VerticalLayout { private static final long serialVersionUID = 379271838736314055L; - public DocumentTemplateSection(DocumentWorkflow documentWorkflow) { + public DocumentTemplateSection(DocumentWorkflow documentWorkflow, boolean hasDisease) { HorizontalLayout sectionHeader = new HorizontalLayout(); - DocumentTemplatesGrid documentTemplatesGrid = new DocumentTemplatesGrid(documentWorkflow); + DocumentTemplatesGrid documentTemplatesGrid = new DocumentTemplatesGrid(documentWorkflow, hasDisease); documentTemplatesGrid.setWidth(700, Unit.PIXELS); Label quarantineTemplatesLabel = new Label(documentWorkflow.toString()); quarantineTemplatesLabel.addStyleName(H3); - Button uploadButton = buildUploadButton(documentWorkflow, documentTemplatesGrid); + Button uploadButton = buildUploadButton(documentWorkflow, documentTemplatesGrid, hasDisease); sectionHeader.addComponents(quarantineTemplatesLabel, uploadButton); sectionHeader.setComponentAlignment(uploadButton, Alignment.MIDDLE_RIGHT); sectionHeader.setWidth(700, Unit.PIXELS); @@ -57,9 +57,9 @@ public DocumentTemplateSection(DocumentWorkflow documentWorkflow) { setExpandRatio(documentTemplatesGrid, 1F); } - private Button buildUploadButton(DocumentWorkflow documentWorkflow, DocumentTemplatesGrid documentTemplatesGrid) { + private Button buildUploadButton(DocumentWorkflow documentWorkflow, DocumentTemplatesGrid documentTemplatesGrid, boolean hasDisease) { return ButtonHelper.createIconButton(I18nProperties.getCaption(Captions.DocumentTemplate_uploadTemplate), VaadinIcons.UPLOAD, e -> { - Window window = VaadinUiUtil.showPopupWindow(new DocumentTemplateUploadLayout(documentWorkflow)); + Window window = VaadinUiUtil.showPopupWindow(new DocumentTemplateUploadLayout(documentWorkflow, hasDisease)); window.setCaption(String.format(I18nProperties.getCaption(Captions.DocumentTemplate_uploadWorkflowTemplate), documentWorkflow)); window.addCloseListener(c -> documentTemplatesGrid.reload()); }, ValoTheme.BUTTON_PRIMARY); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateUploadLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateUploadLayout.java index bf8a1c4fb78..05391e5c8ff 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateUploadLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplateUploadLayout.java @@ -16,15 +16,19 @@ package de.symeda.sormas.ui.configuration.docgeneration; import java.util.Map; +import java.util.function.Supplier; import com.vaadin.icons.VaadinIcons; import com.vaadin.server.ClassResource; import com.vaadin.server.FileDownloader; import com.vaadin.ui.Button; +import com.vaadin.ui.ComboBox; import com.vaadin.ui.VerticalLayout; import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.ui.Upload; +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -42,6 +46,7 @@ public class DocumentTemplateUploadLayout extends VerticalLayout { protected Upload upload; private ImportLayoutComponent importGuideComponent; private final DocumentWorkflow documentWorkflow; + private final boolean hasDisease; private static final Map templateInfoData = Map.ofEntries( Map.entry( @@ -79,9 +84,10 @@ public class DocumentTemplateUploadLayout extends VerticalLayout { DocumentTemplateInfoData .forEmailTemplate(Captions.DocumentTemplate_exampleTemplateTravelEntryEmail, "ExampleDocumentTemplateTravelEntryEmail.txt"))); - public DocumentTemplateUploadLayout(DocumentWorkflow documentWorkflow) { + public DocumentTemplateUploadLayout(DocumentWorkflow documentWorkflow, boolean hasDisease) { super(); this.documentWorkflow = documentWorkflow; + this.hasDisease = hasDisease; addDownloadResourcesComponent(); addUploadResourceComponent(); } @@ -116,13 +122,30 @@ private void addUploadResourceComponent() { ImportLayoutComponent uploadTemplateComponent = new ImportLayoutComponent(2, headline, infoText, null, null); addComponent(uploadTemplateComponent); - DocumentTemplateReceiver receiver = new DocumentTemplateReceiver(documentWorkflow); + VerticalLayout uploadLatout = new VerticalLayout(); + uploadLatout.setMargin(false); + uploadLatout.setSpacing(false); + uploadLatout.setSizeFull(); + addComponent(uploadLatout); + + Supplier diseaseSupplier = () -> null; + if (hasDisease) { + ComboBox diseaseComboBox = new ComboBox<>(); + diseaseComboBox.setCaption(I18nProperties.getCaption(Captions.disease)); + diseaseComboBox.setItems(FacadeProvider.getDiseaseConfigurationFacade().getAllDiseases(true, true, true)); + diseaseComboBox.setPlaceholder(I18nProperties.getString(Strings.all)); + diseaseComboBox.setEmptySelectionAllowed(true); + uploadLatout.addComponent(diseaseComboBox); + diseaseSupplier = diseaseComboBox::getValue; + } + + DocumentTemplateReceiver receiver = new DocumentTemplateReceiver(documentWorkflow, diseaseSupplier); upload = new Upload("", receiver); upload.setButtonCaption(I18nProperties.getCaption(Captions.DocumentTemplate_buttonUploadTemplate)); CssStyles.style(upload, CssStyles.VSPACE_2); upload.addStartedListener(receiver); upload.addSucceededListener(receiver); - addComponent(upload); + uploadLatout.addComponent(upload); } private void addDownloadResource(String caption, VaadinIcons icon, ClassResource resource) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java index ef5e3f6182b..706301cd92e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesGrid.java @@ -46,7 +46,7 @@ public class DocumentTemplatesGrid extends Grid { private final DocumentWorkflow documentWorkflow; - public DocumentTemplatesGrid(DocumentWorkflow documentWorkflow) { + public DocumentTemplatesGrid(DocumentWorkflow documentWorkflow, boolean hasDisease) { super(DocumentTemplateDto.class); this.documentWorkflow = documentWorkflow; setSizeFull(); @@ -55,7 +55,10 @@ public DocumentTemplatesGrid(DocumentWorkflow documentWorkflow) { ListDataProvider dataProvider = DataProvider.fromStream(availableTemplates.stream()); setDataProvider(dataProvider); - setColumns(DocumentTemplateDto.FILE_NAME, DocumentTemplateDto.DISEASE); + setColumns(DocumentTemplateDto.FILE_NAME); + if (hasDisease) { + addColumn(DocumentTemplateDto.DISEASE); + } for (Column column : getColumns()) { column.setCaption(I18nProperties.getPrefixCaption(DocumentTemplateDto.I18N_PREFIX, column.getId(), column.getCaption())); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesView.java index 3f5fcaf0220..fea59946f8b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/DocumentTemplatesView.java @@ -37,11 +37,11 @@ public DocumentTemplatesView() { super(VIEW_NAME); gridLayout = new VerticalLayout( - new DocumentTemplateSection(QUARANTINE_ORDER_CASE), - new DocumentTemplateSection(QUARANTINE_ORDER_CONTACT), - new DocumentTemplateSection(QUARANTINE_ORDER_EVENT_PARTICIPANT), - new DocumentTemplateSection(QUARANTINE_ORDER_TRAVEL_ENTRY), - new DocumentTemplateSection(EVENT_HANDOUT)); + new DocumentTemplateSection(QUARANTINE_ORDER_CASE, true), + new DocumentTemplateSection(QUARANTINE_ORDER_CONTACT, true), + new DocumentTemplateSection(QUARANTINE_ORDER_EVENT_PARTICIPANT, true), + new DocumentTemplateSection(QUARANTINE_ORDER_TRAVEL_ENTRY, true), + new DocumentTemplateSection(EVENT_HANDOUT, true)); gridLayout.setWidth(100, Unit.PERCENTAGE); gridLayout.setMargin(true); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/emailtemplate/EmailTemplatesView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/emailtemplate/EmailTemplatesView.java index 3ae757f1033..4e989fef819 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/emailtemplate/EmailTemplatesView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/configuration/docgeneration/emailtemplate/EmailTemplatesView.java @@ -37,10 +37,10 @@ public EmailTemplatesView() { super(VIEW_NAME); VerticalLayout gridLayout = new VerticalLayout( - new DocumentTemplateSection(CASE_EMAIL), - new DocumentTemplateSection(CONTACT_EMAIL), - new DocumentTemplateSection(EVENT_PARTICIPANT_EMAIL), - new DocumentTemplateSection(TRAVEL_ENTRY_EMAIL)); + new DocumentTemplateSection(CASE_EMAIL, false), + new DocumentTemplateSection(CONTACT_EMAIL, false), + new DocumentTemplateSection(EVENT_PARTICIPANT_EMAIL, false), + new DocumentTemplateSection(TRAVEL_ENTRY_EMAIL, false)); gridLayout.setWidth(100, Unit.PERCENTAGE); gridLayout.setMargin(true); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataView.java index f0745c2c774..a374a345851 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataView.java @@ -28,7 +28,6 @@ import de.symeda.sormas.api.EditPermissionType; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.caze.CaseDataDto; -import de.symeda.sormas.api.caze.CaseReferenceDto; import de.symeda.sormas.api.contact.ContactClassification; import de.symeda.sormas.api.contact.ContactDto; import de.symeda.sormas.api.contact.ContactLogic; @@ -293,7 +292,12 @@ protected void initView(String params) { layout.addSidePanelComponent(new SideComponentLayout(documentList), DOCUMENTS_LOC); } - QuarantineOrderDocumentsComponent.addComponentToLayout(layout, contactDto, documentList); + Disease disease = contactDto.getDisease(); + if (disease == null && caseDto != null) { + disease = caseDto.getDisease(); + } + + QuarantineOrderDocumentsComponent.addComponentToLayout(layout, contactDto, disease, documentList); if (UiUtil.permitted(FeatureType.EXTERNAL_EMAILS, UserRight.EXTERNAL_EMAIL_SEND)) { ExternalEmailSideComponent externalEmailSideComponent = diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java index 7ed3718a08b..742c36004ed 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/AbstractDocgenerationLayout.java @@ -40,8 +40,11 @@ import com.vaadin.ui.Window; import com.vaadin.ui.themes.ValoTheme; +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -63,16 +66,17 @@ public abstract class AbstractDocgenerationLayout extends VerticalLayout { public DocumentVariables documentVariables; public CheckBox checkBoxUploadGeneratedDoc; public HorizontalLayout buttonBar; + private Disease defaultDisease; - public boolean isCheckBoxAfterTemplateSelector; - - public AbstractDocgenerationLayout( + protected AbstractDocgenerationLayout( + Disease defaultDisease, String captionTemplateSelector, - Function fileNameFunction, + Function fileNameFunction, boolean isMultiFilesMode, - boolean isCheckBoxAfterTemplateSelector) { + boolean uploadCheckboxFirst) { + + this.defaultDisease = defaultDisease; - this.isCheckBoxAfterTemplateSelector = isCheckBoxAfterTemplateSelector; additionalVariablesComponent = new VerticalLayout(); additionalVariablesComponent.setSpacing(false); additionalVariablesComponent.setMargin(new MarginInfo(false, false, true, false)); @@ -84,11 +88,13 @@ public AbstractDocgenerationLayout( hideTextfields(); hideAdditionalParameters(); - if (isCheckBoxAfterTemplateSelector) { + if (uploadCheckboxFirst) { + addDiseaseSelector(defaultDisease); addTemplateSelector(captionTemplateSelector, fileNameFunction); addCheckboxUploadButton(isMultiFilesMode); } else { addCheckboxUploadButton(isMultiFilesMode); + addDiseaseSelector(defaultDisease); addTemplateSelector(captionTemplateSelector, fileNameFunction); } @@ -125,19 +131,40 @@ private void addCheckboxUploadButton(boolean isMultiFilesMode) { } } - private void addTemplateSelector(String captionTemplateSelector, Function fileNameFunction) { + private void addDiseaseSelector(Disease defaultDisease) { + ComboBox diseaseSelector = new ComboBox<>(); + diseaseSelector.setCaption(I18nProperties.getCaption(Captions.disease)); + diseaseSelector.setItems(FacadeProvider.getDiseaseConfigurationFacade().getAllDiseases(true, true, true)); + diseaseSelector.setPlaceholder(I18nProperties.getString(Strings.all)); + diseaseSelector.setEmptySelectionAllowed(true); + diseaseSelector.setWidth(100F, Unit.PERCENTAGE); + diseaseSelector.addStyleName(CssStyles.SOFT_REQUIRED); + diseaseSelector.setValue(defaultDisease); + + diseaseSelector.addValueChangeListener(e -> { + Disease disease = e.getValue(); + templateSelector.setValue(null); + + templateSelector.setItems(getAvailableTemplates(disease)); + templateSelector.setItemCaptionGenerator(DocumentTemplateDto::getFileName); + }); + + addComponent(diseaseSelector); + } + + private void addTemplateSelector(String captionTemplateSelector, Function fileNameFunction) { templateSelector = new ComboBox<>(captionTemplateSelector); templateSelector.setWidth(100F, Unit.PERCENTAGE); templateSelector.addValueChangeListener(e -> { - DocumentTemplateDto templateFile = e.getValue(); - boolean isValidTemplateFile = templateFile != null; + DocumentTemplateDto template = e.getValue(); + boolean isValidTemplateFile = template != null; createButton.setEnabled(isValidTemplateFile); additionalVariablesComponent.removeAllComponents(); hideTextfields(); documentVariables = null; if (isValidTemplateFile) { try { - documentVariables = getDocumentVariables(templateFile.getUuid()); + documentVariables = getDocumentVariables(template.toReference()); List additionalVariables = documentVariables.getAdditionalVariables(); if (additionalVariables != null && !additionalVariables.isEmpty()) { for (String variable : additionalVariables) { @@ -149,7 +176,7 @@ private void addTemplateSelector(String captionTemplateSelector, Function getAvailableTemplates(); + protected abstract List getAvailableTemplates(Disease disease); - protected abstract DocumentVariables getDocumentVariables(String templateFile) throws IOException, DocumentTemplateException; + protected abstract DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) + throws IOException, DocumentTemplateException; protected abstract StreamResource createStreamResource(DocumentTemplateDto template, String filename); protected abstract String getWindowCaption(); - - public boolean isCheckBoxAfterTemplateSelector() { - return isCheckBoxAfterTemplateSelector; - } - - public void setCheckBoxAfterTemplateSelector(boolean checkBoxAfterTemplateSelector) { - isCheckBoxAfterTemplateSelector = checkBoxAfterTemplateSelector; - } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java index 29fa05d0b70..cd73d296e73 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/DocGenerationController.java @@ -39,11 +39,12 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.ReferenceDto; -import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EventDocumentFacade; import de.symeda.sormas.api.docgeneneration.QuarantineOrderFacade; import de.symeda.sormas.api.docgeneneration.RootEntityType; +import de.symeda.sormas.api.event.EventDto; import de.symeda.sormas.api.event.EventParticipantReferenceDto; import de.symeda.sormas.api.event.EventReferenceDto; import de.symeda.sormas.api.i18n.Captions; @@ -66,21 +67,23 @@ public void showQuarantineOrderDocumentDialog( RootEntityType rootEntityType, ReferenceDto referenceDto, DocumentWorkflow workflow, + Disease defaultDisease, SampleCriteria sampleCriteria, VaccinationCriteria vaccinationCriteria, DocumentListComponent documentListComponent) { showDialog( new QuarantineOrderLayout( workflow, + defaultDisease, sampleCriteria, vaccinationCriteria, documentListComponent, - (templateReference, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { + (template, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { QuarantineOrderFacade quarantineOrderFacade = FacadeProvider.getQuarantineOrderFacade(); return new ByteArrayInputStream( - quarantineOrderFacade.getGeneratedDocument(, - templateReference, + quarantineOrderFacade.getGeneratedDocument( + template.toReference(), rootEntityType, referenceDto, sample, @@ -89,7 +92,7 @@ public void showQuarantineOrderDocumentDialog( extraProperties, shouldUploadGeneratedDoc)); }, - (templateFile) -> getDocumentFileName(referenceDto, templateReference))); + (template) -> getDocumentFileName(referenceDto, template))); } public void showBulkQuarantineOrderDocumentDialog(List referenceDtos, DocumentWorkflow workflow) { @@ -100,13 +103,14 @@ public void showBulkQuarantineOrderDocumentDialog(List referenceDt null, null, null, - (templateReference, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { + null, + (template, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { QuarantineOrderFacade quarantineOrderFacade = FacadeProvider.getQuarantineOrderFacade(); Map generatedDocumentContents = - quarantineOrderFacade.getGeneratedDocuments(templateReference, referenceDtos, extraProperties, shouldUploadGeneratedDoc); + quarantineOrderFacade.getGeneratedDocuments(template.toReference(), referenceDtos, extraProperties, shouldUploadGeneratedDoc); - return generateZip(templateReference, shouldUploadGeneratedDoc, generatedDocumentContents); + return generateZip(template, shouldUploadGeneratedDoc, generatedDocumentContents); }, (templateFile) -> filename)); @@ -117,27 +121,28 @@ public void showBulkEventParticipantQuarantineOrderDocumentDialog(List { + (template, sample, pathogenTest, vaccination, extraProperties, shouldUploadGeneratedDoc) -> { QuarantineOrderFacade quarantineOrderFacade = FacadeProvider.getQuarantineOrderFacade(); Map generatedDocumentContents = quarantineOrderFacade.getGeneratedDocumentsForEventParticipants( - templateReference, + template.toReference(), referenceDtos, eventDisease, extraProperties, shouldUploadGeneratedDoc); - return generateZip(templateReference, shouldUploadGeneratedDoc, generatedDocumentContents); + return generateZip(template, shouldUploadGeneratedDoc, generatedDocumentContents); }, (templateFile) -> filename)); } private ByteArrayInputStream generateZip( - DocumentTemplateReferenceDto templateReference, + DocumentTemplateDto template, Boolean shouldUploadGeneratedDoc, Map generatedDocumentContents) { long fileSizeLimitMB = FacadeProvider.getConfigFacade().getDocumentUploadSizeLimitMb(); @@ -149,7 +154,7 @@ private ByteArrayInputStream generateZip( for (Map.Entry referenceDocumentContent : generatedDocumentContents.entrySet()) { ReferenceDto referenceDto = referenceDocumentContent.getKey(); - ZipEntry entry = new ZipEntry(getDocumentFileName(referenceDto, templateReference)); + ZipEntry entry = new ZipEntry(getDocumentFileName(referenceDto, template)); zos.putNextEntry(entry); byte[] document = referenceDocumentContent.getValue(); @@ -206,16 +211,17 @@ private void buildDocumentUploadWarningWindow(List fileSizeLimitExceeded window.setWidth(1024, Sizeable.Unit.PIXELS); } - public void showEventDocumentDialog(EventReferenceDto eventReferenceDto, DocumentListComponent documentListComponent) { + public void showEventDocumentDialog(EventDto event, DocumentListComponent documentListComponent) { showDialog( new EventDocumentLayout( + event.getDisease(), documentListComponent, - (templateFileName) -> getDocumentFileName(eventReferenceDto, templateFileName), + (template) -> getDocumentFileName(event.toReference(), template), (template, properties, shouldUploadGeneratedDoc) -> { EventDocumentFacade eventDocumentFacade = FacadeProvider.getEventDocumentFacade(); return new ByteArrayInputStream( - eventDocumentFacade.getGeneratedDocument(, template.getUuid(), eventReferenceDto, properties, shouldUploadGeneratedDoc) + eventDocumentFacade.getGeneratedDocument(template.toReference(), event.toReference(), properties, shouldUploadGeneratedDoc) .getBytes(StandardCharsets.UTF_8)); })); } @@ -223,11 +229,11 @@ public void showEventDocumentDialog(EventReferenceDto eventReferenceDto, Documen public void showEventDocumentDialog(List referenceDtos) { String filename = DownloadUtil.createFileNameWithCurrentDate(ExportEntityName.EVENTS, ".zip"); - showDialog(new EventDocumentLayout(null, (templateFile) -> filename, (template, properties, shouldUploadGeneratedDoc) -> { + showDialog(new EventDocumentLayout(null, null, (templateFile) -> filename, (template, properties, shouldUploadGeneratedDoc) -> { EventDocumentFacade eventDocumentFacade = FacadeProvider.getEventDocumentFacade(); Map generatedDocumentContents = - eventDocumentFacade.getGeneratedDocuments(, template, referenceDtos, properties, shouldUploadGeneratedDoc); + eventDocumentFacade.getGeneratedDocuments(template.toReference(), referenceDtos, properties, shouldUploadGeneratedDoc); return generateZip(template, shouldUploadGeneratedDoc, generatedDocumentContents); @@ -240,7 +246,7 @@ private void showDialog(AbstractDocgenerationLayout docgenerationLayout) { window.setCaption(I18nProperties.getCaption(docgenerationLayout.getWindowCaption())); } - private String getDocumentFileName(ReferenceDto eventReferenceDto, DocumentTemplateReferenceDto templateReference) { - return DataHelper.getShortUuid(eventReferenceDto) + '-' + templateReference.getCaption(); + private String getDocumentFileName(ReferenceDto eventReferenceDto, DocumentTemplateDto template) { + return DataHelper.getShortUuid(eventReferenceDto) + '-' + template.getFileName(); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java index 68d8f09e853..2ff77315502 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentLayout.java @@ -27,9 +27,11 @@ import com.vaadin.server.StreamResource; import com.vaadin.ui.Notification; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentVariables; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -42,10 +44,16 @@ public class EventDocumentLayout extends AbstractDocgenerationLayout { private DocumentListComponent documentListComponent; public EventDocumentLayout( + Disease defaultDisease, DocumentListComponent documentListComponent, - Function fileNameFunction, + Function fileNameFunction, DocumentInputStreamSupplier documentInputStreamSupplier) { - super(I18nProperties.getCaption(Captions.DocumentTemplate_EventHandout), fileNameFunction, isNull(documentListComponent), false); + super( + defaultDisease, + I18nProperties.getCaption(Captions.DocumentTemplate_EventHandout), + fileNameFunction, + isNull(documentListComponent), + false); this.documentListComponent = documentListComponent; this.documentInputStreamSupplier = documentInputStreamSupplier; @@ -54,20 +62,20 @@ public EventDocumentLayout( } @Override - protected List getAvailableTemplates() { - return FacadeProvider.getEventDocumentFacade().getAvailableTemplates(); + protected List getAvailableTemplates(Disease disease) { + return FacadeProvider.getEventDocumentFacade().getAvailableTemplates(disease); } @Override - protected DocumentVariables getDocumentVariables(String templateFile) throws DocumentTemplateException { - return FacadeProvider.getEventDocumentFacade().getDocumentVariables(templateFile); + protected DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + return FacadeProvider.getEventDocumentFacade().getDocumentVariables(templateReference); } @Override protected StreamResource createStreamResource(DocumentTemplateDto template, String filename) { return new StreamResource((StreamResource.StreamSource) () -> { try { - return documentInputStreamSupplier.get(templateFile, readAdditionalVariables(), shouldUploadGeneratedDocument()); + return documentInputStreamSupplier.get(template, readAdditionalVariables(), shouldUploadGeneratedDocument()); } catch (Exception e) { new Notification(I18nProperties.getString(Strings.errorProcessingTemplate), e.getMessage(), Notification.Type.ERROR_MESSAGE) .show(Page.getCurrent()); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentsComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentsComponent.java index 334d1f1d3f3..1f474c4724a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentsComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/EventDocumentsComponent.java @@ -1,6 +1,6 @@ package de.symeda.sormas.ui.docgeneration; -import de.symeda.sormas.api.event.EventReferenceDto; +import de.symeda.sormas.api.event.EventDto; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.ui.ControllerProvider; @@ -10,11 +10,11 @@ public class EventDocumentsComponent extends AbstractDocumentGenerationComponent public static final String DOCGENERATION_LOC = "docgeneration"; - public EventDocumentsComponent(EventReferenceDto eventReferenceDto, DocumentListComponent documentListComponent) { + public EventDocumentsComponent(EventDto event, DocumentListComponent documentListComponent) { super(); if (DocGenerationHelper.isDocGenerationAllowed()) { addDocumentBar( - () -> ControllerProvider.getDocGenerationController().showEventDocumentDialog(eventReferenceDto, documentListComponent), + () -> ControllerProvider.getDocGenerationController().showEventDocumentDialog(event, documentListComponent), I18nProperties.getCaption(Captions.DocumentTemplate_EventHandout_create)); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderDocumentsComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderDocumentsComponent.java index b67a41076de..6f1d830e937 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderDocumentsComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderDocumentsComponent.java @@ -17,6 +17,7 @@ import static de.symeda.sormas.ui.docgeneration.DocGenerationHelper.isDocGenerationAllowed; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.ReferenceDto; import de.symeda.sormas.api.caze.CaseDataDto; import de.symeda.sormas.api.contact.ContactDto; @@ -33,18 +34,21 @@ public class QuarantineOrderDocumentsComponent extends AbstractDocumentGenerationComponent { + private static final long serialVersionUID = 85800761803338225L; + public static final String QUARANTINE_LOC = "quarantine"; - public QuarantineOrderDocumentsComponent( + private QuarantineOrderDocumentsComponent( RootEntityType rootEntityType, ReferenceDto referenceDto, DocumentWorkflow workflow, + Disease defaultDisease, SampleCriteria sampleCriteria, VaccinationCriteria vaccinationCriteria) { super(); addDocumentBar( () -> ControllerProvider.getDocGenerationController() - .showQuarantineOrderDocumentDialog(rootEntityType, referenceDto, workflow, sampleCriteria, vaccinationCriteria, null), + .showQuarantineOrderDocumentDialog(rootEntityType, referenceDto, workflow, defaultDisease, sampleCriteria, vaccinationCriteria, null), Captions.DocumentTemplate_QuarantineOrder); } @@ -56,12 +60,17 @@ public static void addComponentToLayout(LayoutWithSidePanel targetLayout, CaseDa RootEntityType.ROOT_CASE, caze.toReference(), DocumentWorkflow.QUARANTINE_ORDER_CASE, + caze.getDisease(), sampleCriteria, vaccinationCriteria, documentList); } - public static void addComponentToLayout(LayoutWithSidePanel targetLayout, ContactDto contact, DocumentListComponent documentList) { + public static void addComponentToLayout( + LayoutWithSidePanel targetLayout, + ContactDto contact, + Disease defaultDisease, + DocumentListComponent documentList) { VaccinationCriteria vaccinationCriteria = new VaccinationCriteria.Builder(contact.getPerson()).withDisease(contact.getDisease()).build(); SampleCriteria sampleCriteria = new SampleCriteria().contact(contact.toReference()); @@ -70,6 +79,7 @@ public static void addComponentToLayout(LayoutWithSidePanel targetLayout, Contac RootEntityType.ROOT_CONTACT, contact.toReference(), DocumentWorkflow.QUARANTINE_ORDER_CONTACT, + defaultDisease, sampleCriteria, vaccinationCriteria, documentList); @@ -81,6 +91,7 @@ public static void addComponentToLayout(LayoutWithSidePanel targetLayout, Travel RootEntityType.ROOT_TRAVEL_ENTRY, travelEntry.toReference(), DocumentWorkflow.QUARANTINE_ORDER_TRAVEL_ENTRY, + travelEntry.getDisease(), null, null, documentList); @@ -91,21 +102,23 @@ public static void addComponentToLayout( RootEntityType rootEntityType, ReferenceDto referenceDto, DocumentWorkflow workflow, + Disease defaultDisease, SampleCriteria sampleCriteria, VaccinationCriteria vaccinationCriteria) { if (isDocGenerationAllowed()) { QuarantineOrderDocumentsComponent docGenerationComponent = - new QuarantineOrderDocumentsComponent(rootEntityType, referenceDto, workflow, sampleCriteria, vaccinationCriteria); + new QuarantineOrderDocumentsComponent(rootEntityType, referenceDto, workflow, defaultDisease, sampleCriteria, vaccinationCriteria); docGenerationComponent.addStyleName(CssStyles.SIDE_COMPONENT); targetLayout.addSidePanelComponent(docGenerationComponent, QUARANTINE_LOC); } } - public static void addComponentToLayout( + private static void addComponentToLayout( LayoutWithSidePanel targetLayout, RootEntityType rootEntityType, ReferenceDto referenceDto, DocumentWorkflow workflow, + Disease defaultDisease, SampleCriteria sampleCriteria, VaccinationCriteria vaccinationCriteria, DocumentListComponent documentListComponent) { @@ -114,6 +127,7 @@ public static void addComponentToLayout( rootEntityType, referenceDto, workflow, + defaultDisease, sampleCriteria, vaccinationCriteria, documentListComponent); @@ -122,10 +136,11 @@ public static void addComponentToLayout( } } - public QuarantineOrderDocumentsComponent( + private QuarantineOrderDocumentsComponent( RootEntityType rootEntityType, ReferenceDto referenceDto, DocumentWorkflow workflow, + Disease defaultDisease, SampleCriteria sampleCriteria, VaccinationCriteria vaccinationCriteria, DocumentListComponent documentListComponent) { @@ -136,6 +151,7 @@ public QuarantineOrderDocumentsComponent( rootEntityType, referenceDto, workflow, + defaultDisease, sampleCriteria, vaccinationCriteria, documentListComponent), diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java index 9dc7126c603..86119927bd9 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/docgeneration/QuarantineOrderLayout.java @@ -34,6 +34,7 @@ import com.vaadin.ui.ComboBox; import com.vaadin.ui.Notification; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentTemplateException; @@ -73,12 +74,14 @@ public class QuarantineOrderLayout extends AbstractDocgenerationLayout { public QuarantineOrderLayout( DocumentWorkflow workflow, + Disease defaultDisease, @Nullable SampleCriteria sampleCriteria, @Nullable VaccinationCriteria vaccinationCriteria, DocumentListComponent documentListComponent, DocumentStreamSupplier documentStreamSupplier, - Function fileNameFunction) { + Function fileNameFunction) { super( + defaultDisease, I18nProperties.getCaption(Captions.DocumentTemplate_QuarantineOrder), fileNameFunction, isNull(sampleCriteria) && isNull(documentListComponent), @@ -101,7 +104,7 @@ public QuarantineOrderLayout( } public QuarantineOrderLayout(DocumentWorkflow workflow) { - super(I18nProperties.getCaption(Captions.DocumentTemplate_QuarantineOrder), null, true, true); + super(null, I18nProperties.getCaption(Captions.DocumentTemplate_QuarantineOrder), null, true, true); this.workflow = workflow; this.documentStreamSupplier = null; @@ -166,9 +169,9 @@ protected void createVaccinationSelector(VaccinationCriteria vaccinationCriteria } @Override - protected List getAvailableTemplates() { + protected List getAvailableTemplates(Disease disease) { try { - return FacadeProvider.getQuarantineOrderFacade().getAvailableTemplates(workflow); + return FacadeProvider.getQuarantineOrderFacade().getAvailableTemplates(workflow, disease); } catch (Exception e) { new Notification(I18nProperties.getString(Strings.errorProcessingTemplate), e.getMessage(), Notification.Type.ERROR_MESSAGE) .show(Page.getCurrent()); @@ -177,8 +180,8 @@ protected List getAvailableTemplates() { } @Override - protected DocumentVariables getDocumentVariables(String templateFile) throws DocumentTemplateException { - return FacadeProvider.getQuarantineOrderFacade().getDocumentVariables(workflow, templateFile); + protected DocumentVariables getDocumentVariables(DocumentTemplateReferenceDto templateReference) throws DocumentTemplateException { + return FacadeProvider.getQuarantineOrderFacade().getDocumentVariables(templateReference); } @Override @@ -194,7 +197,7 @@ protected StreamResource createStreamResource(DocumentTemplateDto template, Stri try { InputStream stream = documentStreamSupplier.getStream( - template.toReference(), + template, sampleReference, pathogenTestReference, vaccinationReference, @@ -277,7 +280,7 @@ public QuarantineOrderDocumentOptionsDto getFieldValues() { public interface DocumentStreamSupplier { InputStream getStream( - DocumentTemplateReferenceDto templateReference, + DocumentTemplateDto template, SampleReferenceDto sample, PathogenTestReferenceDto pathogenTest, VaccinationReferenceDto vaccinationReference, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalBulkEmailOptionsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalBulkEmailOptionsForm.java index d1021d75aa2..1cb156d7f63 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalBulkEmailOptionsForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalBulkEmailOptionsForm.java @@ -44,6 +44,7 @@ import de.symeda.sormas.api.DocumentHelper; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.EmailAttachementDto; import de.symeda.sormas.api.docgeneneration.RootEntityType; @@ -113,7 +114,7 @@ protected String createHtmlLayout() { protected void addFields() { ComboBox templateCombo = addField(ExternalEmailOptionsWithAttachmentsDto.TEMPLATE_NAME, ComboBox.class); templateCombo.setRequired(true); - List templateNames = FacadeProvider.getExternalEmailFacade().getTemplateNames(documentWorkflow); + List templateNames = FacadeProvider.getExternalEmailFacade().getTemplates(documentWorkflow); FieldHelper.updateItems(templateCombo, templateNames); if (Arrays.asList(DocumentWorkflow.CASE_EMAIL, DocumentWorkflow.CONTACT_EMAIL, DocumentWorkflow.TRAVEL_ENTRY_EMAIL) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalEmailOptionsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalEmailOptionsForm.java index 5a9787a8007..507bf0385b4 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalEmailOptionsForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/email/ExternalEmailOptionsForm.java @@ -18,6 +18,7 @@ import static de.symeda.sormas.ui.utils.LayoutUtil.fluidRowLocs; import java.util.List; +import java.util.stream.Collectors; import org.apache.commons.lang3.StringUtils; @@ -27,6 +28,8 @@ import com.vaadin.v7.ui.ComboBox; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateReferenceDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.document.DocumentReferenceDto; import de.symeda.sormas.api.document.DocumentRelatedEntityType; @@ -43,28 +46,28 @@ public class ExternalEmailOptionsForm extends AbstractEditForm { - private static final String ATTACHMENT_NOT_AVAILABLE_INFO_LOC = "attachmentNotAvailableInfoLoc"; - private static final String HTML_LAYOUT = fluidRowLocs(ExternalEmailOptionsDto.TEMPLATE_NAME) - + fluidRowLocs(ExternalEmailOptionsDto.RECIPIENT_EMAIL) - + fluidRowLocs(ExternalEmailOptionsDto.ATTACHED_DOCUMENTS) - + fluidRowLocs(ATTACHMENT_NOT_AVAILABLE_INFO_LOC); + private static final String ATTACHMENT_NOT_AVAILABLE_INFO_LOC = "attachmentNotAvailableInfoLoc"; + private static final String HTML_LAYOUT = fluidRowLocs(ExternalEmailOptionsDto.TEMPLATE) + + fluidRowLocs(ExternalEmailOptionsDto.RECIPIENT_EMAIL) + + fluidRowLocs(ExternalEmailOptionsDto.ATTACHED_DOCUMENTS) + + fluidRowLocs(ATTACHMENT_NOT_AVAILABLE_INFO_LOC); private final DocumentWorkflow documentWorkflow; - private final DocumentRelatedEntityType documentRelatedEntityType; + private final DocumentRelatedEntityType documentRelatedEntityType; private final PersonDto person; - private final boolean isAttachmentAvailable; - private MultiSelect attachedDocumentsField; - - protected ExternalEmailOptionsForm( - DocumentWorkflow documentWorkflow, - DocumentRelatedEntityType documentRelatedEntityType, - PersonDto person, - boolean isAttachmentAvailable) { + private final boolean isAttachmentAvailable; + private MultiSelect attachedDocumentsField; + + protected ExternalEmailOptionsForm( + DocumentWorkflow documentWorkflow, + DocumentRelatedEntityType documentRelatedEntityType, + PersonDto person, + boolean isAttachmentAvailable) { super(ExternalEmailOptionsDto.class, ExternalEmailOptionsDto.I18N_PREFIX, false); this.documentWorkflow = documentWorkflow; - this.documentRelatedEntityType = documentRelatedEntityType; + this.documentRelatedEntityType = documentRelatedEntityType; this.person = person; - this.isAttachmentAvailable = isAttachmentAvailable; + this.isAttachmentAvailable = isAttachmentAvailable; addFields(); hideValidationUntilNextCommit(); @@ -77,10 +80,14 @@ protected String createHtmlLayout() { @Override protected void addFields() { - ComboBox templateCombo = addField(ExternalEmailOptionsDto.TEMPLATE_NAME, ComboBox.class); + ComboBox templateCombo = addField(ExternalEmailOptionsDto.TEMPLATE, ComboBox.class); templateCombo.setRequired(true); - List templateNames = FacadeProvider.getExternalEmailFacade().getTemplateNames(documentWorkflow); - FieldHelper.updateItems(templateCombo, templateNames); + List templates = FacadeProvider.getExternalEmailFacade() + .getTemplates(documentWorkflow) + .stream() + .map(DocumentTemplateDto::toReference) + .collect(Collectors.toList()); + FieldHelper.updateItems(templateCombo, templates); ComboBox recipientEmailCombo = addField(ExternalEmailOptionsDto.RECIPIENT_EMAIL, ComboBox.class); recipientEmailCombo.setRequired(true); @@ -89,30 +96,30 @@ protected void addFields() { String primaryEmailAddress = person.getEmailAddress(true); if (StringUtils.isNotBlank(primaryEmailAddress)) { recipientEmailCombo - .setItemCaption(primaryEmailAddress, primaryEmailAddress + " (" + I18nProperties.getCaption(Captions.primarySuffix) + ")"); + .setItemCaption(primaryEmailAddress, primaryEmailAddress + " (" + I18nProperties.getCaption(Captions.primarySuffix) + ")"); } - if (documentRelatedEntityType != null) { - attachedDocumentsField = addField(ExternalEmailOptionsDto.ATTACHED_DOCUMENTS, MultiSelect.class); - if (!isAttachmentAvailable) { - attachedDocumentsField.setEnabled(false); - - MultilineLabel attachmentUnavailableInfo = new MultilineLabel( - VaadinIcons.INFO_CIRCLE.getHtml() + " " + I18nProperties.getString(Strings.messageExternalEmailAttachmentNotAvailableInfo), - ContentMode.HTML); - attachmentUnavailableInfo.addStyleNames(CssStyles.VSPACE_2, CssStyles.VSPACE_TOP_4); - getContent().addComponent(attachmentUnavailableInfo, ATTACHMENT_NOT_AVAILABLE_INFO_LOC); - } - } - } - - @Override - public void setValue(ExternalEmailOptionsDto newFieldValue) throws ReadOnlyException, Converter.ConversionException { - super.setValue(newFieldValue); - - if (attachedDocumentsField != null) { - attachedDocumentsField.setItems( - FacadeProvider.getExternalEmailFacade().getAttachableDocuments(documentWorkflow, newFieldValue.getRootEntityReference().getUuid())); - } + if (documentRelatedEntityType != null) { + attachedDocumentsField = addField(ExternalEmailOptionsDto.ATTACHED_DOCUMENTS, MultiSelect.class); + if (!isAttachmentAvailable) { + attachedDocumentsField.setEnabled(false); + + MultilineLabel attachmentUnavailableInfo = new MultilineLabel( + VaadinIcons.INFO_CIRCLE.getHtml() + " " + I18nProperties.getString(Strings.messageExternalEmailAttachmentNotAvailableInfo), + ContentMode.HTML); + attachmentUnavailableInfo.addStyleNames(CssStyles.VSPACE_2, CssStyles.VSPACE_TOP_4); + getContent().addComponent(attachmentUnavailableInfo, ATTACHMENT_NOT_AVAILABLE_INFO_LOC); + } + } + } + + @Override + public void setValue(ExternalEmailOptionsDto newFieldValue) throws ReadOnlyException, Converter.ConversionException { + super.setValue(newFieldValue); + + if (attachedDocumentsField != null) { + attachedDocumentsField.setItems( + FacadeProvider.getExternalEmailFacade().getAttachableDocuments(documentWorkflow, newFieldValue.getRootEntityReference().getUuid())); + } } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataView.java index cb2733d9a47..46db076f7db 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataView.java @@ -140,7 +140,7 @@ protected void initView(String params) { layout.addSidePanelComponent(new SideComponentLayout(documentList), DOCUMENTS_LOC); } - EventDocumentsComponent eventDocuments = new EventDocumentsComponent(getEventRef(), documentList); + EventDocumentsComponent eventDocuments = new EventDocumentsComponent(event, documentList); eventDocuments.addStyleName(CssStyles.SIDE_COMPONENT); layout.addSidePanelComponent(eventDocuments, EventDocumentsComponent.DOCGENERATION_LOC); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantDataView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantDataView.java index 0e067d6f6bc..f34e16f0225 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantDataView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventParticipantDataView.java @@ -151,6 +151,7 @@ protected void initView(String params) { RootEntityType.ROOT_EVENT_PARTICIPANT, eventParticipantRef, DocumentWorkflow.QUARANTINE_ORDER_EVENT_PARTICIPANT, + event.getDisease(), sampleCriteria, vaccinationCriteria); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DocumentTemplateReceiver.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DocumentTemplateReceiver.java index b8336e890c4..dda954b2d26 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DocumentTemplateReceiver.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DocumentTemplateReceiver.java @@ -10,13 +10,16 @@ import java.nio.file.Files; import java.nio.file.Paths; import java.util.Date; +import java.util.function.Supplier; import com.vaadin.server.Page; import com.vaadin.ui.Label; import com.vaadin.ui.Notification; import com.vaadin.v7.ui.Upload.StartedEvent; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.docgeneneration.DocumentTemplateDto; import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; @@ -35,9 +38,11 @@ public class DocumentTemplateReceiver private File file; private String fName; private final DocumentWorkflow documentWorkflow; + private final Supplier diseaseSupplier; - public DocumentTemplateReceiver(DocumentWorkflow documentWorkflow) { + public DocumentTemplateReceiver(DocumentWorkflow documentWorkflow, Supplier diseaseSupplier) { this.documentWorkflow = documentWorkflow; + this.diseaseSupplier = diseaseSupplier; } @Override @@ -89,7 +94,7 @@ public void uploadSucceeded(com.vaadin.v7.ui.Upload.SucceededEvent succeededEven } // Check for duplicate files - if (FacadeProvider.getDocumentTemplateFacade().isExistingTemplate(documentWorkflow, fName)) { + if (FacadeProvider.getDocumentTemplateFacade().isExistingTemplateFile(documentWorkflow, diseaseSupplier.get(), fName)) { VaadinUiUtil.showConfirmationPopup( I18nProperties.getString(Strings.headingFileExists), new Label(String.format(I18nProperties.getString(Strings.infoDocumentAlreadyExists), fName)), @@ -108,8 +113,9 @@ public void uploadSucceeded(com.vaadin.v7.ui.Upload.SucceededEvent succeededEven private void writeTemplateFile() { try { - byte[] filecontent = Files.readAllBytes(file.toPath()); - FacadeProvider.getDocumentTemplateFacade().writeDocumentTemplate(documentWorkflow, fName, filecontent); + byte[] fileContent = Files.readAllBytes(file.toPath()); + DocumentTemplateDto documentTemplateDto = DocumentTemplateDto.build(documentWorkflow, fName, diseaseSupplier.get()); + FacadeProvider.getDocumentTemplateFacade().saveDocumentTemplate(documentTemplateDto, fileContent); VaadinUiUtil.showSimplePopupWindow( I18nProperties.getString(Strings.headingUploadSuccess), I18nProperties.getString(Strings.messageUploadSuccessful)); diff --git a/sormas-ui/src/test/resources/META-INF/persistence.xml b/sormas-ui/src/test/resources/META-INF/persistence.xml index f0c6e5b197e..0242f19fa99 100644 --- a/sormas-ui/src/test/resources/META-INF/persistence.xml +++ b/sormas-ui/src/test/resources/META-INF/persistence.xml @@ -89,6 +89,7 @@ de.symeda.sormas.backend.environment.environmentsample.EnvironmentSample de.symeda.sormas.backend.specialcaseaccess.SpecialCaseAccess de.symeda.sormas.backend.selfreport.SelfReport + de.symeda.sormas.backend.docgeneration.DocumentTemplate true From 157b4e4fdde44a6e25c9661210d89a35d8edf2c4 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Tue, 29 Oct 2024 14:29:17 +0200 Subject: [PATCH 17/56] #13160 Add "Disease" Attribute to Document Templates for Filtering - fix migration after merge --- .../de/symeda/sormas/backend/common/StartupShutdownService.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java index b7f4794b7f9..b051759007e 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/StartupShutdownService.java @@ -875,7 +875,7 @@ private void upgrade() { case 516: fillDefaultUserRole(DefaultUserRole.ENVIRONMENTAL_SURVEILLANCE_USER); break; - case 552: + case 553: createEntitiesForDocumentTemplates(); break; default: From bc38f97deb038a25024be307eeb2498d92cff7ca Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Tue, 29 Oct 2024 14:48:26 +0200 Subject: [PATCH 18/56] #13093 - Update Data Protection for certain Data Fields --- .../symeda/sormas/api/caze/CaseExportDto.java | 1 + .../symeda/sormas/api/contact/ContactDto.java | 1 + .../de/symeda/sormas/api/i18n/Captions.java | 13 ++ .../api/vaccination/VaccinationDto.java | 1 + .../src/main/resources/captions.properties | 15 ++ sormas-api/src/main/resources/enum.properties | 15 -- .../{EntityColumn.java => EntityColumns.java} | 210 +++++++++++------- .../sormas/backend/info/InfoFacadeEjb.java | 48 ++-- .../java/de/symeda/sormas/ui/AboutView.java | 9 +- .../sormas/ui/action/ActionEditForm.java | 3 +- .../clinicalcourse/HealthConditionsForm.java | 6 +- .../sormas/ui/utils/RichTextAreaCustom.java | 24 -- .../utils/SormasFieldGroupFieldFactory.java | 5 +- .../symeda/sormas/ui/utils/VaadinUiUtil.java | 3 +- 14 files changed, 200 insertions(+), 154 deletions(-) rename sormas-backend/src/main/java/de/symeda/sormas/backend/info/{EntityColumn.java => EntityColumns.java} (51%) delete mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java index 16265e06647..347b398bfdb 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java @@ -263,6 +263,7 @@ public class CaseExportDto extends AbstractUuidDto implements IsCase { private String vaccineBatchNumber; private String vaccineUniiCode; private String vaccineAtcCode; + @SensitiveData private HealthConditionsDto healthConditions; private int numberOfPrescriptions; private int numberOfTreatments; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactDto.java index fc7e25b832b..615d81675fc 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactDto.java @@ -312,6 +312,7 @@ public class ContactDto extends SormasToSormasShareableDto implements IsContact @Valid private EpiDataDto epiData; @Valid + @SensitiveData private HealthConditionsDto healthConditions; private YesNoUnknown returningTraveler; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index de514090097..78f72bdf467 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -1076,6 +1076,19 @@ public interface Captions { String edit = "edit"; String endDateTime = "endDateTime"; String endOfProcessingDate = "endOfProcessingDate"; + String EntityColumn_CAPTION = "EntityColumn.CAPTION"; + String EntityColumn_DATA_PROTECTION = "EntityColumn.DATA_PROTECTION"; + String EntityColumn_DESCRIPTION = "EntityColumn.DESCRIPTION"; + String EntityColumn_DISEASES = "EntityColumn.DISEASES"; + String EntityColumn_ENTITY = "EntityColumn.ENTITY"; + String EntityColumn_EXCLUSIVE_COUNTRIES = "EntityColumn.EXCLUSIVE_COUNTRIES"; + String EntityColumn_FIELD = "EntityColumn.FIELD"; + String EntityColumn_FIELD_ID = "EntityColumn.FIELD_ID"; + String EntityColumn_IGNORED_COUNTRIES = "EntityColumn.IGNORED_COUNTRIES"; + String EntityColumn_NEW_DISEASE = "EntityColumn.NEW_DISEASE"; + String EntityColumn_OUTBREAKS = "EntityColumn.OUTBREAKS"; + String EntityColumn_REQUIRED = "EntityColumn.REQUIRED"; + String EntityColumn_TYPE = "EntityColumn.TYPE"; String Environment = "Environment"; String Environment_description = "Environment.description"; String Environment_environmentMedia = "Environment.environmentMedia"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/vaccination/VaccinationDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/vaccination/VaccinationDto.java index 8e82f5a4397..2954ad1a01f 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/vaccination/VaccinationDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/vaccination/VaccinationDto.java @@ -69,6 +69,7 @@ public class VaccinationDto extends PseudonymizableDto { @DependingOnFeatureType(featureType = FeatureType.IMMUNIZATION_MANAGEMENT, properties = @FeatureProperty(property = FeatureTypeProperty.REDUCED, value = "true"), hide = true) + @SensitiveData private HealthConditionsDto healthConditions; @NotNull(message = Validations.validReportDateTime) private Date reportDate; diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index a2fdb57909f..ec5a44f2864 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -1134,6 +1134,21 @@ DocumentTemplate.documentUploadWarning=Document upload warning DocumentTemplate.fileTooBig=The documents were successfully generated, but at least one document could not be uploaded to its entity because its file size exceeds the specified file size limit of %dMB DocumentTemplate.notUploaded=Documents could not be uploaded to the following entities: +# Entity columns +EntityColumn.ENTITY = Entity +EntityColumn.FIELD_ID = Field ID +EntityColumn.FIELD = Field +EntityColumn.TYPE = Type +EntityColumn.DATA_PROTECTION = Data protection +EntityColumn.CAPTION = Caption +EntityColumn.DESCRIPTION = Description +EntityColumn.REQUIRED = Required +EntityColumn.NEW_DISEASE = New disease +EntityColumn.DISEASES = Diseases +EntityColumn.OUTBREAKS = Outbreaks +EntityColumn.IGNORED_COUNTRIES = Ignored countries +EntityColumn.EXCLUSIVE_COUNTRIES = Exclusive countries + # Environment Environment=Environment Environment.uuid=Environment ID diff --git a/sormas-api/src/main/resources/enum.properties b/sormas-api/src/main/resources/enum.properties index 35fa86fe9f5..5fad41403f0 100644 --- a/sormas-api/src/main/resources/enum.properties +++ b/sormas-api/src/main/resources/enum.properties @@ -2181,21 +2181,6 @@ MeansOfImmunization.RECOVERY = Recovery MeansOfImmunization.VACCINATION_RECOVERY = Vaccination/Recovery MeansOfImmunization.OTHER = Other -#EntityColumn -EntityColumn.ENTITY = Entity -EntityColumn.FIELD_ID = Field ID -EntityColumn.FIELD = Field -EntityColumn.TYPE = Type -EntityColumn.DATA_PROTECTION = Data protection -EntityColumn.CAPTION = Caption -EntityColumn.DESCRIPTION = Description -EntityColumn.REQUIRED = Required -EntityColumn.NEW_DISEASE = New disease -EntityColumn.DISEASES = Diseases -EntityColumn.OUTBREAKS = Outbreaks -EntityColumn.IGNORED_COUNTRIES = Ignored countries -EntityColumn.EXCLUSIVE_COUNTRIES = Exclusive countries - #EnumColumn EnumColumn.TYPE = Type EnumColumn.VALUE = Value diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumns.java similarity index 51% rename from sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java rename to sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumns.java index 7661514f8fb..4dce653489a 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumn.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/info/EntityColumns.java @@ -21,8 +21,10 @@ import java.util.Arrays; import java.util.Collection; import java.util.Date; +import java.util.List; import java.util.Map; import java.util.function.Function; +import java.util.stream.Collectors; import javax.validation.constraints.NotNull; @@ -42,80 +44,73 @@ import de.symeda.sormas.api.utils.PersonalData; import de.symeda.sormas.api.utils.SensitiveData; -public enum EntityColumn { - - ENTITY(256 * 30, EntityColumn::getEntity, false, true, true), - FIELD_ID(256 * 30, EntityColumn::getFieldId, false, true, false), - FIELD(256 * 30, EntityColumn::getFieldName, false, true, false), - TYPE(256 * 30, EntityColumn::getFieldType, true, true, false), - DATA_PROTECTION(256 * 30, EntityColumn::getDataProtection, false, true, false), - CAPTION(256 * 30, EntityColumn::getCaption, false, true, false), - DESCRIPTION(256 * 60, EntityColumn::getDescription, true, true, false), - REQUIRED(256 * 10, EntityColumn::getNotNull, false, true, false), - NEW_DISEASE(256 * 8, EntityColumn::getNewDisease, false, true, false), - DISEASES(256 * 45, EntityColumn::getDiseases, true, true, false), - OUTBREAKS(256 * 10, EntityColumn::getOutbreaks, false, true, false), - IGNORED_COUNTRIES(256 * 20, EntityColumn::getIgnoredCountries, false, false, false), - EXCLUSIVE_COUNTRIES(256 * 20, EntityColumn::getExclusiveCountries, false, false, false); - - private final int width; - private final Function getValueFromField; - private final boolean hasDefaultStyle; - private final boolean isDataProtectionColumn; - private final boolean isColumnForAllFieldsSheet; - - EntityColumn( - int width, - Function getValueFromField, - boolean hasDefaultStyle, - boolean isDataProtectionColumn, - boolean isColumnForAllFieldsSheet) { - - this.width = width; - this.getValueFromField = getValueFromField; - this.hasDefaultStyle = hasDefaultStyle; - this.isDataProtectionColumn = isDataProtectionColumn; - this.isColumnForAllFieldsSheet = isColumnForAllFieldsSheet; +public class EntityColumns { + + private final EntityColumn ENTITY = new EntityColumn("ENTITY", 256 * 30, this::getEntity, false, false, false, true); + private final EntityColumn FIELD_ID = new EntityColumn("FIELD_ID", 256 * 30, EntityColumns::getFieldId, false, true, true, true); + private final EntityColumn FIELD = new EntityColumn("FIELD", 256 * 30, this::getFieldName, false, true, true, true); + private final EntityColumn TYPE = new EntityColumn("TYPE", 256 * 30, this::getFieldType, true, true, true, true); + private final EntityColumn DATA_PROTECTION = new EntityColumn("DATA_PROTECTION", 256 * 30, this::getDataProtection, false, true, true, true); + private final EntityColumn CAPTION = new EntityColumn("CAPTION", 256 * 30, this::getCaption, false, true, true, true); + private final EntityColumn DESCRIPTION = new EntityColumn("DESCRIPTION", 256 * 60, this::getDescription, true, true, true, true); + private final EntityColumn REQUIRED = new EntityColumn("REQUIRED", 256 * 10, this::getNotNull, false, true, true, true); + private final EntityColumn NEW_DISEASE = new EntityColumn("NEW_DISEASE", 256 * 8, this::getNewDisease, false, true, true, true); + private final EntityColumn DISEASES = new EntityColumn("DISEASES", 256 * 45, this::getDiseases, true, true, true, true); + private final EntityColumn OUTBREAKS = new EntityColumn("OUTBREAKS", 256 * 10, this::getOutbreaks, false, true, true, true); + private final EntityColumn IGNORED_COUNTRIES = + new EntityColumn("IGNORED_COUNTRIES", 256 * 20, this::getIgnoredCountries, false, true, false, false); + private final EntityColumn EXCLUSIVE_COUNTRIES = + new EntityColumn("EXCLUSIVE_COUNTRIES", 256 * 20, this::getExclusiveCountries, false, true, false, false); + + private final List entityColumns = List.of( + ENTITY, + FIELD_ID, + FIELD, + TYPE, + DATA_PROTECTION, + CAPTION, + DESCRIPTION, + REQUIRED, + NEW_DISEASE, + DISEASES, + OUTBREAKS, + IGNORED_COUNTRIES, + EXCLUSIVE_COUNTRIES); + + private final String serverCountry; + private final boolean isDataProtectionFlow; + + public EntityColumns(final String serverCountry, final boolean isDataProtectionFlow) { + this.serverCountry = serverCountry; + this.isDataProtectionFlow = isDataProtectionFlow; } - public int getWidth() { - return width; - } - - public String getGetValueFromField(FieldData fieldData) { - return getValueFromField.apply(fieldData); - } - - public boolean hasDefaultStyle() { - return hasDefaultStyle; - } + public List getEntityColumns() { - public boolean isDataProtectionColumn() { - return isDataProtectionColumn; - } + if (isDataProtectionFlow) { + return entityColumns.stream().filter(EntityColumn::isDataProtectionColumn).collect(Collectors.toList()); + } - public boolean isColumnForAllFieldsSheet() { - return isColumnForAllFieldsSheet; + return entityColumns.stream().filter(EntityColumn::isDataDictionaryColumn).collect(Collectors.toList()); } - @Override - public String toString() { - return I18nProperties.getEnumCaption(this); + public List getColumnsForAllFieldsSheet() { + return entityColumns.stream().filter(EntityColumn::isColumnForAllFieldsSheet).collect(Collectors.toList()); } - private static String getEntity(FieldData fieldData) { + private String getEntity(FieldData fieldData) { return DataHelper.getHumanClassName(fieldData.getEntityClass()); } - private static String getFieldId(FieldData fieldData) { + public static String getFieldId(FieldData fieldData) { return DataHelper.getHumanClassName(fieldData.getEntityClass()) + "." + fieldData.getField().getName(); } - private static String getFieldName(FieldData fieldData) { + private String getFieldName(FieldData fieldData) { return fieldData.getField().getName(); } - private static String getFieldType(FieldData fieldData) { + private String getFieldType(FieldData fieldData) { Class fieldType = fieldData.getField().getType(); if (fieldType.isEnum()) { // use enum type name - values are added below @@ -153,13 +148,16 @@ private static String getFieldType(FieldData fieldData) { return fieldType.getSimpleName(); } - private static String getDataProtection(FieldData fieldData) { + private String getDataProtection(FieldData fieldData) { Field field = fieldData.getField(); - if (field.getAnnotation(PersonalData.class) != null) { + if (field.getAnnotation(PersonalData.class) != null + && (!isDataProtectionFlow || !Arrays.asList(field.getAnnotation(PersonalData.class).excludeForCountries()).contains(serverCountry))) { return "personal"; } else { - if (field.getAnnotation(SensitiveData.class) != null) { + if (field.getAnnotation(SensitiveData.class) != null + && (!isDataProtectionFlow + || !Arrays.asList(field.getAnnotation(SensitiveData.class).excludeForCountries()).contains(serverCountry))) { return "sensitive"; } } @@ -167,12 +165,16 @@ private static String getDataProtection(FieldData fieldData) { return null; } - private static String getCaption(FieldData fieldData) { + private String getCaption(FieldData fieldData) { return I18nProperties.getPrefixCaption(fieldData.getI18NPrefix(), fieldData.getField().getName(), ""); } - private static String getDescription(FieldData fieldData) { - String prefixDescription = I18nProperties.getPrefixDescription(fieldData.getI18NPrefix(), fieldData.getField().getName(), ""); + private String getDescription(FieldData fieldData) { + String fieldDescription = I18nProperties.getPrefixDescription(fieldData.getI18NPrefix(), fieldData.getField().getName(), ""); + + if (isDataProtectionFlow) { + return fieldDescription; + } String[] excludedForCountries = null; @@ -180,24 +182,23 @@ private static String getDescription(FieldData fieldData) { if (field.getAnnotation(PersonalData.class) != null) { excludedForCountries = field.getAnnotation(PersonalData.class).excludeForCountries(); - } else { - if (field.getAnnotation(SensitiveData.class) != null) { - excludedForCountries = field.getAnnotation(SensitiveData.class).excludeForCountries(); - } + } else if (field.getAnnotation(SensitiveData.class) != null) { + excludedForCountries = field.getAnnotation(SensitiveData.class).excludeForCountries(); + } String description; if (excludedForCountries != null && excludedForCountries.length > 0) { - description = prefixDescription + I18nProperties.getString(Strings.messageCountriesExcludedFromDataProtection) + " " + description = fieldDescription + I18nProperties.getString(Strings.messageCountriesExcludedFromDataProtection) + " " + Arrays.toString(excludedForCountries); } else { - description = prefixDescription; + description = fieldDescription; } return description; } - private static String getNotNull(FieldData fieldData) { + private String getNotNull(FieldData fieldData) { if (fieldData.getField().getAnnotation(NotNull.class) == null) { return null; } @@ -205,11 +206,11 @@ private static String getNotNull(FieldData fieldData) { return Boolean.TRUE.toString(); } - private static String getNewDisease(FieldData fieldData) { + private String getNewDisease(FieldData fieldData) { return null; } - private static String getDiseases(FieldData fieldData) { + private String getDiseases(FieldData fieldData) { Diseases diseases = fieldData.getField().getAnnotation(Diseases.class); if (diseases == null) { return "All"; @@ -224,7 +225,7 @@ private static String getDiseases(FieldData fieldData) { } } - private static String getOutbreaks(FieldData fieldData) { + private String getOutbreaks(FieldData fieldData) { if (fieldData.getField().getAnnotation(Outbreaks.class) == null) { return null; } @@ -232,7 +233,7 @@ private static String getOutbreaks(FieldData fieldData) { return Boolean.TRUE.toString(); } - private static String getIgnoredCountries(FieldData fieldData) { + private String getIgnoredCountries(FieldData fieldData) { HideForCountries hideForCountries = fieldData.getField().getAnnotation(HideForCountries.class); if (hideForCountries == null) { return null; @@ -248,7 +249,7 @@ private static String getIgnoredCountries(FieldData fieldData) { return hideForCountriesString.toString(); } - private static String getExclusiveCountries(FieldData fieldData) { + private String getExclusiveCountries(FieldData fieldData) { HideForCountriesExcept hideForCountriesExcept = fieldData.getField().getAnnotation(HideForCountriesExcept.class); if (hideForCountriesExcept == null) { return null; @@ -263,4 +264,63 @@ private static String getExclusiveCountries(FieldData fieldData) { return hideForCountriesExceptString.toString(); } + + public class EntityColumn { + + private final String name; + private final int width; + private final Function getValueFromField; + private final boolean hasDefaultStyle; + private final boolean isDataDictionaryColumn; + private final boolean isDataProtectionColumn; + private final boolean isColumnForAllFieldsSheet; + + EntityColumn( + String name, + int width, + Function getValueFromField, + boolean hasDefaultStyle, + boolean isDataDictionaryColumn, + boolean isDataProtectionColumn, + boolean isColumnForAllFieldsSheet) { + + this.name = name; + this.width = width; + this.getValueFromField = getValueFromField; + this.hasDefaultStyle = hasDefaultStyle; + this.isDataDictionaryColumn = isDataDictionaryColumn; + this.isDataProtectionColumn = isDataProtectionColumn; + this.isColumnForAllFieldsSheet = isColumnForAllFieldsSheet; + } + + public int getWidth() { + return width; + } + + public String getGetValueFromField(FieldData fieldData) { + return getValueFromField.apply(fieldData); + } + + public boolean hasDefaultStyle() { + return hasDefaultStyle; + } + + public boolean isDataDictionaryColumn() { + return isDataDictionaryColumn; + } + + public boolean isDataProtectionColumn() { + return isDataProtectionColumn; + } + + public boolean isColumnForAllFieldsSheet() { + return isColumnForAllFieldsSheet; + } + + @Override + public String toString() { + return I18nProperties.getPrefixCaption("EntityColumn", name); + } + + } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/info/InfoFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/info/InfoFacadeEjb.java index 44caabb4c59..3e0c1fa8ff5 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/info/InfoFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/info/InfoFacadeEjb.java @@ -27,7 +27,6 @@ import java.util.Collection; import java.util.Collections; import java.util.Date; -import java.util.EnumSet; import java.util.HashMap; import java.util.List; import java.util.Map; @@ -41,7 +40,6 @@ import javax.ejb.Stateless; import org.apache.commons.lang3.reflect.TypeUtils; -import org.apache.poi.openxml4j.exceptions.InvalidFormatException; import org.apache.poi.ss.SpreadsheetVersion; import org.apache.poi.ss.usermodel.CellCopyPolicy; import org.apache.poi.ss.usermodel.CellStyle; @@ -110,6 +108,7 @@ import de.symeda.sormas.backend.common.ConfigFacadeEjb.ConfigFacadeEjbLocal; import de.symeda.sormas.backend.disease.DiseaseConfigurationFacadeEjb.DiseaseConfigurationFacadeEjbLocal; import de.symeda.sormas.backend.feature.FeatureConfigurationFacadeEjb.FeatureConfigurationFacadeEjbLocal; +import de.symeda.sormas.backend.info.EntityColumns.EntityColumn; import de.symeda.sormas.backend.user.UserService; import de.symeda.sormas.backend.util.XssfHelper; @@ -170,35 +169,17 @@ public class InfoFacadeEjb implements InfoFacade { @EJB private FeatureConfigurationFacadeEjbLocal featureConfigurationFacade; - private static EnumSet getColumnsForDataDictionary() { - EnumSet enumSet = EnumSet.allOf(EntityColumn.class); - enumSet.remove(EntityColumn.ENTITY); - return enumSet; - } - @Override public boolean isGenerateDataProtectionDictionaryAllowed() { return userService.hasRight(UserRight.EXPORT_DATA_PROTECTION_DATA) && getDataProtectionFile().exists(); } - private static EnumSet getColumnsForDataProtectionDictionary() { - EnumSet enumSet = EnumSet.allOf(EntityColumn.class); - return enumSet.stream() - .filter(EntityColumn::isDataProtectionColumn) - .collect(Collectors.toCollection(() -> EnumSet.noneOf(EntityColumn.class))); - } - - private static EnumSet getColumnsForAllFieldsSheet(EnumSet enumSet) { - return enumSet.stream() - .filter(column -> !column.isColumnForAllFieldsSheet()) - .collect(Collectors.toCollection(() -> EnumSet.noneOf(EntityColumn.class))); - } - @Override public String generateDataDictionary() throws IOException { + EntityColumns entityColumns = new EntityColumns(configFacade.getCountryLocale(), false); return generateDataDictionary( DATA_DICTIONARY_ENTITIES, - getColumnsForDataDictionary(), + entityColumns, new VisibilityChecks((c, f) -> true, FieldVisibilityCheckers.getNoop()), Collections.emptyList(), Collections.emptyMap(), @@ -218,6 +199,8 @@ public String generateDataProtectionDictionary() throws IOException { .filter(e -> isVisibleByFeatureConfiguration(e.getEntityClass(), featureConfigurations)) .collect(Collectors.toList()); + EntityColumns entityColumns = new EntityColumns(configFacade.getCountryLocale(), true); + try (XSSFWorkbook dataProtectionInputWorkbook = new XSSFWorkbook(getDataProtectionFile())) { XSSFSheet dataProtectionSheet = dataProtectionInputWorkbook.getSheetAt(0); @@ -226,7 +209,7 @@ public String generateDataProtectionDictionary() throws IOException { Map> dataProtectionData = getDataProtectionCellData(dataProtectionSheet); return generateDataDictionary( entities, - getColumnsForDataProtectionDictionary(), + entityColumns, new VisibilityChecks( (dtoClass, field) -> fieldVisibilityCheckers.isVisible(dtoClass, field) && isVisibleByFeatureConfiguration(field.getType(), featureConfigurations), @@ -235,14 +218,15 @@ && isVisibleByFeatureConfiguration(field.getType(), featureConfigurations), dataProtectionData, true); - } catch (InvalidFormatException e) { + } catch (Exception e) { + e.printStackTrace(); throw new IOException(e); } } private String generateDataDictionary( List entities, - EnumSet entityColumns, + EntityColumns entityColumns, VisibilityChecks visibilityChecks, List extraColumns, Map> extraCells, @@ -256,7 +240,7 @@ private String generateDataDictionary( workbook, defaultCellStyle, entities, - entityColumns, + entityColumns.getColumnsForAllFieldsSheet(), visibilityChecks.fieldVisibilityPredicate, extraColumns, extraCells); @@ -266,7 +250,7 @@ private String generateDataDictionary( workbook, defaultCellStyle, entities, - getColumnsForAllFieldsSheet(entityColumns), + entityColumns.getEntityColumns(), visibilityChecks, extraColumns, extraCells, @@ -468,7 +452,7 @@ private void createEntitySheets( XSSFWorkbook workbook, CellStyle defaultCellStyle, List entityInfoList, - EnumSet entityColumns, + List entityColumns, VisibilityChecks visibilityChecks, List extraColumns, Map> extraCells, @@ -491,7 +475,7 @@ private void createAllFieldsSheet( XSSFWorkbook workbook, CellStyle defaultCellStyle, List entityInfoList, - EnumSet entityColumns, + List entityColumns, BiPredicate, Field> fieldVisibilityPredicate, List extraColumns, Map> extraCells) { @@ -529,7 +513,7 @@ private void createEntitySheet( XSSFWorkbook workbook, CellStyle defaultCellStyle, EntityInfo entityInfo, - EnumSet entityColumns, + List entityColumns, VisibilityChecks visibilityChecks, List extraColumns, Map> extraCells, @@ -620,7 +604,7 @@ private XSSFCell createCellsForRow(EntityColumn c, XSSFRow row, FieldData fieldD } private void copyCellsFromExtraCells(XSSFRow row, FieldData fieldData, Map> extraCells) { - String fieldId = EntityColumn.FIELD_ID.getGetValueFromField(fieldData); + String fieldId = EntityColumns.getFieldId(fieldData); if (extraCells.containsKey(fieldId)) { extraCells.get(fieldId).forEach(extraCell -> { XSSFCell newCell = row.createCell(row.getLastCellNum()); @@ -631,7 +615,7 @@ private void copyCellsFromExtraCells(XSSFRow row, FieldData fieldData, Map entityColumns, List extraColumns) { + private void buildHeader(XSSFSheet sheet, List entityColumns, List extraColumns) { XSSFRow headerRow = sheet.createRow(0); entityColumns.forEach(column -> { int colIndex = Math.max(headerRow.getLastCellNum(), 0); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java index bb03f41e445..2413fb29510 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java @@ -348,8 +348,13 @@ private String getCustomDocumentsPath() { public void attachDataProtectionDictionaryDownloader(AbstractComponent target) { new FileDownloader(new StreamResource(() -> new DownloadUtil.DelayedInputStream((out) -> { - String documentPath = FacadeProvider.getInfoFacade().generateDataProtectionDictionary(); - IOUtils.copy(Files.newInputStream(new File(documentPath).toPath()), out); + try { + String documentPath = FacadeProvider.getInfoFacade().generateDataProtectionDictionary(); + IOUtils.copy(Files.newInputStream(new File(documentPath).toPath()), out); + } catch (Exception e) { + e.printStackTrace(); + } + }, (e) -> { }), DownloadUtil.createFileNameWithCurrentDate(ExportEntityName.DATA_PROTECTION_DICTIONARY, ".xlsx"))).extend(target); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java index 018be86c216..69a354b2339 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/action/ActionEditForm.java @@ -41,7 +41,6 @@ import de.symeda.sormas.ui.utils.DateTimeField; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; -import de.symeda.sormas.ui.utils.RichTextAreaCustom; import de.symeda.sormas.ui.utils.VaadinUiUtil; public class ActionEditForm extends AbstractEditForm { @@ -101,7 +100,7 @@ protected void addFields() { addField(ActionDto.ACTION_MEASURE, TextField.class); TextField title = addField(ActionDto.TITLE, TextField.class); title.addStyleName(SOFT_REQUIRED); - RichTextAreaCustom description = addField(ActionDto.DESCRIPTION, RichTextAreaCustom.class); + RichTextArea description = addField(ActionDto.DESCRIPTION, RichTextArea.class); description.setNullRepresentation(""); description.setImmediate(true); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java index cad1c6f6615..944c89a10e9 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java @@ -85,7 +85,7 @@ protected void addFields() { healthConditionsHeadingLabel.addStyleName(H3); getContent().addComponent(healthConditionsHeadingLabel, HEALTH_CONDITIONS_HEADINGS_LOC); - if (UiUtil.permitted(UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION)) { + if (displayFieldsAllowed()) { addFields( TUBERCULOSIS, ASPLENIA, @@ -137,4 +137,8 @@ protected F addFieldToLayout(CustomLayout layout, String prope return super.addFieldToLayout(layout, propertyId, field); } + + private boolean displayFieldsAllowed() { + return UiUtil.permitted(UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION); + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java deleted file mode 100644 index 8f26e2c24b9..00000000000 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/RichTextAreaCustom.java +++ /dev/null @@ -1,24 +0,0 @@ -package de.symeda.sormas.ui.utils; - -import com.vaadin.event.FieldEvents; -import com.vaadin.v7.ui.RichTextArea; - -public class RichTextAreaCustom extends RichTextArea { - - public void addFocusListener(FieldEvents.FocusListener listener) { - addListener(FieldEvents.FocusEvent.EVENT_ID, FieldEvents.FocusEvent.class, listener, FieldEvents.FocusListener.focusMethod); - } - - public void removeFocusListener(FieldEvents.FocusListener listener) { - removeListener(FieldEvents.FocusEvent.EVENT_ID, FieldEvents.FocusEvent.class, listener); - - } - - public void addBlurListener(FieldEvents.BlurListener listener) { - addListener(FieldEvents.BlurEvent.EVENT_ID, FieldEvents.BlurEvent.class, listener, FieldEvents.BlurListener.blurMethod); - } - - public void removeBlurListener(FieldEvents.BlurListener listener) { - removeListener(FieldEvents.BlurEvent.EVENT_ID, FieldEvents.BlurEvent.class, listener); - } -} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java index d8109192d5e..ac2b992d67e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java @@ -15,6 +15,7 @@ import com.vaadin.v7.ui.DateField; import com.vaadin.v7.ui.Field; import com.vaadin.v7.ui.OptionGroup; +import com.vaadin.v7.ui.RichTextArea; import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.FacadeProvider; @@ -179,8 +180,8 @@ public T createField(Class type, Class fieldType) { return (T) new UserField(); } else if (CheckBoxTree.class.isAssignableFrom(fieldType)) { return (T) new CheckBoxTree<>(); - } else if (RichTextAreaCustom.class.isAssignableFrom(fieldType)) { - return (T) new RichTextAreaCustom(); + } else if (RichTextArea.class.isAssignableFrom(fieldType)) { + return (T) new RichTextArea(); } return super.createField(type, fieldType); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java index 84509ae6295..6f53c037eb8 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/VaadinUiUtil.java @@ -46,6 +46,7 @@ import com.vaadin.v7.data.util.PropertyValueGenerator; import com.vaadin.v7.ui.AbstractField; import com.vaadin.v7.ui.Grid; +import com.vaadin.v7.ui.RichTextArea; import com.vaadin.v7.ui.TextArea; import com.vaadin.v7.ui.renderers.HtmlRenderer; @@ -652,7 +653,7 @@ public static void addGdprMessageOnClick(TextArea textArea) { }); } - public static void addGdprMessageOnClick(RichTextAreaCustom richTextArea) { + public static void addGdprMessageOnClick(RichTextArea richTextArea) { AtomicBoolean gdprMessageTriggered = new AtomicBoolean(false); Window subWindowGdpR = VaadinUiUtil.createPopupWindow(); From b74ceb5c11224a963869e00d73213ceed1f17b28 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Tue, 29 Oct 2024 15:59:18 +0200 Subject: [PATCH 19/56] #13159 Automatically (Soft-)Delete Samples & Pathogen Tests with Negative Test Results for COVID-19 - rename config property --- .../src/main/java/de/symeda/sormas/api/ConfigFacade.java | 2 +- .../de/symeda/sormas/backend/common/ConfigFacadeEjb.java | 6 +++--- .../java/de/symeda/sormas/backend/sample/SampleService.java | 2 +- .../sample/DeleteOldPathogenTestsAndSamplesTest.java | 4 ++-- sormas-base/setup/sormas.properties | 2 +- 5 files changed, 8 insertions(+), 8 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java index 496c696ef02..415200b29e2 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/ConfigFacade.java @@ -162,5 +162,5 @@ public interface ConfigFacade { boolean isAnyCaseClassificationCalculationEnabled(); - Integer getNegaiveCovidSamplesMaxAgeDays(); + Integer getNegaiveCovidTestsMaxAgeDays(); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java index 082182554ce..96ccbf3388a 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/common/ConfigFacadeEjb.java @@ -187,7 +187,7 @@ public class ConfigFacadeEjb implements ConfigFacade { public static final int DEFAULT_DOCUMENT_UPLOAD_SIZE_LIMIT_MB = 20; public static final String IMPORT_FILE_SIZE_LIMIT_MB = "importFileSizeLimitMb"; public static final int DEFAULT_IMPOR_FILE_SIZE_LIMIT_MB = 20; - public static final String NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS = "negaiveCovidSamplesMaxAgeDays"; + public static final String NEGATIVE_COVID_TESTS_MAX_AGE_DAYS = "negativeCovidTestsMaxAgeDays"; private final Logger logger = LoggerFactory.getLogger(getClass()); @@ -840,8 +840,8 @@ public void resetRequestContext() { } @Override - public Integer getNegaiveCovidSamplesMaxAgeDays() { - return parseProperty(NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS, null, Integer::parseInt); + public Integer getNegaiveCovidTestsMaxAgeDays() { + return parseProperty(NEGATIVE_COVID_TESTS_MAX_AGE_DAYS, null, Integer::parseInt); } @LocalBean diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java index 8bb70bb1994..c2f06d0ef7f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java @@ -1286,7 +1286,7 @@ public List getAssociatedDiseaseVariants(String sampleUuid) { } public void cleanupOldCovidSamples() { - final Integer maxAgeDays = configFacade.getNegaiveCovidSamplesMaxAgeDays(); + final Integer maxAgeDays = configFacade.getNegaiveCovidTestsMaxAgeDays(); if (maxAgeDays == null) { return; } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java index a5d70e9b207..7ef341689c4 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/sample/DeleteOldPathogenTestsAndSamplesTest.java @@ -54,7 +54,7 @@ public void init() { user = creator.createUser(rdcf).toReference(); caze = creator.createCase(user, creator.createPerson().toReference(), rdcf); - MockProducer.getProperties().setProperty(ConfigFacadeEjb.NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS, String.valueOf(negativeCovidTestMaxAge)); + MockProducer.getProperties().setProperty(ConfigFacadeEjb.NEGATIVE_COVID_TESTS_MAX_AGE_DAYS, String.valueOf(negativeCovidTestMaxAge)); } @Test @@ -242,7 +242,7 @@ public void testDeletionReferenceDate() { @Test public void testNotConfigured() { - MockProducer.getProperties().remove(ConfigFacadeEjb.NEGAIVE_COVID_SAMPLES_MAX_AGE_DAYS); + MockProducer.getProperties().remove(ConfigFacadeEjb.NEGATIVE_COVID_TESTS_MAX_AGE_DAYS); SampleDto sample = creator.createSample(caze.toReference(), user, rdcf.facility, s -> { s.setPathogenTestResult(PathogenTestResultType.NEGATIVE); diff --git a/sormas-base/setup/sormas.properties b/sormas-base/setup/sormas.properties index 287bf8b66f5..cbf7653b9e9 100644 --- a/sormas-base/setup/sormas.properties +++ b/sormas-base/setup/sormas.properties @@ -143,7 +143,7 @@ app.url= # Number of days after which negative CORONAVIRUS pathogen tests and their samples are soft deleted # default: not set, meaning no delete # possible values: any integer representing the number of days -# negativeCovidSamplesMaxAgeDays +# negativeCovidTestsMaxAgeDays # The similarity threshold after which two names are identified as similar enough to consider them for duplicate detection. # The default value should work for most servers. If you need to change it, please change it carefully as slightly higher or lower values already lead to significant differences. From 0591b1103f399db8274225e7058277f069606911 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Tue, 29 Oct 2024 17:51:16 +0200 Subject: [PATCH 20/56] #13093 - Update Data Protection for certain Data Fields - changes after review --- .../java/de/symeda/sormas/ui/AboutView.java | 9 +- .../symeda/sormas/ui/caze/CaseDataForm.java | 4 +- .../clinicalcourse/HealthConditionsForm.java | 93 ++++++++++--------- .../sormas/ui/utils/AbstractEditForm.java | 3 + 4 files changed, 54 insertions(+), 55 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java index 2413fb29510..bb03f41e445 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/AboutView.java @@ -348,13 +348,8 @@ private String getCustomDocumentsPath() { public void attachDataProtectionDictionaryDownloader(AbstractComponent target) { new FileDownloader(new StreamResource(() -> new DownloadUtil.DelayedInputStream((out) -> { - try { - String documentPath = FacadeProvider.getInfoFacade().generateDataProtectionDictionary(); - IOUtils.copy(Files.newInputStream(new File(documentPath).toPath()), out); - } catch (Exception e) { - e.printStackTrace(); - } - + String documentPath = FacadeProvider.getInfoFacade().generateDataProtectionDictionary(); + IOUtils.copy(Files.newInputStream(new File(documentPath).toPath()), out); }, (e) -> { }), DownloadUtil.createFileNameWithCurrentDate(ExportEntityName.DATA_PROTECTION_DICTIONARY, ".xlsx"))).extend(target); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java index b2f181ee080..f17aa0c7c42 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java @@ -1022,6 +1022,8 @@ protected void addFields() { } }); + addField(CaseDataDto.HEALTH_CONDITIONS, HealthConditionsForm.class).setCaption(null); + // Set initial visibilities & accesses initializeVisibilitiesAndAllowedVisibilities(); initializeAccessAndAllowedAccesses(); @@ -1206,8 +1208,6 @@ protected void addFields() { List medicalInformationFields = Arrays.asList(CaseDataDto.PREGNANT, CaseDataDto.VACCINATION_STATUS, CaseDataDto.SMALLPOX_VACCINATION_RECEIVED); - addField(CaseDataDto.HEALTH_CONDITIONS, HealthConditionsForm.class).setCaption(null); - for (String medicalInformationField : medicalInformationFields) { if (getFieldGroup().getField(medicalInformationField).isVisible()) { Label medicalInformationCaptionLabel = new Label(I18nProperties.getString(Strings.headingMedicalInformation)); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java index 944c89a10e9..c99568bef0b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/clinicalcourse/HealthConditionsForm.java @@ -31,6 +31,7 @@ import static de.symeda.sormas.ui.utils.LayoutUtil.locs; import java.util.Arrays; +import java.util.List; import com.vaadin.ui.CustomLayout; import com.vaadin.ui.Label; @@ -42,11 +43,9 @@ import de.symeda.sormas.api.i18n.Descriptions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; -import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.FieldHelper; @@ -56,7 +55,7 @@ public class HealthConditionsForm extends AbstractEditForm private static final long serialVersionUID = 1L; private static final String HEALTH_CONDITIONS_HEADINGS_LOC = "healthConditionsHeadingLoc"; - private static final String CONFIDENTIAL_LABEL = "confidentialLabel"; + private static final String CONFIDENTIAL_LABEL_LOC = "confidentialLabel"; //@formatter:off private static final String HTML_LAYOUT = @@ -70,10 +69,33 @@ public class HealthConditionsForm extends AbstractEditForm CHRONIC_HEART_FAILURE, CHRONIC_PULMONARY_DISEASE, CHRONIC_KIDNEY_DISEASE, CHRONIC_NEUROLOGIC_CONDITION, CARDIOVASCULAR_DISEASE_INCLUDING_HYPERTENSION, OBESITY, CURRENT_SMOKER, FORMER_SMOKER, ASTHMA, SICKLE_CELL_DISEASE)) - ) + - loc(OTHER_CONDITIONS) + loc(CONFIDENTIAL_LABEL); + ) + loc(OTHER_CONDITIONS) + loc(CONFIDENTIAL_LABEL_LOC); //@formatter:on + private static final List fieldsList = List.of( + TUBERCULOSIS, + ASPLENIA, + HEPATITIS, + DIABETES, + HIV, + HIV_ART, + CHRONIC_LIVER_DISEASE, + MALIGNANCY_CHEMOTHERAPY, + CHRONIC_HEART_FAILURE, + CHRONIC_PULMONARY_DISEASE, + CHRONIC_KIDNEY_DISEASE, + CHRONIC_NEUROLOGIC_CONDITION, + DOWN_SYNDROME, + CONGENITAL_SYPHILIS, + IMMUNODEFICIENCY_OTHER_THAN_HIV, + CARDIOVASCULAR_DISEASE_INCLUDING_HYPERTENSION, + OBESITY, + CURRENT_SMOKER, + FORMER_SMOKER, + ASTHMA, + SICKLE_CELL_DISEASE, + IMMUNODEFICIENCY_INCLUDING_HIV); + public HealthConditionsForm(FieldVisibilityCheckers fieldVisibilityCheckers, UiFieldAccessCheckers fieldAccessCheckers) { super(HealthConditionsDto.class, I18N_PREFIX, true, fieldVisibilityCheckers, fieldAccessCheckers); } @@ -85,45 +107,18 @@ protected void addFields() { healthConditionsHeadingLabel.addStyleName(H3); getContent().addComponent(healthConditionsHeadingLabel, HEALTH_CONDITIONS_HEADINGS_LOC); - if (displayFieldsAllowed()) { - addFields( - TUBERCULOSIS, - ASPLENIA, - HEPATITIS, - DIABETES, - HIV, - HIV_ART, - CHRONIC_LIVER_DISEASE, - MALIGNANCY_CHEMOTHERAPY, - CHRONIC_HEART_FAILURE, - CHRONIC_PULMONARY_DISEASE, - CHRONIC_KIDNEY_DISEASE, - CHRONIC_NEUROLOGIC_CONDITION, - DOWN_SYNDROME, - CONGENITAL_SYPHILIS, - IMMUNODEFICIENCY_OTHER_THAN_HIV, - CARDIOVASCULAR_DISEASE_INCLUDING_HYPERTENSION, - OBESITY, - CURRENT_SMOKER, - FORMER_SMOKER, - ASTHMA, - SICKLE_CELL_DISEASE, - IMMUNODEFICIENCY_INCLUDING_HIV); - TextArea otherConditions = addField(OTHER_CONDITIONS, TextArea.class); - otherConditions.setRows(6); - otherConditions.setDescription( - I18nProperties.getPrefixDescription(HealthConditionsDto.I18N_PREFIX, OTHER_CONDITIONS, "") + "\n" - + I18nProperties.getDescription(Descriptions.descGdpr)); - - initializeVisibilitiesAndAllowedVisibilities(); - initializeAccessAndAllowedAccesses(); - - FieldHelper.setVisibleWhen(getFieldGroup(), HIV_ART, HIV, Arrays.asList(YesNoUnknown.YES), true); - } else { - Label confidentialLabel = new Label(I18nProperties.getCaption(Captions.inaccessibleValue)); - confidentialLabel.addStyleName(CssStyles.INACCESSIBLE_LABEL); - getContent().addComponent(confidentialLabel, CONFIDENTIAL_LABEL); - } + addFields(fieldsList); + + TextArea otherConditions = addField(OTHER_CONDITIONS, TextArea.class); + otherConditions.setRows(6); + otherConditions.setDescription( + I18nProperties.getPrefixDescription(HealthConditionsDto.I18N_PREFIX, OTHER_CONDITIONS, "") + "\n" + + I18nProperties.getDescription(Descriptions.descGdpr)); + + FieldHelper.setVisibleWhen(getFieldGroup(), HIV_ART, HIV, Arrays.asList(YesNoUnknown.YES), true); + + initializeVisibilitiesAndAllowedVisibilities(); + initializeAccessAndAllowedAccesses(); } @Override @@ -138,7 +133,13 @@ protected F addFieldToLayout(CustomLayout layout, String prope return super.addFieldToLayout(layout, propertyId, field); } - private boolean displayFieldsAllowed() { - return UiUtil.permitted(UserRight.SEE_SENSITIVE_DATA_IN_JURISDICTION); + public void setInaccessible() { + fieldsList.stream().forEach(field -> { + getContent().getComponent(field).setVisible(false); + }); + getContent().getComponent(OTHER_CONDITIONS).setVisible(false); + Label confidentialLabel = new Label(I18nProperties.getCaption(Captions.inaccessibleValue)); + confidentialLabel.addStyleName(CssStyles.INACCESSIBLE_LABEL); + getContent().addComponent(confidentialLabel, CONFIDENTIAL_LABEL_LOC); } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java index 1162f807383..d5db8ba4ae1 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java @@ -45,6 +45,7 @@ import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; +import de.symeda.sormas.ui.clinicalcourse.HealthConditionsForm; import de.symeda.sormas.ui.utils.components.NotBlankTextValidator; public abstract class AbstractEditForm extends AbstractForm implements FieldGroup.CommitHandler {// implements DtoEditForm { @@ -568,6 +569,8 @@ protected void initializeAccessAndAllowedAccesses() { if (field instanceof NullableOptionGroup) { ((NullableOptionGroup) field).setInaccessible(); } + if (field instanceof HealthConditionsForm) + ((HealthConditionsForm) field).setInaccessible(); } } From f86513308da9cedcfeb203252a729ca06dd29156 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Wed, 30 Oct 2024 09:56:56 +0200 Subject: [PATCH 21/56] #13160 Add "Disease" Attribute to Document Templates for Filtering - re-generated captions --- .../src/main/java/de/symeda/sormas/api/i18n/Captions.java | 2 ++ 1 file changed, 2 insertions(+) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index 233647e5b4f..e97958091c2 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -1394,6 +1394,7 @@ public interface Captions { String documentNoDocuments = "documentNoDocuments"; String DocumentTemplate = "DocumentTemplate"; String DocumentTemplate_buttonUploadTemplate = "DocumentTemplate.buttonUploadTemplate"; + String DocumentTemplate_disease = "DocumentTemplate.disease"; String DocumentTemplate_documentTemplateGuide = "DocumentTemplate.documentTemplateGuide"; String DocumentTemplate_documentUploadWarning = "DocumentTemplate.documentUploadWarning"; String DocumentTemplate_EventHandout = "DocumentTemplate.EventHandout"; @@ -1407,6 +1408,7 @@ public interface Captions { String DocumentTemplate_exampleTemplateEventParticipants = "DocumentTemplate.exampleTemplateEventParticipants"; String DocumentTemplate_exampleTemplateTravelEntries = "DocumentTemplate.exampleTemplateTravelEntries"; String DocumentTemplate_exampleTemplateTravelEntryEmail = "DocumentTemplate.exampleTemplateTravelEntryEmail"; + String DocumentTemplate_fileName = "DocumentTemplate.fileName"; String DocumentTemplate_fileTooBig = "DocumentTemplate.fileTooBig"; String DocumentTemplate_notUploaded = "DocumentTemplate.notUploaded"; String DocumentTemplate_plural = "DocumentTemplate.plural"; From 7fe877008a9df56b773fe9dd8337340ef9d89403 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Wed, 30 Oct 2024 10:37:25 +0200 Subject: [PATCH 22/56] #13093 - Update Data Protection for certain Data Fields - fix after merge with development --- .../AefiInvestigationDataForm.java | 3 ++- .../components/form/AefiDataForm.java | 5 ++--- .../components/information/AefiPersonInfo.java | 11 ++--------- 3 files changed, 6 insertions(+), 13 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java index 1ba7beb0475..416e6fba5a4 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/AefiInvestigationDataForm.java @@ -60,6 +60,7 @@ import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.UserField; @@ -294,7 +295,7 @@ public AefiInvestigationDataForm(boolean isCreateAction, boolean isPseudonymized AefiInvestigationDto.I18N_PREFIX, false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UserProvider.getCurrent().getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.isCreateAction = isCreateAction; this.actionCallback = actionCallback; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java index 78ee9dcb888..4b1411c561a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/form/AefiDataForm.java @@ -47,14 +47,13 @@ import de.symeda.sormas.api.i18n.Validations; import de.symeda.sormas.api.immunization.ImmunizationDto; import de.symeda.sormas.api.utils.YesNoUnknown; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; -import de.symeda.sormas.ui.UserProvider; import de.symeda.sormas.ui.adverseeventsfollowingimmunization.components.fields.vaccines.AefiVaccinationsField; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.ButtonHelper; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.DateTimeField; +import de.symeda.sormas.ui.utils.FieldAccessHelper; import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.NullableOptionGroup; import de.symeda.sormas.ui.utils.UserField; @@ -127,7 +126,7 @@ public AefiDataForm(boolean isCreateAction, boolean isPseudonymized, boolean inJ AefiDto.I18N_PREFIX, false, FieldVisibilityCheckers.withCountry(FacadeProvider.getConfigFacade().getCountryLocale()), - UiFieldAccessCheckers.forDataAccessLevel(UserProvider.getCurrent().getPseudonymizableDataAccessLevel(inJurisdiction), isPseudonymized)); + FieldAccessHelper.getFieldAccessCheckers(inJurisdiction, isPseudonymized)); this.isCreateAction = isCreateAction; this.actionCallback = actionCallback; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java index 3f378ccf0b0..c8bc7373e0a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/adverseeventsfollowingimmunization/components/information/AefiPersonInfo.java @@ -1,17 +1,14 @@ /* * SORMAS® - Surveillance Outbreak Response Management & Analysis System * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) - * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. - * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. - * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ @@ -31,10 +28,10 @@ import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.user.UserRight; import de.symeda.sormas.api.utils.DataHelper; -import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.ui.AbstractInfoLayout; import de.symeda.sormas.ui.UserProvider; import de.symeda.sormas.ui.utils.CssStyles; +import de.symeda.sormas.ui.utils.FieldAccessHelper; @SuppressWarnings("serial") public class AefiPersonInfo extends AbstractInfoLayout { @@ -46,11 +43,7 @@ public class AefiPersonInfo extends AbstractInfoLayout { private Disease disease; public AefiPersonInfo(PersonDto personDto, Disease disease) { - super( - PersonDto.class, - UiFieldAccessCheckers.forDataAccessLevel( - UserProvider.getCurrent().getPseudonymizableDataAccessLevel(personDto.isInJurisdiction()), - personDto.isPseudonymized())); + super(PersonDto.class, FieldAccessHelper.getFieldAccessCheckers(personDto)); this.personDto = personDto; this.disease = disease; From 7e54cf8e8383d46d9ab5678a4f6a4cb6815fa64e Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Wed, 30 Oct 2024 11:00:27 +0200 Subject: [PATCH 23/56] #13160 Add "Disease" Attribute to Document Templates for Filtering - fixed failing tests --- .../backend/docgeneration/DocumentTemplateFacadeEjb.java | 1 + .../backend/importexport/DatabaseExportServiceTest.java | 4 +++- 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java index 8b592fb5d46..79f2dd24828 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/docgeneration/DocumentTemplateFacadeEjb.java @@ -526,6 +526,7 @@ private static EmailTemplateTexts splitTemplateContent(String templateString, bo return new EmailTemplateTexts(cleanupSubject ? subjectLine.substring(1).trim() : subjectLine, content); } + @PermitAll public DocumentTemplateDto getByUuid(String uuid) { return toDto(documentTemplateService.getByUuid(uuid)); } diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java index e6b42ec468c..0746186095b 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/importexport/DatabaseExportServiceTest.java @@ -26,6 +26,7 @@ import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AdverseEvents; import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.Aefi; import de.symeda.sormas.backend.adverseeventsfollowingimmunization.entity.AefiInvestigation; +import de.symeda.sormas.backend.docgeneration.DocumentTemplate; import de.symeda.sormas.backend.environment.Environment; import de.symeda.sormas.backend.environment.environmentsample.EnvironmentSample; import de.symeda.sormas.backend.immunization.entity.DirectoryImmunization; @@ -70,7 +71,8 @@ public void testGetConfigFullyDefined() { SelfReport.class, Aefi.class, AdverseEvents.class, - AefiInvestigation.class); + AefiInvestigation.class, + DocumentTemplate.class); @Test public void test_all_entities_have_export_configuration() { From e777733fc22a04e50ccab2e46066d54dbe79f03a Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Wed, 30 Oct 2024 12:57:16 +0200 Subject: [PATCH 24/56] #13093 - Update Data Protection for certain Data Fields - fix issues on Android app --- .../sormas/app/campaign/edit/CampaignFormMetaDialog.java | 3 ++- .../de/symeda/sormas/app/caze/edit/CaseEditFragment.java | 2 +- .../app/caze/edit/CaseEditHealthConditionsFragment.java | 2 +- .../app/caze/edit/CaseEditHospitalizationFragment.java | 3 ++- .../app/caze/edit/CaseEditMaternalHistoryFragment.java | 3 ++- .../app/caze/edit/PreviousHospitalizationDialog.java | 3 ++- .../de/symeda/sormas/app/caze/read/CaseReadFragment.java | 2 +- .../app/caze/read/CaseReadHospitalizationFragment.java | 3 ++- .../clinicalcourse/edit/ClinicalVisitEditFragment.java | 3 ++- .../sormas/app/contact/edit/ContactEditFragment.java | 2 +- .../app/environment/read/EnvironmentReadFragment.java | 2 +- .../edit/EnvironmentSampleEditFragment.java | 2 +- .../symeda/sormas/app/epidata/ActivityAsCaseDialog.java | 2 +- .../app/epidata/EpidemiologicalDataEditFragment.java | 3 ++- .../app/epidata/EpidemiologicalDataReadFragment.java | 8 ++++---- .../java/de/symeda/sormas/app/epidata/ExposureDialog.java | 2 +- .../symeda/sormas/app/event/edit/EventEditFragment.java | 2 +- .../edit/EventParticipantEditFragment.java | 3 ++- .../read/EventParticipantReadFragment.java | 3 ++- .../symeda/sormas/app/event/read/EventReadFragment.java | 2 +- .../app/immunization/edit/ImmunizationEditFragment.java | 2 +- .../app/immunization/read/ImmunizationReadFragment.java | 2 +- .../immunization/vaccination/VaccinationEditFragment.java | 2 +- .../VaccinationEditHealthConditionsFragment.java | 2 +- .../app/pathogentest/edit/PathogenTestEditFragment.java | 2 +- .../app/pathogentest/read/PathogenTestReadFragment.java | 2 +- .../symeda/sormas/app/person/edit/PersonEditFragment.java | 8 ++++---- .../symeda/sormas/app/person/read/PersonReadFragment.java | 8 ++++---- .../symeda/sormas/app/sample/edit/SampleEditFragment.java | 2 +- .../symeda/sormas/app/symptoms/SymptomsEditFragment.java | 4 ++-- .../sormas/app/therapy/edit/PrescriptionEditFragment.java | 2 +- .../sormas/app/therapy/edit/TreatmentEditFragment.java | 3 ++- .../symeda/sormas/app/visit/edit/VisitEditFragment.java | 3 ++- 33 files changed, 54 insertions(+), 43 deletions(-) diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/campaign/edit/CampaignFormMetaDialog.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/campaign/edit/CampaignFormMetaDialog.java index 29673179c3b..27e92f75b31 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/campaign/edit/CampaignFormMetaDialog.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/campaign/edit/CampaignFormMetaDialog.java @@ -30,6 +30,7 @@ import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.campaign.Campaign; import de.symeda.sormas.app.backend.campaign.form.CampaignFormMeta; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.component.dialog.FormDialog; import de.symeda.sormas.app.component.validation.FragmentValidator; import de.symeda.sormas.app.core.notification.NotificationHelper; @@ -51,7 +52,7 @@ public CampaignFormMetaDialog(final FragmentActivity activity, Campaign campaign R.layout.dialog_root_two_button_panel_layout, R.string.heading_campaign_form_meta_select, -1, - UiFieldAccessCheckers.forSensitiveData(campaign.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(campaign.isPseudonymized(), ConfigProvider.getServerCountryCode())); this.campaign = campaign; } diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditFragment.java index a745274ebc9..7824dbb0fb1 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditFragment.java @@ -130,7 +130,7 @@ public static CaseEditFragment newInstance(Case activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); caseEditFragment.differentPlaceOfStayJurisdiction = activityRootData.getRegion() != null || activityRootData.getDistrict() != null || activityRootData.getCommunity() != null; diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHealthConditionsFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHealthConditionsFragment.java index ce6f13e5bfa..3e44e0f4300 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHealthConditionsFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHealthConditionsFragment.java @@ -33,7 +33,7 @@ public static CaseEditHealthConditionsFragment newInstance(Case activityRootData activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } @Override diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHospitalizationFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHospitalizationFragment.java index 40412340395..4b6aa3afcb1 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHospitalizationFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditHospitalizationFragment.java @@ -33,6 +33,7 @@ import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.caze.Case; import de.symeda.sormas.app.backend.common.DatabaseHelper; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.hospitalization.Hospitalization; import de.symeda.sormas.app.backend.hospitalization.PreviousHospitalization; import de.symeda.sormas.app.component.Item; @@ -57,7 +58,7 @@ public static CaseEditHospitalizationFragment newInstance(Case activityRootData) null, activityRootData, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Instance methods diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditMaternalHistoryFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditMaternalHistoryFragment.java index 68fdfac57af..da101799ed6 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditMaternalHistoryFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/CaseEditMaternalHistoryFragment.java @@ -25,6 +25,7 @@ import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.caze.Case; import de.symeda.sormas.app.backend.caze.maternalhistory.MaternalHistory; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.component.Item; import de.symeda.sormas.app.databinding.FragmentCaseEditMaternalHistoryLayoutBinding; import de.symeda.sormas.app.util.InfrastructureDaoHelper; @@ -44,7 +45,7 @@ public static CaseEditMaternalHistoryFragment newInstance(Case activityRootData) null, activityRootData, null, - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Overrides diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/PreviousHospitalizationDialog.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/PreviousHospitalizationDialog.java index e7eba2bdd69..affbc8e7b06 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/PreviousHospitalizationDialog.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/edit/PreviousHospitalizationDialog.java @@ -30,6 +30,7 @@ import de.symeda.sormas.api.utils.ValidationException; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.app.R; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.hospitalization.PreviousHospitalization; import de.symeda.sormas.app.backend.infrastructure.InfrastructureHelper; import de.symeda.sormas.app.component.Item; @@ -60,7 +61,7 @@ public class PreviousHospitalizationDialog extends FormDialog { R.layout.dialog_root_three_button_panel_layout, R.string.heading_previous_hospitalization, -1, - UiFieldAccessCheckers.forSensitiveData(previousHospitalization.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(previousHospitalization.isPseudonymized(), ConfigProvider.getServerCountryCode())); this.data = previousHospitalization; diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadFragment.java index 077d9d9aaa3..4bd14c8f7d8 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadFragment.java @@ -61,7 +61,7 @@ public static CaseReadFragment newInstance(Case activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); return caseReadFragment; } diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadHospitalizationFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadHospitalizationFragment.java index 4564ed6e324..979c4da09d7 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadHospitalizationFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/caze/read/CaseReadHospitalizationFragment.java @@ -28,6 +28,7 @@ import de.symeda.sormas.app.BaseReadFragment; import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.caze.Case; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.hospitalization.Hospitalization; import de.symeda.sormas.app.backend.hospitalization.PreviousHospitalization; import de.symeda.sormas.app.databinding.FragmentCaseReadHospitalizationLayoutBinding; @@ -48,7 +49,7 @@ public static CaseReadHospitalizationFragment newInstance(Case activityRootData) null, activityRootData, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Overrides diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/clinicalcourse/edit/ClinicalVisitEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/clinicalcourse/edit/ClinicalVisitEditFragment.java index 08890964398..14f2cdd48d9 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/clinicalcourse/edit/ClinicalVisitEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/clinicalcourse/edit/ClinicalVisitEditFragment.java @@ -20,6 +20,7 @@ import de.symeda.sormas.app.BaseEditFragment; import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.clinicalcourse.ClinicalVisit; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.databinding.FragmentClinicalVisitEditLayoutBinding; public class ClinicalVisitEditFragment extends BaseEditFragment { @@ -32,7 +33,7 @@ public static ClinicalVisitEditFragment newInstance(ClinicalVisit activityRootDa null, activityRootData, null, - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } @Override diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/contact/edit/ContactEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/contact/edit/ContactEditFragment.java index aa7562c8386..25cc150e405 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/contact/edit/ContactEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/contact/edit/ContactEditFragment.java @@ -85,7 +85,7 @@ public static ContactEditFragment newInstance(Contact activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } private void setUpControlListeners(FragmentContactEditLayoutBinding contentBinding) { diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/environment/read/EnvironmentReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/environment/read/EnvironmentReadFragment.java index e867e084bf2..2b8552be226 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/environment/read/EnvironmentReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/environment/read/EnvironmentReadFragment.java @@ -27,7 +27,7 @@ public static EnvironmentReadFragment newInstance(Environment activityRootData) null, activityRootData, FieldVisibilityCheckers.getNoop(), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); return environmentReadFragment; } diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/environmentsample/edit/EnvironmentSampleEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/environmentsample/edit/EnvironmentSampleEditFragment.java index 43962036f0a..62779c16cfe 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/environmentsample/edit/EnvironmentSampleEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/environmentsample/edit/EnvironmentSampleEditFragment.java @@ -68,7 +68,7 @@ public static EnvironmentSampleEditFragment newInstance(EnvironmentSample activi null, activityRootData, FieldVisibilityCheckers.withCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized()), + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode()), UserRight.ENVIRONMENT_SAMPLE_EDIT); } diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ActivityAsCaseDialog.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ActivityAsCaseDialog.java index b8cad52e02e..3677e6d4042 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ActivityAsCaseDialog.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ActivityAsCaseDialog.java @@ -60,7 +60,7 @@ public class ActivityAsCaseDialog extends FormDialog { R.string.heading_activityAsCase, -1, false, - UiFieldAccessCheckers.forSensitiveData(activityAsCase.isPseudonymized()), + UiFieldAccessCheckers.forSensitiveData(activityAsCase.isPseudonymized(), ConfigProvider.getServerCountryCode()), FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(activityRootData)).andWithCountry(ConfigProvider.getServerCountryCode())); this.data = activityAsCase; diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataEditFragment.java index d07c6db21d6..c4a422a7db1 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataEditFragment.java @@ -41,6 +41,7 @@ import de.symeda.sormas.app.backend.caze.Case; import de.symeda.sormas.app.backend.common.DatabaseHelper; import de.symeda.sormas.app.backend.common.PseudonymizableAdo; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.contact.Contact; import de.symeda.sormas.app.backend.epidata.EpiData; import de.symeda.sormas.app.backend.exposure.Exposure; @@ -65,7 +66,7 @@ public static EpidemiologicalDataEditFragment newInstance(PseudonymizableAdo act null, activityRootData, FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(activityRootData)), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } private void setUpControlListeners(final FragmentEditEpidLayoutBinding contentBinding) { diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataReadFragment.java index bf56e2d4d13..8dd77dc8803 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/EpidemiologicalDataReadFragment.java @@ -69,7 +69,7 @@ public static EpidemiologicalDataReadFragment newInstance(Case activityRootData) null, activityRootData, FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(activityRootData)).andWithCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static EpidemiologicalDataReadFragment newInstance(Contact activityRootData) { @@ -78,7 +78,7 @@ public static EpidemiologicalDataReadFragment newInstance(Contact activityRootDa null, activityRootData, FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(activityRootData)).andWithCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } private void setUpControlListeners() { @@ -107,7 +107,7 @@ private void setUpControlListeners() { ExposureDto.class, (ViewGroup) infoDialog.getBinding().getRoot(), FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(getActivityRootData())), - UiFieldAccessCheckers.forSensitiveData(((PseudonymizableAdo) getActivityRootData()).isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(((PseudonymizableAdo) getActivityRootData()).isPseudonymized(), ConfigProvider.getServerCountryCode())); infoDialog.show(); }; @@ -141,7 +141,7 @@ private void setUpControlListeners() { (ViewGroup) infoDialog.getBinding().getRoot(), FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(getActivityRootData())) .andWithCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(((PseudonymizableAdo) getActivityRootData()).isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(((PseudonymizableAdo) getActivityRootData()).isPseudonymized(), ConfigProvider.getServerCountryCode())); infoDialog.show(); }; diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ExposureDialog.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ExposureDialog.java index b7125090159..69e91abd80c 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ExposureDialog.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/epidata/ExposureDialog.java @@ -58,7 +58,7 @@ public class ExposureDialog extends FormDialog { R.string.heading_exposure, -1, false, - UiFieldAccessCheckers.forSensitiveData(exposure.isPseudonymized()), + UiFieldAccessCheckers.forSensitiveData(exposure.isPseudonymized(), ConfigProvider.getServerCountryCode()), FieldVisibilityCheckers.withDisease(getDiseaseOfCaseOrContact(activityRootData)).andWithCountry(ConfigProvider.getServerCountryCode())); this.data = exposure; diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/edit/EventEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/edit/EventEditFragment.java index 846c622e9db..cb4c79e2c80 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/edit/EventEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/edit/EventEditFragment.java @@ -101,7 +101,7 @@ public static EventEditFragment newInstance(Event activityRootData) { null, activityRootData, FieldVisibilityCheckers.withCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); fragment.isMultiDayEvent = activityRootData.getEndDate() != null; diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/edit/EventParticipantEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/edit/EventParticipantEditFragment.java index 9e501d2d93d..8843394a6a0 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/edit/EventParticipantEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/edit/EventParticipantEditFragment.java @@ -27,6 +27,7 @@ import de.symeda.sormas.app.BaseEditFragment; import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.common.DatabaseHelper; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.event.EventParticipant; import de.symeda.sormas.app.backend.location.Location; import de.symeda.sormas.app.caze.edit.CaseNewActivity; @@ -50,7 +51,7 @@ public static EventParticipantEditFragment newInstance(EventParticipant activity null, activityRootData, new FieldVisibilityCheckers(), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Instance methods diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/read/EventParticipantReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/read/EventParticipantReadFragment.java index 81c805acd9c..310496bb817 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/read/EventParticipantReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/eventparticipant/read/EventParticipantReadFragment.java @@ -25,6 +25,7 @@ import de.symeda.sormas.app.BaseReadFragment; import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.common.DatabaseHelper; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.event.EventParticipant; import de.symeda.sormas.app.caze.read.CaseReadActivity; import de.symeda.sormas.app.databinding.FragmentEventParticipantReadLayoutBinding; @@ -41,7 +42,7 @@ public static EventParticipantReadFragment newInstance(EventParticipant activity null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getEvent().getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Instance methods diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/read/EventReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/read/EventReadFragment.java index e7d2419f264..7fce89cb3de 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/event/read/EventReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/event/read/EventReadFragment.java @@ -56,7 +56,7 @@ public static EventReadFragment newInstance(Event activityRootData) { null, activityRootData, FieldVisibilityCheckers.withCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Overrides diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/edit/ImmunizationEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/edit/ImmunizationEditFragment.java index 8d6c55f3ef9..3d9827e5c54 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/edit/ImmunizationEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/edit/ImmunizationEditFragment.java @@ -89,7 +89,7 @@ public static ImmunizationEditFragment newInstance(Immunization activityRootData activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); immunizationEditFragment.setMeansOfImmunizationChange(meansOfImmunizationChange); diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/read/ImmunizationReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/read/ImmunizationReadFragment.java index b3d0dab8cf8..c407adf7c0f 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/read/ImmunizationReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/read/ImmunizationReadFragment.java @@ -47,7 +47,7 @@ public static ImmunizationReadFragment newInstance(Immunization activityRootData activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); return immunizationReadFragment; } diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditFragment.java index 27bc4b6aa33..c7aa6016a3d 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditFragment.java @@ -57,7 +57,7 @@ public static VaccinationEditFragment newInstance(Vaccination activityRootData) null, activityRootData, null, - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } @Override diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditHealthConditionsFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditHealthConditionsFragment.java index aa00912d6f5..3bfe576d255 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditHealthConditionsFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/immunization/vaccination/VaccinationEditHealthConditionsFragment.java @@ -33,7 +33,7 @@ public static VaccinationEditHealthConditionsFragment newInstance(Vaccination ac activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getImmunization().getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } @Override diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/edit/PathogenTestEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/edit/PathogenTestEditFragment.java index e090db0d7ba..007abe578d4 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/edit/PathogenTestEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/edit/PathogenTestEditFragment.java @@ -75,7 +75,7 @@ public static PathogenTestEditFragment newInstance(PathogenTest activityRootData null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getTestedDisease()).andWithCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Overrides diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/read/PathogenTestReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/read/PathogenTestReadFragment.java index 28bece5d2cc..9a85f3ee6d3 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/read/PathogenTestReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/pathogentest/read/PathogenTestReadFragment.java @@ -44,7 +44,7 @@ public static PathogenTestReadFragment newInstance(PathogenTest activityRootData null, activityRootData, FieldVisibilityCheckers.withCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Overrides diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/person/edit/PersonEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/person/edit/PersonEditFragment.java index 263e3de8287..7de84d96b90 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/person/edit/PersonEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/person/edit/PersonEditFragment.java @@ -101,7 +101,7 @@ public static PersonEditFragment newInstance(Case activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static PersonEditFragment newInstance(Contact activityRootData) { @@ -111,7 +111,7 @@ public static PersonEditFragment newInstance(Contact activityRootData) { null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static PersonEditFragment newInstance(Immunization activityRootData) { @@ -121,7 +121,7 @@ public static PersonEditFragment newInstance(Immunization activityRootData) { null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static PersonEditFragment newInstance(EventParticipant activityRootData) { @@ -131,7 +131,7 @@ public static PersonEditFragment newInstance(EventParticipant activityRootData) null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getEvent().getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } private void setUpLayoutBinding(final BaseEditFragment fragment, final Person record, final FragmentPersonEditLayoutBinding contentBinding) { diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/person/read/PersonReadFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/person/read/PersonReadFragment.java index 7422906c113..bae6e313cb4 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/person/read/PersonReadFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/person/read/PersonReadFragment.java @@ -70,7 +70,7 @@ public static PersonReadFragment newInstance(Case activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static PersonReadFragment newInstance(Contact activityRootData) { @@ -79,7 +79,7 @@ public static PersonReadFragment newInstance(Contact activityRootData) { null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static PersonReadFragment newInstance(Immunization activityRootData) { @@ -88,7 +88,7 @@ public static PersonReadFragment newInstance(Immunization activityRootData) { null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static PersonReadFragment newInstance(EventParticipant activityRootData) { @@ -97,7 +97,7 @@ public static PersonReadFragment newInstance(EventParticipant activityRootData) null, activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getEvent().getDisease()), - UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.getDefault(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } private void setUpControlListeners() { diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/sample/edit/SampleEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/sample/edit/SampleEditFragment.java index 2c594ecf493..08d8db7b61b 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/sample/edit/SampleEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/sample/edit/SampleEditFragment.java @@ -85,7 +85,7 @@ public static SampleEditFragment newInstance(Sample activityRootData) { null, activityRootData, FieldVisibilityCheckers.withDisease(getDiseaseOfAssociatedEntity(activityRootData)).andWithCountry(ConfigProvider.getServerCountryCode()), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized()), + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode()), UserRight.SAMPLE_EDIT); } diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/symptoms/SymptomsEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/symptoms/SymptomsEditFragment.java index 093b827d26f..fbdf67b65b0 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/symptoms/SymptomsEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/symptoms/SymptomsEditFragment.java @@ -84,7 +84,7 @@ public static SymptomsEditFragment newInstance(Case activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static SymptomsEditFragment newInstance(Visit activityRootData) { @@ -94,7 +94,7 @@ public static SymptomsEditFragment newInstance(Visit activityRootData) { activityRootData, FieldVisibilityCheckers.withDisease(activityRootData.getDisease()) .add(new CountryFieldVisibilityChecker(ConfigProvider.getServerLocale())), - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } public static SymptomsEditFragment newInstance(ClinicalVisit activityRootData, String caseUuid) { diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/PrescriptionEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/PrescriptionEditFragment.java index f34a8e10ecc..d179737e7df 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/PrescriptionEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/PrescriptionEditFragment.java @@ -53,7 +53,7 @@ public static PrescriptionEditFragment newInstance(Prescription activityRootData null, activityRootData, null, - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Instance methods diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/TreatmentEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/TreatmentEditFragment.java index 199558ec248..147b1bb3dfc 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/TreatmentEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/therapy/edit/TreatmentEditFragment.java @@ -27,6 +27,7 @@ import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.app.BaseEditFragment; import de.symeda.sormas.app.R; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.therapy.Treatment; import de.symeda.sormas.app.component.Item; import de.symeda.sormas.app.databinding.FragmentTreatmentEditLayoutBinding; @@ -51,7 +52,7 @@ public static TreatmentEditFragment newInstance(Treatment activityRootData) { null, activityRootData, null, - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } // Instance methods diff --git a/sormas-app/app/src/main/java/de/symeda/sormas/app/visit/edit/VisitEditFragment.java b/sormas-app/app/src/main/java/de/symeda/sormas/app/visit/edit/VisitEditFragment.java index d6d5bc548ed..4377301a9ae 100644 --- a/sormas-app/app/src/main/java/de/symeda/sormas/app/visit/edit/VisitEditFragment.java +++ b/sormas-app/app/src/main/java/de/symeda/sormas/app/visit/edit/VisitEditFragment.java @@ -27,6 +27,7 @@ import de.symeda.sormas.app.BaseEditFragment; import de.symeda.sormas.app.R; import de.symeda.sormas.app.backend.common.DatabaseHelper; +import de.symeda.sormas.app.backend.config.ConfigProvider; import de.symeda.sormas.app.backend.contact.Contact; import de.symeda.sormas.app.backend.visit.Visit; import de.symeda.sormas.app.databinding.FragmentVisitEditLayoutBinding; @@ -44,7 +45,7 @@ public static VisitEditFragment newInstance(Visit activityRootData, String conta new Bundler().setContactUuid(contactUuid).get(), activityRootData, null, - UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized())); + UiFieldAccessCheckers.forSensitiveData(activityRootData.isPseudonymized(), ConfigProvider.getServerCountryCode())); } @Override From 09fe66806f2d5126c311bd949c588a97276a191e Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Wed, 30 Oct 2024 11:33:33 +0000 Subject: [PATCH 25/56] [GITFLOW]updating poms for 1.99.0 branch with snapshot versions From 9907bd6d99411471db5048cb0b6daddc780396e7 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Wed, 30 Oct 2024 11:33:33 +0000 Subject: [PATCH 26/56] [GITFLOW]updating poms for 1.100.0-SNAPSHOT development --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 3096708844c..55e897ce46c 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index f64314439ef..518447fdc15 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index ca267b688bf..f8b626e57bc 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index 90ec7bded5c..1abac7db51f 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 96d5fc57b5d..3d0a47e1f67 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index 7bc40057f54..9a7dbfc5f2d 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 522f79343bb..58dfb145183 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index 7f4271781ae..ae6d8243402 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index d620d49979a..2205ebaec18 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index e44c2105da5..a76762d90f0 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 7a1cfb76348..2fb2a7852c6 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 From 66b8b16b909de494d94a6829d9e112d6642fb706 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Wed, 30 Oct 2024 11:33:34 +0000 Subject: [PATCH 27/56] [GITFLOW]updating poms for branch'release-1.99.0' with non-snapshot versions --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 3096708844c..26dfda0937e 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index f64314439ef..0c1114dd40c 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index ca267b688bf..e8ef8eb1fb4 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index 90ec7bded5c..c497f85b584 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.99.0-SNAPSHOT + 1.99.0 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 96d5fc57b5d..8e796c884fa 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index 7bc40057f54..6b4a6503182 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 522f79343bb..9700fadbd33 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index 7f4271781ae..414595a25ca 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index d620d49979a..0c8d22cdbac 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index e44c2105da5..d09e8e5ecdc 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 7a1cfb76348..58872c4109c 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 From 92925583212853281a8e11b20b3d03f5aa653285 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Wed, 30 Oct 2024 11:40:22 +0000 Subject: [PATCH 28/56] [GITFLOW]updating develop poms to master versions to avoid merge conflicts --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 55e897ce46c..26dfda0937e 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index 518447fdc15..0c1114dd40c 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index f8b626e57bc..e8ef8eb1fb4 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index 1abac7db51f..c497f85b584 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.100.0-SNAPSHOT + 1.99.0 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 3d0a47e1f67..8e796c884fa 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index 9a7dbfc5f2d..6b4a6503182 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 58dfb145183..9700fadbd33 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index ae6d8243402..414595a25ca 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index 2205ebaec18..0c8d22cdbac 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index a76762d90f0..d09e8e5ecdc 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 2fb2a7852c6..58872c4109c 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.99.0 ../sormas-base 4.0.0 From 247e01fb52a290d194896a3c521cd701f868a843 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Wed, 30 Oct 2024 11:40:23 +0000 Subject: [PATCH 29/56] [GITFLOW]Updating develop poms back to pre merge state --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 26dfda0937e..55e897ce46c 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index 0c1114dd40c..518447fdc15 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index e8ef8eb1fb4..f8b626e57bc 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index c497f85b584..1abac7db51f 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.99.0 + 1.100.0-SNAPSHOT 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 8e796c884fa..3d0a47e1f67 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index 6b4a6503182..9a7dbfc5f2d 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 9700fadbd33..58dfb145183 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index 414595a25ca..ae6d8243402 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index 0c8d22cdbac..2205ebaec18 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index d09e8e5ecdc..a76762d90f0 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 58872c4109c..2fb2a7852c6 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.99.0 + 1.100.0-SNAPSHOT ../sormas-base 4.0.0 From 902363d668be6d3492ba924bb3f49a0a2748600b Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Fri, 1 Nov 2024 16:35:29 +0200 Subject: [PATCH 30/56] #13157 - Disable Contact Management for RSV Cases --- .../sormas/ui/caze/BulkCaseDataForm.java | 2 +- .../symeda/sormas/ui/caze/CaseCreateForm.java | 2 +- .../symeda/sormas/ui/caze/CaseDataForm.java | 2 +- .../sormas/ui/contact/ContactCreateForm.java | 2 +- .../sormas/ui/contact/ContactDataForm.java | 2 +- .../sormas/ui/events/EventDataForm.java | 2 +- .../form/ImmunizationCreationForm.java | 2 +- .../components/form/ImmunizationDataForm.java | 2 +- .../sormas/ui/person/PersonEditForm.java | 2 +- .../sormas/ui/samples/PathogenTestForm.java | 2 +- .../ui/selfreport/SelfReportDataForm.java | 2 +- .../ui/travelentry/TravelEntryDataForm.java | 2 +- .../components/TravelEntryCreateForm.java | 2 +- .../sormas/ui/utils/AbstractEditForm.java | 26 ++++++++++++++++--- 14 files changed, 36 insertions(+), 16 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/BulkCaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/BulkCaseDataForm.java index 1df2dc6f717..4cdcf912dd1 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/BulkCaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/BulkCaseDataForm.java @@ -169,7 +169,7 @@ protected void addFields() { diseaseCheckBox = new CheckBox(I18nProperties.getCaption(Captions.bulkDisease)); diseaseCheckBox.setReadOnly(!UiUtil.permitted(UserRight.CASE_CHANGE_DISEASE)); getContent().addComponent(diseaseCheckBox, DISEASE_CHECKBOX); - ComboBox diseaseField = addDiseaseField(CaseBulkEditData.DISEASE, false); + ComboBox diseaseField = addDiseaseField(CaseBulkEditData.DISEASE, false, false); diseaseField.setEnabled(false); addField(CaseBulkEditData.DISEASE_DETAILS, TextField.class); addField(CaseBulkEditData.PLAGUE_TYPE, NullableOptionGroup.class); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java index d118187e0f1..5554342567e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java @@ -192,7 +192,7 @@ protected void addFields() { addField(CaseDataDto.CASE_REFERENCE_NUMBER, TextField.class); addField(CaseDataDto.REPORT_DATE, DateField.class); - ComboBox diseaseField = addDiseaseField(CaseDataDto.DISEASE, false, true); + ComboBox diseaseField = addDiseaseField(CaseDataDto.DISEASE, false, true, false); diseaseVariantField = addField(CaseDataDto.DISEASE_VARIANT, ComboBox.class); diseaseVariantDetailsField = addField(CaseDataDto.DISEASE_VARIANT_DETAILS, TextField.class); diseaseVariantDetailsField.setVisible(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java index 8b23e63c822..c30e6cb8328 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java @@ -453,7 +453,7 @@ protected void addFields() { Collections.singletonList(CaseIdentificationSource.SCREENING), true); - ComboBox diseaseField = addDiseaseField(CaseDataDto.DISEASE, false); + ComboBox diseaseField = addDiseaseField(CaseDataDto.DISEASE, false, false); ComboBox diseaseVariantField = addCustomizableEnumField(CaseDataDto.DISEASE_VARIANT); TextField diseaseVariantDetailsField = addField(CaseDataDto.DISEASE_VARIANT_DETAILS, TextField.class); diseaseVariantDetailsField.setVisible(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java index bb00efe417d..715ea4072f3 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java @@ -151,7 +151,7 @@ protected void addFields() { reportDate = addField(ContactDto.REPORT_DATE_TIME, DateField.class); addField(CaseDataDto.CASE_REFERENCE_NUMBER, TextField.class); - ComboBox cbDisease = addDiseaseField(ContactDto.DISEASE, false, true); + ComboBox cbDisease = addDiseaseField(ContactDto.DISEASE, false, true, true); addField(ContactDto.DISEASE_DETAILS, TextField.class); personCreateForm = new PersonCreateForm(false, false, false, showPersonSearchButton); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java index e76a5f9ad9b..e981850aca6 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactDataForm.java @@ -299,7 +299,7 @@ protected void addFields() { } ComboBox relationToCase = addField(ContactDto.RELATION_TO_CASE, ComboBox.class); addField(ContactDto.RELATION_DESCRIPTION, TextField.class); - cbDisease = addDiseaseField(ContactDto.DISEASE, false); + cbDisease = addDiseaseField(ContactDto.DISEASE, false, true); cbDisease.setNullSelectionAllowed(false); addField(ContactDto.DISEASE_DETAILS, TextField.class); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java index 6fdfd864d4b..b75bfbe3bcc 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/events/EventDataForm.java @@ -222,7 +222,7 @@ protected void addFields() { getContent().addComponent(locationHeadingLabel, LOCATION_HEADING_LOC); addField(EventDto.UUID, TextField.class); - ComboBox diseaseField = addDiseaseField(EventDto.DISEASE, false, isCreateForm); + ComboBox diseaseField = addDiseaseField(EventDto.DISEASE, false, isCreateForm, false); addField(EventDto.DISEASE_DETAILS, TextField.class); ComboBox diseaseVariantField = addCustomizableEnumField(EventDto.DISEASE_VARIANT); diseaseVariantField.setNullSelectionAllowed(true); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java index f8d955c3ee1..d71f82c6f1f 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java @@ -125,7 +125,7 @@ protected void addFields() { TextField externalIdField = addField(ImmunizationDto.EXTERNAL_ID, TextField.class); style(externalIdField, ERROR_COLOR_PRIMARY); - ComboBox diseaseField = addDiseaseField(ImmunizationDto.DISEASE, false, true); + ComboBox diseaseField = addDiseaseField(ImmunizationDto.DISEASE, false, true, false); addField(ImmunizationDto.DISEASE_DETAILS, TextField.class); ComboBox meansOfImmunizationField = addField(ImmunizationDto.MEANS_OF_IMMUNIZATION, ComboBox.class); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java index fbe86c47094..09baebb7a43 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationDataForm.java @@ -164,7 +164,7 @@ protected void addFields() { UserField reportingUser = addField(ImmunizationDto.REPORTING_USER, UserField.class); reportingUser.setParentPseudonymizedSupplier(() -> getValue().isPseudonymized()); - ComboBox cbDisease = addDiseaseField(ImmunizationDto.DISEASE, false); + ComboBox cbDisease = addDiseaseField(ImmunizationDto.DISEASE, false, false); addField(ImmunizationDto.DISEASE_DETAILS, TextField.class); ComboBox meansOfImmunizationField = addField(ImmunizationDto.MEANS_OF_IMMUNIZATION, ComboBox.class); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index 4367a571630..c223d42ccba 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -407,7 +407,7 @@ protected void addFields() { TextField tfPlaceOfBirthFacilityDetails = addField(PersonDto.PLACE_OF_BIRTH_FACILITY_DETAILS, TextField.class); causeOfDeathField = addField(PersonDto.CAUSE_OF_DEATH, ComboBox.class); - causeOfDeathDiseaseField = addDiseaseField(PersonDto.CAUSE_OF_DEATH_DISEASE, true); + causeOfDeathDiseaseField = addDiseaseField(PersonDto.CAUSE_OF_DEATH_DISEASE, true, false); causeOfDeathDetailsField = addField(PersonDto.CAUSE_OF_DEATH_DETAILS, TextField.class); // Set requirements that don't need visibility changes and read only status diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java index 281313c0b50..0d21235cd02 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/samples/PathogenTestForm.java @@ -253,7 +253,7 @@ protected void addFields() { typingIdField.setVisible(false); // Tested Desease or Tested Pathogen, depending on sample type - ComboBox diseaseField = addDiseaseField(PathogenTestDto.TESTED_DISEASE, true, create); + ComboBox diseaseField = addDiseaseField(PathogenTestDto.TESTED_DISEASE, true, create, false); addField(PathogenTestDto.TESTED_DISEASE_DETAILS, TextField.class); ComboBox diseaseVariantField = addCustomizableEnumField(PathogenTestDto.TESTED_DISEASE_VARIANT); diseaseVariantField.setNullSelectionAllowed(true); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java index 870f42355e0..902941e6a3b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/selfreport/SelfReportDataForm.java @@ -114,7 +114,7 @@ protected void addFields() { addField(SelfReportDto.CASE_REFERENCE); //disease related fieldss - ComboBox diseaseField = addDiseaseField(SelfReportDto.DISEASE, false); + ComboBox diseaseField = addDiseaseField(SelfReportDto.DISEASE, false, false); addField(SelfReportDto.DISEASE_DETAILS, TextField.class); diseaseField.setRequired(true); ComboBox diseaseVariantField = addCustomizableEnumField(SelfReportDto.DISEASE_VARIANT); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java index 3c0f0d024d9..f73ced0d30d 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryDataForm.java @@ -153,7 +153,7 @@ protected void addFields() { TextField externalIdField = addField(TravelEntryDto.EXTERNAL_ID, TextField.class); style(externalIdField, ERROR_COLOR_PRIMARY); - ComboBox diseaseField = addDiseaseField(TravelEntryDto.DISEASE, false); + ComboBox diseaseField = addDiseaseField(TravelEntryDto.DISEASE, false, false); ComboBox diseaseVariantField = addCustomizableEnumField(TravelEntryDto.DISEASE_VARIANT); diseaseVariantField.setNullSelectionAllowed(true); diseaseVariantField.setVisible(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java index cbf92e0a21f..f515eba3dfd 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java @@ -113,7 +113,7 @@ protected void addFields() { TextField externalIdField = addField(TravelEntryDto.EXTERNAL_ID, TextField.class); style(externalIdField, ERROR_COLOR_PRIMARY); - ComboBox diseaseField = addDiseaseField(TravelEntryDto.DISEASE, false, true); + ComboBox diseaseField = addDiseaseField(TravelEntryDto.DISEASE, false, true, false); ComboBox diseaseVariantField = addField(TravelEntryDto.DISEASE_VARIANT, ComboBox.class); diseaseVariantField.setNullSelectionAllowed(true); diseaseVariantField.setVisible(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java index d5db8ba4ae1..ecc6ea80b00 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java @@ -197,8 +197,8 @@ public void discard() throws SourceException { super.discard(); } - protected ComboBox addDiseaseField(String fieldId, boolean showNonPrimaryDiseases) { - return addDiseaseField(fieldId, showNonPrimaryDiseases, false); + protected ComboBox addDiseaseField(String fieldId, boolean showNonPrimaryDiseases, boolean hideFollowUpDisabledDiseases) { + return addDiseaseField(fieldId, showNonPrimaryDiseases, false, hideFollowUpDisabledDiseases); } /** @@ -213,7 +213,11 @@ protected ComboBox addDiseaseField(String fieldId, boolean showNonPrimaryDisease * If only a single diseases is active on the server, set it as the default value */ @SuppressWarnings("unchecked") - protected ComboBox addDiseaseField(String fieldId, boolean showNonPrimaryDiseases, boolean setServerDiseaseAsDefault) { + protected ComboBox addDiseaseField( + String fieldId, + boolean showNonPrimaryDiseases, + boolean setServerDiseaseAsDefault, + boolean hideFollowUpDisabledDiseases) { diseaseField = addField(fieldId, ComboBox.class); this.setServerDiseaseAsDefault = setServerDiseaseAsDefault; @@ -233,6 +237,12 @@ protected ComboBox addDiseaseField(String fieldId, boolean showNonPrimaryDisease newItem.getItemProperty(SormasFieldGroupFieldFactory.CAPTION_PROPERTY_ID).setValue(value.toString()); } }); + + // + if (hideFollowUpDisabledDiseases) { + removeFollowUpDisabledDiseases(diseaseField); + } + return diseaseField; } @@ -503,6 +513,16 @@ protected void addNonPrimaryDiseasesTo(ComboBox diseaseField) { } } + protected void removeFollowUpDisabledDiseases(ComboBox diseaseField) { + List allActiveDiseases = FacadeProvider.getDiseaseConfigurationFacade().getAllActiveDiseases(); + + for (Disease disease : allActiveDiseases) { + if (diseaseField.getItem(disease) != null && !disease.isDefaultFollowUpEnabled()) { + diseaseField.removeItem(disease); + } + } + } + /** * Sets the initial visibilities based on annotations and builds a list of all fields in a form that are allowed to be visible - * this is either because the @Diseases and @Outbreaks annotations are not relevant or at least one of these annotations are present on From edd72b4e12e40b45e0c713d2585dc41eee8e0ef7 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Wed, 6 Nov 2024 09:51:00 +0200 Subject: [PATCH 31/56] Birthdate month does not show "Empty" instead of "confidential" when restrictions apply #13179 --- .../de/symeda/sormas/api/person/PersonDto.java | 2 ++ .../symeda/sormas/ui/person/PersonEditForm.java | 17 ++++++++++++----- 2 files changed, 14 insertions(+), 5 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonDto.java index 27363cfeeed..a4895af076b 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonDto.java @@ -195,6 +195,8 @@ public class PersonDto extends PseudonymizableDto implements IsPerson { @SensitiveData private Integer birthdateDD; @Outbreaks + @PersonalData + @SensitiveData private Integer birthdateMM; @Outbreaks private Integer birthdateYYYY; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index 4367a571630..b423a712dd5 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -296,13 +296,13 @@ protected void addFields() { ComboBox birthDateMonth = addField(PersonDto.BIRTH_DATE_MM, ComboBox.class); // @TODO: Done for nullselection Bug, fixed in Vaadin 7.7.3 birthDateMonth.setNullSelectionAllowed(true); - birthDateMonth.addItems(DateHelper.getMonthsInYear()); - birthDateMonth.setPageLength(12); +// birthDateMonth.addItems(DateHelper.getMonthsInYear()); +// birthDateMonth.setPageLength(12); birthDateMonth.setInputPrompt(I18nProperties.getString(Strings.month)); birthDateMonth.setCaption(""); - DateHelper.getMonthsInYear() - .forEach(month -> birthDateMonth.setItemCaption(month, de.symeda.sormas.api.Month.values()[month - 1].toString())); - setItemCaptionsForMonths(birthDateMonth); +// DateHelper.getMonthsInYear() +// .forEach(month -> birthDateMonth.setItemCaption(month, de.symeda.sormas.api.Month.values()[month - 1].toString())); +// setItemCaptionsForMonths(birthDateMonth); ComboBox birthDateYear = addField(PersonDto.BIRTH_DATE_YYYY, ComboBox.class); birthDateYear.setCaption(I18nProperties.getPrefixCaption(PersonDto.I18N_PREFIX, PersonDto.BIRTH_DATE)); // @TODO: Done for nullselection Bug, fixed in Vaadin 7.7.3 @@ -446,6 +446,13 @@ protected void addFields() { initializeVisibilitiesAndAllowedVisibilities(); initializeAccessAndAllowedAccesses(); + if (isEditableAllowed(PersonDto.BIRTH_DATE_MM)) { + birthDateMonth.addItems(DateHelper.getMonthsInYear()); + birthDateMonth.setPageLength(13); + DateHelper.getMonthsInYear() + .forEach(month -> birthDateMonth.setItemCaption(month, de.symeda.sormas.api.Month.values()[month - 1].toString())); + } + if (!getField(PersonDto.OCCUPATION_TYPE).isVisible() && !getField(PersonDto.ARMED_FORCES_RELATION_TYPE).isVisible() && !getField(PersonDto.EDUCATION_TYPE).isVisible()) From 66df1460cdfb83f412efc6c90c4d2b3ccf26cfaa Mon Sep 17 00:00:00 2001 From: cazacmarin Date: Thu, 7 Nov 2024 11:14:26 +0200 Subject: [PATCH 32/56] =?UTF-8?q?=20Configure=20and=20Restrict=20Sample=20?= =?UTF-8?q?Types=20and=20Pathogen=20Test=20Types=20for=20RSV=20=E2=80=A6?= =?UTF-8?q?=20(#13175)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Configure and Restrict Sample Types and Pathogen Test Types for RSV #13158 * Configure and Restrict Sample Types and Pathogen Test Types for RSV #13158 - after Serviu review --------- Co-authored-by: Marin --- .../sormas/api/sample/PathogenTestType.java | 36 ++++++++++++++--- .../sormas/api/sample/SampleMaterial.java | 40 +++++++++++++++---- 2 files changed, 62 insertions(+), 14 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestType.java b/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestType.java index b06c250f0b3..0eb491c7c4d 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestType.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/sample/PathogenTestType.java @@ -24,38 +24,62 @@ public enum PathogenTestType { + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) ANTIBODY_DETECTION, ANTIGEN_DETECTION, RAPID_TEST, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) CULTURE, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) HISTOPATHOLOGY, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) ISOLATION, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) IGM_SERUM_ANTIBODY, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) IGG_SERUM_ANTIBODY, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) IGA_SERUM_ANTIBODY, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) INCUBATION_TIME, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) INDIRECT_FLUORESCENT_ANTIBODY, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) DIRECT_FLUORESCENT_ANTIBODY, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) MICROSCOPY, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) NEUTRALIZING_ANTIBODIES, PCR_RT_PCR, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) GRAM_STAIN, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) LATEX_AGGLUTINATION, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) CQ_VALUE_DETECTION, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) SEQUENCING, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) DNA_MICROARRAY, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) TMA, OTHER; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/sample/SampleMaterial.java b/sormas-api/src/main/java/de/symeda/sormas/api/sample/SampleMaterial.java index 7369d33bf82..4ec4e3fbd1e 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/sample/SampleMaterial.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/sample/SampleMaterial.java @@ -24,44 +24,68 @@ public enum SampleMaterial { + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) BLOOD, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) SERA, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) STOOL, NASAL_SWAB, THROAT_SWAB, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) NP_SWAB, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) RECTAL_SWAB, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) CEREBROSPINAL_FLUID, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) CRUST, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) TISSUE, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) URINE, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) CORNEA_PM, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) SALIVA, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) URINE_PM, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) NUCHAL_SKIN_BIOPSY, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) SPUTUM, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) ENDOTRACHEAL_ASPIRATE, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) BRONCHOALVEOLAR_LAVAGE, @Diseases(value = { - Disease.CORONAVIRUS }, hide = true) + Disease.CORONAVIRUS, Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) BRAIN_TISSUE, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) ANTERIOR_NARES_SWAB, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) OP_ASPIRATE, NP_ASPIRATE, + @Diseases(value = { + Disease.RESPIRATORY_SYNCYTIAL_VIRUS }, hide = true) PLEURAL_FLUID, OTHER; From a682d07f52efd8bad3d393d52952bd43e18a4016 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Thu, 7 Nov 2024 11:16:59 +0200 Subject: [PATCH 33/56] Birthdate month does not show "Empty" instead of "confidential" when restrictions apply #13179 --- .../main/java/de/symeda/sormas/ui/person/PersonEditForm.java | 5 ----- 1 file changed, 5 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index b423a712dd5..5507feb83ec 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -296,13 +296,8 @@ protected void addFields() { ComboBox birthDateMonth = addField(PersonDto.BIRTH_DATE_MM, ComboBox.class); // @TODO: Done for nullselection Bug, fixed in Vaadin 7.7.3 birthDateMonth.setNullSelectionAllowed(true); -// birthDateMonth.addItems(DateHelper.getMonthsInYear()); -// birthDateMonth.setPageLength(12); birthDateMonth.setInputPrompt(I18nProperties.getString(Strings.month)); birthDateMonth.setCaption(""); -// DateHelper.getMonthsInYear() -// .forEach(month -> birthDateMonth.setItemCaption(month, de.symeda.sormas.api.Month.values()[month - 1].toString())); -// setItemCaptionsForMonths(birthDateMonth); ComboBox birthDateYear = addField(PersonDto.BIRTH_DATE_YYYY, ComboBox.class); birthDateYear.setCaption(I18nProperties.getPrefixCaption(PersonDto.I18N_PREFIX, PersonDto.BIRTH_DATE)); // @TODO: Done for nullselection Bug, fixed in Vaadin 7.7.3 From b99f4bb1fddd3286c78a6df89e250d821681fdf3 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Mon, 11 Nov 2024 13:55:29 +0200 Subject: [PATCH 34/56] #13157 - Disable Contact Management for RSV Cases --- .../sormas/ui/utils/AbstractEditForm.java | 27 ++++++++++--------- 1 file changed, 14 insertions(+), 13 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java index ecc6ea80b00..fb3e4203bac 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java @@ -15,6 +15,8 @@ package de.symeda.sormas.ui.utils; +import static com.vaadin.v7.data.fieldgroup.DefaultFieldGroupFieldFactory.CAPTION_PROPERTY_ID; + import java.util.ArrayList; import java.util.Collection; import java.util.List; @@ -221,6 +223,11 @@ protected ComboBox addDiseaseField( diseaseField = addField(fieldId, ComboBox.class); this.setServerDiseaseAsDefault = setServerDiseaseAsDefault; + + if (hideFollowUpDisabledDiseases) { + removeFollowUpDisabledDiseases(diseaseField); + } + if (showNonPrimaryDiseases) { addNonPrimaryDiseasesTo(diseaseField); } @@ -234,15 +241,10 @@ protected ComboBox addDiseaseField( Object value = e.getProperty().getValue(); if (value != null && !diseaseField.containsId(value)) { Item newItem = diseaseField.addItem(value); - newItem.getItemProperty(SormasFieldGroupFieldFactory.CAPTION_PROPERTY_ID).setValue(value.toString()); + newItem.getItemProperty(CAPTION_PROPERTY_ID).setValue(value.toString()); } }); - // - if (hideFollowUpDisabledDiseases) { - removeFollowUpDisabledDiseases(diseaseField); - } - return diseaseField; } @@ -509,17 +511,16 @@ protected void addNonPrimaryDiseasesTo(ComboBox diseaseField) { } Item newItem = diseaseField.addItem(disease); - newItem.getItemProperty(SormasFieldGroupFieldFactory.CAPTION_PROPERTY_ID).setValue(disease.toString()); + newItem.getItemProperty(CAPTION_PROPERTY_ID).setValue(disease.toString()); } } protected void removeFollowUpDisabledDiseases(ComboBox diseaseField) { - List allActiveDiseases = FacadeProvider.getDiseaseConfigurationFacade().getAllActiveDiseases(); - - for (Disease disease : allActiveDiseases) { - if (diseaseField.getItem(disease) != null && !disease.isDefaultFollowUpEnabled()) { - diseaseField.removeItem(disease); - } + List allDiseasesWithFollowUp = FacadeProvider.getDiseaseConfigurationFacade().getAllDiseasesWithFollowUp(true, true, true); + diseaseField.removeAllItems(); + for (Object r : allDiseasesWithFollowUp) { + Item newItem = diseaseField.addItem(r); + newItem.getItemProperty(CAPTION_PROPERTY_ID).setValue(r.toString()); } } From d79fbfa4b4f06ce516231f9dcec2b2af392210d6 Mon Sep 17 00:00:00 2001 From: Marin Date: Tue, 3 Dec 2024 07:49:35 +0200 Subject: [PATCH 35/56] Automate Case Details and Fields Preselection for Influenza Cases #13184 --- .../src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java | 2 +- .../src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java | 2 ++ 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java index aae97fc38d0..a6b0089068d 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java @@ -638,7 +638,7 @@ public static CaseDataDto build(PersonReferenceDto person, Disease disease, Heal caze.setPortHealthInfo(PortHealthInfoDto.build()); caze.setDisease(disease); caze.setInvestigationStatus(InvestigationStatus.PENDING); - caze.setCaseClassification(CaseClassification.NOT_CLASSIFIED); + caze.setCaseClassification(CaseClassification.CONFIRMED); caze.setOutcome(CaseOutcome.NO_OUTCOME); caze.setCaseOrigin(CaseOrigin.IN_COUNTRY); // TODO This is a workaround for transferring the followup comment while converting a contact to a case. This can be removed if the followup for cases is implemented in the mobile app diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java index 5554342567e..2b8e7811857 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java @@ -282,6 +282,7 @@ protected void addFields() { facilityOrHome.setId("facilityOrHome"); facilityOrHome.setWidth(100, Unit.PERCENTAGE); CssStyles.style(facilityOrHome, ValoTheme.OPTIONGROUP_HORIZONTAL); + facilityOrHome.setValue(Sets.newHashSet(TypeOfPlace.HOME)); facilityTypeGroup = ComboBoxHelper.createComboBoxV7(); facilityTypeGroup.setId("typeGroup"); facilityTypeGroup.setCaption(I18nProperties.getCaption(Captions.Facility_typeGroup)); @@ -550,6 +551,7 @@ private void setNoneFacility() { FacilityReferenceDto noFacilityRef = FacadeProvider.getFacilityFacade().getByUuid(FacilityDto.NONE_FACILITY_UUID).toReference(); facilityCombo.addItem(noFacilityRef); facilityCombo.setValue(noFacilityRef); + facilityType.setRequired(false); } private void updateFacility() { From cb8c136673b0fbcadf797d20ea5d68eeed2692e7 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Tue, 3 Dec 2024 16:31:35 +0200 Subject: [PATCH 36/56] #13181 - Enhance National Health ID Search Functionality Across Modules and Address Duplication Issues - v3 --- .../de/symeda/sormas/api/i18n/Strings.java | 1 + .../api/person/PersonSimilarityCriteria.java | 18 ++++- .../src/main/resources/strings.properties | 1 + .../ExternalMessageService.java | 2 + .../AutomaticLabMessageProcessor.java | 2 +- .../sormas/backend/person/PersonService.java | 22 +++--- .../sormas/backend/sample/SampleService.java | 1 + .../backend/person/PersonFacadeEjbTest.java | 2 +- .../symeda/sormas/ui/caze/CaseController.java | 79 ++++++++++++++++--- .../symeda/sormas/ui/caze/CaseCreateForm.java | 12 +++ .../sormas/ui/person/PersonController.java | 9 ++- .../sormas/ui/person/PersonCreateForm.java | 19 ++++- .../sormas/ui/person/PersonEditForm.java | 23 +++++- .../sormas/ui/person/PersonFormHelper.java | 54 +++++++++++++ .../ui/person/PersonSelectionField.java | 2 +- .../sormas/ui/person/PersonSelectionGrid.java | 7 ++ .../sormas/ui/utils/AbstractEditForm.java | 12 +++ .../utils/SormasFieldGroupFieldFactory.java | 3 + .../ui/utils/components/TextFieldCustom.java | 61 ++++++++++++++ 19 files changed, 299 insertions(+), 31 deletions(-) create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index ec0559d766b..484d972c404 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -825,6 +825,7 @@ public interface Strings { String headingShowExternalMessage = "headingShowExternalMessage"; String headingSignsAndSymptoms = "headingSignsAndSymptoms"; String headingSimilarImmunization = "headingSimilarImmunization"; + String headingSimilarPerson = "headingSimilarPerson"; String headingSomeCasesAlreadyInEvent = "headingSomeCasesAlreadyInEvent"; String headingSomeCasesNotRestored = "headingSomeCasesNotRestored"; String headingSomeContactsAlreadyInEvent = "headingSomeContactsAlreadyInEvent"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonSimilarityCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonSimilarityCriteria.java index fd0ca37565c..e782b271914 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonSimilarityCriteria.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonSimilarityCriteria.java @@ -20,6 +20,7 @@ public class PersonSimilarityCriteria extends BaseCriteria implements Cloneable private String passportNumber; private String nationalHealthId; private String nameUuidExternalIdExternalTokenLike; + private boolean checkOnlyForNationalHealthId; /** * If true, compare the name of the person only to the first and last name fields of the database; if false, compare the * name of the person to other fields like UUID and external ID as well. @@ -106,6 +107,14 @@ public PersonSimilarityCriteria nationalHealthId(String nationalHealthId) { return this; } + public boolean isCheckOnlyForNationalHealthId() { + return checkOnlyForNationalHealthId; + } + + public void setCheckOnlyForNationalHealthId(boolean checkOnlyForNationalHealthId) { + this.checkOnlyForNationalHealthId = checkOnlyForNationalHealthId; + } + public String getNameUuidExternalIdExternalTokenLike() { return nameUuidExternalIdExternalTokenLike; } @@ -140,7 +149,11 @@ public void setBirthdateDD(Integer birthdateDD) { } public static PersonSimilarityCriteria forPerson(PersonDto person) { - return forPerson(person, false); + return forPerson(person, false, false); + } + + public static PersonSimilarityCriteria forPerson(PersonDto person, boolean checkOnlyForNationalHealthId){ + return forPerson(person, false, checkOnlyForNationalHealthId); } /** @@ -148,7 +161,7 @@ public static PersonSimilarityCriteria forPerson(PersonDto person) { * If true, compares the name of the person only to the first and last name fields of the database; if false, compares the * name of the person to other fields like UUID and external ID as well. */ - public static PersonSimilarityCriteria forPerson(PersonDto person, boolean strictNameComparison) { + public static PersonSimilarityCriteria forPerson(PersonDto person, boolean strictNameComparison, boolean checkOnlyForNationalHealthId) { PersonSimilarityCriteria personSimilarityCriteria = new PersonSimilarityCriteria().sex(person.getSex()) .birthdateDD(person.getBirthdateDD()) @@ -156,6 +169,7 @@ public static PersonSimilarityCriteria forPerson(PersonDto person, boolean stric .birthdateYYYY(person.getBirthdateYYYY()) .passportNumber(person.getPassportNumber()) .nationalHealthId(person.getNationalHealthId()); + personSimilarityCriteria.setCheckOnlyForNationalHealthId(checkOnlyForNationalHealthId); if (strictNameComparison) { personSimilarityCriteria.firstName(person.getFirstName()).lastName(person.getLastName()).strictNameComparison(Boolean.TRUE); } else { diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index aa9c0c3ea24..7f03cdf3778 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -697,6 +697,7 @@ headingShowExternalMessage = Message headingSelfReportSideComponent = Self reports headingSignsAndSymptoms = Clinical Signs and Symptoms headingSimilarImmunization = Similar immunizaton +headingSimilarPerson = Similar persons found based on person introduced data headingSyncUsers = Sync Users headingTasksDeleted = Tasks deleted headingTasksNotDeleted = None of the tasks were deleted diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java index d163cb10330..9ce612c32cd 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java @@ -18,6 +18,7 @@ import javax.persistence.criteria.Predicate; import javax.persistence.criteria.Root; +import de.symeda.sormas.backend.person.Person; import org.apache.commons.lang3.StringUtils; import de.symeda.sormas.api.ReferenceDto; @@ -123,6 +124,7 @@ public Predicate buildCriteriaFilter(CriteriaBuilder cb, Root l CriteriaBuilderHelper.ilike(cb, labMessage.get(ExternalMessage.UUID), textFilter), CriteriaBuilderHelper.unaccentedIlike(cb, labMessage.get(ExternalMessage.PERSON_FIRST_NAME), textFilter), CriteriaBuilderHelper.unaccentedIlike(cb, labMessage.get(ExternalMessage.PERSON_LAST_NAME), textFilter), + CriteriaBuilderHelper.ilike(cb, labMessage.get(ExternalMessage.PERSON_NATIONAL_HEALTH_ID), textFilter), CriteriaBuilderHelper.ilike(cb, labMessage.get(ExternalMessage.PERSON_POSTAL_CODE), textFilter), CriteriaBuilderHelper.unaccentedIlike(cb, labMessage.get(ExternalMessage.REPORTER_NAME), textFilter), CriteriaBuilderHelper.ilike(cb, labMessage.get(ExternalMessage.REPORTER_POSTAL_CODE), textFilter)); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java index 74f141f2b50..f660bce1805 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java @@ -149,7 +149,7 @@ protected void handlePickOrCreatePerson(PersonDto person, HandlerCallback caseCreateComponent; + public CaseController() { } @@ -187,7 +192,7 @@ public void registerViews(Navigator navigator) { } public void create() { - CommitDiscardWrapperComponent caseCreateComponent = getCaseCreateComponent(null, null, null, null, null, false); + caseCreateComponent = getCaseCreateComponent(null, null, null, null, null, false); VaadinUiUtil.showModalPopupWindow(caseCreateComponent, I18nProperties.getString(Strings.headingCreateNewCase)); } @@ -740,6 +745,12 @@ public CommitDiscardWrapperComponent getCaseCreateComponent( dto.getSymptoms().setOnsetDate(createForm.getOnsetDate()); dto.setWasInQuarantineBeforeIsolation(YesNoUnknown.YES); +// if (StringUtils.isNotEmpty(person.getNationalHealthId())) { +// PersonFormHelper.warningSimilarPersons(person.getNationalHealthId(), null, () -> { +// transferDataToPerson(createForm, person); +// FacadeProvider.getPersonFacade().save(person); +// }); +// } transferDataToPerson(createForm, person); FacadeProvider.getPersonFacade().save(person); @@ -835,18 +846,60 @@ public CommitDiscardWrapperComponent getCaseCreateComponent( } else { // look for potential duplicate final PersonDto duplicatePerson = PersonDto.build(); - transferDataToPerson(createForm, duplicatePerson); - ControllerProvider.getPersonController() - .selectOrCreatePerson( - duplicatePerson, - I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), - selectedPerson -> { - if (selectedPerson != null) { - dto.setPerson(selectedPerson); - selectOrCreateCase(createForm, dto, selectedPerson); - } - }, - true); + + if (createForm.getWarningSimilarPersons() != null) { + CommitDiscardWrapperComponent warningPopUpDuplicatePerson = + (CommitDiscardWrapperComponent) editView.getWrappedComponent() + .getWarningSimilarPersons() + .getContent(); + + warningPopUpDuplicatePerson.getDiscardButton().setVisible(true); + warningPopUpDuplicatePerson.getCommitButton().setCaption("Continue to save"); + warningPopUpDuplicatePerson.addDoneListener(()-> { + VaadinUiUtil.showModalPopupWindow(caseCreateComponent, I18nProperties.getString(Strings.headingCreateNewCase)); + }); + warningPopUpDuplicatePerson.addCommitListener(() -> { + + transferDataToPerson(createForm, duplicatePerson); + ControllerProvider.getPersonController() + .selectOrCreatePerson( + duplicatePerson, + I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), + selectedPerson -> { + if (selectedPerson != null) { + dto.setPerson(selectedPerson); + selectOrCreateCase(createForm, dto, selectedPerson); + } + }, + true); + }); + } else { + transferDataToPerson(createForm, duplicatePerson); + ControllerProvider.getPersonController() + .selectOrCreatePerson( + duplicatePerson, + I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), + selectedPerson -> { + if (selectedPerson != null) { + dto.setPerson(selectedPerson); + selectOrCreateCase(createForm, dto, selectedPerson); + } + }, + true); + + } +// transferDataToPerson(createForm, duplicatePerson); +// ControllerProvider.getPersonController() +// .selectOrCreatePerson( +// duplicatePerson, +// I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), +// selectedPerson -> { +// if (selectedPerson != null) { +// dto.setPerson(selectedPerson); +// selectOrCreateCase(createForm, dto, selectedPerson); +// } +// }, +// true); } } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java index 5554342567e..6283b61ea39 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java @@ -31,6 +31,8 @@ import java.util.Date; import java.util.List; +import com.vaadin.ui.Window; +import de.symeda.sormas.ui.person.PersonFormHelper; import org.apache.commons.collections4.CollectionUtils; import com.google.common.collect.Sets; @@ -111,6 +113,7 @@ public class CaseCreateForm extends AbstractEditForm { private NullableOptionGroup ogCaseOrigin; private PersonCreateForm personCreateForm; + private Window warningSimilarPersons; private final boolean showHomeAddressForm; private final boolean showPersonSearchButton; @@ -209,6 +212,11 @@ protected void addFields() { personCreateForm.setWidth(100, Unit.PERCENTAGE); getContent().addComponent(personCreateForm, CaseDataDto.PERSON); + personCreateForm.getNationalHealthIdField().addTextFieldValueChangeListener(e -> { + warningSimilarPersons = PersonFormHelper + .warningSimilarPersons(personCreateForm.getNationalHealthIdField().getValue(), null, () -> warningSimilarPersons = null); + }); + // Jurisdiction fields Label jurisdictionHeadingLabel = new Label(I18nProperties.getString(Strings.headingCaseResponsibleJurisidction)); jurisdictionHeadingLabel.addStyleName(H3); @@ -730,4 +738,8 @@ public void setValue(CaseDataDto caseDataDto) throws com.vaadin.v7.data.Property public void setSearchedPerson(PersonDto searchedPerson) { personCreateForm.setSearchedPerson(searchedPerson); } + + public Window getWarningSimilarPersons() { + return warningSimilarPersons; + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonController.java index 03e117f37fa..88f53a0b9e9 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonController.java @@ -474,7 +474,14 @@ public CommitDiscardWrapperComponent getPersonEditComponent( editView.addCommitListener(() -> { if (!editForm.getFieldGroup().isModified()) { PersonDto dto = editForm.getValue(); - savePerson(dto); + if (editForm.getWarningSimilarPersons() != null) { + CommitDiscardWrapperComponent content = + (CommitDiscardWrapperComponent) editView.getWrappedComponent().getWarningSimilarPersons().getContent(); + content.getDiscardButton().setVisible(true); + content.addCommitListener(() -> savePerson(dto)); + } else { + savePerson(dto); + } } }); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java index 9f6cab9751c..02311ca226e 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java @@ -28,6 +28,8 @@ import java.util.List; import java.util.stream.Collectors; +import com.vaadin.ui.Window; +import de.symeda.sormas.ui.utils.components.TextFieldCustom; import org.apache.commons.lang3.StringUtils; import com.vaadin.icons.VaadinIcons; @@ -86,6 +88,8 @@ public class PersonCreateForm extends AbstractEditForm { private final boolean showPresentCondition; private final boolean showSymptomsOnsetDate; private final boolean showPersonSearchButton; + private TextFieldCustom nationalHealthIdField; + private Window warningSimilarPersons; private static final String HTML_LAYOUT = "%s" + fluidRow(fluidRowLocs(PersonDto.BIRTH_DATE_YYYY, PersonDto.BIRTH_DATE_MM, PersonDto.BIRTH_DATE_DD), fluidRowLocs(PersonDto.SEX)) @@ -186,7 +190,12 @@ protected void addFields() { ComboBox sex = addField(PersonDto.SEX, ComboBox.class); addField(PersonDto.PASSPORT_NUMBER, TextField.class); - addField(PersonDto.NATIONAL_HEALTH_ID, TextField.class); + + nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, TextFieldCustom.class); +// nationalHealthIdField.addTextFieldValueChangeListener(e -> { +// warningSimilarPersons = +// PersonFormHelper.warningSimilarPersons(nationalHealthIdField.getValue(), null, () -> getContent()); +// }); ComboBox presentCondition = addField(PersonDto.PRESENT_CONDITION, ComboBox.class); presentCondition.setVisible(showPresentCondition); @@ -498,4 +507,12 @@ public PersonDto getSearchedPerson() { public void setSearchedPerson(PersonDto searchedPerson) { this.person = searchedPerson; } + + public Window getWarningSimilarPersons() { + return warningSimilarPersons; + } + + public TextFieldCustom getNationalHealthIdField() { + return nationalHealthIdField; + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index 283d00c1d3c..6be5d4f0e0b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -33,8 +33,12 @@ import java.util.Date; import java.util.GregorianCalendar; import java.util.List; +import java.util.concurrent.atomic.AtomicBoolean; import java.util.stream.Collectors; +import com.vaadin.shared.Registration; +import com.vaadin.ui.Window; +import de.symeda.sormas.ui.utils.components.TextFieldCustom; import org.apache.commons.lang3.StringUtils; import com.vaadin.shared.ui.ErrorLevel; @@ -184,6 +188,8 @@ public class PersonEditForm extends AbstractEditForm { private boolean isPseudonymized; private LocationEditForm addressForm; private PresentConditionChangeListener presentConditionChangeListener; + private TextFieldCustom nationalHealthIdField; + private Window warningSimilarPersons; //@formatter:on public PersonEditForm( @@ -367,7 +373,7 @@ protected void addFields() { addField(PersonDto.PASSPORT_NUMBER); - TextField nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID); + nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, TextFieldCustom.class); Label nationalHealthIdWarningLabel = new Label(I18nProperties.getString(Strings.messagePersonNationalHealthIdInvalid)); nationalHealthIdWarningLabel.addStyleNames(VSPACE_3, LABEL_WHITE_SPACE_NORMAL); nationalHealthIdWarningLabel.setVisible(false); @@ -383,6 +389,17 @@ protected void addFields() { getContent().addComponent(externalTokenWarningLabel, EXTERNAL_TOKEN_WARNING_LOC); addField(PersonDto.INTERNAL_TOKEN); + AtomicBoolean nationalHealthIdFirstLoading = new AtomicBoolean(true); + nationalHealthIdField.addTextFieldValueChangeListener(e -> { + String currentPersonUuid = this.getValue().getUuid(); + if (nationalHealthIdFirstLoading.get()) { + nationalHealthIdFirstLoading.set(false); + } else { + warningSimilarPersons = + PersonFormHelper.warningSimilarPersons(nationalHealthIdField.getValue(), currentPersonUuid, () -> warningSimilarPersons = null); + } + }); + addField(PersonDto.HAS_COVID_APP).addStyleName(CssStyles.FORCE_CAPTION_CHECKBOX); addField(PersonDto.COVID_CODE_DELIVERED).addStyleName(CssStyles.FORCE_CAPTION_CHECKBOX); @@ -905,6 +922,10 @@ public Field getLastNameField() { return lastNameField; } + public Window getWarningSimilarPersons() { + return warningSimilarPersons; + } + @Override protected F addFieldToLayout(CustomLayout layout, String propertyId, F field) { field.addValueChangeListener(e -> fireValueChange(false)); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java new file mode 100644 index 00000000000..21afff244d3 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java @@ -0,0 +1,54 @@ +package de.symeda.sormas.ui.person; + +import java.util.List; +import java.util.stream.Collectors; + +import org.apache.commons.lang3.StringUtils; + +import com.vaadin.server.Sizeable; +import com.vaadin.ui.Window; + +import de.symeda.sormas.api.FacadeProvider; +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.person.PersonSimilarityCriteria; +import de.symeda.sormas.api.person.SimilarPersonDto; +import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; +import de.symeda.sormas.ui.utils.VaadinUiUtil; + +public class PersonFormHelper { + + public static Window warningSimilarPersons(String nationalHealthId, String currentPersonUuid, Runnable callback) { + if (StringUtils.isNoneBlank(nationalHealthId)) { + PersonDto currentPerson = new PersonDto(); + currentPerson.setNationalHealthId(nationalHealthId); + List similarPersonDtos = + FacadeProvider.getPersonFacade().getSimilarPersonDtos(PersonSimilarityCriteria.forPerson(currentPerson, true)); + List filteredSimilarPersons = similarPersonDtos.stream() + .filter(similarPersonDto -> !similarPersonDto.getUuid().equals(currentPersonUuid)) + .collect(Collectors.toList()); + + if (!filteredSimilarPersons.isEmpty()) { + PersonSelectionGrid similarPersonGrid = new PersonSelectionGrid(); + similarPersonGrid.loadData(filteredSimilarPersons); + + final CommitDiscardWrapperComponent component = new CommitDiscardWrapperComponent<>(similarPersonGrid); + component.getCommitButton().setCaption(I18nProperties.getCaption(Captions.actionDone)); + component.getDiscardButton().setVisible(false); + component.getWrappedComponent().setWidth(800, Sizeable.Unit.PIXELS); + + Window popupWindow = VaadinUiUtil.showPopupWindow(component); + component.addDoneListener(() -> { + popupWindow.close(); + callback.run(); + }); + popupWindow.setCaption(I18nProperties.getString(Strings.headingSimilarPerson)); + popupWindow.setWidth(900, Sizeable.Unit.PIXELS); + return popupWindow; + } + } + return null; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionField.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionField.java index 3cfdbf8d70f..8640ddf7df2 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionField.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionField.java @@ -269,7 +269,7 @@ protected void addFilterForm() { * Load a grid of all persons similar to the given reference person. */ protected void initializeGrid() { - defaultCriteria = PersonSimilarityCriteria.forPerson(referencePerson, true); + defaultCriteria = PersonSimilarityCriteria.forPerson(referencePerson, true, false); personGrid = new PersonSelectionGrid(); personGrid.addSelectionListener(e -> { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionGrid.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionGrid.java index 0762c264e40..e4fe4f63b59 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionGrid.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonSelectionGrid.java @@ -104,4 +104,11 @@ public void loadData(PersonSimilarityCriteria criteria) { getContainer().addAll(similarPersons); setHeightByRows(similarPersons.size() > 0 ? (Math.min(similarPersons.size(), 10)) : 1); } + + public void loadData(List similarPersons) { + getContainer().removeAllItems(); + getContainer().addAll(similarPersons); + setHeightByRows(similarPersons.size() > 0 ? (Math.min(similarPersons.size(), 10)) : 1); + } + } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java index fb3e4203bac..dba76e34844 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java @@ -21,9 +21,12 @@ import java.util.Collection; import java.util.List; import java.util.Optional; +import java.util.Set; import java.util.stream.Collectors; import java.util.stream.Stream; +import com.vaadin.navigator.ViewLeaveAction; +import com.vaadin.ui.Window; import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.data.Item; import com.vaadin.v7.data.Validator; @@ -45,10 +48,17 @@ import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.person.PersonDto; +import de.symeda.sormas.api.person.PersonSimilarityCriteria; +import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.person.SimilarPersonDto; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.clinicalcourse.HealthConditionsForm; +import de.symeda.sormas.ui.person.PersonSelectionGrid; import de.symeda.sormas.ui.utils.components.NotBlankTextValidator; +import org.apache.commons.lang3.ObjectUtils; +import org.apache.commons.lang3.StringUtils; public abstract class AbstractEditForm extends AbstractForm implements FieldGroup.CommitHandler {// implements DtoEditForm { @@ -613,4 +623,6 @@ protected boolean isEditableAllowed(String propertyId) { public void setHeading(String heading) { throw new RuntimeException("setHeading should be implemented in " + getClass().getSimpleName()); } + + } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java index f23d145988d..d7cd5d95382 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java @@ -43,6 +43,7 @@ import de.symeda.sormas.ui.utils.components.JsonForm; import de.symeda.sormas.ui.utils.components.MultiSelect; import de.symeda.sormas.ui.utils.components.MultiSelectFiles; +import de.symeda.sormas.ui.utils.components.TextFieldCustom; import de.symeda.sormas.ui.vaccination.VaccinationsField; public class SormasFieldGroupFieldFactory extends DefaultFieldGroupFieldFactory { @@ -191,6 +192,8 @@ public T createField(Class type, Class fieldType) { return (T) new CheckBoxTree<>(); } else if (RichTextArea.class.isAssignableFrom(fieldType)) { return (T) new RichTextArea(); + }else if (TextFieldCustom.class.isAssignableFrom(fieldType)) { + return (T) new TextFieldCustom(); } return super.createField(type, fieldType); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java new file mode 100644 index 00000000000..4371c0716d2 --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java @@ -0,0 +1,61 @@ +package de.symeda.sormas.ui.utils.components; + +import java.lang.reflect.Method; + +import com.vaadin.event.SerializableEventListener; +import com.vaadin.shared.Registration; +import com.vaadin.ui.Component; +import com.vaadin.util.ReflectTools; +import com.vaadin.v7.data.Property; +import com.vaadin.v7.ui.AbstractField; +import com.vaadin.v7.ui.TextField; + +public class TextFieldCustom extends TextField { + + /* Value change events */ + + private static final Method VALUE_CHANGE_METHOD; + + static { + try { + VALUE_CHANGE_METHOD = Property.ValueChangeListener.class + .getDeclaredMethod("valueChange", + new Class[] { Property.ValueChangeEvent.class }); + } catch (final NoSuchMethodException e) { + // This should never happen + throw new RuntimeException( + "Internal error finding methods in AbstractField"); + } + } + + public Registration addTextFieldValueChangeListener(Property.ValueChangeListener listener) { + Registration registration = addListener(ValueChangeEvent.class, listener, + VALUE_CHANGE_METHOD); + // ensure "automatic immediate handling" works + markAsDirty(); + return registration; + } + + public interface TextFieldValueChangeListener extends SerializableEventListener { + + Method TEXTFIELD_CHANGE_METHOD = + ReflectTools.findMethod(TextFieldValueChangeListener.class, "textFieldValueChange", TextFieldValueChangeEvent.class); + + void textFieldValueChange(TextFieldValueChangeEvent event); + + } + + public static final class TextFieldValueChangeEvent extends Component.Event { + + private static final long serialVersionUID = 7689979222604215471L; + + public TextFieldValueChangeEvent(TextField source) { + super(source); + } + + public TextField getTextField() { + return (TextField) getComponent(); + } + } + +} From 89685ea58313afe6e4f6bf331f94d641d0992c8e Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Thu, 5 Dec 2024 12:10:54 +0200 Subject: [PATCH 37/56] #13181 - Enhance National Health ID Search Functionality Across Modules and Address Duplication Issues --- .../java/de/symeda/sormas/ui/MainScreen.java | 2 +- .../symeda/sormas/ui/caze/CaseController.java | 34 ++----- .../symeda/sormas/ui/caze/CaseCreateForm.java | 4 +- .../sormas/ui/contact/ContactController.java | 90 ++++++++++++++----- .../sormas/ui/contact/ContactCreateForm.java | 11 +++ .../immunization/ImmunizationController.java | 55 +++++++++--- .../form/ImmunizationCreationForm.java | 12 +++ .../sormas/ui/person/PersonCreateForm.java | 9 +- .../sormas/ui/person/PersonEditForm.java | 6 +- .../ui/travelentry/TravelEntryController.java | 46 ++++++++-- .../components/TravelEntryCreateForm.java | 11 +++ .../ui/utils/components/TextFieldCustom.java | 62 ++++--------- 12 files changed, 220 insertions(+), 122 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java index 260e045d60f..00ecea96b08 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/MainScreen.java @@ -315,7 +315,7 @@ && permitted(FeatureType.ADVERSE_EVENTS_FOLLOWING_IMMUNIZATION_MANAGEMENT, UserR } if (permitted(FeatureType.TRAVEL_ENTRIES, UserRight.TRAVEL_ENTRY_MANAGEMENT_ACCESS) - && FacadeProvider.getConfigFacade().isConfiguredCountry(CountryHelper.COUNTRY_CODE_LUXEMBOURG)) { + && FacadeProvider.getConfigFacade().isConfiguredCountry(CountryHelper.COUNTRY_CODE_GERMANY)) { ControllerProvider.getTravelEntryController().registerViews(navigator); menu.addView( TravelEntriesView.class, diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java index 1e83418c063..c30af8de040 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java @@ -26,7 +26,6 @@ import java.util.stream.Collectors; import org.apache.commons.collections4.CollectionUtils; -import org.apache.commons.lang3.StringUtils; import com.vaadin.navigator.Navigator; import com.vaadin.server.ExternalResource; @@ -131,7 +130,6 @@ import de.symeda.sormas.ui.externalsurveillanceservice.ExternalSurveillanceServiceGateway; import de.symeda.sormas.ui.hospitalization.HospitalizationForm; import de.symeda.sormas.ui.hospitalization.HospitalizationView; -import de.symeda.sormas.ui.person.PersonFormHelper; import de.symeda.sormas.ui.person.PersonSelectionGrid; import de.symeda.sormas.ui.symptoms.SymptomsForm; import de.symeda.sormas.ui.therapy.TherapyView; @@ -155,6 +153,7 @@ public class CaseController { CommitDiscardWrapperComponent caseCreateComponent; + boolean caseSaveTriggered; public CaseController() { @@ -723,6 +722,7 @@ public CommitDiscardWrapperComponent getCaseCreateComponent( editView.addFieldGroups(createForm.getHomeAddressForm().getFieldGroup()); } + caseSaveTriggered = false; editView.addCommitListener(() -> { if (!createForm.getFieldGroup().isModified()) { final CaseDataDto dto = createForm.getValue(); @@ -745,12 +745,6 @@ public CommitDiscardWrapperComponent getCaseCreateComponent( dto.getSymptoms().setOnsetDate(createForm.getOnsetDate()); dto.setWasInQuarantineBeforeIsolation(YesNoUnknown.YES); -// if (StringUtils.isNotEmpty(person.getNationalHealthId())) { -// PersonFormHelper.warningSimilarPersons(person.getNationalHealthId(), null, () -> { -// transferDataToPerson(createForm, person); -// FacadeProvider.getPersonFacade().save(person); -// }); -// } transferDataToPerson(createForm, person); FacadeProvider.getPersonFacade().save(person); @@ -854,12 +848,14 @@ public CommitDiscardWrapperComponent getCaseCreateComponent( .getContent(); warningPopUpDuplicatePerson.getDiscardButton().setVisible(true); - warningPopUpDuplicatePerson.getCommitButton().setCaption("Continue to save"); - warningPopUpDuplicatePerson.addDoneListener(()-> { - VaadinUiUtil.showModalPopupWindow(caseCreateComponent, I18nProperties.getString(Strings.headingCreateNewCase)); + warningPopUpDuplicatePerson.getCommitButton().setCaption(I18nProperties.getCaption(Captions.actionContinue)); + warningPopUpDuplicatePerson.addDoneListener(() -> { + if (!caseSaveTriggered) { + VaadinUiUtil.showModalPopupWindow(caseCreateComponent, I18nProperties.getString(Strings.headingCreateNewCase)); + } }); warningPopUpDuplicatePerson.addCommitListener(() -> { - + caseSaveTriggered = true; transferDataToPerson(createForm, duplicatePerson); ControllerProvider.getPersonController() .selectOrCreatePerson( @@ -886,27 +882,13 @@ public CommitDiscardWrapperComponent getCaseCreateComponent( } }, true); - } -// transferDataToPerson(createForm, duplicatePerson); -// ControllerProvider.getPersonController() -// .selectOrCreatePerson( -// duplicatePerson, -// I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), -// selectedPerson -> { -// if (selectedPerson != null) { -// dto.setPerson(selectedPerson); -// selectOrCreateCase(createForm, dto, selectedPerson); -// } -// }, -// true); } } } }); return editView; - } private void selectOrCreateCase(CaseCreateForm createForm, CaseDataDto dto, PersonReferenceDto selectedPerson) { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java index 6283b61ea39..7c2b71d69a5 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java @@ -31,12 +31,11 @@ import java.util.Date; import java.util.List; -import com.vaadin.ui.Window; -import de.symeda.sormas.ui.person.PersonFormHelper; import org.apache.commons.collections4.CollectionUtils; import com.google.common.collect.Sets; import com.vaadin.ui.Label; +import com.vaadin.ui.Window; import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.data.util.converter.Converter; import com.vaadin.v7.ui.AbstractSelect.ItemCaptionMode; @@ -74,6 +73,7 @@ import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.location.LocationEditForm; import de.symeda.sormas.ui.person.PersonCreateForm; +import de.symeda.sormas.ui.person.PersonFormHelper; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.ComboBoxHelper; import de.symeda.sormas.ui.utils.CssStyles; diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java index 926e6d61662..5b0754a3562 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java @@ -93,6 +93,7 @@ import de.symeda.sormas.ui.contact.components.linelisting.layout.LineListingLayout; import de.symeda.sormas.ui.epidata.ContactEpiDataView; import de.symeda.sormas.ui.epidata.EpiDataForm; +import de.symeda.sormas.ui.person.PersonSelectionGrid; import de.symeda.sormas.ui.utils.AbstractView; import de.symeda.sormas.ui.utils.ArchiveHandlers; import de.symeda.sormas.ui.utils.BulkOperationHandler; @@ -110,6 +111,8 @@ public class ContactController { protected final Logger logger = LoggerFactory.getLogger(getClass()); + boolean contactSaveTriggered; + CommitDiscardWrapperComponent createComponent; public ContactController() { @@ -270,8 +273,7 @@ public void create(CaseReferenceDto caseRef, boolean asResultingCase, Runnable a if (caseRef != null) { caze = FacadeProvider.getCaseFacade().getCaseDataByUuid(caseRef.getUuid()); } - CommitDiscardWrapperComponent createComponent = - getContactCreateComponent(caze, asResultingCase, alternativeCallback, false); + createComponent = getContactCreateComponent(caze, asResultingCase, alternativeCallback, false); VaadinUiUtil.showModalPopupWindow(createComponent, I18nProperties.getString(Strings.headingCreateNewContact)); } @@ -452,6 +454,7 @@ public CommitDiscardWrapperComponent getContactCreateComponen final CommitDiscardWrapperComponent createComponent = new CommitDiscardWrapperComponent(createForm, UiUtil.permitted(UserRight.CONTACT_CREATE), createForm.getFieldGroup()); + contactSaveTriggered = false; createComponent.addCommitListener(() -> { if (!createForm.getFieldGroup().isModified()) { final ContactDto dto = createForm.getValue(); @@ -504,26 +507,73 @@ public CommitDiscardWrapperComponent getContactCreateComponen } else { final PersonDto person = PersonDto.build(); - transferDataToPerson(createForm, person); - ControllerProvider.getPersonController() - .selectOrCreatePerson(person, I18nProperties.getString(Strings.infoSelectOrCreatePersonForContact), selectedPerson -> { - if (selectedPerson != null) { - dto.setPerson(selectedPerson); - - selectOrCreateContact(dto, selectedPerson, selectedContactUuid -> { - if (selectedContactUuid != null) { - editData(selectedContactUuid); - } - }); - } - if (createForm.adoptAddressLayout.isAdoptAddress()) { - FacadeProvider.getPersonFacade() - .copyHomeAddress( - FacadeProvider.getCaseFacade().getByUuid(dto.getCaze().getUuid()).getPerson(), - dto.getPerson()); + if (createForm.getWarningSimilarPersons() != null) { + CommitDiscardWrapperComponent warningPopUpDuplicatePerson = + (CommitDiscardWrapperComponent) createComponent.getWrappedComponent() + .getWarningSimilarPersons() + .getContent(); + warningPopUpDuplicatePerson.getDiscardButton().setVisible(true); + warningPopUpDuplicatePerson.getCommitButton().setCaption(I18nProperties.getCaption(Captions.actionContinue)); + warningPopUpDuplicatePerson.addDoneListener(() -> { + if (!contactSaveTriggered) { + VaadinUiUtil.showModalPopupWindow(createComponent, I18nProperties.getString(Strings.headingCreateNewContact)); } - }, true); + }); + + warningPopUpDuplicatePerson.addCommitListener(() -> { + contactSaveTriggered = true; + transferDataToPerson(createForm, person); + + ControllerProvider.getPersonController() + .selectOrCreatePerson( + person, + I18nProperties.getString(Strings.infoSelectOrCreatePersonForContact), + selectedPerson -> { + if (selectedPerson != null) { + dto.setPerson(selectedPerson); + + selectOrCreateContact(dto, selectedPerson, selectedContactUuid -> { + if (selectedContactUuid != null) { + editData(selectedContactUuid); + } + }); + } + if (createForm.adoptAddressLayout.isAdoptAddress()) { + FacadeProvider.getPersonFacade() + .copyHomeAddress( + FacadeProvider.getCaseFacade().getByUuid(dto.getCaze().getUuid()).getPerson(), + dto.getPerson()); + } + }, + true); + }); + } else { + transferDataToPerson(createForm, person); + + ControllerProvider.getPersonController() + .selectOrCreatePerson( + person, + I18nProperties.getString(Strings.infoSelectOrCreatePersonForContact), + selectedPerson -> { + if (selectedPerson != null) { + dto.setPerson(selectedPerson); + + selectOrCreateContact(dto, selectedPerson, selectedContactUuid -> { + if (selectedContactUuid != null) { + editData(selectedContactUuid); + } + }); + } + if (createForm.adoptAddressLayout.isAdoptAddress()) { + FacadeProvider.getPersonFacade() + .copyHomeAddress( + FacadeProvider.getCaseFacade().getByUuid(dto.getCaze().getUuid()).getPerson(), + dto.getPerson()); + } + }, + true); + } } } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java index 715ea4072f3..e7c57c7d226 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactCreateForm.java @@ -27,6 +27,7 @@ import com.vaadin.shared.ui.ContentMode; import com.vaadin.ui.Button; import com.vaadin.ui.Label; +import com.vaadin.ui.Window; import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.ui.AbstractField; import com.vaadin.v7.ui.CheckBox; @@ -55,6 +56,7 @@ import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.person.PersonCreateForm; +import de.symeda.sormas.ui.person.PersonFormHelper; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.ButtonHelper; import de.symeda.sormas.ui.utils.CssStyles; @@ -119,6 +121,7 @@ public class ContactCreateForm extends AbstractEditForm { ComboBox community; private final boolean showPersonSearchButton; + private Window warningSimilarPersons; /** * TODO use disease and case relation information given in ContactDto @@ -159,6 +162,11 @@ protected void addFields() { personCreateForm.setValue(new PersonDto()); getContent().addComponent(personCreateForm, ContactDto.PERSON); + personCreateForm.getNationalHealthIdField().addTextFieldValueChangeListener(e -> { + warningSimilarPersons = PersonFormHelper + .warningSimilarPersons(personCreateForm.getNationalHealthIdField().getValue(), null, () -> warningSimilarPersons = null); + }); + addField(ContactDto.RETURNING_TRAVELER, NullableOptionGroup.class); region = addInfrastructureField(ContactDto.REGION); district = addInfrastructureField(ContactDto.DISTRICT); @@ -430,4 +438,7 @@ private void updateDateComparison() { } } + public Window getWarningSimilarPersons() { + return warningSimilarPersons; + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java index e39ccc96b84..6288c273cb2 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java @@ -30,6 +30,7 @@ import de.symeda.sormas.ui.immunization.components.fields.popup.SimilarImmunizationPopup; import de.symeda.sormas.ui.immunization.components.form.ImmunizationCreationForm; import de.symeda.sormas.ui.immunization.components.form.ImmunizationDataForm; +import de.symeda.sormas.ui.person.PersonSelectionGrid; import de.symeda.sormas.ui.utils.ArchiveHandlers; import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; import de.symeda.sormas.ui.utils.NotificationHelper; @@ -40,6 +41,9 @@ public class ImmunizationController { + boolean immunizationSaveTriggered; + CommitDiscardWrapperComponent immunizationCreateComponent; + public void registerViews(Navigator navigator) { navigator.addView(ImmunizationsView.VIEW_NAME, ImmunizationsView.class); navigator.addView(ImmunizationDataView.VIEW_NAME, ImmunizationDataView.class); @@ -47,7 +51,7 @@ public void registerViews(Navigator navigator) { } public void create() { - CommitDiscardWrapperComponent immunizationCreateComponent = getImmunizationCreateComponent(); + immunizationCreateComponent = getImmunizationCreateComponent(); if (immunizationCreateComponent != null) { VaadinUiUtil.showModalPopupWindow(immunizationCreateComponent, I18nProperties.getString(Strings.headingCreateNewImmunization)); } @@ -81,6 +85,7 @@ private CommitDiscardWrapperComponent getImmunizationC currentUserProvider.hasUserRight(UserRight.IMMUNIZATION_CREATE), createForm.getFieldGroup()); + immunizationSaveTriggered = false; viewComponent.addCommitListener(() -> { if (!createForm.getFieldGroup().isModified()) { final ImmunizationDto dto = createForm.getValue(); @@ -90,16 +95,44 @@ private CommitDiscardWrapperComponent getImmunizationC selectOrCreateimmunizationForPerson(dto, searchedPerson.toReference()); } else { final PersonDto person = createForm.getPerson(); - ControllerProvider.getPersonController() - .selectOrCreatePerson( - person, - I18nProperties.getString(Strings.infoSelectOrCreatePersonForImmunization), - selectedPerson -> { - if (selectedPerson != null) { - selectOrCreateimmunizationForPerson(dto, selectedPerson); - } - }, - true); + if (createForm.getWarningSimilarPersons() != null) { + CommitDiscardWrapperComponent warningPopUpDuplicatePerson = + (CommitDiscardWrapperComponent) viewComponent.getWrappedComponent() + .getWarningSimilarPersons() + .getContent(); + warningPopUpDuplicatePerson.getDiscardButton().setVisible(true); + warningPopUpDuplicatePerson.getCommitButton().setCaption(I18nProperties.getCaption(Captions.actionContinue)); + warningPopUpDuplicatePerson.addDoneListener(() -> { + if (!immunizationSaveTriggered) { + VaadinUiUtil.showModalPopupWindow(viewComponent, I18nProperties.getString(Strings.headingCreateNewImmunization)); + } + }); + + warningPopUpDuplicatePerson.addCommitListener(() -> { + immunizationSaveTriggered = true; + ControllerProvider.getPersonController() + .selectOrCreatePerson( + person, + I18nProperties.getString(Strings.infoSelectOrCreatePersonForImmunization), + selectedPerson -> { + if (selectedPerson != null) { + selectOrCreateimmunizationForPerson(dto, selectedPerson); + } + }, + true); + }); + } else { + ControllerProvider.getPersonController() + .selectOrCreatePerson( + person, + I18nProperties.getString(Strings.infoSelectOrCreatePersonForImmunization), + selectedPerson -> { + if (selectedPerson != null) { + selectOrCreateimmunizationForPerson(dto, selectedPerson); + } + }, + true); + } } } }); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java index d71f82c6f1f..87265176ea6 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/components/form/ImmunizationCreationForm.java @@ -27,6 +27,7 @@ import java.util.Collections; import com.vaadin.ui.Label; +import com.vaadin.ui.Window; import com.vaadin.v7.ui.CheckBox; import com.vaadin.v7.ui.ComboBox; import com.vaadin.v7.ui.DateField; @@ -56,6 +57,7 @@ import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.person.PersonCreateForm; +import de.symeda.sormas.ui.person.PersonFormHelper; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.ComboBoxHelper; import de.symeda.sormas.ui.utils.DateComparisonValidator; @@ -96,6 +98,7 @@ public class ImmunizationCreationForm extends AbstractEditForm private ComboBox responsibleRegion; private ComboBox responsibleDistrict; private ComboBox responsibleCommunity; + private Window warningSimilarPersons; public ImmunizationCreationForm() { this(null, null); @@ -191,6 +194,11 @@ protected void addFields() { personCreateForm = new PersonCreateForm(false, true, false); personCreateForm.setWidth(100, Unit.PERCENTAGE); personCreateForm.setValue(new PersonDto()); + personCreateForm.getNationalHealthIdField().addTextFieldValueChangeListener(e -> { + warningSimilarPersons = PersonFormHelper + .warningSimilarPersons(personCreateForm.getNationalHealthIdField().getValue(), null, () -> warningSimilarPersons = null); + }); + diseaseField.addValueChangeListener( (ValueChangeListener) valueChangeEvent -> personCreateForm .updatePresentConditionEnum((Disease) valueChangeEvent.getProperty().getValue())); @@ -406,4 +414,8 @@ protected void setInternalValue(ImmunizationDto newValue) { hideAndFillJurisdictionFields(); } } + + public Window getWarningSimilarPersons() { + return warningSimilarPersons; + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java index 02311ca226e..a1e9d757967 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java @@ -28,13 +28,12 @@ import java.util.List; import java.util.stream.Collectors; -import com.vaadin.ui.Window; -import de.symeda.sormas.ui.utils.components.TextFieldCustom; import org.apache.commons.lang3.StringUtils; import com.vaadin.icons.VaadinIcons; import com.vaadin.ui.Button; import com.vaadin.ui.Label; +import com.vaadin.ui.Window; import com.vaadin.v7.data.validator.EmailValidator; import com.vaadin.v7.ui.AbstractSelect; import com.vaadin.v7.ui.AbstractSelect.ItemCaptionMode; @@ -67,6 +66,7 @@ import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.PhoneNumberValidator; import de.symeda.sormas.ui.utils.VaadinUiUtil; +import de.symeda.sormas.ui.utils.components.TextFieldCustom; public class PersonCreateForm extends AbstractEditForm { @@ -192,10 +192,7 @@ protected void addFields() { addField(PersonDto.PASSPORT_NUMBER, TextField.class); nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, TextFieldCustom.class); -// nationalHealthIdField.addTextFieldValueChangeListener(e -> { -// warningSimilarPersons = -// PersonFormHelper.warningSimilarPersons(nationalHealthIdField.getValue(), null, () -> getContent()); -// }); + nationalHealthIdField.setNullRepresentation(""); ComboBox presentCondition = addField(PersonDto.PRESENT_CONDITION, ComboBox.class); presentCondition.setVisible(showPresentCondition); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index 6be5d4f0e0b..ec3427adb6f 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -36,14 +36,12 @@ import java.util.concurrent.atomic.AtomicBoolean; import java.util.stream.Collectors; -import com.vaadin.shared.Registration; -import com.vaadin.ui.Window; -import de.symeda.sormas.ui.utils.components.TextFieldCustom; import org.apache.commons.lang3.StringUtils; import com.vaadin.shared.ui.ErrorLevel; import com.vaadin.ui.CustomLayout; import com.vaadin.ui.Label; +import com.vaadin.ui.Window; import com.vaadin.v7.data.Item; import com.vaadin.v7.data.Property; import com.vaadin.v7.ui.AbstractSelect; @@ -98,6 +96,7 @@ import de.symeda.sormas.ui.utils.SormasFieldGroupFieldFactory; import de.symeda.sormas.ui.utils.ValidationUtils; import de.symeda.sormas.ui.utils.ViewMode; +import de.symeda.sormas.ui.utils.components.TextFieldCustom; public class PersonEditForm extends AbstractEditForm { @@ -374,6 +373,7 @@ protected void addFields() { addField(PersonDto.PASSPORT_NUMBER); nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, TextFieldCustom.class); + nationalHealthIdField.setNullRepresentation(""); Label nationalHealthIdWarningLabel = new Label(I18nProperties.getString(Strings.messagePersonNationalHealthIdInvalid)); nationalHealthIdWarningLabel.addStyleNames(VSPACE_3, LABEL_WHITE_SPACE_NORMAL); nationalHealthIdWarningLabel.setVisible(false); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java index 8432275a15c..78507946a95 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java @@ -20,6 +20,7 @@ import de.symeda.sormas.api.docgeneneration.DocumentWorkflow; import de.symeda.sormas.api.docgeneneration.RootEntityType; import de.symeda.sormas.api.document.DocumentRelatedEntityType; +import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.person.PersonDto; @@ -32,6 +33,7 @@ import de.symeda.sormas.ui.ControllerProvider; import de.symeda.sormas.ui.SormasUI; import de.symeda.sormas.ui.UiUtil; +import de.symeda.sormas.ui.person.PersonSelectionGrid; import de.symeda.sormas.ui.travelentry.components.TravelEntryCreateForm; import de.symeda.sormas.ui.utils.ArchiveHandlers; import de.symeda.sormas.ui.utils.CommitDiscardWrapperComponent; @@ -43,6 +45,9 @@ public class TravelEntryController { + boolean travelEntrySaveTriggered; + CommitDiscardWrapperComponent travelEntryCreateComponent; + public void registerViews(Navigator navigator) { navigator.addView(TravelEntriesView.VIEW_NAME, TravelEntriesView.class); navigator.addView(TravelEntryDataView.VIEW_NAME, TravelEntryDataView.class); @@ -50,7 +55,7 @@ public void registerViews(Navigator navigator) { } public void create() { - CommitDiscardWrapperComponent travelEntryCreateComponent = getTravelEntryCreateComponent(null, null); + travelEntryCreateComponent = getTravelEntryCreateComponent(null, null); VaadinUiUtil.showModalPopupWindow(travelEntryCreateComponent, I18nProperties.getString(Strings.headingCreateNewTravelEntry)); } @@ -106,14 +111,39 @@ private CommitDiscardWrapperComponent getTravelEntryCreat if (dto.getPerson() == null) { final PersonDto person = createForm.getPerson(); - ControllerProvider.getPersonController() - .selectOrCreatePerson(person, I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), selectedPerson -> { - if (selectedPerson != null) { - dto.setPerson(selectedPerson); - FacadeProvider.getTravelEntryFacade().save(dto); - navigateToTravelEntry(dto.getUuid()); + if (createForm.getWarningSimilarPersons() != null) { + CommitDiscardWrapperComponent warningPopUpDuplicatePerson = + (CommitDiscardWrapperComponent) editView.getWrappedComponent() + .getWarningSimilarPersons() + .getContent(); + warningPopUpDuplicatePerson.getDiscardButton().setVisible(true); + warningPopUpDuplicatePerson.getCommitButton().setCaption(I18nProperties.getCaption(Captions.actionContinue)); + warningPopUpDuplicatePerson.addDoneListener(() -> { + if (!travelEntrySaveTriggered) { + VaadinUiUtil.showModalPopupWindow(editView, I18nProperties.getString(Strings.headingCreateNewImmunization)); } - }, true); + }); + warningPopUpDuplicatePerson.addCommitListener(() -> { + travelEntrySaveTriggered = true; + ControllerProvider.getPersonController() + .selectOrCreatePerson(person, I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), selectedPerson -> { + if (selectedPerson != null) { + dto.setPerson(selectedPerson); + FacadeProvider.getTravelEntryFacade().save(dto); + navigateToTravelEntry(dto.getUuid()); + } + }, true); + }); + } else { + ControllerProvider.getPersonController() + .selectOrCreatePerson(person, I18nProperties.getString(Strings.infoSelectOrCreatePersonForCase), selectedPerson -> { + if (selectedPerson != null) { + dto.setPerson(selectedPerson); + FacadeProvider.getTravelEntryFacade().save(dto); + navigateToTravelEntry(dto.getUuid()); + } + }, true); + } } else { FacadeProvider.getTravelEntryFacade().save(dto); navigateToTravelEntry(dto.getUuid()); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java index f515eba3dfd..0e8e974759a 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/components/TravelEntryCreateForm.java @@ -17,6 +17,7 @@ import org.apache.commons.collections4.CollectionUtils; import com.vaadin.ui.Label; +import com.vaadin.ui.Window; import com.vaadin.v7.data.util.converter.Converter; import com.vaadin.v7.ui.CheckBox; import com.vaadin.v7.ui.ComboBox; @@ -44,6 +45,7 @@ import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.UserProvider; import de.symeda.sormas.ui.person.PersonCreateForm; +import de.symeda.sormas.ui.person.PersonFormHelper; import de.symeda.sormas.ui.travelentry.DEAFormBuilder; import de.symeda.sormas.ui.utils.AbstractEditForm; import de.symeda.sormas.ui.utils.FieldHelper; @@ -86,6 +88,7 @@ public class TravelEntryCreateForm extends AbstractEditForm { private PersonCreateForm personCreateForm; private final PersonReferenceDto personDto; + private Window warningSimilarPersons; public TravelEntryCreateForm() { this(null); @@ -156,6 +159,10 @@ protected void addFields() { personCreateForm.setWidth(100, Unit.PERCENTAGE); personCreateForm.setValue(new PersonDto()); getContent().addComponent(personCreateForm, TravelEntryDto.PERSON); + personCreateForm.getNationalHealthIdField().addTextFieldValueChangeListener(e -> { + warningSimilarPersons = PersonFormHelper + .warningSimilarPersons(personCreateForm.getNationalHealthIdField().getValue(), null, () -> warningSimilarPersons = null); + }); regionCombo.addItems(FacadeProvider.getRegionFacade().getAllActiveByServerCountry()); regionCombo.addValueChangeListener(e -> { @@ -315,4 +322,8 @@ public void setValue(TravelEntryDto newFieldValue) throws ReadOnlyException, Con responsibleCommunity.setReadOnly(true); } } + + public Window getWarningSimilarPersons() { + return warningSimilarPersons; + } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java index 4371c0716d2..49d41d45e09 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java @@ -2,60 +2,32 @@ import java.lang.reflect.Method; -import com.vaadin.event.SerializableEventListener; import com.vaadin.shared.Registration; -import com.vaadin.ui.Component; -import com.vaadin.util.ReflectTools; import com.vaadin.v7.data.Property; -import com.vaadin.v7.ui.AbstractField; import com.vaadin.v7.ui.TextField; public class TextFieldCustom extends TextField { - /* Value change events */ + /* Value change events */ - private static final Method VALUE_CHANGE_METHOD; + private static final Method VALUE_CHANGE_METHOD; - static { - try { - VALUE_CHANGE_METHOD = Property.ValueChangeListener.class - .getDeclaredMethod("valueChange", - new Class[] { Property.ValueChangeEvent.class }); - } catch (final NoSuchMethodException e) { - // This should never happen - throw new RuntimeException( - "Internal error finding methods in AbstractField"); - } - } - - public Registration addTextFieldValueChangeListener(Property.ValueChangeListener listener) { - Registration registration = addListener(ValueChangeEvent.class, listener, - VALUE_CHANGE_METHOD); - // ensure "automatic immediate handling" works - markAsDirty(); - return registration; - } - - public interface TextFieldValueChangeListener extends SerializableEventListener { - - Method TEXTFIELD_CHANGE_METHOD = - ReflectTools.findMethod(TextFieldValueChangeListener.class, "textFieldValueChange", TextFieldValueChangeEvent.class); - - void textFieldValueChange(TextFieldValueChangeEvent event); - - } - - public static final class TextFieldValueChangeEvent extends Component.Event { - - private static final long serialVersionUID = 7689979222604215471L; - - public TextFieldValueChangeEvent(TextField source) { - super(source); - } - - public TextField getTextField() { - return (TextField) getComponent(); + static { + try { + VALUE_CHANGE_METHOD = Property.ValueChangeListener.class.getDeclaredMethod( + "valueChange", + new Class[] { + Property.ValueChangeEvent.class }); + } catch (final NoSuchMethodException e) { + // This should never happen + throw new RuntimeException("Internal error finding methods in AbstractField"); } } + public Registration addTextFieldValueChangeListener(Property.ValueChangeListener listener) { + Registration registration = addListener(ValueChangeEvent.class, listener, VALUE_CHANGE_METHOD); + // ensure "automatic immediate handling" works + markAsDirty(); + return registration; + } } From 6c9f438f81839e772f055633aa3a33c806f4ee87 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Fri, 6 Dec 2024 13:55:22 +0200 Subject: [PATCH 38/56] #13181 - Enhance National Health ID Search Functionality Across Modules and Address Duplication Issues - changes after review --- sormas-api/src/main/resources/strings.properties | 2 +- .../de/symeda/sormas/ui/caze/CaseController.java | 4 ++-- .../symeda/sormas/ui/contact/ContactController.java | 4 ++-- .../ui/immunization/ImmunizationController.java | 4 ++-- .../de/symeda/sormas/ui/person/PersonCreateForm.java | 8 ++++---- .../de/symeda/sormas/ui/person/PersonEditForm.java | 6 +++--- .../sormas/ui/travelentry/TravelEntryController.java | 4 ++-- .../de/symeda/sormas/ui/utils/AbstractEditForm.java | 12 ------------ .../ui/utils/SormasFieldGroupFieldFactory.java | 6 +++--- .../{TextFieldCustom.java => SormasTextField.java} | 2 +- 10 files changed, 20 insertions(+), 32 deletions(-) rename sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/{TextFieldCustom.java => SormasTextField.java} (94%) diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index 7f03cdf3778..2e9723c6cf6 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -697,7 +697,7 @@ headingShowExternalMessage = Message headingSelfReportSideComponent = Self reports headingSignsAndSymptoms = Clinical Signs and Symptoms headingSimilarImmunization = Similar immunizaton -headingSimilarPerson = Similar persons found based on person introduced data +headingSimilarPerson = There are other persons with similar national health Id headingSyncUsers = Sync Users headingTasksDeleted = Tasks deleted headingTasksNotDeleted = None of the tasks were deleted diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java index c30af8de040..2c0b8db9493 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseController.java @@ -152,8 +152,8 @@ public class CaseController { - CommitDiscardWrapperComponent caseCreateComponent; - boolean caseSaveTriggered; + private CommitDiscardWrapperComponent caseCreateComponent; + private boolean caseSaveTriggered; public CaseController() { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java index 5b0754a3562..24a49fb16e7 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactController.java @@ -111,8 +111,8 @@ public class ContactController { protected final Logger logger = LoggerFactory.getLogger(getClass()); - boolean contactSaveTriggered; - CommitDiscardWrapperComponent createComponent; + private boolean contactSaveTriggered; + private CommitDiscardWrapperComponent createComponent; public ContactController() { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java index 6288c273cb2..7e535b5e0cd 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/immunization/ImmunizationController.java @@ -41,8 +41,8 @@ public class ImmunizationController { - boolean immunizationSaveTriggered; - CommitDiscardWrapperComponent immunizationCreateComponent; + private boolean immunizationSaveTriggered; + private CommitDiscardWrapperComponent immunizationCreateComponent; public void registerViews(Navigator navigator) { navigator.addView(ImmunizationsView.VIEW_NAME, ImmunizationsView.class); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java index a1e9d757967..0827f474a00 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonCreateForm.java @@ -66,7 +66,7 @@ import de.symeda.sormas.ui.utils.FieldHelper; import de.symeda.sormas.ui.utils.PhoneNumberValidator; import de.symeda.sormas.ui.utils.VaadinUiUtil; -import de.symeda.sormas.ui.utils.components.TextFieldCustom; +import de.symeda.sormas.ui.utils.components.SormasTextField; public class PersonCreateForm extends AbstractEditForm { @@ -88,7 +88,7 @@ public class PersonCreateForm extends AbstractEditForm { private final boolean showPresentCondition; private final boolean showSymptomsOnsetDate; private final boolean showPersonSearchButton; - private TextFieldCustom nationalHealthIdField; + private SormasTextField nationalHealthIdField; private Window warningSimilarPersons; private static final String HTML_LAYOUT = @@ -191,7 +191,7 @@ protected void addFields() { addField(PersonDto.PASSPORT_NUMBER, TextField.class); - nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, TextFieldCustom.class); + nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, SormasTextField.class); nationalHealthIdField.setNullRepresentation(""); ComboBox presentCondition = addField(PersonDto.PRESENT_CONDITION, ComboBox.class); @@ -509,7 +509,7 @@ public Window getWarningSimilarPersons() { return warningSimilarPersons; } - public TextFieldCustom getNationalHealthIdField() { + public SormasTextField getNationalHealthIdField() { return nationalHealthIdField; } } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java index ec3427adb6f..c1d2af88566 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonEditForm.java @@ -96,7 +96,7 @@ import de.symeda.sormas.ui.utils.SormasFieldGroupFieldFactory; import de.symeda.sormas.ui.utils.ValidationUtils; import de.symeda.sormas.ui.utils.ViewMode; -import de.symeda.sormas.ui.utils.components.TextFieldCustom; +import de.symeda.sormas.ui.utils.components.SormasTextField; public class PersonEditForm extends AbstractEditForm { @@ -187,7 +187,7 @@ public class PersonEditForm extends AbstractEditForm { private boolean isPseudonymized; private LocationEditForm addressForm; private PresentConditionChangeListener presentConditionChangeListener; - private TextFieldCustom nationalHealthIdField; + private SormasTextField nationalHealthIdField; private Window warningSimilarPersons; //@formatter:on @@ -372,7 +372,7 @@ protected void addFields() { addField(PersonDto.PASSPORT_NUMBER); - nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, TextFieldCustom.class); + nationalHealthIdField = addField(PersonDto.NATIONAL_HEALTH_ID, SormasTextField.class); nationalHealthIdField.setNullRepresentation(""); Label nationalHealthIdWarningLabel = new Label(I18nProperties.getString(Strings.messagePersonNationalHealthIdInvalid)); nationalHealthIdWarningLabel.addStyleNames(VSPACE_3, LABEL_WHITE_SPACE_NORMAL); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java index 78507946a95..579d7702030 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/travelentry/TravelEntryController.java @@ -45,8 +45,8 @@ public class TravelEntryController { - boolean travelEntrySaveTriggered; - CommitDiscardWrapperComponent travelEntryCreateComponent; + private boolean travelEntrySaveTriggered; + private CommitDiscardWrapperComponent travelEntryCreateComponent; public void registerViews(Navigator navigator) { navigator.addView(TravelEntriesView.VIEW_NAME, TravelEntriesView.class); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java index dba76e34844..fb3e4203bac 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/AbstractEditForm.java @@ -21,12 +21,9 @@ import java.util.Collection; import java.util.List; import java.util.Optional; -import java.util.Set; import java.util.stream.Collectors; import java.util.stream.Stream; -import com.vaadin.navigator.ViewLeaveAction; -import com.vaadin.ui.Window; import com.vaadin.ui.themes.ValoTheme; import com.vaadin.v7.data.Item; import com.vaadin.v7.data.Validator; @@ -48,17 +45,10 @@ import de.symeda.sormas.api.i18n.Captions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; -import de.symeda.sormas.api.person.PersonDto; -import de.symeda.sormas.api.person.PersonSimilarityCriteria; -import de.symeda.sormas.api.person.Sex; -import de.symeda.sormas.api.person.SimilarPersonDto; import de.symeda.sormas.api.utils.fieldaccess.UiFieldAccessCheckers; import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.clinicalcourse.HealthConditionsForm; -import de.symeda.sormas.ui.person.PersonSelectionGrid; import de.symeda.sormas.ui.utils.components.NotBlankTextValidator; -import org.apache.commons.lang3.ObjectUtils; -import org.apache.commons.lang3.StringUtils; public abstract class AbstractEditForm extends AbstractForm implements FieldGroup.CommitHandler {// implements DtoEditForm { @@ -623,6 +613,4 @@ protected boolean isEditableAllowed(String propertyId) { public void setHeading(String heading) { throw new RuntimeException("setHeading should be implemented in " + getClass().getSimpleName()); } - - } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java index d7cd5d95382..11d08a40b3d 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/SormasFieldGroupFieldFactory.java @@ -43,7 +43,7 @@ import de.symeda.sormas.ui.utils.components.JsonForm; import de.symeda.sormas.ui.utils.components.MultiSelect; import de.symeda.sormas.ui.utils.components.MultiSelectFiles; -import de.symeda.sormas.ui.utils.components.TextFieldCustom; +import de.symeda.sormas.ui.utils.components.SormasTextField; import de.symeda.sormas.ui.vaccination.VaccinationsField; public class SormasFieldGroupFieldFactory extends DefaultFieldGroupFieldFactory { @@ -192,8 +192,8 @@ public T createField(Class type, Class fieldType) { return (T) new CheckBoxTree<>(); } else if (RichTextArea.class.isAssignableFrom(fieldType)) { return (T) new RichTextArea(); - }else if (TextFieldCustom.class.isAssignableFrom(fieldType)) { - return (T) new TextFieldCustom(); + }else if (SormasTextField.class.isAssignableFrom(fieldType)) { + return (T) new SormasTextField(); } return super.createField(type, fieldType); } diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/SormasTextField.java similarity index 94% rename from sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java rename to sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/SormasTextField.java index 49d41d45e09..7245a6c9b8b 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/TextFieldCustom.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/components/SormasTextField.java @@ -6,7 +6,7 @@ import com.vaadin.v7.data.Property; import com.vaadin.v7.ui.TextField; -public class TextFieldCustom extends TextField { +public class SormasTextField extends TextField { /* Value change events */ From bac6746ad7cca0f56b8ef9b0204b5925bd8e36ba Mon Sep 17 00:00:00 2001 From: Levente Gal <62599627+leventegal-she@users.noreply.github.com> Date: Fri, 6 Dec 2024 14:58:57 +0200 Subject: [PATCH 39/56] =?UTF-8?q?#13183=20Add=20New=20Influenza=20Disease?= =?UTF-8?q?=20Types=20and=20Modify=20Display=20for=20SORMAS-=E2=80=A6=20(#?= =?UTF-8?q?13188)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * #13183 Add New Influenza Disease Types and Modify Display for SORMAS-Luxembourg * #13183 Add New Influenza Disease Types and Modify Display for SORMAS-Luxembourg --------- Co-authored-by: Levente Gal --- .../src/main/java/de/symeda/sormas/api/Disease.java | 5 +++-- sormas-api/src/main/resources/enum.properties | 1 + .../src/main/resources/sql/sormas_schema.sql | 13 +++++++++++++ 3 files changed, 17 insertions(+), 2 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/Disease.java b/sormas-api/src/main/java/de/symeda/sormas/api/Disease.java index e1c5c9f7349..f257bcbe37b 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/Disease.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/Disease.java @@ -69,8 +69,9 @@ public enum Disease YAWS_ENDEMIC_SYPHILIS(true, false, false, true, false, 0, true, false, false), MATERNAL_DEATHS(true, false, false, true, false, 0, true, false, false), PERINATAL_DEATHS(true, false, false, true, false, 0, true, false, false), - INFLUENZA_A(true, false, true, false, false, 0, true, false, false), - INFLUENZA_B(true, false, true, false, false, 0, true, false, false), + INFLUENZA(true, true, true, false, false, 0, true, false, false), + INFLUENZA_A(false, false, true, false, false, 0, true, false, false), + INFLUENZA_B(false, false, true, false, false, 0, true, false, false), H_METAPNEUMOVIRUS(true, false, true, false, false, 0, true, false, false), RESPIRATORY_SYNCYTIAL_VIRUS(true, false, true, false, false, 0, true, false, false), PARAINFLUENZA_1_4(true, false, true, false, false, 0, true, false, false), diff --git a/sormas-api/src/main/resources/enum.properties b/sormas-api/src/main/resources/enum.properties index 3cb4942c2ac..f9ed852643b 100644 --- a/sormas-api/src/main/resources/enum.properties +++ b/sormas-api/src/main/resources/enum.properties @@ -501,6 +501,7 @@ Disease.YAWS_ENDEMIC_SYPHILIS = Yaws and Endemic Syphilis Disease.MATERNAL_DEATHS = Maternal Deaths Disease.PERINATAL_DEATHS = Perinatal Deaths Disease.CORONAVIRUS = COVID-19 +Disease.INFLUENZA=Influenza Disease.INFLUENZA_A = Influenza A Disease.INFLUENZA_B = Influenza B Disease.H_METAPNEUMOVIRUS = H.metapneumovirus diff --git a/sormas-backend/src/main/resources/sql/sormas_schema.sql b/sormas-backend/src/main/resources/sql/sormas_schema.sql index 5b55069ae30..f5def4af2a5 100644 --- a/sormas-backend/src/main/resources/sql/sormas_schema.sql +++ b/sormas-backend/src/main/resources/sql/sormas_schema.sql @@ -13698,4 +13698,17 @@ CREATE TRIGGER delete_history_trigger ALTER TABLE documenttemplates_history OWNER TO sormas_user; INSERT INTO schema_version (version_number, comment, upgradeneeded) VALUES (553, 'Add "Disease" Attribute to Document Templates for Filtering #13160', true); + +-- 2024-11-18 Add New Influenza Disease Types and Modify Display for SORMAS-LuxembourgAdd #13183 +INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases, properties) +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', 'A', 'Type A', + 'INFLUENZA', jsonb_build_object('hasDetails', true)); +INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', 'B', 'Type B', + 'INFLUENZA'); +INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases, properties) +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', 'AB', 'Type A+B', + 'INFLUENZA', jsonb_build_object('hasDetails', true)); + +INSERT INTO schema_version (version_number, comment) VALUES (554, 'Add New Influenza Disease Types and Modify Display for SORMAS-LuxembourgAdd #13183'); -- *** Insert new sql commands BEFORE this line. Remember to always consider _history tables. *** From 0bb1fc5c264e56f07d49aa481dac57e855d7ecf4 Mon Sep 17 00:00:00 2001 From: Marin Date: Tue, 10 Dec 2024 11:50:53 +0200 Subject: [PATCH 40/56] Automate Case Details and Fields Preselection for Influenza Cases #13184 - after review comments --- .../java/de/symeda/sormas/api/caze/CaseDataDto.java | 2 +- .../de/symeda/sormas/ui/caze/CaseCreateForm.java | 13 +++++++++++-- 2 files changed, 12 insertions(+), 3 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java index a6b0089068d..aae97fc38d0 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java @@ -638,7 +638,7 @@ public static CaseDataDto build(PersonReferenceDto person, Disease disease, Heal caze.setPortHealthInfo(PortHealthInfoDto.build()); caze.setDisease(disease); caze.setInvestigationStatus(InvestigationStatus.PENDING); - caze.setCaseClassification(CaseClassification.CONFIRMED); + caze.setCaseClassification(CaseClassification.NOT_CLASSIFIED); caze.setOutcome(CaseOutcome.NO_OUTCOME); caze.setCaseOrigin(CaseOrigin.IN_COUNTRY); // TODO This is a workaround for transferring the followup comment while converting a contact to a case. This can be removed if the followup for cases is implemented in the mobile app diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java index 2b8e7811857..bdc58dbb03d 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java @@ -30,7 +30,9 @@ import java.util.Collections; import java.util.Date; import java.util.List; +import java.util.Objects; +import de.symeda.sormas.api.caze.CaseClassification; import org.apache.commons.collections4.CollectionUtils; import com.google.common.collect.Sets; @@ -282,7 +284,6 @@ protected void addFields() { facilityOrHome.setId("facilityOrHome"); facilityOrHome.setWidth(100, Unit.PERCENTAGE); CssStyles.style(facilityOrHome, ValoTheme.OPTIONGROUP_HORIZONTAL); - facilityOrHome.setValue(Sets.newHashSet(TypeOfPlace.HOME)); facilityTypeGroup = ComboBoxHelper.createComboBoxV7(); facilityTypeGroup.setId("typeGroup"); facilityTypeGroup.setCaption(I18nProperties.getCaption(Captions.Facility_typeGroup)); @@ -545,13 +546,21 @@ private void updateDiseaseVariant(Disease disease) { FieldHelper.updateItems(diseaseVariantField, diseaseVariants); diseaseVariantField .setVisible(disease != null && isVisibleAllowed(CaseDataDto.DISEASE_VARIANT) && CollectionUtils.isNotEmpty(diseaseVariants)); + if (Objects.nonNull(disease) && (Disease.INFLUENZA.compareTo(disease) == 0)) { + facilityOrHome.setValue(Sets.newHashSet(TypeOfPlace.HOME)); + facilityOrHome.select(TypeOfPlace.HOME); + getValue().setCaseClassification(CaseClassification.CONFIRMED); + } else { + facilityOrHome.setValue(null); + facilityOrHome.unselect(TypeOfPlace.HOME); + getValue().setCaseClassification(CaseClassification.NOT_CLASSIFIED); + } } private void setNoneFacility() { FacilityReferenceDto noFacilityRef = FacadeProvider.getFacilityFacade().getByUuid(FacilityDto.NONE_FACILITY_UUID).toReference(); facilityCombo.addItem(noFacilityRef); facilityCombo.setValue(noFacilityRef); - facilityType.setRequired(false); } private void updateFacility() { From 206e1ead71f5fd137f1c5a0c4309918070c79e95 Mon Sep 17 00:00:00 2001 From: Marin Date: Tue, 10 Dec 2024 14:58:48 +0200 Subject: [PATCH 41/56] Automate Case Details and Fields Preselection for Influenza Cases #13184 - after review comments. Simplified 1 if statement. --- .../src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java index bdc58dbb03d..6f26274534c 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseCreateForm.java @@ -30,7 +30,6 @@ import java.util.Collections; import java.util.Date; import java.util.List; -import java.util.Objects; import de.symeda.sormas.api.caze.CaseClassification; import org.apache.commons.collections4.CollectionUtils; @@ -546,7 +545,7 @@ private void updateDiseaseVariant(Disease disease) { FieldHelper.updateItems(diseaseVariantField, diseaseVariants); diseaseVariantField .setVisible(disease != null && isVisibleAllowed(CaseDataDto.DISEASE_VARIANT) && CollectionUtils.isNotEmpty(diseaseVariants)); - if (Objects.nonNull(disease) && (Disease.INFLUENZA.compareTo(disease) == 0)) { + if (disease == Disease.INFLUENZA) { facilityOrHome.setValue(Sets.newHashSet(TypeOfPlace.HOME)); facilityOrHome.select(TypeOfPlace.HOME); getValue().setCaseClassification(CaseClassification.CONFIRMED); From 4a46f94073f3cf6c96023bf5308530f60c9c6f36 Mon Sep 17 00:00:00 2001 From: Marin Date: Thu, 12 Dec 2024 13:33:41 +0200 Subject: [PATCH 42/56] RSV disease variants #13204 --- .../src/main/resources/sql/sormas_schema.sql | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/sormas-backend/src/main/resources/sql/sormas_schema.sql b/sormas-backend/src/main/resources/sql/sormas_schema.sql index f5def4af2a5..b8e97a8d26f 100644 --- a/sormas-backend/src/main/resources/sql/sormas_schema.sql +++ b/sormas-backend/src/main/resources/sql/sormas_schema.sql @@ -13711,4 +13711,17 @@ VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VA 'INFLUENZA', jsonb_build_object('hasDetails', true)); INSERT INTO schema_version (version_number, comment) VALUES (554, 'Add New Influenza Disease Types and Modify Display for SORMAS-LuxembourgAdd #13183'); + +-- 2024-12-12 RSV disease variants #13204 +INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', ' A', 'Type A', + 'RESPIRATORY_SYNCYTIAL_VIRUS'); +INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', ' B', 'Type B', + 'RESPIRATORY_SYNCYTIAL_VIRUS'); +INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', ' AB', 'Type A+B', + 'RESPIRATORY_SYNCYTIAL_VIRUS'); + +INSERT INTO schema_version (version_number, comment) VALUES (555, 'RSV disease variants #13204'); -- *** Insert new sql commands BEFORE this line. Remember to always consider _history tables. *** From 922ffdaadc646bfe06179a36cf3f11e9eb8a42ab Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Sat, 14 Dec 2024 15:43:05 +0100 Subject: [PATCH 43/56] fix view title for aefi investigations directory --- .../src/main/java/de/symeda/sormas/api/i18n/Captions.java | 2 +- sormas-api/src/main/resources/captions.properties | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index cac81380ac0..ae028b9b556 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -3127,8 +3127,8 @@ public interface Captions { String versionIsMissing = "versionIsMissing"; String view = "view"; String View_actions = "View.actions"; - String View_adverseeventinvestigations = "View.adverseeventinvestigations"; String View_adverseevents = "View.adverseevents"; + String View_adverseevents_investigations = "View.adverseevents.investigations"; String View_aggregatereports = "View.aggregatereports"; String View_aggregatereports_aggregatereporting = "View.aggregatereports.aggregatereporting"; String View_aggregatereports_reportdata = "View.aggregatereports.reportdata"; diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index 1a1c0b040a9..4c7f34f7279 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -3122,7 +3122,7 @@ View.samples.sub= View.travelEntries=Travel Entries Directory View.immunizations=Immunization Directory View.adverseevents=Adverse Events Directory -View.adverseeventinvestigations=AEFI Investigations Directory +View.adverseevents.investigations=AEFI Investigations Directory View.statistics=Statistics View.statistics.database-export=Database export View.tasks=Task Management From 8439dd109c501ef5f7fb944363c93d70e00cd6b7 Mon Sep 17 00:00:00 2001 From: Obinna Henry <55580796+obinna-h-n@users.noreply.github.com> Date: Sat, 14 Dec 2024 15:48:42 +0100 Subject: [PATCH 44/56] remove giz logo from login screen --- .../main/java/de/symeda/sormas/ui/login/LoginScreen.java | 6 ------ 1 file changed, 6 deletions(-) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/login/LoginScreen.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/login/LoginScreen.java index bd13ce6b7f3..f5c100950db 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/login/LoginScreen.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/login/LoginScreen.java @@ -315,12 +315,6 @@ private Layout buildLoginSidebarLayout() { imgSormasFoundation.setWidth(isCustomBranding ? CUSTOM_BRANDING_LOGO_WIDTH : LOGO_WIDTH, Unit.PIXELS); poweredByLayout.addComponent(imgSormasFoundation); - if (!isCustomBranding) { - Image imgGiz = new Image(null, new ThemeResource("img/giz-logo.png")); - imgGiz.setWidth(isCustomBranding ? CUSTOM_BRANDING_LOGO_WIDTH : LOGO_WIDTH, Unit.PIXELS); - poweredByLayout.addComponent(imgGiz); - } - loginSidebarLayout.addComponent(poweredByLayout); Label customSidebarLabel = new Label(); From b692cbf3916e49879636956b175149a4796eeeb1 Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Tue, 17 Dec 2024 10:44:37 +0200 Subject: [PATCH 45/56] #13190 - Add date range filter for birthdate (from-to) to Persons, Cases, and Contacts --- .../symeda/sormas/api/caze/CaseCriteria.java | 28 ++++ .../sormas/api/contact/ContactCriteria.java | 28 ++++ .../de/symeda/sormas/api/i18n/Captions.java | 2 + .../de/symeda/sormas/api/i18n/Strings.java | 5 + .../sormas/api/person/PersonCriteria.java | 29 +++++ .../src/main/resources/captions.properties | 2 + .../src/main/resources/strings.properties | 5 + .../sormas/backend/caze/CaseService.java | 10 ++ .../backend/contact/ContactService.java | 10 ++ .../sormas/backend/person/PersonService.java | 10 ++ .../util/BirthdateRangeFilterPredicate.java | 87 +++++++++++++ .../symeda/sormas/ui/caze/CaseFilterForm.java | 35 ++++- .../sormas/ui/contact/ContactsFilterForm.java | 35 ++++- .../sormas/ui/person/PersonFilterForm.java | 47 +++++++ .../utils/BirthdateRangeFilterComponent.java | 120 ++++++++++++++++++ 15 files changed, 449 insertions(+), 4 deletions(-) create mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/util/BirthdateRangeFilterPredicate.java create mode 100644 sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseCriteria.java index 7e6e398c3d9..0920c1f4874 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseCriteria.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseCriteria.java @@ -106,6 +106,9 @@ public class CaseCriteria extends CriteriaWithDateType implements ExternalShareC private Date newCaseDateTo; private Date creationDateFrom; private Date creationDateTo; + private Date birthdateFrom; + private Date birthdateTo; + private boolean includePartialMatch; private CriteriaDateType newCaseDateType; // Used to re-construct whether users have filtered by epi weeks or dates private DateFilterOption dateFilterOption = DateFilterOption.DATE; @@ -552,6 +555,31 @@ public CaseCriteria creationDateTo(Date creationDateTo) { return this; } + public Date getBirthdateFrom() { + return birthdateFrom; + } + + public void setBirthdateFrom(Date birthdateFrom) { + this.birthdateFrom = birthdateFrom; + } + + public Date getBirthdateTo() { + return birthdateTo; + } + + public void setBirthdateTo(Date birthdateTo) { + this.birthdateTo = birthdateTo; + } + + @IgnoreForUrl + public boolean isIncludePartialMatch() { + return includePartialMatch; + } + + public void setIncludePartialMatch(boolean includePartialMatch) { + this.includePartialMatch = includePartialMatch; + } + public Date getQuarantineTo() { return quarantineTo; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactCriteria.java index 6f1086e30ea..c96d4349842 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactCriteria.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactCriteria.java @@ -136,6 +136,9 @@ public class ContactCriteria extends BaseCriteria implements Serializable { private Boolean onlyContactsFromOtherInstances; private Date creationDateFrom; private Date creationDateTo; + private Date birthdateFrom; + private Date birthdateTo; + private boolean includePartialMatch; private String reportingUserLike; private String personLike; private boolean excludeLimitedSyncRestrictions; @@ -665,6 +668,31 @@ public ContactCriteria creationDateTo(Date creationDateTo) { return this; } + public Date getBirthdateFrom() { + return birthdateFrom; + } + + public void setBirthdateFrom(Date birthdateFrom) { + this.birthdateFrom = birthdateFrom; + } + + public Date getBirthdateTo() { + return birthdateTo; + } + + public void setBirthdateTo(Date birthdateTo) { + this.birthdateTo = birthdateTo; + } + + @IgnoreForUrl + public boolean isIncludePartialMatch() { + return includePartialMatch; + } + + public void setIncludePartialMatch(boolean includePartialMatch) { + this.includePartialMatch = includePartialMatch; + } + public String getReportingUserLike() { return reportingUserLike; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index cac81380ac0..8ba4c4b30b8 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -557,6 +557,7 @@ public interface Captions { String assignmentDate = "assignmentDate"; String assignToMe = "assignToMe"; String BAGExport = "BAGExport"; + String birthdateFilter = "birthdateFilter"; String bulkActionCreatDocuments = "bulkActionCreatDocuments"; String bulkActions = "bulkActions"; String bulkCancelFollowUp = "bulkCancelFollowUp"; @@ -1993,6 +1994,7 @@ public interface Captions { String importSkips = "importSkips"; String importValueSeparator = "importValueSeparator"; String inaccessibleValue = "inaccessibleValue"; + String includePartialBirthdates = "includePartialBirthdates"; String info = "info"; String infrastructureImportAllowOverwrite = "infrastructureImportAllowOverwrite"; String lastName = "lastName"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index 484d972c404..3582175d870 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -705,6 +705,7 @@ public interface Strings { String headingImportSelfReports = "headingImportSelfReports"; String headingImportSubcontinents = "headingImportSubcontinents"; String headingImportTravelEntries = "headingImportTravelEntries"; + String headingIncorrectDateRange = "headingIncorrectDateRange"; String headingInformationSource = "headingInformationSource"; String headingInfrastructureLocked = "headingInfrastructureLocked"; String headingIntroduction = "headingIntroduction"; @@ -921,6 +922,7 @@ public interface Strings { String infoAutomaticDeletionTooltipYears = "infoAutomaticDeletionTooltipYears"; String infoBAGExport = "infoBAGExport"; String infoBasicExport = "infoBasicExport"; + String infoBirthdateFilter = "infoBirthdateFilter"; String infoBulkProcess = "infoBulkProcess"; String infoBulkProcessCancelled = "infoBulkProcessCancelled"; String infoBulkProcessFinished = "infoBulkProcessFinished"; @@ -1419,6 +1421,7 @@ public interface Strings { String messageImportSuccessful = "messageImportSuccessful"; String messageImportSuccessfulWithSkips = "messageImportSuccessfulWithSkips"; String messageIncompleteGpsCoordinates = "messageIncompleteGpsCoordinates"; + String messageIncorrectDateRange = "messageIncorrectDateRange"; String messageInfrastructureLocked = "messageInfrastructureLocked"; String messageInvalidDatesLineListing = "messageInvalidDatesLineListing"; String messageLaboratoriesArchived = "messageLaboratoriesArchived"; @@ -1680,6 +1683,8 @@ public interface Strings { String promptAllDistricts = "promptAllDistricts"; String promptAllRegions = "promptAllRegions"; String promptArea = "promptArea"; + String promptBirthdateFrom = "promptBirthdateFrom"; + String promptBirthdateTo = "promptBirthdateTo"; String promptCampaign = "promptCampaign"; String promptCampaignSearch = "promptCampaignSearch"; String promptCaseOrContactEventSearchField = "promptCaseOrContactEventSearchField"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonCriteria.java b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonCriteria.java index 0ef155b8d25..a78d4eadbef 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonCriteria.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonCriteria.java @@ -1,5 +1,6 @@ package de.symeda.sormas.api.person; +import java.util.Date; import java.util.Set; import de.symeda.sormas.api.infrastructure.community.CommunityReferenceDto; @@ -37,6 +38,9 @@ public class PersonCriteria extends BaseCriteria implements Cloneable { private CommunityReferenceDto community; private PersonAssociation personAssociation; private Set uuids; + private Date birthdateFrom; + private Date birthdateTo; + private boolean includePartialMatch; public PersonCriteria() { @@ -127,6 +131,31 @@ public PersonCriteria personAssociation(PersonAssociation personAssociation) { return this; } + public Date getBirthdateFrom() { + return birthdateFrom; + } + + public void setBirthdateFrom(Date birthdateFrom) { + this.birthdateFrom = birthdateFrom; + } + + public Date getBirthdateTo() { + return birthdateTo; + } + + public void setBirthdateTo(Date birthdateTo) { + this.birthdateTo = birthdateTo; + } + + @IgnoreForUrl + public boolean isIncludePartialMatch() { + return includePartialMatch; + } + + public void setIncludePartialMatch(boolean includePartialMatch) { + this.includePartialMatch = includePartialMatch; + } + @IgnoreForUrl public Set getUuids() { return uuids; diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index 1a1c0b040a9..66f6dd768ca 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -52,6 +52,7 @@ creationDate=Creation date changeDate=Date of last change notAvailableShort=NA inaccessibleValue=Confidential +includePartialBirthdates = Include partial birthdates numberOfCharacters=Number of characters: %d / %d remove=Remove notTestedYet=Not tested yet @@ -71,6 +72,7 @@ adoptHomeAddressOfCasePersonIfRelationMatches=Adopt home address of the case per casePersonAddress=Address of the case person viewMessage=View message primarySuffix=primary +birthdateFilter = Birthdate filter # About about=About aboutAdditionalInfo=Additional Info diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index 2e9723c6cf6..79d6cf14f59 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -620,6 +620,7 @@ headingImportRegions= Import Regions headingImportTravelEntries = Import Travel Entries headingImportEnvironments = Import Environments headingImportSelfReports = Import Self Reports +headingIncorrectDateRange = Incorrect date range headingInformationSource = Source of Information headingInfrastructureLocked = Infrastructure locked headingIntroduction = Introduction @@ -1115,6 +1116,7 @@ infoAefiSelectPrimarySuspectVaccine = The list below contains all vaccinations o infoArchivedAefiEntries = Adverse event entries are automatically archived after %d days without changes to the data. infoNoAefiInvestigations = No investigations have been created for this adverse event infoHeadingAefiDashboardMap=Adverse events are shown using the GPS coordinate of the facility or person's home address. +infoBirthdateFilter = If checked the search will include also the persons that have incomplete birthdate and have only higher level match eg. only year or only year and month # Messages messageActionOutsideJurisdictionDeletionDenied = The action outside user's jurisdiction cannot be deleted @@ -1314,6 +1316,7 @@ messageImportSuccessful = Import successful!
All rows have been impor messageImportSuccessfulWithSkips = Import successful!
The import has been successful, but some of the rows were skipped. You can now close this window. messageUploadSuccessful = Upload successful! You can now close this window. messageIncompleteGpsCoordinates = GPS coordinates are incomplete +messageIncorrectDateRange = Date from is after date to messageExternalMessagesAssigned = The assignee has been changed for all selected messages messageLoginFailed = Please check your username and password and try again messageMissingCases = Please generate some cases before generating contacts @@ -1651,6 +1654,8 @@ promptActionChangeDateFrom = Date of action change from... promptActionChangeDateTo = ... to promptActionChangeEpiWeekFrom = Date of action change from epi week... promptActionChangeEpiWeekTo = ... to epi week +promptBirthdateFrom = Birthdate from +promptBirthdateTo = Birthdate to promptCampaignSearch = ID, name promptCasesDateFrom = New cases from... promptCasesEpiWeekFrom = New cases from epi week... diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java index fd359454347..ebcd4065d2e 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java @@ -177,6 +177,7 @@ import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.user.UserRole; import de.symeda.sormas.backend.user.UserService; +import de.symeda.sormas.backend.util.BirthdateRangeFilterPredicate; import de.symeda.sormas.backend.util.ExternalDataUtil; import de.symeda.sormas.backend.util.IterableHelper; import de.symeda.sormas.backend.util.JurisdictionHelper; @@ -912,6 +913,15 @@ public Predicate createCriteriaFilter(CaseCrite filter = CriteriaBuilderHelper.and(cb, filter, likeFilters); } } + + filter = BirthdateRangeFilterPredicate.createBirthdateRangeFilter( + caseCriteria.getBirthdateFrom(), + caseCriteria.getBirthdateTo(), + caseCriteria.isIncludePartialMatch(), + cb, + joins.getPerson(), + filter); + if (caseCriteria.getBirthdateYYYY() != null) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(joins.getPerson().get(Person.BIRTHDATE_YYYY), caseCriteria.getBirthdateYYYY())); } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java index 07b51eb8efa..3dfcd3377bf 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java @@ -137,6 +137,7 @@ import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.user.UserRole; import de.symeda.sormas.backend.user.UserService; +import de.symeda.sormas.backend.util.BirthdateRangeFilterPredicate; import de.symeda.sormas.backend.util.ExternalDataUtil; import de.symeda.sormas.backend.util.IterableHelper; import de.symeda.sormas.backend.util.JurisdictionHelper; @@ -1419,6 +1420,15 @@ public Predicate buildCriteriaFilter(ContactCriteria contactCriteria, ContactQue if (contactCriteria.getPerson() != null) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(joins.getPerson().get(Person.UUID), contactCriteria.getPerson().getUuid())); } + + filter = BirthdateRangeFilterPredicate.createBirthdateRangeFilter( + contactCriteria.getBirthdateFrom(), + contactCriteria.getBirthdateTo(), + contactCriteria.isIncludePartialMatch(), + cb, + joins.getPerson(), + filter); + if (contactCriteria.getBirthdateYYYY() != null) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(joins.getPerson().get(Person.BIRTHDATE_YYYY), contactCriteria.getBirthdateYYYY())); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java index 06c40972314..57d37c49f9f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonService.java @@ -114,6 +114,7 @@ import de.symeda.sormas.backend.travelentry.services.TravelEntryService; import de.symeda.sormas.backend.user.User; import de.symeda.sormas.backend.user.UserService; +import de.symeda.sormas.backend.util.BirthdateRangeFilterPredicate; import de.symeda.sormas.backend.util.ExternalDataUtil; import de.symeda.sormas.backend.util.IterableHelper; import de.symeda.sormas.backend.util.JurisdictionHelper; @@ -386,6 +387,15 @@ public Predicate buildCriteriaFilter(PersonCriteria personCriteria, PersonQueryC filter = andEquals(cb, personFrom, filter, personCriteria.getBirthdateYYYY(), Person.BIRTHDATE_YYYY); filter = andEquals(cb, personFrom, filter, personCriteria.getBirthdateMM(), Person.BIRTHDATE_MM); filter = andEquals(cb, personFrom, filter, personCriteria.getBirthdateDD(), Person.BIRTHDATE_DD); + + filter = BirthdateRangeFilterPredicate.createBirthdateRangeFilter( + personCriteria.getBirthdateFrom(), + personCriteria.getBirthdateTo(), + personCriteria.isIncludePartialMatch(), + cb, + personFrom, + filter); + if (personCriteria.getNameAddressPhoneEmailLike() != null) { String[] textFilters = personCriteria.getNameAddressPhoneEmailLike().split("\\s+"); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/util/BirthdateRangeFilterPredicate.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/util/BirthdateRangeFilterPredicate.java new file mode 100644 index 00000000000..b2dcfcbd287 --- /dev/null +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/util/BirthdateRangeFilterPredicate.java @@ -0,0 +1,87 @@ +package de.symeda.sormas.backend.util; + +import java.util.Calendar; +import java.util.Date; + +import javax.persistence.criteria.CriteriaBuilder; +import javax.persistence.criteria.From; +import javax.persistence.criteria.Predicate; + +import de.symeda.sormas.backend.common.CriteriaBuilderHelper; +import de.symeda.sormas.backend.person.Person; + +public class BirthdateRangeFilterPredicate { + + public static Predicate createBirthdateRangeFilter( + Date birthdateFrom, + Date birthdateTo, + boolean includePartialMatch, + CriteriaBuilder cb, + From personFrom, + Predicate filter) { + if (birthdateFrom != null) { + Calendar calendarBirthdateFrom = Calendar.getInstance(); + calendarBirthdateFrom.setTime(birthdateFrom); + int birthdateFromCriteriaYear = calendarBirthdateFrom.get(Calendar.YEAR); + int birthdateFromCriteriaMonth = calendarBirthdateFrom.get(Calendar.MONTH) + 1; + int birthdateFromCriteriaDay = calendarBirthdateFrom.get(Calendar.DAY_OF_MONTH); + + Predicate yearPredicate = cb.greaterThan(personFrom.get(Person.BIRTHDATE_YYYY), birthdateFromCriteriaYear); + + Predicate monthPredicate = cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateFromCriteriaYear); + monthPredicate = cb.and(monthPredicate, cb.greaterThan(personFrom.get(Person.BIRTHDATE_MM), birthdateFromCriteriaMonth)); + + Predicate dayPredicate = cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateFromCriteriaYear); + dayPredicate = cb.and(dayPredicate, cb.equal(personFrom.get(Person.BIRTHDATE_MM), birthdateFromCriteriaMonth)); + dayPredicate = cb.and(dayPredicate, cb.greaterThanOrEqualTo(personFrom.get(Person.BIRTHDATE_DD), birthdateFromCriteriaDay)); + + if (includePartialMatch) { + Predicate sameYearPartialMatchPredicate = cb + .and(cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateFromCriteriaYear), cb.isNull(personFrom.get(Person.BIRTHDATE_MM))); + Predicate sameYearMonthPartialMatchPredicate = cb.and( + cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateFromCriteriaYear), + cb.equal(personFrom.get(Person.BIRTHDATE_MM), birthdateFromCriteriaMonth), + cb.isNull(personFrom.get(Person.BIRTHDATE_DD))); + filter = CriteriaBuilderHelper.and( + cb, + filter, + cb.or(yearPredicate, monthPredicate, dayPredicate, sameYearPartialMatchPredicate, sameYearMonthPartialMatchPredicate)); + } else { + filter = CriteriaBuilderHelper.and(cb, filter, cb.or(yearPredicate, monthPredicate, dayPredicate)); + } + } + + if (birthdateTo != null) { + Calendar calendarBirthdateTo = Calendar.getInstance(); + calendarBirthdateTo.setTime(birthdateTo); + int birthdateToCriteriaYear = calendarBirthdateTo.get(Calendar.YEAR); + int birthdateToCriteriaMonth = calendarBirthdateTo.get(Calendar.MONTH) + 1; + int birthdateToCriteriaDay = calendarBirthdateTo.get(Calendar.DAY_OF_MONTH); + + Predicate yearPredicate = cb.lessThan(personFrom.get(Person.BIRTHDATE_YYYY), birthdateToCriteriaYear); + + Predicate monthPredicate = cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateToCriteriaYear); + monthPredicate = cb.and(monthPredicate, cb.lessThan(personFrom.get(Person.BIRTHDATE_MM), birthdateToCriteriaMonth)); + + Predicate dayPredicate = cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateToCriteriaYear); + dayPredicate = cb.and(dayPredicate, cb.equal(personFrom.get(Person.BIRTHDATE_MM), birthdateToCriteriaMonth)); + dayPredicate = cb.and(dayPredicate, cb.lessThanOrEqualTo(personFrom.get(Person.BIRTHDATE_DD), birthdateToCriteriaDay)); + + if (includePartialMatch) { + Predicate sameYearPartialMatchPredicate = + cb.and(cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateToCriteriaYear), cb.isNull(personFrom.get(Person.BIRTHDATE_MM))); + Predicate sameYearMonthPartialMatchPredicate = cb.and( + cb.equal(personFrom.get(Person.BIRTHDATE_YYYY), birthdateToCriteriaYear), + cb.equal(personFrom.get(Person.BIRTHDATE_MM), birthdateToCriteriaMonth), + cb.isNull(personFrom.get(Person.BIRTHDATE_DD))); + filter = CriteriaBuilderHelper.and( + cb, + filter, + cb.or(yearPredicate, monthPredicate, dayPredicate, sameYearPartialMatchPredicate, sameYearMonthPartialMatchPredicate)); + } else { + filter = CriteriaBuilderHelper.and(cb, filter, cb.or(yearPredicate, monthPredicate, dayPredicate)); + } + } + return filter; + } +} diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseFilterForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseFilterForm.java index cd651e8c8d2..c495eb0d07d 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseFilterForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseFilterForm.java @@ -62,6 +62,7 @@ import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractFilterForm; +import de.symeda.sormas.ui.utils.BirthdateRangeFilterComponent; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.EpiWeekAndDateFilterComponent; import de.symeda.sormas.ui.utils.FieldConfiguration; @@ -72,6 +73,7 @@ public class CaseFilterForm extends AbstractFilterForm { private static final long serialVersionUID = -8326451364091398731L; private static final String WEEK_AND_DATE_FILTER = "moreFilters"; + private static final String BIRTHDATE_RANGE_FILTER = "birthdateRangeFilter"; private static final String MORE_FILTERS_HTML_LAYOUT = filterLocs( CaseCriteria.PRESENT_CONDITION, @@ -114,7 +116,8 @@ public class CaseFilterForm extends AbstractFilterForm { CaseCriteria.ONLY_ENTITIES_SHARED_WITH_EXTERNAL_SURV_TOOL, CaseCriteria.ONLY_ENTITIES_CHANGED_SINCE_LAST_SHARED_WITH_EXTERNAL_SURV_TOOL, CaseCriteria.ONLY_CASES_WITH_DONT_SHARE_WITH_EXTERNAL_SURV_TOOL) - + loc(WEEK_AND_DATE_FILTER); + + loc(WEEK_AND_DATE_FILTER) + + loc(BIRTHDATE_RANGE_FILTER); protected CaseFilterForm() { super( @@ -365,7 +368,8 @@ public void addMoreFilters(CustomLayout moreFiltersContainer) { CaseCriteria.ONLY_CASES_WITH_EVENTS, I18nProperties.getCaption(Captions.caseFilterRelatedToEvent), I18nProperties.getDescription(Descriptions.descCaseFilterRelatedToEvent), - CssStyles.CHECKBOX_FILTER_INLINE)).setVisible(UiUtil.permitted(UserRight.EVENT_VIEW)); + CssStyles.CHECKBOX_FILTER_INLINE)) + .setVisible(UiUtil.permitted(UserRight.EVENT_VIEW)); addField( moreFiltersContainer, @@ -448,6 +452,8 @@ public void addMoreFilters(CustomLayout moreFiltersContainer) { } moreFiltersContainer.addComponent(buildWeekAndDateFilter(isExternalShareEnabled), WEEK_AND_DATE_FILTER); + + moreFiltersContainer.addComponent(buildBirthdayRangeFilter(), BIRTHDATE_RANGE_FILTER); } @Override @@ -829,6 +835,19 @@ private HorizontalLayout buildWeekAndDateFilter(boolean isExternalShareEnabled) return dateFilterRowLayout; } + private HorizontalLayout buildBirthdayRangeFilter() { + BirthdateRangeFilterComponent birthdateRangeFilterComponent = new BirthdateRangeFilterComponent(false, this); + addApplyHandler(e -> onApplyClick(birthdateRangeFilterComponent)); + + HorizontalLayout dateFilterRowLayout = new HorizontalLayout(); + dateFilterRowLayout.setSpacing(true); + dateFilterRowLayout.setSizeUndefined(); + + dateFilterRowLayout.addComponent(birthdateRangeFilterComponent); + + return dateFilterRowLayout; + } + private void onApplyClick(EpiWeekAndDateFilterComponent weekAndDateFilter) { DateFilterOption dateFilterOption = (DateFilterOption) weekAndDateFilter.getDateFilterOptionFilter().getValue(); Date fromDate, toDate; @@ -852,6 +871,18 @@ private void onApplyClick(EpiWeekAndDateFilterComponent weekAn } } + private void onApplyClick(BirthdateRangeFilterComponent birthdateRangeFilter) { + Date birthdateFrom, birthdateTo; + Date dateFrom = birthdateRangeFilter.getDateFromFilter().getValue(); + birthdateFrom = dateFrom != null ? DateHelper.getStartOfDay(dateFrom) : null; + Date dateTo = birthdateRangeFilter.getDateToFilter().getValue(); + birthdateTo = dateTo != null ? DateHelper.getEndOfDay(dateTo) : null; + CaseCriteria criteria = getValue(); + criteria.setBirthdateFrom(birthdateFrom); + criteria.setBirthdateTo(birthdateTo); + criteria.setIncludePartialMatch(birthdateRangeFilter.getIncludePartialMatch().getValue()); + } + @Override public void setValue(CaseCriteria newCriteria) throws ReadOnlyException, Converter.ConversionException { diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactsFilterForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactsFilterForm.java index c7744674782..82167788cf3 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactsFilterForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/contact/ContactsFilterForm.java @@ -49,6 +49,7 @@ import de.symeda.sormas.api.utils.fieldvisibility.FieldVisibilityCheckers; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractFilterForm; +import de.symeda.sormas.ui.utils.BirthdateRangeFilterComponent; import de.symeda.sormas.ui.utils.CssStyles; import de.symeda.sormas.ui.utils.EpiWeekAndDateFilterComponent; import de.symeda.sormas.ui.utils.FieldConfiguration; @@ -60,6 +61,7 @@ public class ContactsFilterForm extends AbstractFilterForm { private static final String DISTRICT_INFO_LABEL_ID = "infoContactsViewRegionDistrictFilter"; private static final String WEEK_AND_DATE_FILTER = "moreFilters"; + private static final String BIRTHDATE_RANGE_FILTER = "birthdateRangeFilter"; private static final String CHECKBOX_STYLE = CssStyles.CHECKBOX_FILTER_INLINE + " " + CssStyles.VSPACE_3; @@ -92,7 +94,8 @@ public class ContactsFilterForm extends AbstractFilterForm { ContactCriteria.ONLY_CONTACTS_FROM_OTHER_INSTANCES, ContactCriteria.INCLUDE_CONTACTS_FROM_OTHER_JURISDICTIONS) - + loc(WEEK_AND_DATE_FILTER); + + loc(WEEK_AND_DATE_FILTER) + + loc(BIRTHDATE_RANGE_FILTER); protected ContactsFilterForm() { super( @@ -320,7 +323,8 @@ public void addMoreFilters(CustomLayout moreFiltersContainer) { ContactCriteria.ONLY_CONTACTS_SHARING_EVENT_WITH_SOURCE_CASE, I18nProperties.getCaption(Captions.contactOnlyWithSharedEventWithSourceCase), null, - CHECKBOX_STYLE)).setVisible(UiUtil.permitted(UserRight.EVENT_VIEW)); + CHECKBOX_STYLE)) + .setVisible(UiUtil.permitted(UserRight.EVENT_VIEW)); addField( moreFiltersContainer, @@ -344,6 +348,8 @@ public void addMoreFilters(CustomLayout moreFiltersContainer) { } moreFiltersContainer.addComponent(buildWeekAndDateFilter(), WEEK_AND_DATE_FILTER); + + moreFiltersContainer.addComponent(buildBirthdayRangeFilter(), BIRTHDATE_RANGE_FILTER); } @Override @@ -526,6 +532,19 @@ private HorizontalLayout buildWeekAndDateFilter() { return dateFilterRowLayout; } + private HorizontalLayout buildBirthdayRangeFilter() { + BirthdateRangeFilterComponent birthdateRangeFilterComponent = new BirthdateRangeFilterComponent(false, this); + addApplyHandler(e -> onApplyClick(birthdateRangeFilterComponent)); + + HorizontalLayout dateFilterRowLayout = new HorizontalLayout(); + dateFilterRowLayout.setSpacing(true); + dateFilterRowLayout.setSizeUndefined(); + + dateFilterRowLayout.addComponent(birthdateRangeFilterComponent); + + return dateFilterRowLayout; + } + private void onApplyClick(EpiWeekAndDateFilterComponent weekAndDateFilter) { ContactCriteria criteria = getValue(); @@ -555,6 +574,18 @@ private void onApplyClick(EpiWeekAndDateFilterComponent weekAnd } } + private void onApplyClick(BirthdateRangeFilterComponent birthdateRangeFilter) { + Date birthdateFrom, birthdateTo; + Date dateFrom = birthdateRangeFilter.getDateFromFilter().getValue(); + birthdateFrom = dateFrom != null ? DateHelper.getStartOfDay(dateFrom) : null; + Date dateTo = birthdateRangeFilter.getDateToFilter().getValue(); + birthdateTo = dateTo != null ? DateHelper.getEndOfDay(dateTo) : null; + ContactCriteria criteria = getValue(); + criteria.setBirthdateFrom(birthdateFrom); + criteria.setBirthdateTo(birthdateTo); + criteria.setIncludePartialMatch(birthdateRangeFilter.getIncludePartialMatch().getValue()); + } + private void populateContactResponsiblesForRegion(RegionReferenceDto regionReferenceDto) { List items = fetchContactResponsiblesByRegion(regionReferenceDto != null ? regionReferenceDto : currentUserDto().getRegion()); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFilterForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFilterForm.java index 399151f8558..4b727d62fdd 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFilterForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFilterForm.java @@ -1,5 +1,11 @@ package de.symeda.sormas.ui.person; +import static de.symeda.sormas.ui.utils.LayoutUtil.loc; + +import java.util.Date; + +import com.vaadin.ui.CustomLayout; +import com.vaadin.ui.HorizontalLayout; import com.vaadin.v7.data.Property; import com.vaadin.v7.ui.ComboBox; import com.vaadin.v7.ui.TextField; @@ -19,11 +25,16 @@ import de.symeda.sormas.api.utils.DateHelper; import de.symeda.sormas.ui.UiUtil; import de.symeda.sormas.ui.utils.AbstractFilterForm; +import de.symeda.sormas.ui.utils.BirthdateRangeFilterComponent; import de.symeda.sormas.ui.utils.FieldConfiguration; import de.symeda.sormas.ui.utils.FieldHelper; public class PersonFilterForm extends AbstractFilterForm { + private static final String BIRTHDATE_RANGE_FILTER = "birthdateRangeFilter"; + + private static final String MORE_FILTERS_HTML_LAYOUT = loc(BIRTHDATE_RANGE_FILTER); + protected PersonFilterForm() { super( PersonCriteria.class, @@ -44,6 +55,11 @@ protected String[] getMainFilterLocators() { PersonCriteria.COMMUNITY }; } + @Override + protected String createMoreFiltersHtmlLayout() { + return MORE_FILTERS_HTML_LAYOUT; + } + @Override protected void addFields() { @@ -78,6 +94,37 @@ protected void addFields() { FieldConfiguration.withCaptionAndPixelSized(PersonCriteria.COMMUNITY, I18nProperties.getCaption(Captions.personCommunityPrompt), 140)); } + @Override + public void addMoreFilters(CustomLayout moreFiltersContainer) { + + moreFiltersContainer.addComponent(buildBirthdayRangeFilter(), BIRTHDATE_RANGE_FILTER); + } + + private HorizontalLayout buildBirthdayRangeFilter() { + BirthdateRangeFilterComponent birthdateRangeFilterComponent = new BirthdateRangeFilterComponent(false, this); + addApplyHandler(e -> onApplyClick(birthdateRangeFilterComponent)); + + HorizontalLayout dateFilterRowLayout = new HorizontalLayout(); + dateFilterRowLayout.setSpacing(true); + dateFilterRowLayout.setSizeUndefined(); + + dateFilterRowLayout.addComponent(birthdateRangeFilterComponent); + + return dateFilterRowLayout; + } + + private void onApplyClick(BirthdateRangeFilterComponent birthdateRangeFilter) { + Date birthdateFrom, birthdateTo; + Date dateFrom = birthdateRangeFilter.getDateFromFilter().getValue(); + birthdateFrom = dateFrom != null ? DateHelper.getStartOfDay(dateFrom) : null; + Date dateTo = birthdateRangeFilter.getDateToFilter().getValue(); + birthdateTo = dateTo != null ? DateHelper.getEndOfDay(dateTo) : null; + PersonCriteria criteria = getValue(); + criteria.setBirthdateFrom(birthdateFrom); + criteria.setBirthdateTo(birthdateTo); + criteria.setIncludePartialMatch(birthdateRangeFilter.getIncludePartialMatch().getValue()); + } + @Override protected void applyDependenciesOnFieldChange(String propertyId, Property.ValueChangeEvent event) { super.applyDependenciesOnFieldChange(propertyId, event); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java new file mode 100644 index 00000000000..6d68a157d4a --- /dev/null +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java @@ -0,0 +1,120 @@ +package de.symeda.sormas.ui.utils; + +import static de.symeda.sormas.ui.utils.CssStyles.CHECKBOX_FILTER_INLINE; +import static de.symeda.sormas.ui.utils.CssStyles.LABEL_BOLD; +import static de.symeda.sormas.ui.utils.CssStyles.VSPACE_TOP_4; + +import java.util.Calendar; +import java.util.Date; + +import com.vaadin.icons.VaadinIcons; +import com.vaadin.server.Page; +import com.vaadin.shared.ui.ContentMode; +import com.vaadin.ui.HorizontalLayout; +import com.vaadin.ui.Label; +import com.vaadin.ui.Notification; +import com.vaadin.v7.ui.CheckBox; +import com.vaadin.v7.ui.PopupDateField; + +import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.I18nProperties; +import de.symeda.sormas.api.i18n.Strings; +import de.symeda.sormas.api.utils.DateHelper; + +public class BirthdateRangeFilterComponent extends HorizontalLayout { + + private static final long serialVersionUID = 8752630393182144501L; + + private final PopupDateField dateFromFilter; + private final PopupDateField dateToFilter; + private final CheckBox includePartialMatch; + + public BirthdateRangeFilterComponent(boolean showCaption, AbstractFilterForm parentFilterForm) { + setSpacing(true); + + Calendar c = Calendar.getInstance(); + c.setTime(new Date()); + + dateFromFilter = new PopupDateField(); + dateToFilter = new PopupDateField(); + includePartialMatch = new CheckBox(); + + Label birthdateLabel = new Label(); + birthdateLabel.setValue(I18nProperties.getCaption(Captions.birthdateFilter)); + birthdateLabel.addStyleName(CHECKBOX_FILTER_INLINE); + birthdateLabel.addStyleName(LABEL_BOLD); + birthdateLabel.addStyleName(VSPACE_TOP_4); + addComponent(birthdateLabel); + + addComponent(dateFromFilter); + addComponent(dateToFilter); + + // Date filter + dateFromFilter.setDateFormat(DateFormatHelper.getDateFormatPattern()); + dateFromFilter.setId("dateFrom"); + dateFromFilter.setWidth(200, Unit.PIXELS); + if (showCaption) { + dateFromFilter.setCaption(I18nProperties.getCaption(Captions.from)); + } + dateFromFilter.setInputPrompt(I18nProperties.getString(Strings.promptBirthdateFrom)); + + dateToFilter.setDateFormat(DateFormatHelper.getDateFormatPattern()); + dateToFilter.setId("dateTo"); + dateToFilter.setWidth(200, Unit.PIXELS); + if (showCaption) { + dateToFilter.setCaption(I18nProperties.getCaption(Captions.to)); + } + dateToFilter.setInputPrompt(I18nProperties.getString(Strings.promptBirthdateTo)); + + dateFromFilter.addValueChangeListener(e -> { + Date dateFrom = (Date) e.getProperty().getValue(); + Date dateTo = dateToFilter.getValue(); + notifyIfIncorrectRange(dateFrom, dateTo); + parentFilterForm.onChange(); + }); + + dateToFilter.addValueChangeListener(e -> { + Date dateTo = (Date) e.getProperty().getValue(); + Date dateFrom = dateFromFilter.getValue(); + notifyIfIncorrectRange(dateFrom, dateTo); + parentFilterForm.onChange(); + }); + + includePartialMatch.setCaption(I18nProperties.getCaption(Captions.includePartialBirthdates)); + includePartialMatch.addStyleName(VSPACE_TOP_4); + addComponent(includePartialMatch); + + Label infoLabel = new Label(VaadinIcons.INFO_CIRCLE.getHtml(), ContentMode.HTML); + infoLabel.setSizeUndefined(); + infoLabel.setDescription(I18nProperties.getString(Strings.infoBirthdateFilter)); + CssStyles.style(infoLabel, CssStyles.LABEL_XLARGE, CssStyles.LABEL_SECONDARY); + addComponent(infoLabel); + } + + private static void notifyIfIncorrectRange(Date dateFrom, Date dateTo) { + if (dateFrom != null & dateTo != null) { + if (DateHelper.isDateAfter(dateFrom, dateTo)) { + Notification notification = new Notification( + I18nProperties.getString(Strings.headingIncorrectDateRange), + I18nProperties.getString(Strings.messageIncorrectDateRange), + Notification.Type.WARNING_MESSAGE, + false); + + notification.setDelayMsec(-1); + notification.show(Page.getCurrent()); + } + } + } + + public PopupDateField getDateFromFilter() { + return dateFromFilter; + } + + public PopupDateField getDateToFilter() { + return dateToFilter; + } + + public CheckBox getIncludePartialMatch() { + return includePartialMatch; + } +} From aef93eaf6973ce921dff7a52584efe7cba9178c3 Mon Sep 17 00:00:00 2001 From: Levente Gal <62599627+leventegal-she@users.noreply.github.com> Date: Tue, 17 Dec 2024 15:02:38 +0200 Subject: [PATCH 46/56] #13203 Additional lab message fields (#13214) Co-authored-by: Levente Gal --- .../symeda/sormas/api/caze/CaseDataDto.java | 1 + .../externalmessage/ExternalMessageDto.java | 22 ++++++++++++++++++ .../processing/AbstractProcessingFlow.java | 3 +++ .../externalmessage/ExternalMessage.java | 23 +++++++++++++++++++ .../ExternalMessageFacadeEjb.java | 6 +++++ .../src/main/resources/sql/sormas_schema.sql | 10 ++++++++ .../symeda/sormas/ui/caze/CaseDataForm.java | 3 ++- 7 files changed, 67 insertions(+), 1 deletion(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java index aae97fc38d0..34b1a8d986a 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseDataDto.java @@ -365,6 +365,7 @@ public class CaseDataDto extends SormasToSormasShareableDto implements IsCase { Disease.UNSPECIFIED_VHF, Disease.ANTHRAX, Disease.CORONAVIRUS, + Disease.RESPIRATORY_SYNCYTIAL_VIRUS, Disease.OTHER }) @Outbreaks private VaccinationStatus vaccinationStatus; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageDto.java index 207ec0ae899..ecd24b99bee 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageDto.java @@ -25,6 +25,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.audit.AuditIncludeProperty; import de.symeda.sormas.api.audit.AuditedClass; +import de.symeda.sormas.api.caze.VaccinationStatus; import de.symeda.sormas.api.caze.surveillancereport.SurveillanceReportReferenceDto; import de.symeda.sormas.api.disease.DiseaseVariant; import de.symeda.sormas.api.externalmessage.labmessage.SampleReportDto; @@ -41,6 +42,7 @@ import de.symeda.sormas.api.utils.DependingOnFeatureType; import de.symeda.sormas.api.utils.FieldConstraints; import de.symeda.sormas.api.utils.HideForCountriesExcept; +import de.symeda.sormas.api.utils.YesNoUnknown; @AuditedClass @DependingOnFeatureType(featureType = FeatureType.EXTERNAL_MESSAGES) @@ -161,6 +163,10 @@ public class ExternalMessageDto extends SormasToSormasShareableDto { @Size(max = FieldConstraints.CHARACTER_LIMIT_TEXT, message = Validations.textTooLong) private String personAdditionalDetails; + private VaccinationStatus vaccinationStatus; + + private YesNoUnknown admittedToHealthFacility; + public ExternalMessageType getType() { return type; } @@ -433,6 +439,22 @@ public void setSurveillanceReport(SurveillanceReportReferenceDto surveillanceRep this.surveillanceReport = surveillanceReport; } + public VaccinationStatus getVaccinationStatus() { + return vaccinationStatus; + } + + public void setVaccinationStatus(VaccinationStatus vaccinationStatus) { + this.vaccinationStatus = vaccinationStatus; + } + + public YesNoUnknown getAdmittedToHealthFacility() { + return admittedToHealthFacility; + } + + public void setAdmittedToHealthFacility(YesNoUnknown admittedToHealthFacility) { + this.admittedToHealthFacility = admittedToHealthFacility; + } + public static ExternalMessageDto build() { ExternalMessageDto message = new ExternalMessageDto(); diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/processing/AbstractProcessingFlow.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/processing/AbstractProcessingFlow.java index fefeb32f8ab..5fc5b5bf63f 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/processing/AbstractProcessingFlow.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/processing/AbstractProcessingFlow.java @@ -175,6 +175,9 @@ protected CaseDataDto buildCase(PersonDto person, ExternalMessageDto externalMes caseDto.setHealthFacility(processingFacade.getFacilityReferenceByUuid(FacilityDto.NONE_FACILITY_UUID)); } + caseDto.setVaccinationStatus(externalMessageDto.getVaccinationStatus()); + caseDto.getHospitalization().setAdmittedToHealthFacility(externalMessageDto.getAdmittedToHealthFacility()); + return caseDto; } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java index 4785bc2b3e5..0e084a75d4d 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java @@ -26,12 +26,14 @@ import com.vladmihalcea.hibernate.type.array.ListArrayType; import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.caze.VaccinationStatus; import de.symeda.sormas.api.disease.DiseaseVariant; import de.symeda.sormas.api.externalmessage.ExternalMessageStatus; import de.symeda.sormas.api.externalmessage.ExternalMessageType; import de.symeda.sormas.api.person.PhoneNumberType; import de.symeda.sormas.api.person.PresentCondition; import de.symeda.sormas.api.person.Sex; +import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.backend.caze.surveillancereport.SurveillanceReport; import de.symeda.sormas.backend.common.AbstractDomainObject; import de.symeda.sormas.backend.disease.DiseaseVariantConverter; @@ -126,6 +128,9 @@ public class ExternalMessage extends AbstractDomainObject { private String tsv; private String personAdditionalDetails; + private VaccinationStatus vaccinationStatus; + private YesNoUnknown admittedToHealthFacility; + @Enumerated(EnumType.STRING) public ExternalMessageType getType() { return type; @@ -460,4 +465,22 @@ public String getPersonAdditionalDetails() { public void setPersonAdditionalDetails(String personAdditionalDetails) { this.personAdditionalDetails = personAdditionalDetails; } + + @Enumerated(EnumType.STRING) + public VaccinationStatus getVaccinationStatus() { + return vaccinationStatus; + } + + public void setVaccinationStatus(VaccinationStatus vaccinationStatus) { + this.vaccinationStatus = vaccinationStatus; + } + + @Enumerated(EnumType.STRING) + public YesNoUnknown getAdmittedToHealthFacility() { + return admittedToHealthFacility; + } + + public void setAdmittedToHealthFacility(YesNoUnknown admittedToHealthFacility) { + this.admittedToHealthFacility = admittedToHealthFacility; + } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java index b21a7c3e732..a71d7d99e0f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java @@ -192,6 +192,9 @@ ExternalMessage fillOrBuildEntity(@NotNull ExternalMessageDto source, ExternalMe target.setSampleReports(sampleReports); } target.setSurveillanceReport(surveillanceReportService.getByReferenceDto(source.getSurveillanceReport())); + target.setVaccinationStatus(source.getVaccinationStatus()); + target.setAdmittedToHealthFacility(source.getAdmittedToHealthFacility()); + return target; } @@ -368,6 +371,9 @@ public ExternalMessageDto toDto(ExternalMessage source) { target.setAssignee(source.getAssignee().toReference()); } + target.setVaccinationStatus(source.getVaccinationStatus()); + target.setAdmittedToHealthFacility(source.getAdmittedToHealthFacility()); + return target; } diff --git a/sormas-backend/src/main/resources/sql/sormas_schema.sql b/sormas-backend/src/main/resources/sql/sormas_schema.sql index b8e97a8d26f..02d53bd8d33 100644 --- a/sormas-backend/src/main/resources/sql/sormas_schema.sql +++ b/sormas-backend/src/main/resources/sql/sormas_schema.sql @@ -13724,4 +13724,14 @@ VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VA 'RESPIRATORY_SYNCYTIAL_VIRUS'); INSERT INTO schema_version (version_number, comment) VALUES (555, 'RSV disease variants #13204'); + +-- 2024-12-12 Additional lab message fields #13203 +ALTER TABLE externalmessage + ADD COLUMN vaccinationstatus varchar(255), + ADD COLUMN admittedtohealthfacility varchar(255); +ALTER TABLE externalmessage_history + ADD COLUMN vaccinationstatus varchar(255), + ADD COLUMN admittedtohealthfacility varchar(255); + +INSERT INTO schema_version (version_number, comment) VALUES (556, 'Additional lab message fields #13203'); -- *** Insert new sql commands BEFORE this line. Remember to always consider _history tables. *** diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java index c30e6cb8328..94a3cac8154 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/caze/CaseDataForm.java @@ -941,7 +941,8 @@ protected void addFields() { addField(CaseDataDto.TRIMESTER, NullableOptionGroup.class); FieldHelper.setVisibleWhen(getFieldGroup(), CaseDataDto.TRIMESTER, CaseDataDto.PREGNANT, Arrays.asList(YesNoUnknown.YES), true); - addField(CaseDataDto.VACCINATION_STATUS); + addField(CaseDataDto.VACCINATION_STATUS, TextField.class); +// getContent().addComponent(new Label("Debug vaccination"), CaseDataDto.VACCINATION_STATUS); addFields(CaseDataDto.SMALLPOX_VACCINATION_SCAR, CaseDataDto.SMALLPOX_VACCINATION_RECEIVED); addDateField(CaseDataDto.SMALLPOX_LAST_VACCINATION_DATE, DateField.class, 0); From 82cb7060acf5d4f0422e4a88d3b47d5f49130e6c Mon Sep 17 00:00:00 2001 From: SergiuPacurariu Date: Tue, 17 Dec 2024 15:28:27 +0200 Subject: [PATCH 47/56] #13190 - Add date range filter for birthdate (from-to) to Persons, Cases, and Contacts - changes after review --- .../de/symeda/sormas/api/i18n/Captions.java | 1 - .../symeda/sormas/api/i18n/Descriptions.java | 1 + .../de/symeda/sormas/api/i18n/Strings.java | 1 - .../src/main/resources/captions.properties | 1 - .../main/resources/descriptions.properties | 1 + .../src/main/resources/strings.properties | 3 +-- .../utils/BirthdateRangeFilterComponent.java | 20 ++----------------- 7 files changed, 5 insertions(+), 23 deletions(-) diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java index 8ba4c4b30b8..80605df463c 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Captions.java @@ -557,7 +557,6 @@ public interface Captions { String assignmentDate = "assignmentDate"; String assignToMe = "assignToMe"; String BAGExport = "BAGExport"; - String birthdateFilter = "birthdateFilter"; String bulkActionCreatDocuments = "bulkActionCreatDocuments"; String bulkActions = "bulkActions"; String bulkCancelFollowUp = "bulkCancelFollowUp"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java index e45b067dca1..52a5bfebaef 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Descriptions.java @@ -13,6 +13,7 @@ public interface Descriptions { String aefiDashboardDiseaseFilter = "aefiDashboardDiseaseFilter"; String aefiDashboardDistrictFilter = "aefiDashboardDistrictFilter"; String aefiDashboardRegionFilter = "aefiDashboardRegionFilter"; + String birthdateFilterPartialMatchDescription = "birthdateFilterPartialMatchDescription"; String Campaign_calculatedBasedOn = "Campaign.calculatedBasedOn"; String Campaign_campaignPhase = "Campaign.campaignPhase"; String CaseData_caseClassification = "CaseData.caseClassification"; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java index 3582175d870..20a31f30878 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/i18n/Strings.java @@ -922,7 +922,6 @@ public interface Strings { String infoAutomaticDeletionTooltipYears = "infoAutomaticDeletionTooltipYears"; String infoBAGExport = "infoBAGExport"; String infoBasicExport = "infoBasicExport"; - String infoBirthdateFilter = "infoBirthdateFilter"; String infoBulkProcess = "infoBulkProcess"; String infoBulkProcessCancelled = "infoBulkProcessCancelled"; String infoBulkProcessFinished = "infoBulkProcessFinished"; diff --git a/sormas-api/src/main/resources/captions.properties b/sormas-api/src/main/resources/captions.properties index 66f6dd768ca..9c380fd3bec 100644 --- a/sormas-api/src/main/resources/captions.properties +++ b/sormas-api/src/main/resources/captions.properties @@ -72,7 +72,6 @@ adoptHomeAddressOfCasePersonIfRelationMatches=Adopt home address of the case per casePersonAddress=Address of the case person viewMessage=View message primarySuffix=primary -birthdateFilter = Birthdate filter # About about=About aboutAdditionalInfo=Additional Info diff --git a/sormas-api/src/main/resources/descriptions.properties b/sormas-api/src/main/resources/descriptions.properties index 66bc7df81fd..7ecec3129c4 100644 --- a/sormas-api/src/main/resources/descriptions.properties +++ b/sormas-api/src/main/resources/descriptions.properties @@ -80,6 +80,7 @@ descContactOnlyWithReducedQuarantine = Only list contacts whose quarantine perio descContactIncludeContactsFromOtherJurisdictions = Include all contacts from other jurisdictions that you have access to, e.g. because you created them or their source case is in your jurisdiction descGdpr = Reminder: All comments entered must comply with GDPR rules as described during connection. discardDescription = Discards any unsaved changes +birthdateFilterPartialMatchDescription = If checked the search will include also the persons that have incomplete birthdate and have only higher level match eg. only year or only year and month # EpiData EpiData.bats = Did you have contact with live or dead bats or their excreta during the incubation period? diff --git a/sormas-api/src/main/resources/strings.properties b/sormas-api/src/main/resources/strings.properties index 79d6cf14f59..b091bed4353 100644 --- a/sormas-api/src/main/resources/strings.properties +++ b/sormas-api/src/main/resources/strings.properties @@ -1116,7 +1116,6 @@ infoAefiSelectPrimarySuspectVaccine = The list below contains all vaccinations o infoArchivedAefiEntries = Adverse event entries are automatically archived after %d days without changes to the data. infoNoAefiInvestigations = No investigations have been created for this adverse event infoHeadingAefiDashboardMap=Adverse events are shown using the GPS coordinate of the facility or person's home address. -infoBirthdateFilter = If checked the search will include also the persons that have incomplete birthdate and have only higher level match eg. only year or only year and month # Messages messageActionOutsideJurisdictionDeletionDenied = The action outside user's jurisdiction cannot be deleted @@ -1655,7 +1654,7 @@ promptActionChangeDateTo = ... to promptActionChangeEpiWeekFrom = Date of action change from epi week... promptActionChangeEpiWeekTo = ... to epi week promptBirthdateFrom = Birthdate from -promptBirthdateTo = Birthdate to +promptBirthdateTo = ... to promptCampaignSearch = ID, name promptCasesDateFrom = New cases from... promptCasesEpiWeekFrom = New cases from epi week... diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java index 6d68a157d4a..3bfded5b7e0 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/utils/BirthdateRangeFilterComponent.java @@ -1,22 +1,18 @@ package de.symeda.sormas.ui.utils; -import static de.symeda.sormas.ui.utils.CssStyles.CHECKBOX_FILTER_INLINE; -import static de.symeda.sormas.ui.utils.CssStyles.LABEL_BOLD; import static de.symeda.sormas.ui.utils.CssStyles.VSPACE_TOP_4; import java.util.Calendar; import java.util.Date; -import com.vaadin.icons.VaadinIcons; import com.vaadin.server.Page; -import com.vaadin.shared.ui.ContentMode; import com.vaadin.ui.HorizontalLayout; -import com.vaadin.ui.Label; import com.vaadin.ui.Notification; import com.vaadin.v7.ui.CheckBox; import com.vaadin.v7.ui.PopupDateField; import de.symeda.sormas.api.i18n.Captions; +import de.symeda.sormas.api.i18n.Descriptions; import de.symeda.sormas.api.i18n.I18nProperties; import de.symeda.sormas.api.i18n.Strings; import de.symeda.sormas.api.utils.DateHelper; @@ -39,13 +35,6 @@ public BirthdateRangeFilterComponent(boolean showCaption, AbstractFilterForm par dateToFilter = new PopupDateField(); includePartialMatch = new CheckBox(); - Label birthdateLabel = new Label(); - birthdateLabel.setValue(I18nProperties.getCaption(Captions.birthdateFilter)); - birthdateLabel.addStyleName(CHECKBOX_FILTER_INLINE); - birthdateLabel.addStyleName(LABEL_BOLD); - birthdateLabel.addStyleName(VSPACE_TOP_4); - addComponent(birthdateLabel); - addComponent(dateFromFilter); addComponent(dateToFilter); @@ -82,13 +71,8 @@ public BirthdateRangeFilterComponent(boolean showCaption, AbstractFilterForm par includePartialMatch.setCaption(I18nProperties.getCaption(Captions.includePartialBirthdates)); includePartialMatch.addStyleName(VSPACE_TOP_4); + includePartialMatch.setDescription(I18nProperties.getDescription(Descriptions.birthdateFilterPartialMatchDescription)); addComponent(includePartialMatch); - - Label infoLabel = new Label(VaadinIcons.INFO_CIRCLE.getHtml(), ContentMode.HTML); - infoLabel.setSizeUndefined(); - infoLabel.setDescription(I18nProperties.getString(Strings.infoBirthdateFilter)); - CssStyles.style(infoLabel, CssStyles.LABEL_XLARGE, CssStyles.LABEL_SECONDARY); - addComponent(infoLabel); } private static void notifyIfIncorrectRange(Date dateFrom, Date dateTo) { From 027eaab4badb5c6d12a72abb6909440b2f992961 Mon Sep 17 00:00:00 2001 From: sergiupacurariu <62688603+sergiupacurariu@users.noreply.github.com> Date: Wed, 18 Dec 2024 11:06:52 +0200 Subject: [PATCH 48/56] #13181 - Enhance National Health ID Search Functionality Across Modules and Address Duplication Issues (#13216) Co-authored-by: SergiuPacurariu --- .../main/java/de/symeda/sormas/ui/person/PersonFormHelper.java | 1 + 1 file changed, 1 insertion(+) diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java index 21afff244d3..427a3536fad 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/person/PersonFormHelper.java @@ -36,6 +36,7 @@ public static Window warningSimilarPersons(String nationalHealthId, String curre final CommitDiscardWrapperComponent component = new CommitDiscardWrapperComponent<>(similarPersonGrid); component.getCommitButton().setCaption(I18nProperties.getCaption(Captions.actionDone)); + component.getCommitButton().setDescription(I18nProperties.getCaption(Captions.actionOkay)); component.getDiscardButton().setVisible(false); component.getWrappedComponent().setWidth(800, Sizeable.Unit.PIXELS); From 9d38e9b940128344d7df655d017bcdbddbd13b16 Mon Sep 17 00:00:00 2001 From: Levente Gal Date: Wed, 18 Dec 2024 11:29:41 +0200 Subject: [PATCH 49/56] #13204 RSV disease variants - fix enum values --- sormas-backend/src/main/resources/sql/sormas_schema.sql | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/sormas-backend/src/main/resources/sql/sormas_schema.sql b/sormas-backend/src/main/resources/sql/sormas_schema.sql index 02d53bd8d33..57f222938b2 100644 --- a/sormas-backend/src/main/resources/sql/sormas_schema.sql +++ b/sormas-backend/src/main/resources/sql/sormas_schema.sql @@ -13714,13 +13714,13 @@ INSERT INTO schema_version (version_number, comment) VALUES (554, 'Add New Influ -- 2024-12-12 RSV disease variants #13204 INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) -VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', ' A', 'Type A', +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', 'A', 'Type A', 'RESPIRATORY_SYNCYTIAL_VIRUS'); INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) -VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', ' B', 'Type B', +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', 'B', 'Type B', 'RESPIRATORY_SYNCYTIAL_VIRUS'); INSERT INTO customizableenumvalue(id, uuid, changedate, creationdate, datatype, value, caption, diseases) -VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', ' AB', 'Type A+B', +VALUES (nextval('entity_seq'), generate_base32_uuid(), now(), now(), 'DISEASE_VARIANT', 'AB', 'Type A+B', 'RESPIRATORY_SYNCYTIAL_VIRUS'); INSERT INTO schema_version (version_number, comment) VALUES (555, 'RSV disease variants #13204'); From bb67f2a2a722126b29b6b955634237d068132e06 Mon Sep 17 00:00:00 2001 From: Levente Gal <62599627+leventegal-she@users.noreply.github.com> Date: Wed, 18 Dec 2024 12:26:27 +0200 Subject: [PATCH 50/56] #13217 [Automatic message processing] Check sample assignment threshold against sample data instead of current date (#13218) Co-authored-by: Levente Gal --- .../sormas/backend/caze/CaseService.java | 7 +++- .../AutomaticLabMessageProcessor.java | 13 ++++++- .../AutomaticLabMessageProcessorTest.java | 38 +++++++++++++++++++ 3 files changed, 55 insertions(+), 3 deletions(-) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java index ebcd4065d2e..f2438dcb32b 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java @@ -31,6 +31,7 @@ import java.util.function.Function; import java.util.stream.Collectors; +import javax.annotation.Nullable; import javax.ejb.EJB; import javax.ejb.LocalBean; import javax.ejb.Stateless; @@ -2345,7 +2346,9 @@ protected String getDeleteReferenceField(DeletionReference deletionReference) { return super.getDeleteReferenceField(deletionReference); } - public String getCaseUuidForAutomaticSampleAssignment(Set uuids, Disease disease, int threshold) { + public String getCaseUuidForAutomaticSampleAssignment(Set uuids, Disease disease, @Nullable Date sampleDateTime, int threshold) { + Date dateToCompareTo = sampleDateTime != null ? sampleDateTime : new Date(); + CriteriaBuilder cb = em.getCriteriaBuilder(); CriteriaQuery cq = cb.createQuery(String.class); Root caseRoot = cq.from(Case.class); @@ -2366,7 +2369,7 @@ public String getCaseUuidForAutomaticSampleAssignment(Set uuids, Disease ExtendedPostgreSQL94Dialect.DATE, Date.class, CriteriaBuilderHelper.coalesce(cb, Date.class, earliestSampleSq, caseRoot.get(Case.REPORT_DATE))), - cb.function(ExtendedPostgreSQL94Dialect.DATE, Date.class, cb.literal(new Date()))), + cb.function(ExtendedPostgreSQL94Dialect.DATE, Date.class, cb.literal(dateToCompareTo))), Long.valueOf(TimeUnit.DAYS.toSeconds(threshold)).doubleValue())); cq.orderBy(cb.desc(caseRoot.get(Case.REPORT_DATE))); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java index f660bce1805..763177459f2 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessor.java @@ -16,7 +16,10 @@ package de.symeda.sormas.backend.externalmessage.labmessage; import java.text.Collator; +import java.util.Comparator; +import java.util.Date; import java.util.List; +import java.util.Objects; import java.util.Set; import java.util.concurrent.CompletableFuture; import java.util.concurrent.CompletionStage; @@ -43,6 +46,7 @@ import de.symeda.sormas.api.event.EventParticipantDto; import de.symeda.sormas.api.event.SimilarEventParticipantDto; import de.symeda.sormas.api.externalmessage.ExternalMessageDto; +import de.symeda.sormas.api.externalmessage.labmessage.SampleReportDto; import de.symeda.sormas.api.externalmessage.processing.ExternalMessageMapper; import de.symeda.sormas.api.externalmessage.processing.ExternalMessageProcessingFacade; import de.symeda.sormas.api.externalmessage.processing.ExternalMessageProcessingResult; @@ -183,7 +187,14 @@ protected void handlePickOrCreateEntry( } Set similarCaseUuids = similarCases.stream().map(CaseSelectionDto::getUuid).collect(Collectors.toSet()); - String caseUuid = caseService.getCaseUuidForAutomaticSampleAssignment(similarCaseUuids, disease, automaticSampleAssignmentThreshold); + Date sampleDate = externalMessageDto.getSampleReports() + .stream() + .map(SampleReportDto::getSampleDateTime) + .filter(Objects::nonNull) + .min(Comparator.comparing(Date::getTime)) + .orElse(null); + String caseUuid = + caseService.getCaseUuidForAutomaticSampleAssignment(similarCaseUuids, disease, sampleDate, automaticSampleAssignmentThreshold); if (caseUuid == null) { logger.debug( diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java index e3e22f325c7..85da716cf6f 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/labmessage/AutomaticLabMessageProcessorTest.java @@ -31,6 +31,7 @@ import org.junit.jupiter.api.Test; import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.caze.CaseCriteria; import de.symeda.sormas.api.caze.CaseDataDto; import de.symeda.sormas.api.externalmessage.ExternalMessageDto; import de.symeda.sormas.api.externalmessage.ExternalMessageStatus; @@ -46,6 +47,7 @@ import de.symeda.sormas.api.sample.PathogenTestDto; import de.symeda.sormas.api.sample.PathogenTestResultType; import de.symeda.sormas.api.sample.PathogenTestType; +import de.symeda.sormas.api.sample.SampleCriteria; import de.symeda.sormas.api.sample.SampleDto; import de.symeda.sormas.api.sample.SampleMaterial; import de.symeda.sormas.api.sample.SamplePurpose; @@ -213,6 +215,42 @@ public void testProcessWithExistingPersonAndCase() throws ExecutionException, In assertThat(samples, hasSize(2)); } + /** + * External message with sample date in the threshold period should generate a new sample to the existing case + * @throws ExecutionException + * @throws InterruptedException + */ + @Test + public void testThresholdAgainstSampleDate() throws ExecutionException, InterruptedException { + ExternalMessageDto externalMessage = createExternalMessage(e -> { + e.getSampleReports().get(0).setSampleDateTime(DateHelper.subtractDays(new Date(), 10)); + }); + + PersonDto person = + creator.createPerson(externalMessage.getPersonFirstName(), externalMessage.getPersonLastName(), externalMessage.getPersonSex(), p -> { + p.setNationalHealthId(externalMessage.getPersonNationalHealthId()); + }); + + CaseDataDto caze = creator.createCase(reportingUser.toReference(), person.toReference(), rdcf, c -> { + c.setDisease(externalMessage.getDisease()); + c.setReportDate(DateHelper.subtractDays(new Date(), 15)); + }); + creator.createSample(caze.toReference(), reportingUser.toReference(), rdcf.facility, s -> { + s.setSampleDateTime(DateHelper.subtractDays(new Date(), 15)); + }); + + // set the threshold + creator.updateDiseaseConfiguration(externalMessage.getDisease(), true, true, true, true, null, 10); + getBean(DiseaseConfigurationFacadeEjb.DiseaseConfigurationFacadeEjbLocal.class).loadData(); + + ProcessingResult result = runFlow(externalMessage); + assertThat(result.getStatus(), is(DONE)); + assertThat(externalMessage.getStatus(), is(ExternalMessageStatus.PROCESSED)); + + assertThat(getCaseFacade().count(new CaseCriteria().person(caze.getPerson())), is(1L)); + assertThat(getSampleFacade().count(new SampleCriteria().caze(caze.toReference())), is(2L)); + } + @Test public void testProcessWithExistingPersonAndCaseWithBySampleDate() throws ExecutionException, InterruptedException { ExternalMessageDto externalMessage = createExternalMessage(null); From 7865ae31997637ea325979a72afe228996085d40 Mon Sep 17 00:00:00 2001 From: sergiupacurariu <62688603+sergiupacurariu@users.noreply.github.com> Date: Thu, 19 Dec 2024 14:15:47 +0200 Subject: [PATCH 51/56] =?UTF-8?q?#13204=20-=20RSV=20disease=20variants=20-?= =?UTF-8?q?=20add=20support=20for=20similar=20Customizable=20=E2=80=A6=20(?= =?UTF-8?q?#13219)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * #13204 - RSV disease variants - add support for similar Customizable disease values * #13204 RSV disease variants - fix values without disease * #13204 - RSV disease variants - add support for similar Customizable disease values * #13204 RSV disease variants - basic disease variant converter tests --------- Co-authored-by: SergiuPacurariu Co-authored-by: Levente Gal --- .../api/bagexport/BAGExportCaseDto.java | 6 +- .../api/bagexport/BAGExportContactDto.java | 5 +- .../symeda/sormas/api/caze/CaseExportDto.java | 10 +- .../sormas/api/caze/CaseIndexDetailedDto.java | 2 +- .../symeda/sormas/api/caze/CaseIndexDto.java | 5 +- .../sormas/api/caze/CaseMergeIndexDto.java | 2 +- .../sormas/api/caze/PreviousCaseDto.java | 6 +- .../sormas/api/contact/ContactExportDto.java | 5 +- .../CustomizableEnumConverter.java | 19 ++- .../CustomizableEnumFacade.java | 6 +- .../api}/disease/DiseaseVariantConverter.java | 5 +- .../api}/disease/PathogenConverter.java | 4 +- .../api/event/EventActionExportDto.java | 5 +- .../sormas/api/event/EventActionIndexDto.java | 5 +- .../sormas/api/event/EventExportDto.java | 9 +- .../sormas/api/event/EventIndexDto.java | 9 +- .../api}/event/SpecificRiskConverter.java | 5 +- .../ExternalMessageIndexDto.java | 5 +- .../api/person/OccupationTypeConverter.java | 10 ++ .../sormas/api/person/PersonExportDto.java | 4 +- .../api/selfreport/SelfReportExportDto.java | 5 +- .../resources/doc/SORMAS_Data_Dictionary.xlsx | Bin 321069 -> 276347 bytes .../main/resources/doc/SORMAS_User_Roles.xlsx | Bin 45326 -> 38001 bytes .../sormas/backend/action/ActionService.java | 6 +- ...EventActionIndexDtoReasultTransformer.java | 2 +- .../backend/bagexport/BAGExportFacadeEjb.java | 4 +- .../de/symeda/sormas/backend/caze/Case.java | 22 +++- .../sormas/backend/caze/CaseFacadeEjb.java | 4 +- ...CaseIndexDetailedDtoResultTransformer.java | 2 +- .../caze/CaseIndexDtoResultTransformer.java | 2 +- .../backend/caze/CaseListCriteriaBuilder.java | 2 +- .../sormas/backend/caze/CaseService.java | 5 +- .../backend/contact/ContactFacadeEjb.java | 2 +- .../backend/contact/ContactService.java | 3 +- .../CustomizableEnumFacadeEjb.java | 70 +++++++---- .../CustomizableEnumValue.java | 1 + .../environmentsample/EnvironmentSample.java | 17 ++- .../EnvironmentSampleFacadeEjb.java | 12 +- .../EnvironmentSampleService.java | 6 +- .../RequestedPathogensConverter.java | 15 ++- .../de/symeda/sormas/backend/event/Event.java | 39 ++++-- .../sormas/backend/event/EventFacadeEjb.java | 8 +- .../event/EventIndexDtoResultTransformer.java | 4 +- .../sormas/backend/event/EventService.java | 4 +- .../externalmessage/ExternalMessage.java | 21 +++- .../ExternalMessageFacadeEjb.java | 2 +- ...ernalMessageIndexDtoResultTransformer.java | 2 +- .../ExternalMessageService.java | 4 +- .../person/OccupationTypeConverter.java | 11 -- .../symeda/sormas/backend/person/Person.java | 19 ++- .../backend/person/PersonFacadeEjb.java | 2 +- .../sormas/backend/sample/PathogenTest.java | 42 +++++-- .../sormas/backend/sample/SampleService.java | 11 +- .../sormas/backend/selfreport/SelfReport.java | 29 +++-- .../SelfReportExportDtoResultTransformer.java | 2 +- .../selfreport/SelfReportFacadeEjb.java | 2 +- .../backend/selfreport/SelfReportService.java | 22 ++-- .../backend/travelentry/TravelEntry.java | 21 +++- .../sormas/backend/TestDataCreator.java | 4 +- .../backend/caze/CaseFacadeEjbTest.java | 8 +- .../CustomizableEnumConverterTest.java | 114 ++++++++++++++++++ .../EnvironmentSampleFacadeEjbTest.java | 6 +- .../ExternalMessageFacadeEjbTest.java | 8 +- .../AbstractSelfReportProcessingFlowTest.java | 4 +- .../sormas/ui/importer/DataImporter.java | 2 +- 65 files changed, 490 insertions(+), 208 deletions(-) rename {sormas-backend/src/main/java/de/symeda/sormas/backend => sormas-api/src/main/java/de/symeda/sormas/api}/customizableenum/CustomizableEnumConverter.java (81%) rename {sormas-backend/src/main/java/de/symeda/sormas/backend => sormas-api/src/main/java/de/symeda/sormas/api}/disease/DiseaseVariantConverter.java (85%) rename {sormas-backend/src/main/java/de/symeda/sormas/backend => sormas-api/src/main/java/de/symeda/sormas/api}/disease/PathogenConverter.java (89%) rename {sormas-backend/src/main/java/de/symeda/sormas/backend => sormas-api/src/main/java/de/symeda/sormas/api}/event/SpecificRiskConverter.java (85%) create mode 100644 sormas-api/src/main/java/de/symeda/sormas/api/person/OccupationTypeConverter.java delete mode 100644 sormas-backend/src/main/java/de/symeda/sormas/backend/person/OccupationTypeConverter.java create mode 100644 sormas-backend/src/test/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverterTest.java diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportCaseDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportCaseDto.java index fc62080b4ef..b7ed1d30462 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportCaseDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportCaseDto.java @@ -24,6 +24,7 @@ import de.symeda.sormas.api.contact.QuarantineType; import de.symeda.sormas.api.infrastructure.facility.FacilityType; import de.symeda.sormas.api.person.OccupationType; +import de.symeda.sormas.api.person.OccupationTypeConverter; import de.symeda.sormas.api.person.Sex; import de.symeda.sormas.api.sample.PathogenTestResultType; import de.symeda.sormas.api.sample.PathogenTestType; @@ -32,6 +33,7 @@ import de.symeda.sormas.api.utils.YesNoUnknown; public class BAGExportCaseDto implements Serializable { + private Integer caseIdIsm; private Long caseId; private Long personId; @@ -124,7 +126,7 @@ public BAGExportCaseDto(Integer caseIdIsm, Long caseId, Long personId, String homeAddressStreet, String homeAddressHouseNumber, String homeAddressCity, String homeAddressPostalCode, String homeAddressCountry, String phoneNumber, String mobileNumber, String emailAddress, Sex sex, Integer birthdateDD, Integer birthdateMM, Integer birthdateYYYY, - OccupationType occupationType, + String occupationType, boolean symptomatic, Date symptomOnsetDate, String activityMappingYn, Date contactTracingContactDate, @@ -148,7 +150,7 @@ public BAGExportCaseDto(Integer caseIdIsm, Long caseId, Long personId, this.emailAddress = emailAddress; this.sex = sex; this.birthDate = new BirthDateDto(birthdateDD, birthdateMM, birthdateYYYY); - this.occupationType = occupationType; + this.occupationType = new OccupationTypeConverter().convertToEntityAttribute(null, occupationType); this.symptomatic = symptomatic ? YesNoUnknown.YES : YesNoUnknown.NO; this.symptomOnsetDate = symptomOnsetDate; this.activityMappingYn = activityMappingYn; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportContactDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportContactDto.java index bc097ab09e4..f6a08ab2dac 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportContactDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/bagexport/BAGExportContactDto.java @@ -23,6 +23,7 @@ import de.symeda.sormas.api.contact.QuarantineType; import de.symeda.sormas.api.infrastructure.facility.FacilityType; import de.symeda.sormas.api.person.OccupationType; +import de.symeda.sormas.api.person.OccupationTypeConverter; import de.symeda.sormas.api.person.Sex; import de.symeda.sormas.api.sample.PathogenTestResultType; import de.symeda.sormas.api.sample.PathogenTestType; @@ -91,7 +92,7 @@ public BAGExportContactDto(Long contactId, Long personId, String lastName, Strin String homeAddressStreet, String homeAddressHouseNumber, String homeAddressCity, String homeAddressPostalCode, String phoneNumber, String mobileNumber, Sex sex, Integer birthdateDD, Integer birthdateMM, Integer birthdateYYYY, - OccupationType occupationType, + String occupationType, QuarantineType quarantineType, String quarantineDetails, Integer caseLinkCaseIdIsm, Long caseLinkCaseId, Date caseLinkContactDate, Date startOfQuarantineDate, Date endOfQuarantineDate, EndOfQuarantineReason endOfQuarantineReason, String endOfQuarantineReasonDetails @@ -112,7 +113,7 @@ public BAGExportContactDto(Long contactId, Long personId, String lastName, Strin this.mobileNumber = mobileNumber; this.sex = sex; this.birthDate = new BirthDateDto(birthdateDD, birthdateMM, birthdateYYYY); - this.occupationType = occupationType; + this.occupationType = new OccupationTypeConverter().convertToEntityAttribute(null, occupationType); this.quarantineType = quarantineType; this.quarantineDetails = quarantineDetails; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java index 347b398bfdb..8a2d221a6db 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseExportDto.java @@ -28,6 +28,7 @@ import de.symeda.sormas.api.contact.FollowUpStatus; import de.symeda.sormas.api.contact.QuarantineType; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.epidata.EpiDataDto; import de.symeda.sormas.api.event.EventStatus; import de.symeda.sormas.api.feature.FeatureType; @@ -49,6 +50,7 @@ import de.symeda.sormas.api.person.BurialConductor; import de.symeda.sormas.api.person.EducationType; import de.symeda.sormas.api.person.OccupationType; +import de.symeda.sormas.api.person.OccupationTypeConverter; import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.person.PresentCondition; import de.symeda.sormas.api.person.Salutation; @@ -362,7 +364,7 @@ public class CaseExportDto extends AbstractUuidDto implements IsCase { @SuppressWarnings("unchecked") public CaseExportDto(long id, long personId, Double personAddressLatitude, Double personAddressLongitude, Float personAddressLatLonAcc, long epiDataId, long symptomsId, long hospitalizationId, long healthConditionsId, String uuid, String epidNumber, - Disease disease, DiseaseVariant diseaseVariant, String diseaseDetails, String diseaseVariantDetails, + Disease disease, String diseaseVariant, String diseaseDetails, String diseaseVariantDetails, String personUuid, String firstName, String lastName, Salutation salutation, String otherSalutation, Sex sex, YesNoUnknown pregnant, Integer approximateAge, ApproximateAgeType approximateAgeType, Integer birthdateDD, Integer birthdateMM, Integer birthdateYYYY, Date reportDate, String region, String district, String community, @@ -389,7 +391,7 @@ public CaseExportDto(long id, long personId, Double personAddressLatitude, Doubl String addressRegion, String addressDistrict, String addressCommunity, String city, String street, String houseNumber, String additionalInformation, String postalCode, String facility, String facilityUuid, String facilityDetails, String phone, String phoneOwner, String emailAddress, String otherContactDetails, EducationType educationType, String educationDetails, - OccupationType occupationType, String occupationDetails, ArmedForcesRelationType ArmedForcesRelationType, YesNoUnknown contactWithSourceCaseKnown, + String occupationType, String occupationDetails, ArmedForcesRelationType ArmedForcesRelationType, YesNoUnknown contactWithSourceCaseKnown, //Date onsetDate, VaccinationStatus vaccinationStatus, YesNoUnknown postpartum, Trimester trimester, long eventCount, Long prescriptionCount, Long treatmentCount, Long clinicalVisitCount, @@ -419,7 +421,7 @@ public CaseExportDto(long id, long personId, Double personAddressLatitude, Doubl this.armedForcesRelationType = ArmedForcesRelationType; this.disease = disease; this.diseaseDetails = diseaseDetails; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.diseaseVariantDetails = diseaseVariantDetails; this.personUuid = personUuid; this.firstName = firstName; @@ -501,7 +503,7 @@ public CaseExportDto(long id, long personId, Double personAddressLatitude, Doubl this.otherContactDetails = otherContactDetails; this.educationType = educationType; this.educationDetails = educationDetails; - this.occupationType = occupationType; + this.occupationType = new OccupationTypeConverter().convertToEntityAttribute(null, occupationType); this.occupationDetails = occupationDetails; this.contactWithSourceCaseKnown = contactWithSourceCaseKnown; // this.onsetDate = onsetDate; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDetailedDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDetailedDto.java index b1575091f22..1774a17d66a 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDetailedDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDetailedDto.java @@ -75,7 +75,7 @@ public class CaseIndexDetailedDto extends CaseIndexDto { //@formatter:off public CaseIndexDetailedDto(long id, String uuid, String epidNumber, String externalID, String externalToken, String internalToken,String caseReferenceNumber, String personUuid, String personFirstName, String personLastName, - Disease disease, DiseaseVariant diseaseVariant, String diseaseDetails, CaseClassification caseClassification, InvestigationStatus investigationStatus, + Disease disease, String diseaseVariant, String diseaseDetails, CaseClassification caseClassification, InvestigationStatus investigationStatus, PresentCondition presentCondition, Date reportDate, Date creationDate, String regionUuid, String districtUuid, String healthFacilityUuid, String healthFacilityName, String healthFacilityDetails, diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDto.java index 6ab5de5d058..9f1831345e2 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseIndexDto.java @@ -25,6 +25,7 @@ import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.contact.FollowUpStatus; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.infrastructure.InfrastructureHelper; import de.symeda.sormas.api.infrastructure.facility.FacilityHelper; import de.symeda.sormas.api.person.ApproximateAgeType; @@ -141,7 +142,7 @@ public class CaseIndexDto extends PseudonymizableIndexDto implements MergeableIn //@formatter:off public CaseIndexDto(long id, String uuid, String epidNumber, String externalID, String externalToken, String internalToken, String caseReferenceNumber, String personUuid, String personFirstName, String personLastName, Disease disease, - DiseaseVariant diseaseVariant, String diseaseDetails, CaseClassification caseClassification, InvestigationStatus investigationStatus, + String diseaseVariant, String diseaseDetails, CaseClassification caseClassification, InvestigationStatus investigationStatus, PresentCondition presentCondition, Date reportDate, Date creationDate, String regionUuid, String districtUuid, String healthFacilityUuid, String healthFacilityName, String healthFacilityDetails, String pointOfEntryUuid, String pointOfEntryName, String pointOfEntryDetails, String surveillanceOfficerUuid, CaseOutcome outcome, @@ -166,7 +167,7 @@ public CaseIndexDto(long id, String uuid, String epidNumber, String externalID, this.personFirstName = personFirstName; this.personLastName = personLastName; this.disease = disease; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.diseaseDetails = diseaseDetails; this.caseClassification = caseClassification; this.investigationStatus = investigationStatus; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseMergeIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseMergeIndexDto.java index c00c3d5f593..5beb43b4936 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseMergeIndexDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/CaseMergeIndexDto.java @@ -21,7 +21,7 @@ public class CaseMergeIndexDto extends CaseIndexDto { //@formatter:off public CaseMergeIndexDto( long id, String uuid, String epidNumber, String externalID, String externalToken, String internalToken, String caseReferenceNumber, String personUuid, String personFirstName, String personLastName, Disease disease, - DiseaseVariant diseaseVariant, String diseaseDetails, CaseClassification caseClassification, InvestigationStatus investigationStatus, + String diseaseVariant, String diseaseDetails, CaseClassification caseClassification, InvestigationStatus investigationStatus, PresentCondition presentCondition, Date reportDate, Date creationDate, String regionUuid, String districtUuid, String healthFacilityUuid, String healthFacilityName, String healthFacilityDetails, String pointOfEntryUuid, String pointOfEntryName, String pointOfEntryDetails, String surveillanceOfficerUuid, CaseOutcome outcome, diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/caze/PreviousCaseDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/caze/PreviousCaseDto.java index 2e1b5e21c74..59e960bd5ce 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/caze/PreviousCaseDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/caze/PreviousCaseDto.java @@ -17,7 +17,9 @@ import java.util.Date; +import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.uuid.AbstractUuidDto; public class PreviousCaseDto extends AbstractUuidDto { @@ -29,11 +31,11 @@ public class PreviousCaseDto extends AbstractUuidDto { private final DiseaseVariant diseaseVariant; private final Date onsetDate; - public PreviousCaseDto(String uuid, Date reportDate, String externalToken, DiseaseVariant diseaseVariant, Date onsetDate) { + public PreviousCaseDto(String uuid, Date reportDate, String externalToken, Disease disease, String diseaseVariant, Date onsetDate) { super(uuid); this.reportDate = reportDate; this.externalToken = externalToken; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.onsetDate = onsetDate; } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactExportDto.java index 36f4dbe34a2..33f49816dbe 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/contact/ContactExportDto.java @@ -46,6 +46,7 @@ import de.symeda.sormas.api.person.ApproximateAgeType.ApproximateAgeHelper; import de.symeda.sormas.api.person.ArmedForcesRelationType; import de.symeda.sormas.api.person.OccupationType; +import de.symeda.sormas.api.person.OccupationTypeConverter; import de.symeda.sormas.api.person.PersonDto; import de.symeda.sormas.api.person.PresentCondition; import de.symeda.sormas.api.person.Salutation; @@ -259,7 +260,7 @@ public ContactExportDto(long id, long personId, String uuid, String sourceCaseUu PresentCondition presentCondition, Date deathDate, String addressRegion, String addressDistrict, String addressCommunity, String city, String street, String houseNumber, String additionalInformation, String postalCode, String facility, String facilityUuid, String facilityDetails, - String phone, String phoneOwner, String emailAddress, String otherContactDetails, OccupationType occupationType, String occupationDetails, ArmedForcesRelationType armedForcesRelationType, + String phone, String phoneOwner, String emailAddress, String otherContactDetails, String occupationType, String occupationDetails, ArmedForcesRelationType armedForcesRelationType, String region, String district, String community, long epiDataId, YesNoUnknown contactWithSourceCaseKnown, YesNoUnknown returningTraveler, VaccinationStatus vaccinationStatus, String externalID, String externalToken, String internalToken, String caseReferenceNumber, @@ -337,7 +338,7 @@ public ContactExportDto(long id, long personId, String uuid, String sourceCaseUu // this.otherContactDetails += this.otherContactDetails.equals("") ? otherContactDetail : ", " + otherContactDetail; // } this.otherContactDetails = otherContactDetails; - this.occupationType = occupationType; + this.occupationType = new OccupationTypeConverter().convertToEntityAttribute(null, occupationType); this.occupationDetails = occupationDetails; this.armedForcesRelationType = armedForcesRelationType; this.region = region; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverter.java b/sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumConverter.java similarity index 81% rename from sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverter.java rename to sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumConverter.java index 37ed910f221..3a7ecfd3f88 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverter.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumConverter.java @@ -13,17 +13,14 @@ * along with this program. If not, see . */ -package de.symeda.sormas.backend.customizableenum; +package de.symeda.sormas.api.customizableenum; import javax.naming.InitialContext; import javax.naming.NamingException; -import javax.persistence.AttributeConverter; import org.apache.commons.lang3.StringUtils; -import de.symeda.sormas.api.customizableenum.CustomizableEnum; -import de.symeda.sormas.api.customizableenum.CustomizableEnumFacade; -import de.symeda.sormas.api.customizableenum.CustomizableEnumType; +import de.symeda.sormas.api.Disease; /** * JPA Converter that converts a JSON String stored in the database to an instance of {@link CustomizableEnum} and vice versa. @@ -36,7 +33,7 @@ * @param * The specific extension of {@link CustomizableEnum} for type safety */ -public abstract class CustomizableEnumConverter implements AttributeConverter { +public abstract class CustomizableEnumConverter { private final Class enumClass; private CustomizableEnumFacade customizableEnumFacade; @@ -45,13 +42,11 @@ public CustomizableEnumConverter(Class enumClass) { this.enumClass = enumClass; } - @Override public String convertToDatabaseColumn(T enumValue) { return enumValue != null ? enumValue.getValue() : null; } - @Override - public T convertToEntityAttribute(String enumString) { + public T convertToEntityAttribute(Disease disease, String enumString) { if (StringUtils.isBlank(enumString)) { return null; } @@ -66,7 +61,11 @@ public T convertToEntityAttribute(String enumString) { throw new RuntimeException("No CustomizableEnumType for given enumClass " + enumClass + "found"); } - return customizableEnumFacade.getEnumValue(CustomizableEnumType.getByEnumClass(enumClass), enumString); + T enumValue = customizableEnumFacade.getEnumValue(enumType, disease, enumString); + if (enumValue == null && disease != null) { + enumValue = customizableEnumFacade.getEnumValue(enumType, null, enumString); + } + return enumValue; } catch (NamingException e) { throw new RuntimeException(e); } diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumFacade.java b/sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumFacade.java index 9cf74c08a5b..6db95dfd48d 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumFacade.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/customizableenum/CustomizableEnumFacade.java @@ -46,11 +46,11 @@ public interface CustomizableEnumFacade * The specific extension of {@link CustomizableEnum} for type safety * @return The enum instance containing its value, internationalized caption, and optional properties */ - T getEnumValue(CustomizableEnumType type, String value); + T getEnumValue(CustomizableEnumType type, Disease disease, String value); /** * Works similar to the {@link CustomizableEnumFacade#getEnumValues(CustomizableEnumType, Disease)}, but looks up a specific value. - * Unlike the {@link CustomizableEnumFacade#getEnumValue(CustomizableEnumType, String)}, this method does not throw a RuntimeException + * Unlike the {@link CustomizableEnumFacade#getEnumValue(CustomizableEnumType, Disease, String)}, this method does not throw a RuntimeException * when an enum can not be found. * * @param type @@ -106,7 +106,7 @@ public interface CustomizableEnumFacade /** * Clears the caches and reloads the customizable enum values from the database. Does not load enum values by language - * or disease as those are retrieved on demand by using {@link #getEnumValue(CustomizableEnumType, String)} and + * or disease as those are retrieved on demand by using {@link #getEnumValue(CustomizableEnumType, Disease, String)} and * {@link #getEnumValues(CustomizableEnumType, Disease)}. Exposed to this facade to allow reloading the caches without * having to restart the server. */ diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/disease/DiseaseVariantConverter.java b/sormas-api/src/main/java/de/symeda/sormas/api/disease/DiseaseVariantConverter.java similarity index 85% rename from sormas-backend/src/main/java/de/symeda/sormas/backend/disease/DiseaseVariantConverter.java rename to sormas-api/src/main/java/de/symeda/sormas/api/disease/DiseaseVariantConverter.java index b94fecb9c6f..d89ffd0d523 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/disease/DiseaseVariantConverter.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/disease/DiseaseVariantConverter.java @@ -13,10 +13,9 @@ * along with this program. If not, see . */ -package de.symeda.sormas.backend.disease; +package de.symeda.sormas.api.disease; -import de.symeda.sormas.api.disease.DiseaseVariant; -import de.symeda.sormas.backend.customizableenum.CustomizableEnumConverter; +import de.symeda.sormas.api.customizableenum.CustomizableEnumConverter; public class DiseaseVariantConverter extends CustomizableEnumConverter { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/disease/PathogenConverter.java b/sormas-api/src/main/java/de/symeda/sormas/api/disease/PathogenConverter.java similarity index 89% rename from sormas-backend/src/main/java/de/symeda/sormas/backend/disease/PathogenConverter.java rename to sormas-api/src/main/java/de/symeda/sormas/api/disease/PathogenConverter.java index 92aca033266..43533256f88 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/disease/PathogenConverter.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/disease/PathogenConverter.java @@ -13,10 +13,10 @@ * along with this program. If not, see . */ -package de.symeda.sormas.backend.disease; +package de.symeda.sormas.api.disease; import de.symeda.sormas.api.environment.environmentsample.Pathogen; -import de.symeda.sormas.backend.customizableenum.CustomizableEnumConverter; +import de.symeda.sormas.api.customizableenum.CustomizableEnumConverter; public class PathogenConverter extends CustomizableEnumConverter { diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionExportDto.java index f10a26d6eb8..147c45813bd 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionExportDto.java @@ -25,6 +25,7 @@ import de.symeda.sormas.api.action.ActionPriority; import de.symeda.sormas.api.action.ActionStatus; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.importexport.format.ExportFormat; import de.symeda.sormas.api.importexport.format.ImportExportFormat; import de.symeda.sormas.api.user.UserReferenceDto; @@ -64,7 +65,7 @@ public EventActionExportDto( String eventUuid, String eventTitle, Disease eventDisease, - DiseaseVariant eventDiseaseVariant, + String eventDiseaseVariant, String eventDiseaseDetails, String eventDesc, EventIdentificationSource eventIdentificationSource, @@ -98,7 +99,7 @@ public EventActionExportDto( this.eventUuid = eventUuid; this.eventTitle = eventTitle; this.eventDisease = eventDisease; - this.eventDiseaseVariant = eventDiseaseVariant; + this.eventDiseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(eventDisease, eventDiseaseVariant); this.eventDiseaseDetails = eventDiseaseDetails; this.eventDesc = eventDesc; this.eventIdentificationSource = eventIdentificationSource; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionIndexDto.java index 0250bb7708f..6317fec5e5b 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionIndexDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventActionIndexDto.java @@ -25,6 +25,7 @@ import de.symeda.sormas.api.action.ActionStatus; import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.user.UserReferenceDto; public class EventActionIndexDto implements Serializable { @@ -87,7 +88,7 @@ public EventActionIndexDto( String eventUuid, String eventTitle, Disease eventDisease, - DiseaseVariant eventDiseaseVariant, + String eventDiseaseVariant, String eventDiseaseDetails, EventIdentificationSource eventIdentificationSource, Date eventStartDate, @@ -113,7 +114,7 @@ public EventActionIndexDto( this.eventUuid = eventUuid; this.eventTitle = eventTitle; this.eventDisease = eventDisease; - this.eventDiseaseVariant = eventDiseaseVariant; + this.eventDiseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(eventDisease, eventDiseaseVariant); this.eventDiseaseDetails = eventDiseaseDetails; this.eventIdentificationSource = eventIdentificationSource; this.eventStartDate = eventStartDate; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventExportDto.java index 7ef44af77cf..1c5450ac12a 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventExportDto.java @@ -19,6 +19,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.importexport.ExportEntity; import de.symeda.sormas.api.importexport.ExportGroup; import de.symeda.sormas.api.importexport.ExportGroupType; @@ -105,12 +106,12 @@ public EventExportDto( String internalToken, EventStatus eventStatus, RiskLevel riskLevel, - SpecificRisk specificRisk, + String specificRisk, EventInvestigationStatus eventInvestigationStatus, Date eventInvestigationStartDate, Date eventInvestigationEndDate, Disease disease, - DiseaseVariant diseaseVariant, + String diseaseVariant, String diseaseDetails, String diseaseVariantDetails, Date startDate, @@ -160,12 +161,12 @@ public EventExportDto( this.internalToken = internalToken; this.eventStatus = eventStatus; this.riskLevel = riskLevel; - this.specificRisk = specificRisk; + this.specificRisk = new SpecificRiskConverter().convertToEntityAttribute(disease, specificRisk); this.eventInvestigationStatus = eventInvestigationStatus; this.eventInvestigationStartDate = eventInvestigationStartDate; this.eventInvestigationEndDate = eventInvestigationEndDate; this.disease = disease; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.diseaseDetails = diseaseDetails; this.diseaseVariantDetails = diseaseVariantDetails; this.startDate = startDate; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventIndexDto.java index 1f92791d04f..5e2d2031767 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/event/EventIndexDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/event/EventIndexDto.java @@ -21,6 +21,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.location.LocationReferenceDto; import de.symeda.sormas.api.share.ExternalShareStatus; import de.symeda.sormas.api.user.UserReferenceDto; @@ -129,11 +130,11 @@ public EventIndexDto( String internalToken, EventStatus eventStatus, RiskLevel riskLevel, - SpecificRisk specificRisk, + String specificRisk, EventInvestigationStatus eventInvestigationStatus, EventManagementStatus eventManagementStatus, Disease disease, - DiseaseVariant diseaseVariant, + String diseaseVariant, String diseaseDetails, Date startDate, Date endDate, @@ -174,11 +175,11 @@ public EventIndexDto( this.internalToken = internalToken; this.eventStatus = eventStatus; this.riskLevel = riskLevel; - this.specificRisk = specificRisk; + this.specificRisk = new SpecificRiskConverter().convertToEntityAttribute(disease, specificRisk); this.eventInvestigationStatus = eventInvestigationStatus; this.eventManagementStatus = eventManagementStatus; this.disease = disease; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.diseaseDetails = diseaseDetails; this.startDate = startDate; this.endDate = endDate; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/SpecificRiskConverter.java b/sormas-api/src/main/java/de/symeda/sormas/api/event/SpecificRiskConverter.java similarity index 85% rename from sormas-backend/src/main/java/de/symeda/sormas/backend/event/SpecificRiskConverter.java rename to sormas-api/src/main/java/de/symeda/sormas/api/event/SpecificRiskConverter.java index 58e2f5067f3..789ebdb8f3e 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/SpecificRiskConverter.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/event/SpecificRiskConverter.java @@ -13,10 +13,9 @@ * along with this program. If not, see . */ -package de.symeda.sormas.backend.event; +package de.symeda.sormas.api.event; -import de.symeda.sormas.api.event.SpecificRisk; -import de.symeda.sormas.backend.customizableenum.CustomizableEnumConverter; +import de.symeda.sormas.api.customizableenum.CustomizableEnumConverter; public class SpecificRiskConverter extends CustomizableEnumConverter { diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageIndexDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageIndexDto.java index 215f48d46d0..c5118d94782 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageIndexDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/externalmessage/ExternalMessageIndexDto.java @@ -20,6 +20,7 @@ import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.user.UserReferenceDto; import de.symeda.sormas.api.uuid.AbstractUuidDto; @@ -60,7 +61,7 @@ public ExternalMessageIndexDto( String reporterName, String reporterPostalCode, Disease disease, - DiseaseVariant diseaseVariant, + String diseaseVariant, String personFirstName, String personLastName, Integer personBirthDateYYYY, @@ -78,7 +79,7 @@ public ExternalMessageIndexDto( this.reporterName = reporterName; this.reporterPostalCode = reporterPostalCode; this.disease = disease; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.personFirstName = personFirstName; this.personLastName = personLastName; this.personPostalCode = personPostalCode; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/person/OccupationTypeConverter.java b/sormas-api/src/main/java/de/symeda/sormas/api/person/OccupationTypeConverter.java new file mode 100644 index 00000000000..b718749a878 --- /dev/null +++ b/sormas-api/src/main/java/de/symeda/sormas/api/person/OccupationTypeConverter.java @@ -0,0 +1,10 @@ +package de.symeda.sormas.api.person; + +import de.symeda.sormas.api.customizableenum.CustomizableEnumConverter; + +public class OccupationTypeConverter extends CustomizableEnumConverter { + + public OccupationTypeConverter() { + super(OccupationType.class); + } +} diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonExportDto.java index 7fd3b5506c3..4361cc8e3ed 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/person/PersonExportDto.java @@ -194,7 +194,7 @@ public PersonExportDto( String otherContactDetails, EducationType educationType, String educationDetails, - OccupationType occupationType, + String occupationType, String occupationDetails, ArmedForcesRelationType armedForcesRelationType, String passportNumber, @@ -247,7 +247,7 @@ public PersonExportDto( this.otherContactDetails = otherContactDetails; this.educationType = educationType; this.educationDetails = educationDetails; - this.occupationType = occupationType; + this.occupationType = new OccupationTypeConverter().convertToEntityAttribute(null, occupationType); this.occupationDetails = occupationDetails; this.armedForcesRelationType = armedForcesRelationType; this.passportNumber = passportNumber; diff --git a/sormas-api/src/main/java/de/symeda/sormas/api/selfreport/SelfReportExportDto.java b/sormas-api/src/main/java/de/symeda/sormas/api/selfreport/SelfReportExportDto.java index 951673427b4..ee28381144f 100644 --- a/sormas-api/src/main/java/de/symeda/sormas/api/selfreport/SelfReportExportDto.java +++ b/sormas-api/src/main/java/de/symeda/sormas/api/selfreport/SelfReportExportDto.java @@ -6,6 +6,7 @@ import de.symeda.sormas.api.caze.BirthDateDto; import de.symeda.sormas.api.common.DeletionReason; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.person.Sex; import de.symeda.sormas.api.user.UserReferenceDto; import de.symeda.sormas.api.utils.Order; @@ -79,7 +80,7 @@ public SelfReportExportDto( String caseReference, Disease disease, String diseaseDetails, - DiseaseVariant diseaseVariant, + String diseaseVariant, String diseaseVariantDetails, String firstName, String lastName, @@ -111,7 +112,7 @@ public SelfReportExportDto( this.caseReference = caseReference; this.disease = disease; this.diseaseDetails = diseaseDetails; - this.diseaseVariant = diseaseVariant; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariant); this.diseaseVariantDetails = diseaseVariantDetails; this.firstName = firstName; this.lastName = lastName; diff --git a/sormas-api/src/main/resources/doc/SORMAS_Data_Dictionary.xlsx b/sormas-api/src/main/resources/doc/SORMAS_Data_Dictionary.xlsx index 5f40f0c4d246af4f0559dfc87915237d53b52e4d..7908fc14d84092eba27fc8950e347f409d0a3e50 100644 GIT binary patch literal 276347 zcmb5Vb8u!s*S8x@a>uqcv29Ik+qP{_Y}?kvPVQ)8Cllw6ZDVrgd7rQ9eE+U33i z?Y>s;UA=x?wKmpnC0R%)3@`uy0LGG;tPS>m74(0{x?+wFZsrbdhHBnU=C1mTUiNm= z8~yvh^FPp5Z(y-rcCh7A!DcG&m$Kv5!Mu!#p}R)T6dR=- z?ca=13H;ix^A)y^n(-_4=dQ`@Q>^yiKIe$uBlRDSBMlv^t=OjJcR07^y7eC}%1^eo zE+dbdb2uk`->tC;k6u4a2^8N$`8XCmxVr;*T!dmPEDQn&iT7Na1MP_P?4(tO--`}Lfb`a`&3Rv|{KD_W3dIH@O?sAU;2j_1Z>$RxIA7*LCZ)x?fIv z>nRW%m%RTvez(XCv{jh=wtsl=JULHez}WklzxL%v|8DHI!+ztk=PB>^xTA19&vEJI zAd-+Z9u8xG#vb_v1ug6Rv-!(wa+&!b}Ki_GICX1V(5#O$XdC<#xWMn zQ?hr3No35OpyB-=sM-}qqYq4FG?vL=E|bk%pp-jBD|dtaFOFq2mdIc(kI0J*jioahOJ%hdDCtkp(%oSHi=!Eh#WNacZW+Ln2521Zltq?X4rr$VI!8O@k>yqZo@s!=(N0BVxs?DeT3}MN zQyE!q6~L1gSQPD4MV4C)Xr~1>MLX4z<<^LL zigxNG%dH2r(*bXzoqEY~8vxIAK=2r+L9*ON02e(FHO6U#EVl{ZNgu(7HAEN#iXU+r zGwCp8*=)kO(n9$Eh3QL8JNq0Ir<($8oIYQ_Mh%e_ZXYRkJqK+Q zf_4B60uTXAJx>6yv4?f?UH`>@&{Odr+!pKo2dN(a!Ox`zQ*g(cfSB5{N>XC}4=in2 zRWZSc7lvQy)e(Nwx4LEw`iQST_u7B!OyJ!Gk2of^rg9wY$Lx^V(%4F+UjGkIK6(E) z{#OLl`+u%eOZwXPU-Q)eF+cs+9QOCW=C%GEvafWGKDqp>ztBpuFo50%ikm_xFff5A zFfgV6HNe6D3vh-m=60@3jQ?rtk_TlVSx_PjDW`gK?7yK}ug3s-qn2QBQr5N_bL6BX zw)%PpW7_OQ2KSlScD`7~J%l2qdsP|pOM=}OLyE1|UjN{G@d(ym31bePIPFP6E2tUu zZwM}isc25@sw$_U#srH{Y}x8x2!qQGMVVZ>y*umWTc|M@Nh z@&Ba`%D)k2j;2a3j!v#j#!gQEiT(2NgK$7*lpoh2Unnk)?LH_;>}4b&RK9b_fYTM| zYz6t|W-@i23>Ew1oFE-U$UqE*mLQ6O%~oRG34J znWmk+M@O`W#6_Ry zF-mb9G{1fIh7^SU{kOa@08WQc(v*MG?f6&!_vJ?Yza27lbTR*rNA`*HGQ-R$;(?S0 zy?N!B`l1`+sozAL2c!3NrAFKSd?$_ac)pfD*AelTb(tmj-o8mR8thki-#d?KZY=?8 zW3AS(5iQ8sZ6+3lEK0|M6Nf2KT(QxWnl8KVbR8>>?ejq4({j9^(E>l{0>;Eg5UxmG zWcp&WNiEvbg;ksASS;kDXyc9IFNY(QvWDSn(S; z7+BN4`oC+4@lQi9J0@2vV;6HXRW}!F2TRxg$hg_Nk-YyqiTb6_;vKn>ldmLITix7{ zeSJQ5$y5q0D=eFWI%rB|dYuR_86jCI7%W`Y!u)Hn@A|%vkgwqw|44NvqtDGjjNJcmYuHatVY8?@RxeW8l<1aJ|%A1#IE^>qhZwj7uZyMd?*3DaD#~9)-uNd>ZyK~^-xwL*p+z?rdg0@bjJh-6;B2hS1y;tp>JXRBl;b(6w zt2l2#4Yn=*A?*|mqPlZ{{#Duty5C<*2&ZRhIDNJKT=ACjdvo**?}u8m)NgX;*?xb- zwGF0*$Vp|O=$AA0x-Lrb;Z1oF=YFO!*Ktc1vDR=?+pg|uohoFg)#f;EI^0vFTVAJv%kDar47fa(h_hWwafI0r-{fb|y40_0^)0R(u_ScUaD38G z*ZUIiM%XpFl|CRKPXyWh^w{9RCVIETa4j~d;}ZJr^N-)$c492C?dBNkcJ|1Dbqk!9gX$ z!Y4(;KPS>oo6Gq@Zm29p-;XdN@eeQAj~|xvL8G5I>ejM^$XKf*zmcKmpo`(9u<>b8 z5C*I##NqJ6`dT+G?NW&nNSv@5zDNHk#Ufcqt)WPOOROEPb}V`l-2B0^1WG?s>LF#W zF$#kAiI^jMq(st}ozg^@)50{yNP#>=thB$@33-pa@p{9yrHi@cVDL@T1fpEHc6aPo z5(VAO6xdwZW>ZjHZr2usD`mGh zb$&Vi6RVIk2m|A6iI9dwZ1wOd4D|#7;>n$^F3e#{v5;)-Rcwx%@W+fXWJZ}Lion7EIYdGh=d^lsg;2zUv2mAa z5AJkN&xs?pq7iLUb0y9&=UYVqDWB^FUwR_A$Y9lZ3>_5Ts>qDsRtX5GtZUX)Ml5a~ zkd5fXJ7_3GyulR2#zOYs1c-rtse+Qa9iE1!IKq{!{+5@(%n9Ta=7x9~y3)sNw$z_! z#q42Fu?q|YU@V(|aLW9IB*dqom1?m`92(HlK)qXVDGUTSqIDAUk3RgEr?<1#S%V9~ zh{~yEOOd-!L(aAR>QWjrCLH`v@KPN?;;Wqq&Ne@L9&)-SADYQ({s8PYEQ;Lkl})S$ z0d6Va(?aUm;C_sxp1SC@j93l596i@7-NEoPv$W&LsM;^kb5hWGugYpojK&+!Wc()_ z2_fURfQ(Nm?nmCR;&L5Ax0xlI-$PU*Ng4L*9!u#vAnocZ zRVYI$BNFT^TkNY{=%hh2@72*eI%n{#&D2d0esWW{55a`*Hs(%Fz(TDU1W!RJVf`ZH z+ooH2xOT;!^WA-m4RHZp;@YyG#o|GxumVD$ZVdwR!8r>X!l79u8VT`>NmBaSS=2^k zEmVe2RHd9il8Sh$_#&*`-I|9<5z2u+car4{Rf562Xv4?f;g*{fXn*B+<#oBvhWM~WBB|2Dp7A@HN!J$n z5tfCkK4G(@-EnA;_R%Up+)ySn*Y{Vj@2*$CwZd%qG`jl+XT%JeNTW^2GWkU1|C$x=Gd2uqW1YD#>DbO#cK_s8*C08!0&uocm(u%baTi9}Ofxn1zx z8zGXn5UmN;{e*`@GErL(Zx~2t6(dJ?w6Fp5Cd35z@8E)hHdK>%5)vit&hSHre@XnE zK+JV#cE*NEO+z)f$u!-{iYE+wll};x+O9_6?s~YLDPc3L2Nht}y~XlszC`99Kj(89 zz{vXYQYNx*6*o%Pv3=iOs;kURLe4F9R#Wz(F#MUq8tx8DzAu-Nin=?FJ@gg^Qg#;c z1`fkxF(%5KA68Tr!qE;G@)CFy?Rk=DnT=ISRc&(J{?X}L$O|y+@P&~Cv+LvDTEJc? z*yNEr2)$$?{N)}9aVl*7&P=D~VtcKGpXZ2Vjn~Y^+j`BvW8fpG9kF;r6TSf__JM#i zwoVk+vo?j_d|?W`YD2fV3(9b)8HQCjC9CXq+C7Z*#(af3;cFoZsMQq>g+(a7(;}YW{Mnib2=$WywP{-Z_y^ zPK!9b6*=xAkX?TT_#u=|5)*ccBogKrkZ^}MAsm=-pt}#IAzHshDAI?yIU-Pe2PoG+ zAeX0WRCh}3S#~+2FY_>B0O6FGqtcQ6RAb=TU;ce%?4<5mM>{)kSj|}xk6-H}j3^ms|9$ zMkzaV_5g{dPFIo#=}E{JTz^Y`3ss0a<_@-==&^F3zu_IBe*6c@L*Dyx=zb;_-e%8a zuS&+$87D*(;qs{yQq7qMm4C&x+y$TWZ$EcH0fIY7N$-_pXwJupIvw{XOS||>EvrpF zh+cnZesLs#Y`W%B-*Wve?eh{6Lx;_s)I6o|z9$F33kE&qZ~;Vg;D z+kD>i1LJOW&yTSZgD*LGtw&DRz9CBZwWUkwgML$~GS}z%;QgSPCEBPKf7ZORzU5km#EVIp#V4lk(m=vMUNilJ={PAhh89No^CnFQ^v9 zD6`lfqS&Ox8|IQOwNQ*5d7(Oyp1%rBcn=5o6uHb!>~GUwisk&X@&N39XGr>CJwTq} zNNMBxEyM_eE?XBkN}(0IHPDyCyZ;EmnagDend6p04^JfN{Sro-aFz#7F;wxBi)=-0 z@Y{s`3AwC;N(}!%Pf3bk{Q(xSz7h2AK;E3Ca^ABsz-3zydiXISp2Wd$)3vc>t@mF~ z(kR(b3q|A8L||qU4n7}ekxQ+p=)%^7Ct*lb!^i_@P&bFfTxA6m;CQN z19~Y!Yv`mbcdFq&b+qUa$K2cj7Mcf{&hzRIb$*iKS*Fh1xd}?5^Wq9l>d?ku>aSwl zIW*Niw)-O?o&v0GL)hI^G})`pd*{oU>k2x{zm8!zTx%m`G66_+_z&6RlTCYq z?;U2t8Wsw|TIQqvAbEW*q%K)r1sZ+RTGG8gxjDXu%Mz z&iwu5XQCN$GE)Cx-DiU9o(sHpD88Zh-KN_ha z^queL0}9b%SjlC~cYYl3@d#$~FzM+sY7@naCrq<{xQ(x?ZdAN9Gb6D@_kOfw zdR=7_R~??8CPG}BL{1RZrO)8~gw|kpzJX|SgMGzB4r4&rv?{*Y8=J`K}+=3%MZ-={5}3x)-%lAvkljFRLAJP+Skp7DF77r85UyaJ3`+q`yI zH=N#!^tV64bEv&H0l$QxZ7mI=+`*mbQBF=nRbP^A^J-Jn?)1Sn;SW@OWX^MlV=<)) zHf`0kbY-k+lzrZcqLq<34E=-G#RZo9KL3&%);4kegvD4|A)DBCWq`;YV6d~+j70J! zk{2W}{+9Oux{po9UF$&F9Qo8yzJa}Rm+jbVSZkZh=AEz+AEeW}Xlavax7m6bBnL4m zNknWsUEWs&wEra(0qf-Bl~vv8{UYShzAjEgM}X1`puVV+31i9Z5GDLVfpy=dJ$dn? zw2z%#&92i054Ir|%*DeE76jp*#UVTf;j)Ua$h*&YQej|eKYmCOi|@+)wf@!~@k({H zg6_`<@=5GTj?(fXLFsBgGw zHMbTJcOcAaf8NfAz(}w3+gjC(H^K1N%uQ4dB0E3olgmC{96zUFKq46+)oo5_ym&~G zgKi5*wqoo6=2?G>eITlrUpi!Cs8y^68!eNct3v2M1h33Am|o*D8XT|vRjU<)dEM$g zQm3_}%W&;C_7^suFGueqI}^N>rBA&waT84Bfa0D@k%1D|V*frz%n%SicFU~T7KeObJ)J*R)%KU$AhDOBZE5=S?r%&%)F z9Nuja%ZN3$a(J^j?f58< zzNI0Fnffq$3-}@SMzSlX$qUI=;P&8v)$2nbY&T?Jzz+9=a?Xr{?hRp@j5Do;B(}(y zkx;|W-|WV0t$b0Ex5#?aBKA5aWB9iYl7K#y!lw(ri`=`Iq!MHCsN)_vS{~@tbZ3<&pbC9O2hde0OPYKi;P? zP2vhaH=oKpQ0L?=Roghw%P1a%qLDrIwcUp+c}9PbuA31bs07DBxC^hgZ}5JlXmbqt zMwt1>-^i-)eyg@2Sc%^2drO_UogbhtHGy)TvlTOhN9s`JCUPbN&ZHi|y$HNNbft(m zQreq+XM@$d&K>-Y&r%%UU``X;*($!qD4TFXyB$ZTp_{(1Dk$;&-SYe^cJfP>f*J}o z-*aw%(q61n@S`?|)fX;hqjn0IxtTjx-CH^9d{Nvw?SQx=E~5<1AcL1ELmQ|CwSt1f zi$enIv_^;2?yZ9bLl;)#Okkm*=NX^;sBy_Q(y`R~Q1;?9hDr_2ju{$+821v`VT0Aq zlxBROsf3HQ$PQkGTW%?qo{1Yx5zc||PS!5qY1vy|);bNurfcC;Zi_OuQ7@+Sc{rJy z{6+fA_3DkJRPZO@lJE0aHVoa^9K=a$JpR=?OlP6HDG>hED%i#aDHI6yAA0B;Q>tx^ESU-uj7}kf2`SAjaDRS3yo#^V%8#7}2^J2w(@o@j*XFj8HKN)arW{kQbbSs&)?c(;Re z-ISbw4f|DwW4v)Fazv@LhRL!@?otsDzV84w1S)~`^1-Go3tpsg9Ao^L0?DXxuVT1T zbZBDzTYT9~jNhsb0q4`CiJrU9*(Pe_9>g^21E0V_XpnV{6h<>`# zW>lV7sLLp=>x38>zqs!oX|koi}Y5EY9*{jTQRB zq~f!e%2=hdGOPO=wAwM6I}VNWxtO1aJB$0B#z%7kYhzD2ka?jPYAd0{zqNnOOA=CP zldBkzl!jBil)i!Ke`B$#m&o{(o<0)MISxCbHN7w5-n88?CBNS#7ViRC`AQ9`kLfCI zd7hbVAbl9Dq=1JJ_Dpiv$1@6`$dYO55(yVH0}j;qQgA?p4uS)9ZHT)YaNA&-Kzs%n#S( zK8tXVGkQz&x$v2ch_$%%;TECZ(2#X4!WK2YdZ z2NSD-6vrY_w~1{BTYq|7st7(usle*D>jb?nPdr-TLEPEMM=`ryptyB#Y!u5yIy!bhSO`E_Fy7;6eeqki1xwc1X zIAxVKrzpBY%e%i|8VuH_wV0;CKXh*K*v;!?HLdPpM0Y@%=dRz zlH`C;%V}vIywZWYPwI?3#Hs$Jg*p8Ew<+fAxx-=O9}$TYc40U14vqjU#wd@I*Y;s> zYM-PQ?(|r-rX4ZB@iHr>T(Iu|c*v}CZjCEKLOM3Wu!*$An58zeBG*X^G1rx)8q&p3 zkgOywcHHX)F0p!qUHDAt(oA|VE!+`!8S)iUxGqn^oC?IT|L9m<8fl<4X)gC)>?Wec zs=+V^l!eY_1MrW{Z2`HAO?;S3w-oFmkbSopyYc#WtqMqSQ#jUXMK!-{y4JGn5K?YH z6B=T5tz$OrukGE~eREw5cF==wh_B$9Nhm7cPOe5~X02{XUoe8dGmu{KwEQ4{YFjUu z^!&87D-^W7y|UbAQp!Cto|A~r0ROAgrjiz|KcEFViM>=MgPm0)wCV?Zzj(H7Z-}uz zek01$Uz%a9w_~`Ks!<|V>#ddWv2`v|*%7~45pUj+289rfi1NCvNFDItEUS~b>pm-& zxiRFbLAQuCRW$PZWM-O&x+hs}%MkS+Zh}Zp9}ZUm7M6+sHwEOp5-qkMdMNk8^NC&qZ6?g_iaZnles@@4KBf$&dxaCm?3u z@nX@f93A|KLk_K#t6!dZaLZocF6|BqSSf&k>g%nh!-bRBybL{i@dW2&VZf>ZVym$L z6;D5+uUoLaaL=9R7o3T2)@9y$S(Xds&~v--KDv%(-G;HM zV%O0cFyN1Jh`nZ+J#V`m9r zb-`l~vgc9#Kp0Y%UYn8MH}%>9f(S9^n|?$>f3T#ZmYKU6nTv*0zt>6f56J)r3Os3vbQGa;wa`7I+|;6hRdm zbo%eJ-t8~*)Z;QdQzLQOU1W(R77j<%;T{HdGCkLn6-T_aUz{N?z3ygry-ZCsdhf8y#$QATxW5QUTaR9h5JB&%mz z1@lf0-Ob$qDFU|ed>9R=_Z|oFMKwK9NHf?*)CQ{rzfah}blc~cV2Mli=6p~XvZjcK zSc08D${Sry{foWZnshX&xSOo#J{Dt<9?{kpQd-f&IxQpid=Pjo2w0fi>z4Sa_WHFy`H#; z-kNhLvvZ1#NmxnNI%Fp?ef>t8o<xKI4Gr z$lvcyX(DHC%;SHCB?XNM?uQ`9r8TmJh;_Z%eQOx!G}R-&((F603*jKn{py=~_pKiLtE@kG`xsCZZ2tGdZ39 zEvMI}pXVrTBR#5W=@v&*+^WCIW52T$sO|j=k)#+IUC?isq9p6QbaB&YMzyUY&hLDN zIM3)*@Fog72c^RTnbxtdMKY6pmgtls;3=fsxNhGABQsa!7>PHD-m@cm3w2cbDKGrE zc@h`O#a?H|$Hnsmj8%?;xg2Kjw`VSoD)J)7a@63;lh+ZD$U=g4-hs;7V5eP_#TbXZ zf{n|VA_;VPdZD8C#2ZUX=$Bn@;aG_HH=}>_aDF0fAthKMo81S%emRwqS%|20weA^O zup-PI#fmtsc_g*INfC25%loj#u|C{XF*(i)CV2R2YuT3RQ$X-5F>jwn!7xoDOe4Zf z_@G?M8W5~tBkzes=hX@9x0<_GFqx(vpWjV&&xg%D{lrVDnISexVKP>nGi4bqbikc3 z()CX<-xwg7qC=Rb;|gKqKP*SldHtc7p|}|=>d6b5obj8YKanM#N?ABvmDca`nu6-- zvvZvE0>pN*as~M_mq!k%yP|frGb7iV!h+GGb!_}hLJPv??8_>!YH4a>1M(Zn#U3D@ zQD>q|kzXez-1Q7{k?QQv6U8x~Le?#GG%?RF9J|!1-_ft=a+|Iiypc3^-1FAZa#ejE|Gv_$1>hf|O)yz4)G4|-_9u%ED22xWAIrBP5g`*^ujc<|zEy!MK5G1~ zIWh5J!Z~B$_v~AmPwVihE2Ff<^_fYKB(ay_$2Iw!{d)dtdUN6Rfgfvdp$d5ZY~Oie zKgv_Xav`1C2tDs6_Q6j|_R%cL=k7m(RqCU5%GT+iJFN4uG!^N~ zWzXG{Y#=`&mqrZUXK=JXEVV1xGySdc3(TB1&t_HMKSrADJX2c!#pVa(DboaN<@;2` za!qzGV}XyGwESX^7u{!0OKiJFG8v5qW>3%X@pl=)bwv_`7Xk^E3jO&l>?%2M0B}zw zH|MPE(koAh1n4YjF*5!vn`y!b@s%KpGl_o8MxZnjbssR=@nUi_WuAuofW-ER)9@mn zTYU!ZXxev!+_}LE$=Eh1+2PUPZU(mGR|{f>@p&!G7=+gwl6)DpDo{xGIK?#KX|2Xy zE%m9w5ketie#OTX-_KMg+FLMSLbNT@wt8L&;i+k0T|+8bCdqc{A#vilG_{|wtCrmc zHJ}pZ@>J_V95d0yj#ZB(|;+Mobs&-zjb{;NCXx?~aHwc9);DI((@^tdI>rBWIR%x{)n6c-( zkIOE3c5m>m^ghbUkGP(Zibp?=?jfZ{&-@wkYretZ64-TK{|h=YuG7<2SeYQ1Kb43^ zCJYjSg1{y6`~8C2>%Py1{(gbt)esKa+GA$Mv;gSl01{95h!#GW=QyeC96RzqH|xQ; z_FnB)PsG-)06(lVB_w`$DPK#0H0;o)7376f*8(WRx1sjy#uJ)lo!byYbak0+2WSaT z-aNj6LnY9lwFZ{_FCQ(mS=vR~bfx3FN#4$nVs3>Slv)MMIxjmOn06^Crs?=r+r#de zy7epEq)Pj`yCvF36k_X$hln#iKN}~aYJ7vV*?bqoq79#V5ImQ26juU{MMG0)l?9&v zp6SE$7#%jkpL(UNtRj>ZJIL#Z^c7Q-7uwO__f=1cSPS#DKVYrr*DhO24y;rNn;~8O zNYBE?2U@s}K&8J*8z|bK{1bv}%|%}=hKG|*rhP(sFa^VQy_nV;V^gYSKOCk^%aOM< z4pm++HYE3sjnh)AIID1;!(aC1Afu)@qqdeaQiwlfX&-+Sl3UoW& zO1g7*Ay+W4R@At=MXu(wHRnpY1F~gg_2_-=3kQ4I&p#HPk#fmRd));(h<2JX23Fw8 zkdlK=%ROl)-G*!bUgI=isbV-1)bW^Zi*C$|l$70Jl!TQ$;Dcv?2(bRJJ(wK7+&Lft zeZyhkUEc3$@Lv4Ufl%uW(B!x`M9S=Kkd<2XA2y zb!t%{P`c8?c@Kz<`3tQutqi_+>%qG$@SMH@Q-BzJjxG;8A(eVr3ASo^58j8^6IIFU z6m1D5dZ`SDzg=$drY2zalVzn)!kO`vI=8yMqHq$ELZ30IgZXG8MyQAe^cR1lXn;IY zvlNrS4@UlO)cmr66_HF@n=8nD>bMK!4O3JuYdq=$BIMx|A&MV!LZ^_8Y+Z;I>dTaX zsmzy%JuEl+XGz_F+qwa7<0ZKYuMrcJ6^Jo+ub7?vu4ozRGNSyC;T%!+z?cYx=F(Au z_l3I1NBQO9`$+hpX?^26N~bs{;+~06!zFPpp_r;mrTGB4uaN!-24UXy$pS-M=}(^m zXpN*GSJS=kl0--i@r-ZR@RN`zV|~pKF&Hehuiu6UmI`nzq30mdiw1 zs~tJvn{#smu9#y;3=MaFy)Kmu+}?5}N?L2Eo`MX$E>^#DxB+O<@pz#d+c!J+jqcoL zsKGe3Q>hgHS@Mj6805XbPLvR|;6{hDy;Zk)s>Np6cBGFxqSFdo zm6(LmQYkTt!=vqbEI$klT54EfTQ$+m* ziEN#^1&u?FC+LKcq1bK!ol6$EJluGjB!&$FaQM3|$K~NRewNMVzni*45gD5G+}2&r zZb?S0Fv}KOzTpKrcDuC={UloH2zSbM$KukU1&S!mBIwp38y{5$*4dfngm##635VrP z8f6h@7hX-enZ%+9VIcQj;nzfR8u<}2v>Z0zY~smV*#6iHm_@Ccw&QtBME&DcDNxc8 zjQ56>^9wOg>~bk8tiO(~pw91Hswx@rj$_Np%p1VA?L7mqJ8?*Qi1DpczBq5Yv#$8i zqs87w*Bh-$NN#TDPj-#hHS9GySGCj~iMz~OXrsJ3j~lSC`Z4=?UT@M$cj5n2W8 z%lJ<(KvlRA2sgyLl0$E%+`uO^R3W^(*z>iq|(PH(a*`8k38_>QYvUP;RDA(}Z z*z{N*s!_0BKD*X;K31{~YkOvpw&;`td0VbruRaP-RA4uI^m;V6$f#`<7ml~xp%Lq^ z!z4On=zLNV;M;Bh0l`?h%Ur85AKx{%mG!gGKhTnWRvVg+$SQ1jubUL z>h%Qh)9pE#AWynH?mBXs+AZarzf$*7B|8rLU{>zD;NE}ygGewcR^=ie@yXQjG3jmb z>d&P`!n}Mf5PztuWwq?3C6o3t%pbw{hhG4#FSAXudM+cxQi?&uc{O_;30|N6krEbQ z@sV%!-BIge4+mk*Mdc8bqwz7-hi>8|o?eAqKYeppnDS9-^6jW%J1O3ru~dVdWAytp z2e?0aqcVFt0|H4IVv;;^#i(t5$grYGK@$zD0eFvo9u;%(Y1Z^^GX@h0vjArXh2U2< zt`0Hu+H3hpv#?mZ3Q?ON_odEWMa0aZ`;vX8Y=@myOZ#AZY&=FlxL;3VT>|G8hXzMk zB})l2q9d@t8bhf3g6szXhN5PDA46?Fby0T9SJ0=G8JXkRIzo_sQ&xUI^g#c`U{^KT z>#D47M=G<|W-Zs|XZ>=ITB}>m}3H{F3(9P4{0;y1MIYuR5l#_7agSA+dh3 znUWXECZiB_4~zw?B6-aX2-CS33;m69@XK#;)%0vgaKwt$wOeWtAM^S_QcU9OLsa zzMzaX-O88~a(8+_46Ki)7+uMeW!(Kt^#xh$w+rPW(LOV8!^jZ!tuE<-@OjojH)Q1&iSe%z)9?l#}rK>7CCqqrFSC@rI>URG5S0|*9|X*Vu}nzswpIEyVi{2wlY zw_7j`kUklS^?b92Fps?xnSNxfrFm;;MJvsASs zCIozXygM;aKiuf@2STrUmn3?-F7!3vli>F_7MW34kRh4c$GXbU5g3FIQ4iSu_m4Fw zeCtdsXYCFzz2%4YuvKE0@<8rW>$`T=s;TJ{h7|-tec?L%7GISH+KO^K?4Q@nhoUOS z#LK}LhT*1lf!{YChOv)d@GY4bo`;K~83>%i4W+9Zp*T(rITD2NBw7#@C{v`1U}}YG zz44yuCB{ng3e+wwdV%sQyB3Kr#;a^4@m?l+|k2I^ww!F7l z*>0O@`}ePoz`>6J955_+9cL(y$+2<1$3m@^dmw1jXMx5;KB zM%U%snO5JO=l=J%%kj;Yneo}cLr6$+bjci6MuTf5dtW0rN=T(qhUmwo-OYHzdsQGJ z<;XJ3Ea`M~1g>V{2E)#8!QHN}IN_5@~?-E%&Xu^-NZHk0qIEao;PRiY9n=?~~JX?ol<*Fu3 zwJPvPp37vp;Sb7XDyL0`q0>59+O&NVE)%MU_fYC^L|YCaGZ)EY37pXWS>Nb6_>Si8 zMz2f|>hMYoF0y1l@}nk5tOqSNjXI$9vTfgF>Q}fOyUxM#*TeN!=;)OFM#3b3@oABD zEkSffyB-@vgu0&jN$Dfq?4_3L5dYYq{rjDJ>iKk%S|DQZdue;A;EUL|F%*`fG*_JC z`iYVbCb>qB*`*N9y*}k#kQ(uHm+ywxAWz?++fS9(AJkvbaSsJe_PfnbTnet9l&Mi!BnAYtk(DP9Hynxv7`!DKxP!`oQ3uu@!>tp}2NXWn2qSM$KknEoW z;YGHMV+qUdgy0WTTMe!{4SPoMLIiC@zTS7_nLm5lyJJ|hc|I9BreX3NQC7g`Te)K~ zf8ztQC}O?FPb+>56u8IAsp(mVnp+9%KhTCgoidAD+A>iWEUFUt&xESVw5ojnqlovE zvuRQkD_GM@-=k~`NaMy=E_y+Y2HmCPPxMglGefSV+;fRbYmikN>-bR^UfM*$~}@r5|qYs@V&eaObKI_LD*dFLqCIeG&e|R>C2+KcH^$`(CV(eQ zH?1_YD=t6BqA_LIO1ZCA#|n4 z)?h1sRQ!Kk;5TD769`=7ANx^1p)S&Yn&OmbZ*b^Q=iiPTuM}UNxvAg+ew0veq)HAt zxKMA?ulF&bq8eV+$%nrU`tb>*zkb))oEC|gVh{62ybh|(*|){H#_f*!TJ{-l=uM)) zxytB~zA1BUd&uxRO+;SWQ4Kt7C=wCnODO#5!KaXC2W10hOBHhT-Aii8z3z5mx=7kr?a{W8mNH<+c7X@n16b+aMFXx;o~xs7#p zU*%Ml{R7~fD8%)OO@cpVaHrEz`Uo3*+Po0yVtfaNAePJtL|He#Zj`ZZj^psc7mK{Z z`Ty<>nA&j@do-kNZuY_oDql_AkjiQHPCS>U_woMA`ZCa1uJ! z?J4_09fxntHrV2KScpNac#JU(c=-9XY^V9mok5s`BfrFkHdcOdot1@I#2Y|h1EUH5 zv}XSmI7hBd`I0oALZ>cc6AV^pZFPSIY!KK|@7BDhXyVby=4ero82DC~$G-%?rh zU^)PyA2@d^uSUooHuUZjSQ3+Q%jKtwKb`LjxB`FA(|Un4h+(;$rP~&k;zApc3`)Zx zU6J>g%UWzrTZXoJeZ?lkF?4=7oKNrlkUa#4=G{|+IK~o^9M{*7;Ays$Z!GVL!ws3} z$w7}87Wz@trJgelddzg;J1KT7@D56uuq+DbhF9EM)K@C*UVSYtu0q>+2`IYFu}&zk z)cpI#t~x1}9udYeo>(|R5mdO5ecS&D(qa|Q=ml(#jxzIFcJN4L2NN;3Uf0n5{!AOU zOyh2#&B)V^*oDHY;3!G`G?)?SF9lN|GP%N>1R&5BwpteX{0- zw$*4Hd84gO>NI|&QHB^@;*a@T7f#4s-XOkjg(Q4F_##;L&>qGq^}IHL@)4ISNWwiD z$Iq^v*Nd9!#6)K z0Gd`KhxW-xuwx-`yD@Ou39>yN*d$% zU7Mv&xQoyfPVA{xkG^s12`n}&H&DjPGhDx|5-w1~z8PXB-Kg*atmnzJvtr$^I;M65 z=OfS$8i_bGes`L>|h#6mEB3{htru|d}E%P;G|huvz8uoD>wjdsKs+w-K}qk2WV zSN{?aONGVCG`f_s7EIsUrwHJnVn!0qdMt5!0AxD!3=0o?S{IM{?RFV!{l$gH@hvW8 z_L};;urearJk-ehF1NuPXnw4>R07H@8O%&!jPt{wWGfQADB>brtqz2oR5w&$k?7cw z@eBf|w#cQ$DZAWJ;SL4!`>{ zM@z&Z3cevu1~svugA~pJE$WA00Z7EsfT+Zyp^+ zW(B*1)52{WCmd)(Ge><~OrIDNef9Rt|HVGVtVL?6WE4>!I~R{8q{{3@n;@$Dt{ret z8y;!QFr(GJj?KE%u>v3I_ayM~FJ}tLJ(9mh6(ZPC0@7e4bKB&-d-(l(OT`<^4sPZ_ zFQL$Vf5qw-5@H1GOu(qQ0Texju%}m;8xQNSHEFv;`e1{i-$AbjsPBl%?`NstE?VJ! zXY5KIlW_{%qs69LQY4KU8hIk=^NYsM72?~Pt5F9Q!~pRIuTzX)pSZ#3=r=CcjVaUr z4`b&PBU-ei;nTKl+qUiQ)3$Bfwr$(CZQHhuGktI7c{0gLRS4T@kQXsB_p!ArLfU5IIQvqUg7R(Yg% z9>|4)tzGFJ6oWskz1K#y$4@PNpKLrP9@I3Ac&gDkW#3qz z_)U7WLe+X0UzwwSwwYqGBhn4o}exVd)WU8G8= zJq2`sg;|RUz1`Z3WY?*v6FLoH*m{3zXwr*kZWMLs_+#-zbKTW`9#`i^*h7jhkCU!c zZadg25pZd(Jvb3eX$qTG_K-G^&R${d-c{8>SmIex=7aQ5Akbo?!Ih7^o1Ov5xG5Is zYTF9!IN~e-nxC~j6im$?RQVEESdan%D&P8@C1keg%R&Ck(4vL3GJZ)f+))u!r;Dq1JDVBe(OpHEm)dsk6Pfgf9;gSU=|&<1CNcS(m2x3G3P&3 zr|VAG5GfPXdL@>OjF1a@EHn0VY_o_7C_R8<{_ozj{U9BUs*Ri2Gc|!u=!Pn6zQ_bj z%9qeR49ovDE%AHqo1py(hXjBZ<_vBHwcQt=Rsm?9I~vs0A9VI3 zwCxS;60W!LC$ei>t=GBd$?$sm$K5v0#xQ#u1_{%q=yDYvyK{h}7sRRu_cObg_loaN zU0)S!?d=|^X3k%B$NAcz_*kFpH|gFXVUV%ijAQSKzZ5U@OH+9+=jfW6DW-{N=^?>E9$&CC6zC=Lfq0suP$z%&BAuY?jt z9tF!B)c`5z+H#~Ho1wDmei`y! zPQDJal6Q+o%zq?0LQe3)lXp5*QkFi4b=f5RDBoHpO3oC?xwmB-Uc~ToKZ7yhwe=&4 zWTIgGsi0QpwLdI_=6pp<+2J(Mb#oo&2Yi}xWHkF<`7EQ@_PDE`?rvit4HM|J_fiUUk=$>WD{8@Kc)WMS~Ab@N4;}v zktbKtuFZf{jslRcZc9)QZ`v642d2**mo7<+16yM8BPrN|soGe?_3eY>trP^@W$YP} zF-mPg1R{K5f{LanNqOkt=0bZy$rcxYfvi{PqcyuJG{`7^AX6ZBdf9JcW>HWFDBVCD zpU##M^34upnOQ;S#X5JL5dn4DcnG-u0=k&6a(xZ5>Or&iksuPf?znK=QhtUg?A5at zD%zfI$~Jr&By%|$T_Ye;xIIYbifVy@2xv{OJ2*9sHhIm$V`;z`{vYx8opArmaGI}y zLl^dm#&H9&X%xR&QH8?&Sb)#PF1gMSibSHqo_fT059KTeiCVu^1IDSRco)fqNYzLP z`ZbRIx*a6A7AZHA&c@UK3TY!*&HIO&rf~qjVey!CUcrJnr5h0IS+@w=O#c~rO|!|zy(MBC$AI{S?g|R_n&o5%9JPWFv-l+Ay@2 zi?UiR@-I`#Gfxy_YdA(fsf)!cn4@}Y6V+Y>iF&|#C?hSSzXf9pFayebSqqLK){HkJ zXv)zGuggSamJsmUs+;HcxoG`2LT?lmz?akRfpEA1vW){AtjM<+5j~Bt>ane#VLl1J z6H)SmLrofjj&x@a!Gl+FTbyp^7ccIsMnGqVaAhj*(C5nN4?a;gV%|9+Q*GA7TFVIUXDCv^dZl1(A{AX3vOgvp(4)YX*PqVRXv%j1YQ-pwo%vF^B8wPbUwoLdX4Z})pN~Cl?91vFDRrFgTJd2NJ z8^~T|=&}-DGhi>5 zR6BOA=qidk;s`?Ka~%$y`!St&-SJ4kucCF<&3IuDom)1HJA1+4)(qP7!e>I7L&C-Kecb}L$oKp*j5;32Gw^;LGF^s%Va}Fu zK+OL42AFaqkzPaL!yteahY4nNWF-vl7hG`#N!&aA6%$tN6M8u@Yj-}W#MualckWhS z!sGzrFKJyY-VVoz_kknsmus#y`yys9j{*i&yZ&@_;( zQKg|wGmM(#0AJY-Q(WA0($K4DO;=I-bL zvf;KCsbj;~p=R?dzZ z1DrFShS1t`l@}gaEkz8cD8!Tp0_vMdb zlCdVF36GtCOP&1{(l-r_oVM8SL;oJTb4N89mF}4>BcxO_!vb$j$SLF*>V(=Fj*hZ* zPApLw?MuBf=^lLVzZ|L3fpjl#2As6YcB6Y~J|$>~M0F5Q>5cLW^R&}3l!uv&{WmOD zW_0>G*kw{Ux)zPaMG|c}zA?+-LpJ_g&xF93{TqKq^l0t7*=C?>eDTul#(6nwM+i2{ zToTQHNF(Q=TUt^6pi7n*JYyR?tV(yQ>K176ky#H_m`QG}jd;P!o*FAtQyS83usi?k zIp>+QsRX#hSCP+3d61!UEvtx>=%@#gbO2yK_bjSNSmk+^ONEBh8iD3^DvrC@naa_p zD(Cr=7kP98utze*M{7UYgon0@YDhdT7x$XPLSxEo8fAa|3ac|&$APs-%8v>a9u ztEyl$Fbg_$h=aP!U&DHzq7WfO`guaw&EKRz2+Sr&r#D1~G&C6vTsa0`W8QI)Yl?d! zmMj7<(rC87&_uaOwMP03q(sjs(q?5~Z0Qd`fZ=^shBkBQGJZg6k=yAZN38&UBmWXu z@60dP#h25o{yf@Ikl;C}E-PWeMn9bU;a31RUQm0G zfOGKrx;&+JHQ9|$a0j`UVY$pxc020q8lHx#Xz(pgV!Uo+=H)l}17O8JhbSvjZwe>b zJN2qh!`vV5(ikcG->Sf-7Bu(B6Aqq+jNh%^1IXR75Hfb{+3uUUN)R1i7D(5D+eBf{)d9-<LD3|V;f zjHc9>5<*i;RE`a-W|JJ#Rb_ z5@1iZg^lc2-u>(VN+ayS>z z{R;v3ae#G$eAbH5g)E&9Hu=TM8CUoY#64m3E!e8=d$_+Fn}>|{r*q9VD&Yj#K^c^Z zuWI1b?olmPV76M@IH6Ml3ST?oh0@q=R#`&w5tt(*&9PtzPFpO7JVFgiHW*d6sjW84 zAUVs})OhS<1UIYPUE4~GGEO!mJ6(lY%`t49Bce(x-pMIsmbCV6=`sH~uRccVoSI8L zwLgOM;i_-3$}EX|Yo;yqJjqLv!qF!+3JY2` zsUk#ZE2Ax}4^dM*C-)T8fSNsezRzjS7X3kz7!OlD_H9m@k)PoJXx{E0&orY_2-$yg zk#_5F7GMS5$Kp$TB1kB?Cz96dVI!(-IxWLvv$2$(1It`(0XdQ6QKRQ2%sA5qa$Cpl zcD93*atnnTMeRBmwu3kEY_wD$edPzLWb^$yJp~hcGr#2~=nQ3D>&fZ7^I9;KSVwv* z+}g@C(!($Oxxo-66mW5Fv)U=QqiEG649HI@9Fy=(Hy&`GKs^a`nsJD#$zG=sZX@(9 zd}34Zzsit=&LuSJdr4<}5NkWdb!@hM*6KbfxCNzUi0XU}Kct8S=XC^Qq7vQ9k1-(D zCUgP2_-fve^OK+FXYiz+#u`7^JrX$E?8T}ric=_QplWR~I9?bb9awp-5V2L>|I*!- zs)(N5WsyncCjK$gMTHX~H!ezs}P@7%<_)nhEfW<3ww16kO~d!|Cn z72C#cy$^2aP>1!clUG*_v@?{|I88ldbpUMQ6NYzo`VGq zl2EW-Vw-BYq$^>)hi&Y^KJ=ZDqi^#45Y?srg?^`FU6Qg7=>2VN3cgdwstepC@;A|B zfV@PHf1ZnA(T58Xd4F^Bh&wE4j$BFt^Su3Y6)Gu^iN_lE;KKOe_OQqI{6~*^{RT-r zyOWT0v)>UxNNQjg&k zs8lBBXSY50ZnMI)rRuX(%*qY?cht}Em+Eo9_U>YHxX7aHBh6pv@{pf$F2~fRYqhlt zhQF|Z&dsA#C^|9?u!t|;2w+bRi!Dv|mq*#peE)9$fnF*{JQ{B^H*4bN2dbDY8s~|N zi8N?s_?sW_O0V=_!M9FGOf~^mt7tm%6aqQbXkRK8so3SIGS%Z?pFOab6_>xpf5%aP zE0o-(d0I8QTU0e?3<9{^en3xG8Z8dFp;+cBpmU6Ap`D=+nHXw;mTOQ7?p<*S9aD5ck4`I{PUq{uN1j2CHR-z3 zk~a3G`WiflP89u zlzpP|JI-C-Lbs6Lw}`xmb;wG_c`N_(3+-t$-hsGb!O4j_BcjZ4baHd%Nte_nT7P;48>!!L3K)!;ngifFP z4C1OANzMo+c22@%Blu42CoEb`t2grM7-Km{qLv=|?@GN2G?4(zFu50ZWvC!4G`EB( zw#A@PeV)KCExP^1ij&aGFA>gQZdhA=<|WIJ!qgmPDx~Id^U;*`Hc7l_Lf(`=DLD`* z@g|--P{jQvuYjTZ%c*vx7&FS=K%L#_YZ#ygM%X@6bA8~}SxA+vpw6?#kC&ruQ?H}% zm0C_Mg6R}n_0kaPBl z(l_ba^T`1;X-(%I#VtT=;aAZ95b@%g%egHwwwF~17*f?I4or@US43kW4ZYu9JaSAC z8UcEQol85KnZyE#Nl0a`%X_Oo0lSy^_IQZzA$$bg3$|^hbpS{=eSb)&Hkh52iG@_I zosqb*GPfPqWLkSH)bN`}6fca&HRIl-GF7!2r<|AjHnx+C_$y^UL~Rvy4ZFF5`|92Jl{WlGIiZ#msY zR5Wnv5w&{K)$f8Aq&zqSSIsluGfPzt_6^OIcEDEE@rPWlJ!sTgO5t@ckVJ8~zpXpI z^Fjqp8hB)1l)1h#b&1Dk+HHI_1m(HEVhkO$+V9fRf2VRM%+kNw4QtVkV4Rao%v>Bl z_!o&b6<_0N6X|1e#s%kt=&LO?+D1+*=YdZT$Xt7R9J@vM4Q!_-rvXZjnCSOQ$$5^S zcstc-RGGVIGcVWEV|kj~(%I)KVG_@o`VF#;G7E;GHjdQ^q)kV(u=HrA#E@olcsX~L zURD)9=Y8;HwvEjjIEwVYs{TSVqJ_@x=a^51+@6|#(L&w7>7Q=r(BbSpe?4!#Wcz8) zyA&vPo7j;J3CnFwyr3S4yQ#Y}NeD5dVzq6={Dq1vZ+s}WZVppUXTh4VxKS}$u&^ef zux(1huh1B%-fO=!8F+Z++H#d7&wx>%3% z7d4Ot%}%CM6T3-r&4U<)P$Alf1%CpnW`Ig~2F_Fo86>PX9r02rKtlzLeLGidwmdUB zK4Ge3WvrQGF2@x5#|rQ3P0(2>cZ-;x7Pi#;?YW%)$z-Of1hp}HEo&v)?TyUODksp( z02>N4qrY4Fwuj(B4|Ozu9eaYl>}}S4q}UB1ilgFI{~>m1(7uoVS8MsXuiv3T7--0P zh0MN*Ho0?a2t75_iSy6++e}V1rt4D7w;p$}O=(5V02?3+aRyt)sl2G>@QGn`-p!`= z*=oJVcF{d|GbN{g$53O_i)0t{UQFg z&mo5i4fGoun`8^zQ;G{*i_(a00WiC@EDL575@Qi1QLcjNJEWWpYGDv+E1S#t8%5DE z{ea|#BBHR$$Rf7uJ@oMbKc{q8hx^)Qz5x8d!nUJssuK=rHx;TplwW% z6$}dVW~h-XmCwLVd2smc4f);_Is^(hWxuNX)7ko2m}?ccCi5a&mLBh3Ws)mR%mDPv z=}I?|6{=enYQ$@z8%Zxt9OrJ{1RH+O(nXT2YBe8Mp-_MvKwy1z5r{Um=drE=kSf8J zv(r{jJb!~cx8rr`23{(*dCx+N^Zw9HF0h;@_s|{`rf>a4e6h4|=pqezBkOlE0O94} z&uTg5JdONjRAZ?Mft!W$4faEw#bb$!xNS3n0iyjZ7Sk<)1P2mGuRnS|kfSVj$B)T5 z)5V&GEmIUngvT-{Dbk2GyrGNJNmjk&)x;IvTc}CT zQ-Q(gN1SihwTN8}!qMHzb_`TC7mC+%bf0mtGYa%KJ-O)NXbj|hs!$V$(tQK%(#YA- zjw?zB@!v34R^ZbFqs#Qsgw7N|IGDcniwr3CAfXhe#LRCo*xfiehC)VdW)ZJSL?udH zogT)neBZ;-XMF8v%8;d=v+rl%tx*z5m2$80Neo4z2N|m+rTngXm;IW{N!dHnZkr}U z$eNN9jylMDU=pCVW%I=S**@fO4{+G6_>jFVmDTbHJI6Omn=kf29Hc6Sllg542L0NX zi{p|nk-$irFa>`Ydc^W1Uoc(PkY$>S$>eE_IE(@J?+>QLqC6>b!_v&%pxf24?U>U$L^MqnyTlj49nDhqsy16!cGMcG)irinNvspk@H8&gFR^8b#AujqA(xv@B&S{9ZeQec@QvCCnS+A=bzd zNZQD$a*|{)L5`cGk{OzxhS`M&B_z$+qWHM!-R6QzOfhoY?_-yyCbc=^%HR7T@7@ht04T?BK;P3-O?XQSzn|+Cr!HHcwWC-KkOU zz!IVb`7K<#&yRGIjrC>fj~vF}n~pT$xUpnLGyOl#nrKzJIF?C&TZ{e8Xy!KTUB?7C0LhrRV zHfcZy6FZUMgAC3G7rOeCz(*+1?1Mzbq`!oilq|rNN3U9#h`2X;EN4?M_4cE-JTEph zsE{^3H(9)XVt0LB&3QCwD$!)_^Cb_}M;pL1qf`gDUWX%N^U^y78MSL`1CF%t{2T~Ly0@(s zZa=vU;$(?#Jugq(vaURLfYW&g3A`D_Bkw9dv#b{IJU)-&NQrL2&KBQ4ubXNYz0*|P zi&DH&3V7d)5h=TbnK46OXy!Z8cLoHuKHu#TuH=>!p_OQJ^T`$%C6wqy8&VvsLmX2s1>7$$&FRveXRn}X-COYZrw=bx0GV)8DN~^ z^YyEK#n83lfXRZ^$Pn0g)%2@(PiN-uG)TO}P%Xph2J>`%RJz+a7qn+l`oLM=4-yA5 za|aDaR%K5yxV+4duNH^Nn_0Zb>rc?MNQL~YAxffyIqs@um?jUfZXC)AT08&Cp^|V= zEp^;cXgxl1ha7PqpCiRpoH;T1t<$?upB_6sm*=nI9zP6mAk;h-SByX?Y*yY=LN;NJ z*LAVHVFv6z=a;Yiq0O7ygwSD@7b39OAeUheM+Q2_@eo?BK+sff8{l6}*0uho@Z;ho&4S zj3bXA+|0U=30s?PlKXB%gHF*FVK{Q;&Z-O=l{7o>cR8Dnvwy*xen~FJe_hP8JvyDI z6%TvnP?*gQI?=9H?Ioc3cFrgjx+y04J!m6ot)rP0op&19O7!-AcDcaR8faSbrWVuz`8BIhHsc#UxtQy;YCY7q|l|ZB1e5*B>ax8TVz_-;U{3?%|Er0PqdD@v zyjRhaikXoAn%L%SSxoWg4^QvNO#@@N^J(3qsnu#;+_3N^fFz{9Tw1zfc1I|6jP^Jt7tUgL%$5Yg2@pGxP*FT8VZ>{-># zrU%@r086C$DL4M@EUoAa`DyvaA-a%) zhk5=>{b$@KhE^dB?$YxJQ~jH_FIB{caP84~ z+X=h{Hc}WcV9ytTd_)~v?|h$`@+_Kn5zrA&@8dj^zW7?ks7}#Z6PLzT3xCo&5~mUL znOl*rJ8hz)rc!<}RH8X=4DP+buN#751Y8$Wm-l z9tGc)aQ4&1apN>$s8p!IW3Lxr>;YBN^Hb&=XO)*8$#WtdxxOJAq=xBp+zA-{{UvDb z0(b_jT;Q*>fDH#F(;uvc5$(Fdi>=GqF}Ih-F$m+st zi-8j58v(8l#Wk(}BiX_ND<8T!LZ7yDvDychz$~|)Wnv&a5!Vi0&eqYS2jm{!QeQng z`4-?oC9)l`l8!E>6E=F3>*GZJ5lqAVHWhyVOC#ky4)F0VCCs(~@! zPX+$DvG8(2$}vJ$OWT9aiveunACO-rbZ_xcK-&R3k|q9VfG-Fs^m2kpT0v|Zn8G># zGY4GhxkZrmTz^Tg7yXgB6|^-+s7{N?&q89)o&}H5sy$1YllTf+-lg8q$0?iqs{yYc z#*R@%s>zvUD9!{Tc`QwK!9!S(5!OlByshVxepl!*V1*=YOKSIh&nX;4vPU1(4W5RE z3_&Fjq&>%IM}wZ~ef`J+WKl~!UC z;ppm*tg*leW5Ee&8pW@O-~p5jx5DzunN-?{be_de(4)h#7^8**??HSs%1)bhXZ2z*`$QZyu2aY+yBul%s0=@mbPwpdb7tLuH$< zAwIAC_IEx=4d54yz!$1=B)18D9KWS*rvirZa{>2EHW!EOriS&VLarP)fy8Ydw{oW= zK0e+fC>v|b{`$&nYOaDc{p(yRrk(EZsa*Ii?s-nM!Hqr=?xdgnee zL7$Tv4Uz+O%T@q6uw#Q>5SNeQBm;oX1fOgmf-4$#mbPA6| zB)jC&b-;8#TVpC`L6)p$*r3dbD=O;7sTh-NQo@*4=cC>^t!lG(-@r?%Bj@AIzt!K1 zVY@ZQ^{@-2EC~ht(9HlL04RRBwzj+);#RN>AhUQC36GEtVX4w_imBa$ zPI?Z;m&@xPEbSo?-$f?$DLFa0e^K-7XPPekb~t??qy?3h&&*s<`^k)#&H;Ve?~Gm9 z*L0|sv0aEdBqH$<%S6fM;<2NQM7Kf9%ME(RQ!A|QZ0Ext-a%QyR98<}&d|v6o==v} z&?skGT|USktXisz%QF%fTxFH*ps~E z4`x0xghz@(i?V3?H!1c5jfO{iy zx>_3K%kd@;3Ep}dd!P$C>^J(v}@8pS@6K0<`wPz-H4gno z_KS!1Gm242qb~Z;BH@*x5oIUzwU0u_(~tId1Cn92Ieh`8xErz@TeChr@Ff%rJi*7! zwFfc7N<$ydoyMNY6HtZ!Uc=wME?C(*NIDE8cc4r&I$gg;tt~)WxQAS^IzjHc7)@>D z%%q2#7X@r#wy}md#U-kjUJ+eA33Az`Ht6y(d10jh2!d*mj3QZa?#j`AsD5&)DCIvJo`%2J#uLis zG&k`|Gd2{Hh(NWR+!6OMD|L;{IzLflKk+IE1Y8iWgL(UNtP8*&h)KXpnYkyW*foDZ zAoApaDBF5laU~IddUsmX16{Noa$F65xEdc&3CWf}iQ7#G+%}_T7RgYH>r=Fh(G(Z9 z1^P7~@Wad8_B$=shYk9tVcVJK7YH{GFgu*raY;9Cu{-i-kuG9CwOb}GY6CW~kSM_5 zWz;jH@cV%p>q7+kpiJY0?R0(YS+;}swnX$cTTc;h<`*uN%DGs~o;!gdJi8Fg$iq_z zhTMfJ0k7Iksc+MT^Q_Dew76z9BUWAaJ1B{$F!!db)}omSo%P6!&>`Ol`tV>F=8ie< z;U1q#rV~Tp<^71S&k|uOjSmA6dyrai-UL#Oz#m&R5PyKLKx_wdie{?GGPW0urH8n9 zP5S)nhcw9=>|@^fd6Lq)szJ8}sK?8U&^-+}r z>^TEBJFbvP%^)E=n!g2VJdq~pgw#eFX;YA(f^;^@AU&+qgG{ksyad(~RS)B0fwKZS zj5?6+tY!TM?wm^UTDE$xlKvg{vn*P`B7OAL;=FxCe%)kAkRdyEAzENeh&fp&+ZQp_ z{(Lczln(n`^%(F(&oEylfNA(uc8TLqxJ(~dTOq1N?kW24t4H+3y%HASutJP?0jNjW zFy_jrag6LgrBtT|5Uq=HfoM(W?|Uh`Ey)>&8zLy4BZBv}$8(L+M*UFwL{M!M2>f+x zJOoU?lweAGt~(+QydU?Ow)5Ai>{G)RjeRvHiMY8JzHwRU{?>R&t$K}$a#U$VYV5B~ z%tZS|MG_lw^4fTph;D^NXMNjW)Ax98+*gpw7DFiU?!U3`B&^VO_D3MIf9~n1%jo%{GbbAwp%25@0GLYy~exW$;q`e{B^l!7pXGDr|?>$m1WS z?pVq@j;~kpEmyS8ywB4nl?H9$}#Uw@7D}f36*; zc#S@;qXD_vx({8&a?U?yj`7K(G^NOX&jzR(vbb@dVq}y~G$rC7ggN=9vt(W^Qlm{fiN}7=7 zFkXlXx3PX!qC$PeZhr!~28LXzV=DfE;0TA;f&Nv-I$ja817nEUmyM^q;YwNAzcDcz zc|=}is?I@~H1}HPo!?0x1eY-2?hor&QP22$e*rdTIqN(=1tN^HVr*IFpRYmXN$mTs z8CJz%-Ngmjjzj`kbcF4w-&>-EeT`E=~TawW-_YzeYS?)3H^mn{Z%} zak4{+5XgU(dvkShjn^y-ze@{FJHNE8<{0+L*j&(hK)0*O@nYT7<-%vMeWj4(_813w z4j*%-c=^0JURlw>R^gHL(y2SiHs4BDbDGJv8&h?D-N~0nez_pX5t#|A|AJ5;R5afh zN-bFyGuIq-=~wS{9O;+^A~m~?|J$wfC)Ce~yflD4*c{zD4Vn^=ks4hqxSSj+6Vg#Yhm1I{EKr3R4d{4cv(_n252z-gu$ zw$?eRxL~uZrN0s)gmPYTSqes&mB2MfaD=G&@&r>ubBceQ3-p_sLLe@P{;fa_V+j3= z9eY{dH0!Ld;Q&V@Hh+@6-7;S zW;l780i&2yNw4n?qU^;pA$*Gl$WO$VK^;`dtf33f&iK>3rqKtAcZbAjw;3a``#t2> zXZ$#{xqwWSue0~#O4o;XxhUVbm(%otM`JB-(V<5jowlWI4E>l>E43iEY(gJS#H{gs z7Z;4lQ0Uz78-rZVCFMuSa~5c87lb{>3!D6Ih^CdYc``^`Pb{dT!|uBFDUwC)RER)( zU6E`Yu_xeXgK7E8qcK+4RXD>tWnjtZBwS8rm-dV&`i)v8BEak}+!P3Wdqc0>L)Cj| z#-m^iEwF!EYL+;&d#rqZnu=FuS z8)DYP!CyiBWbu^eoHvDLjNts?N#=sL;dJHSQacZ5o81*CC+0N6Bgmzcdg{e|zt5d< z00O^^#VrL%H|nb=cS;*V{vf?u-Jxy(;)8||qpROcNniUl*sx8V$2oU%#Y zy^=kv4nasj@HI;~!kZ-Ix&3R7C~0Dh4*W$AY&SO@kjNv*jRPHZ_fS%wsH*b;UNgup z*@3oAz4o`V)W`t|5AQ6%(MdoBEvppH%9&hFQv2eBg4D)aB>nX)q=@#BcbqrTxXQ96 zs@WHrC0qdGaBFvUO`tNYQTHFl;Ti6#fKg#;`Fx_PetOO&MAD)9yg)mThcmh1n9`@!D7{| z4w_U%Ar>v(M3+kmn3t~$Lor9kh|DM*ENyaLTqDFuW`eZMR8OBkvVFE%{SZ852(^t z1Dw`}_yJfiW-TsVhxt8RY-;1~7tYncY$fRXGYW#p4+a2)eLLI>kb@j_uI!=#J8$l@ zRCGu&rAF{vL5ewjnxfhV3z*hgcncaOY@)0mvtvT}2qVD*CdX)}kS#!oGU}J+9I-Hm zF78cB$)BjvVc!6QRKGK&k=MVW@y>aUI2Z;`p!Ph=J?$gul4x%&)#2-CzbKY}MuSp& zy5+ch0HNZx5cRB!?VD0nSx(%!e_^KUNJ^pGN_{;#6Qp>dj72q#szSjuOLM}p?WuKF z?L#ynMGGWJN3c|;^x+UfN$0oFxPj>aj0rTCGq?0jo1V03mb< zdX9N=KPZexGY9bU!s}$c`XHE{iw2Oxo7>M$@_C&uG1mEFRJIluX4Ez#Tk%;B5FOJN zIC+-F**3J%cp4L|#C>VyDQ;l3N~v9*-<=Jz($rIsK3`A|rB#Vrw~3@JBhdzm=!Jdp z6^MZq!!igy6_e@>3NU9fXb-w7A|VEfcvl)j&oi=wyB%mJc8*JpzkRBSZ>%ITDBU#T z?voc#fsLR@t&^lkX( zy6r-GsxMWWw3O~*+h2oZ46#bkU)T{7sDz>Ui5Jp#o~?k{)`f<6f2lwe=v+}5TwqXe zZcQ;gW~b-p!+8?za{Artee&ObEtORTZ9J!4|NIHJ1^hqkfMNdE4%o@v%J{!Nz{Lp+ zwuAWa!LO3vaADp3_wpjiG9i-XoMg!zh@V!41Ns0G*FPTt_K2<(OS0Ipo_KJ%k=J=B z9D&XYdJ?OZQHXsLONg5ZJvc6KSRKc&uECFm0p4JH5_Wl1=BM8OhSDlI=LV{!;m~C< z4Sz*6Ej3R(pbA{(i%=#Tuj0)K;vWFzJW9KXk~#9}%tbNiFaz0)WfJ?KUPWdEK}bxX ztdkW+5W`uZmxxX;d(byCatrIp(IASs|-;7;9+JOBB}-IAj3p zJPml(wLxJqu7=i!CKL#Wrw_nB_mo`3_!THZ&f4}d`GNV5RkIwbSBw6yO)~({ z|HG=;N#Ef2+tB@YWB89%vvb1#aP}2mS?%4{M-Y%kLXi;ZmIi47>29REL%O@9ySrPu zJER+=L%LH05s+{1N6)$Mx#Ny+4BmfXKI@rltvToVZ7XTu8{WOEDL?R&+Y(SP-*pKg ziSYAJ|K;ITwOrTxsm!7;!^+Cu&bqSYaQWmoy}4j&zG!{-uLXKa-K7&jY;O^R-gi$o zB9_lBLyZQGMy4mZ;d&VRv~vj~&TKuaUZMa`8OWxlYni|>csQozc8J(%S_aTFye?D3 z;FnP47nGEyLVFz}t|5Dn-tZn?`b7>FoGfc+6j$lN-3wRDpJ;L{1M;9LFx;O z;q)bRjHyGM>%|gmn`u5_p~qpC@|#}`Fw>6{oJ<2*L*e=E`sJ-^mpSWsx>a1S6$x`O zX6AC|<35$a(G|qb2HAkFKK}YO)gZ{!?K7B9ymgUvz zk7U?xT<4{xB7`N++NQR5xD}GjZq(G*kT1}u)>S3x0>6>aLw_Si@HcXgTo&toizaje zvr0JJx?DFym6t+9`zY&U&6gX7Bxb}L3EoEX$|K6Yvy zMjQctR!9FTVfgaE!|g{(d*8oApJ~WvBf!gxaK)j&JUM=z;=GB7Wg2$GgcE&=F3g^y z^QmUtL74RVO~k9p`-GeTWOS-+7s7Y_)Qs3dpU7~j^S;5d^W4e$Enqy$BGS$IY{!+3 z=sU>Y)DxhQ&m8mIWT)AoiGh`*(Tqksv9tWvHAU3Q!PpGy?0Mp42k!nhk@X^mSt)z^ z89LKJ60-kNi8lr>_n3vNB@6fbF1gZZhD>I<}16NTm@w-|x zCO8Cp6arG{BN>$Huim>_Kwry8AEd5&B-O4bwhf=8H*@vT#;*!h(_H}AYBOU)38Xah z&x>{3IJmR<8o0Ptzq&4pD&p_JPVAHG=w?phl{D%=z`~)uqMOY0ntd}kFR>p_@;RyF zFpkT^+}A6)Fv<4iIZhZ7^lWBW+@v!xC6D`OfeqPbSwbW^&Ww0cjZpxuk+hAqlYZ$1 zX={UfGjkm_v#RLy-n`eHp3~G8SIAKJs3hE{Pc7)sH_}xGwkw8Vw%d$eGHhyUY>{y% zd>Ngh>l|ZaXVtHn{b4RX+nmuAR%{wPnB~x1d+yyN^eAuoQL`H{U9Xj8f~-Qf6YTjB z!%>HP)`qm|9x6yLp8Z){_yD|&4~3Wc54<3~1cLMu?@EXNME40k!!A08Zm7WO>``y> z#n|MEpSx>`UI;~7%PRLQ&+KbuxUy_iJ5AMY9j;(P{+LDnWcEy8FebM!to2}F}u?}G!(sj#y2M0aZHiYI!78!eEm6FWtJS#TX| z)LDde{DO*k>4Z_tF0OvnqdJY5zK_Rrl;nER7;|q^ojyc$up6Sy%f^nPKTYMvoOiSw z_<`6F3&uce-kmC~z%o*RGPdva>c{G{kYyvxX?i#4#_dFhuj9Kv_^7WRgV_EG(sl() zm*k;(?j347 zu}$8oslJw?K8NQKF?)l5@iW$Iz=BZn^WLJNv#s>b!1pgmp97(Y5T%4;ohaBP$v5_< zHoc?yVGDYG2^xRBV9_*?+)u6NEP~-z?KC{`LjNXBASm&*M#xUSmtWKOS(Xv-m9^Lz z;WsbcKl^-p6)XdtsboHP5|Q+5a$KJbr$)R8jZ0j0!(TCs)qAs!!Rf<&6v5RCJ5yog zOI&P+j%Jw=!Sy=gXI(mO{o=zKeO&zgt+z3__UcM%5pX{4tDDS`Ynq+alomT4D9TmE z3{-&&mI;$^=5Xw^{s`mS)Yc4|ixvU0$m?vGR#GO2=^^pFW~5pm#%4x+)bJ+;m0IsHam(RJx>-i^>woCs)e9u=tPzc@L#d7% z*5;H>#@5#Nk+;+R3WVAS!+Mlob>fFJqOGAyWwK_f+@oM&T6&vPNAy3{WIHU*JQ>z3 z>zTDclX?2Zc2qz!D@y09X-%aZhZV`^oYgZK*I2TEUSPLw`ep7w~qr8D+hOYUfY7*-gp;~jnixOY87t^ zg$s-0S8m)d2m#k22`an-5N3NQ!fYVI?ot10Zuz`mbNk{WlRLM6mn@O&znoUVS$TP+ zk&;iLM7NW|)D3(H>!yOtS(t#C9p0oI6}~AI;iwK>lQpD7Ezuz1xxA7UEQW?igLjV} z<=Sg}WOCF!iWz^DTG<RIf^0X=t%VK}cv3KI{4&8ff zQp&u#{ArJs_V}anL1v>M9pPa`_cz-}O#6whNR6|eHXGq;%A40AXsdMWD5cI;rxS70 zjTv2eD@u5A0)ox{m)?tf@Og}n7Tt;UY{zj^6X34>ae?#E`*KtyjWzM3Iw(!Hp%RAS^Pv#R0Y_hn1gPE)C?%-lsLuN}K2N89;#ZpF_iL?782QSUiFhoQ)J>2jvQRDI#vCX_%GhpHsU z5kMPZ|MtC*zsSrfU%O{_*)+;|k$FE|Eg=$@75B~x^NgUu|SzyYDVpNVOpX=YQJDQ1^CboI=+lr{2su*AADE)!~F=qB!eg{&;n}=e| z{)Z~WyTMk6=n?qgL}O+oI6O~)6|M`da_XwG^^^B4<-nIlD=Wjbt1mSrrtC|CuRB)Jhp58qC{6vV{YS-6Bsbau* z7`-O1Bz778vSozMa-N8jHwB}`R6+qO>7!FAOtUySTlEcdn0+39PL}IqUmuHgv(Zmz zB9_dt4M#ZoL3ahu@y*dB&{>#91t=%pVmHEC98QnoW~%(eM-3A$^mzx&YaB1NY@r9M z%u#3?)lK-vsx=+>l?0rnm_7y1;ciCa`1r3W1+n2qbRIx>Lq z-j|FeyRQg)T<++%H)Mg{J{)1Fbq>Ba4C*CmxrKPSi#q@r!@-pB!5w~MNe#_dYRfDP zIK%C)53GTU2utTRH6Q>H_Hm%z0TJeYFv5^+5F}%0kS8FUh9s_9)TH@@DIWzpRT{ie^!#l%&Sf~E=?O_Ftbq2en3hR>+-cQ&P4(`UH@qoP zE=>J)aY;`rylVY$341OyKQdH^_01ICCK(b!73=4qYdD^4(O(~jB}6~Rv>Dh4xm5Hp3-oNDGe#T z7>M$NzwK`}a48>Y6a3o#Xac%?FgK>hK$=ED{2EJ|VI#eAXmP7)q<=uMmnm#t%f zAZHLQGk`q7`(ThE>zw_+J6_@f5zz5!VX{5~9WN2+cz5v47-p+0WhTM8uI#I+FGlI7 z<`X6c$geX~<34bT5E802$jz7e)sLKSzcjr5{w4NIKt6l0Uq8~Rn1oAz-l?iTpz^h% zG4pZ8M5JFr!^phg{Ih=R1H)A`l1m$EoViz7hL&Xfe%WNcI$v?BD&^U&BK7v_iI~$X zkXWAL(pE9!gWDX>=lEqg6EeDUDf?rviVrY)<#tzJENAZJj$FQUdh0ylb z2xGnFmi60(LTAW_i+T&*1wwrpHRo0!z+Y0S{Y#=KkdpiO6B1>|mh>oE ze#YQlE#*6;hd;hw*{rF^^6eE?Yh1a*`!xM2zYx!~ek=}kniioj7iDgX+L=MK`R~~+ zeG-KWmE%`woI4bVv=WP^204JV^n;OxtatuV|HH^OCmN$?yWvo0PSRqbjw%r z0|GSpuIm9A0JP|XfrjjQ?*EqQEGRzug<(hf`DH$GlN@wd#A$Y@WBcXiT9Omevo^GH zcYNIF%7H_G$ak5Etjw6p*H`dZzZCs-b!3~Bp(8X0@YztIeg~*0(lk>!?*worceERx zK#8GSigikfE4>w2HK!)6uZ-^3T_W=eMHJig6s6`%j@kqAWBuCsNIoDmR+X^F7c0vf~2BhJ-s34w?D zQvg$aC2RQlK3vIr;EPUgZD$jGE+vM;RmKmW+5qMKiaL4mihC@3 z`|ug6@fUT*-<;2MN^1wt5d2`I{b-*;Wz2BXi)kfp39syf=?dao{aTrwNz$v zG*8#m8hHZzHyBYz{YX&|JCEUV_PQ$v^UAUC(%esN?&5=qr1s6T|U z@X}O2=(dwp5;yA1a0 z0yfK5NdtU*B9A>*PZq<$+QZh`q%a*5e~6;FKuC4~QR zE2#lzj^sjm9yf+Fs4@w)oL1dmOMSaL(*J3R>ViycM~<5$feIM}}-rz&zywc>A?l0%#{Rrgo)LMbMP%8Ub? zhi$1B(-kT!PE(|fphB4clSU|N0aWS_)^geP(d!c@f`wfDL==Bj36b3WDyxv=OqbM` zd@~!+5mH*jC?K!)TqV3VCl>WOqJ?t7)WhWC>f1@E^2uqVpo93tf#7w-3_VNL?MM8_ z-!qmfwq_8x-O<-wW2U;2!(r3agMbfTzP``|R(1qb+aqX}5K6i{`0-6dXI78i?&NHo z9;;Vc8I!Pwu*j<>zo$gxV-8*#0B*Gh!~J`k{5Khe6@g^*0e-*-{wo~CqJ`Pv!h*x? z^`(;V^NI8GV-LqG^$aPazJM%l3Q3rVvkz#xkGY=1Y$=AW*6%H2k3}!06cKX5etZ`H zLN7Nn^la!uv>n7 zy#d7tH41MLG@ug$U}UkweF9{{V<=6+L@1dueRAcQpO;!R2sIivRxS{^$?iVpq@{#W z%G`p2yU*H%I)8c2%;0`5zV)KmQ!Wjf`f7L*e}vlJODts3xkfy8lp%U$RqN)0Xg563$hmv3U0q+l z;6_b^r{}>)0Z}Cby<&SUeBsib=vkM3f`uRFj2ZAF2>kf~`AVnzwS4kGNKMSjn`La= zXh6Oq9bS`ODVuJ_Urc_h4n6hKNqBjQ#E`qP^QTd>+Hi|e9i~k=tkmsgOAUU*mkV9- z*%ah8sWhr^{w$jviR%uH96xmX%-5@HBTgryaNz{D-~E3D(qA>17Gqiy>r!Lp1{SwU#{|8fi>Tl%`o8MOBBWZ} z=f0D)ykk&MZIu&42k~Ua4}NktFq+i@n)P{krbOZbJ#%fm+8UqLKaa#+XmqkTt&P?)mb_C z`TpI~$2kNT%Dja(tKaZ4{f3vO_QLuE*pUEu>CB>~k#bvO#t-ZW`4XI-Snn;PFM96^AbHQiq@`@yO#vwDzQ|jC|vIHDRblNOH{k zs_9L%45KmzAqDjW)&36zjZ{n?mCt7{>U@g5s0z>lUGUEGN)ylpbLADQwM@IQ-z|vH zwG@@*kbJAvgD3tGML_*swLT}{2l9p5SB<2fR1C&J{>3OqpBTF1o!<`n{EES?U23}U zM)hkjw^3F3w(>QG+U)=kw_Vh16!Ire;yKXRLeLQmo}{?ted;au02U=+-G4J(Q1fWa zH-OhHZxv(+xP@qMP!$w>Fv`D;MPNt)j77+hlGMD+T&E<)CRqF^%c)}DU6O31%foLu zh~<98VhZ!~zV)iJp|hr82YAS{a&k>_faFVEC zav_US^MLs;dMU#WF~I(6CO47kSip*2w`!z`Rhn+KLx!rpAI|nchBD1I9CfBJWo8Uj z_hpr$+6l}3aiR7SC!z|8h3Lh|6iMTn5EbG)f8C}Hs9U+r#{I2ZX(RBS3&fmd+K+#2 z_qO|MB^OvmXQ3;jN80At4H@aLuo23CZRSJQ7o`<&ceb;!zcw$N`e97X){xOvyD)

EcvEDJu`YZrrq)lKM^)28V%c!ZM12N7uQ`;j1Fy?wN#=i~5 z=w-_;dL;i7@+16g;0nt?4{0JBUI`xF0=)w6N!g;1cnpo`wew9L_0d~u$Dq4;p7S$1 z2O9PTUsj1=v>RwtVzeNMq<(?h#xR0jr)SHnlC;a#Z)uRN>3c*3o~>|7jHVk6C1wgM=lvl`Vn15->I*xA7__A1BW-ahs4?L4 z#EjlKlx3x%FX}Tzr$v&)HRHN6-zjx*tUYQIyQTWBhR*<j*!c@bT;y-K3tgs13RmBg7FV2r3+~AcQV@TPB#F4 zr*{wSmW+^9);;oH|AkyiP?Z5pb|>SKD;CgXe+roFx315p-hEyN{|lE%@5^{Y4NpLE z@hykAR~D^C-l1s6P-TOwm%WE;H4&&8h;_I<8rGxz$}keW#X7>M9fwdBe^<>+p-l|^ ztV%&uw!II>9q(huq335~9bd12LJgajgRo&pOtaZxaSNaud>0frlT*s?D+08GipGnzEYpPDb2ID~eW8AT8QghgXSWvb??9u{WUpu-;l8&&&~IhYQjjg$S>#ykGj z#+BRG{?W#n@m3((IF*`>I7Az-ZTy>KUitH0*E>xvmO3Is@xlmkpOV?T#$(?jP0K9J zv2QmSJp)N*YchSvY{x|UbTtMK$7aFAtt#$dIbu!B0^+hS=0RNcxak{~lW%ccxa|!V zQnJV!Y>`%|CU7qi@ja&p+aNC502PS@sN7ON7~#K-%73(RzMd9wQCeYMwYm1_8#7K-V_5Ri;pbcUmc<`=pbUmO(4hW~yI|7AfWJ zOarqNYE*r#k(>)jGN7UYNrpJ2AB6^A){npXdB)c+O#Z+WvL{noh3Xw`c2068WK8QQg7adg#)I(xQ0#azB-VmP5w-bqKkD6A1imY|WKJRlk09OBm=4cIJ0Sp$uBg zh5iKd&m?yhW=#09>{z~)3CRA0jjGWV#am^d&Gn|i!kDyu`*gi(3k{w(xCy^Qjn`5#{_XMg7zjw3I>R!)2qE<3~pDVP_&pcr(5 zAFSIl{&6WKr6CK8$ZcT1V|*R`4{W!uPbpoT$`z&LGsuMUtzJ3takKn-I?EY1{%vk!yvz&1Yg5F6oXwwgqZ>(A4T? zz(JMAmhfH8?-~Z4^#yDA`=1)NblH9i8fZ%mJ2OBH3j#Hq3Oi#r{==La$l`S8E&Q%w zL`V%&fi*0L(o0>hB@dXu$eWAu3*I|`E?#>s4dT1>vh)JX;yk>BAq%1Vze`xAhhi@t zJGRHq3GJe-PQGctgPBu4%JW2w()1mw>zrfyJeL(~N~>B-i_WW~G`|V$C-6;O2@4h^ zT;j!OiC$2QV62TaQCcfwPIQ=QsQItBkWA5zvI)7eB_ zaglS;i5d|IkwYel4r6-1$KX`W2BkUkC(%GTz!nxn1rtiVUWxu;j6H+L#sf~Yh(BlR zSQH17KH<@o(fe#t#Ug$=H!jCAIyCefR5Ffv&=JlL^#>hcpzCeqf{w6qyUyQTFBCP9 zZ5T+7h#QDoW-(_Oy(i~&Y+_+0@i(J|Pwa@eEmRXf3Xe2M#&`;-dodKoyZH)1=9 z2kFioGEUz?Bdx_D{qph6%4S96l+?x~X!8V6*A`I%Hjl+~z^r@*qW~sHfS$+O{GUA! z4(xe65P{=}WCfQ3fLijwQ2)-ZMh!^;j=SKM$gKdg>pqvNUoi(3)jR%TmB_9Yl9=9T zf++5+ixUP(`L{LVtL%CiFX-YkR0RokTWSIX7=sA;<>ytCX#-~^t!?3kZhxBcf#EVU zl=o0kfcxN{O!}?Q4oB5=kCXIrjGs6#?o>qyWx`LK^#+BIF#%!CeuKT{-Et~}R&Akw z#vF=dAv3_pYrA#BjwpghYVRh4^Q_&5Y9e9J{dWhR$vx$WM@^UR9C!*oVq(@jhNxk961s`bZ zX<-aj7c>ad3bLXEh-3Q*wL-vzS=?_%{*{RGwTS){nA;%2DhcB~6F61;|Fb#&hlxQnD)q{5lf`*dY&7% z+Q7Esh=T4&7U9Sl4N*EM%EzhpuW>X|W5FF9yUu6dt2h(P=9sj5(w0u5R27-d(bec6 z5t)&K!4cW$o8C%4hTDO5Y9x-l0{5_2zo1RvxTt~PNC0n{X}FbR@C$o@F=+ANj_`#EoK-(*-#}YOuzO~GPwr-}#QV6IRN!ZuF-E}kz2fZBuFgNlB z%#B>agS31YZ?_FH2FruX7_uR{|A%_LwIr_Ng~Kn5B&N&X@?qC~d9;p9qxmA$yQTFr z!}LeZOWswJw)Ql-6*CNri5Lf^E`TqdSZv!^h{MyReEC946Fn5dUc%-ZHA))VhvrN} zvh+D7WnO>_nYuB*x9~2eShh>fF4ws-agTmzX z?VpLFeM@@%tv?gQIl(BOu+M$!=JannxM>uygTH*rF2tj%*PlyPtd1>&Je0bg@maYaiIVB?z0vSOo56n?zJSpBTzJtwt>#aE4dxgH?NNZW}vFTEw9v| zL*khj^?sd(Z^xlO*NO$0r+)w8WiXJZCI<45$-bD?C%loEtKcEfd4?I-8c|3+xx3UbUB^Wkhcsc*vc`y5kCHzp`hgnuJEI8rXbviN=$h%u%4`+=uj#f!%ae%|#Y76woQp}8@psZX0@Waf{%Y>Y1r?Hpd>61HPdev-D=@GPWPbZ-+&*I z!Kj=d3fd^5FQn18HJ>kwH9RLeuuqVx?+qyY05Amxz!dQ>P&LPH0^Z)jMn9U%1T?!n z<=SnTJv>`(dPg8X@@ta}IsuX&0hz*P#muE54tEJ)3JbI2SU|F)-h#dgL8qGb{^=HG zv_W1!)F_{Gj0EOK{z!I^%(^?pa_N5+&|&%$N&S~vq0NV!C5|M5e#EJ>M)vKhS!LuS{{XmQn0u&q4wHU89b%TJut=|*4c#Al>Sh_)Gndx*wg zh$`?6g8>@@lJQ|)aa1m}j#uCw66$o2txWR08fnXA=V{_yfj9~)LSfLY-!~N?IdaGz z!O9YcC?w|+fE>Y-NuviAvQE3_C{V~?UXnJJ49H6w>;PmTq|^YA1;cGWuks5!)eM|l zHu^&^@}iWScbBzE1$Nzf!-aaE7@qOn4TN!8DmK3*czoYB;Gr$Z8Rl4LlLjOjiX|mQ z*=H=9WeDXsfixe7ZVKJj#E(u3<=AO@cW8gZjJmH`#@D_|zfi!gC)f9w2n6}TQ8T7L z%ijN_*M}MtP{{rytKVVpog6sSvF!CtrU%`+hUjHlYhJ9k?3mo7IyC;AruzAICyxcM zHwT-Dxsf)v=uOLURSJ9QWyHb! zfEdolVQrjamUIEhX&T@>APhJUuo(Yy9-z#y`Uca#{HZ84YtY7vDWcTK*Mj1#F~Q@N zH5lq@UraZ0mg>OBi1M#8-eZ4IxisL}XfkGD zcCaf_kr9*13Mdw*6R~xUyn(pyzhK6?TU;gCr8kA)iDhvCiUm?HV2bBu2CU)s*A~=3 zkA;N;;yxs~ZjCC{^Ir)TOnrb0k#NJ}A^qVT=(5(=N-`j{-656O(H;Ct- z-4Z7GH^#Op>RypB+o&K!MC5n*bdBHzb&(fvr6yUd)D$e2<@+;7=4BIuxr545)aIJvsa|3KBk%?T^{!;l>0|AD?UO?``WIfx zSxwdNY)og4In(s=p=~1o$wJza^@Fn>{t^6T>wpenLE1ul7Je1M=h{+SgeZ5tdg?u{ z?;&_pSzMPtRa4_Kj#vvI!Gi-7Q`DD0Ub6Sk3No@u$p2qeG4mfAPy>v42(H51cQc-_ z-U*3U3l_s5}j~wP16hG_wQ9t zsC;@fQ+sXzClK_g_?4nN$E~pdComOs0@;WZ(&r2JX`qV&Q{Xc8mV!-YJ6cF#B&M+j^?cN)|u8DVDv3z5GsS(Xo-a7NNSw2J%E7 z6$2yK@z@_awlV)%^8QB@8gz`eCqR-ECudIdW46}zbv1^FLUW)fw9D}}R!@^x>Ln*2 zY_bgCf*XhkoPfyqR@=BG-I{ZKTa&kegL9?!cT@Tae?L(K(GIz;Bd7S$lfG(7694># z!VMsls+0G`kxz}J_phJ>#Ep`05{2RK>Q4r#Yl30hgq~C9p(zWKtFK>znl$ z17k6uSIqOX9L_HE9K4&^FKaO=%^|r8(*wrhIM0ECw6S9q;*aeeh0vO@Ccw~kCBo8I z_MIUnyis7#_g6xS!&2S3H!u7d5UcKvj{supdz@Fc4HGxPIgEEl5($P;uA2Kdj8Cl07?AWz%|8lOzNPZk%; zXYrHd21A=Prsz+poQnfA;Tb^)&;`(a(~Qshg7aCbuxtPhplfz}+^3rmBIhDbM2_SVUC>zgekwJs zffv=m6~H4q6}~TG*yXoC0rGpGolT}-8-<SFl70(Ur;%>>ZRCMuv0+6#fO#2-%g zHyYoX56pjQzBm)&F*d?Na&bq~XG$<@Ry=KauPLdWuBkbUt%VGqMvYDSiVF+DGG@To zB}2mU?Iy=G>{WzHln)H7>YJg&vtGStTKmQBlCKUuGmQ;=eWDxF#a>iJQz3CH$AqPI zT_do^dr%7;!GDuMw>H?USZ$BTGDX}&!gUb0NNkZ%yYpv~H1O?L*L&Xy-nneSrmgY` z{DK8w_Q_`#C()vw!7FzO2RCW|;RD`A>%KDao$Hv$Rp1~FxN_O;eD=u}2>!Ox0>R(O zqxc0lOE@e{W2jm%_$&Q8_=`Ko)>i`~9Tv}Pk_Sl*$PsF$gDc*H%8#;>DZ7yUE%~b=FJY30uCywu}BvbH22Kz zL{(J{8PX`f?)_f2FEYB=wkg%SelbRDj!#6mDP7LKc*5-%S?7Mv6hU*w6DJPkVBRH7 zXT-7yJd`DF{?NqAN|IDUD~{b6ar-S+v!s?=;qf}#xRtlj^!N9=r}pnZpCOL{nSsW< zNM}HGGi{XI#OwwkW}>8*V_@d@ z!RZXj7%LA*&}_a%&ct&gka* zoz8HCs!Q1#!5^+ZD0wlU93~M5H3<*7C}MsgNie04^!uWS1{p!6-^)y)uFGe=oL$ru zn*L^$UUd#4BL8EZ{uYs?|KqGEmIT8xw<>Z(UmU3Ctey7-Ks~dtR&>?IKMq|J{`KB) z&eZ}1I4eR9b@?g?WZ1256wBC^ntF1%zjLI{qf}$=Rd#BIdtQCMqUR4Q$~j78v^iNcDDr zR1czjNf9DL2pn%kk`RL^OBx9LM!AI3@$Vx~BoO8NRc8xS764@=4gh7~ydFTA2Sgda zx+BsRAdSNY$u0193qV;%0KDC@OS@rLFVYZ{Wmm9lSKyIprpS!t&`v4e{0x!?kY6AB ztqi?{C_|81F81D!`|nVOYIrlI+;*?z-rn zuOeQ(4s{_)Awn_QJPyO*)rp_t0w1d^@(sWmeEMZShf{hv#B5gI0H-em*Ohab)r*H_ zkI%r1u#s{pkWkdf(!+Y!YyV zxchbqMT{bj(EM9o6L$Hn

xA_+Sj)E2BcjIfz0u6jE&jo@9tWc%1ulhWI~w^+;nv z7alA;J_Ko~@tdYJK;~3gV2bjzvPNR=GFbhV3c-2E_%aCSfiM8hkRU)a)@j}Qr7-Q} z@)sG8OU@=z^3L11uK7&fi?1;FNyil2GKc( zAv)&;pmRz=bWR}A^^eZ!aqo>=M5B)}AE`zG9=!yQrdoy~(>R8aJ`(s5b3cu3Czs+L z)^QIl3fCgOtWI(~*19i{CoZwTV2jkKtlX&6S4O?F_x5yEG91xt!OX2P?*Pz;ft6%I5PZ{M(T$Q0@KM!%`=}{;3qA#WLqMbpa$nR*h3Ey!oDZJnSbi^k(OOnM z^q8Ml6An2e+oXS)a~M6Y-W8D;jxTg&#dkJRwz$ zTqap`4a0F04e=d`9+_N*+i`aw<@?81j<&hND9=LN8CgI2^pKYyn{Olu4BK^@e3!DN z*Y!K>so;=t|5ymC8}w)y>DdDL3tfh43w(n}dG2VKC-S1XgOtfRVip<{nU$k|(eKpl zyusiskciZl5|1U^cru8B6DU3~q;5{J{a9p(=!9=wPq)&&5MJD>f)f_}a2n{Ef;W*^XBV8>UY26Ye_{wig5L>$&rO@^z zTpAKAjk|Yw`t4Qsb9!<_4@w9QG(FM~OhQ>0C|N(4C3G&5uICw&H&|5M3wa!^>U`Fr zMV}9t3c`&yG{)PNNEgUwW)0&Jr6|AGE3+~#Z;9JUOH%c^E-xG`S@6=xTH{pfgs6F> z--YzefL}}Pp}{T|$O0MmKa#1=eDgAEEfJ|GB`BHB?bntLtV@WCQ0^Mk7&Z*8y#^eQlGq#o;xtju6n%?zfBq@g%& zl3V-wCfBdufYh!K2()t-xxgqCCcGzJSi#VMSG9W9Y?Vh8tZs4?2UYPE`c=|9mGstb zs*YZhufxB5gM^-8Xrf;OAJ>c1Cb>+7!zK3z#8bBb-?s#Bw^g7bCv@jdRuPHgw)P~H zHkHIM9!Ih6BNkBWMHzgm!W2IxxK^&@E2%Hx{s54CT}R6F7e(D^bi_SM%?lhR%Ovs+ znnK=2QQ-9J`ECqgW~o1`Q!sO?15}ab4`nE%f(A6pCV5bD)rqJ4rha5vLnUXuX?2)i z0AI|1KblpW4M;0e-$D9m0Dp)sXnjAJKI~Y^P_&vu4A*F&fE9tQdvaxGRfm1HBU1*HD2H* z+QsFmc;EN8n|)Vp_Cn}CNC7M=mQueD1cGs6-BKgI`Veom?$s;&NJvv@5rmKT}^sWxJ`_R*>}I^y-jJ7ShOI~ni&c)V~-dcGeok2^&cf-e<`}g z5V1L;&)C|GBNk;+jKmxvqbnjm!#OenYa9c$2JMqq;C9q4%RVMI=V{Y9EJuR44cr;z z!NkAiR?b3>S-M7lqkg9(yY-Zj3aydZbnZl20e)f736`Zf57>D=Tlo7k*_(*weyYP8 z><0?Fldqt`k}7r|v;~yuzl*L{5kHf$o;##8oMxyODS|vyAs#$d`a(N> z*J``OL$v#4$6QPb1D*ax|Gz-+$tK)sx!XI9U-7Z z6@3WwKkpgQrLh_JL)Pkgf4hl7jWPkeAat`Yf_uhnd@EnN2?;Bmgm1c018B=W7;VUk z351pY^@I?AByc#VEAjC;N-JsKQ79{D!g$|YDfNC7l4v<}O`qbj;!N?x9&rFt&%Y_V zrja^!ne5iSwI%t~*~OWG*a-;cMx9P1GDR44osf0!bsu?E^ZSbld_#iOOO9EfG-X9& zWq<7UIC^)({ve-!6ws!2^#!^}Wm&L@Hqnv&(o4pBMfX}B0q<8}Wt~3}1m0+@!M?nK z5XopJS-k>CByn1AG;ncMj!nfYVS$H)ijg_meSuR8B+jqMcJA>F#~)!7Fjp_|Y^NBm zux$WSQ3yXlbY~BEfdzn?(Uh5Ie$pP4P3oevZ6befK#8a5b{(u*o%26FA2C5Lc-`D3 z)J~a=uVa*cLYm{d5TDimDy+l;*)#uFVY_`q$RdR$O-4iY(Ht0;+9SJ}IF#u8<@#Ja zNJxU~m~+fLosB8U0~`NnJjRCgS*w)tJ7f6-%I2`oi>4m-`k2)Ha}e24l+}A3X_$ zECY~sH%Qun1*>9I-26f?M7Xp!%g15NnWZ9RGL zG9b(Ub)N_xOYVxmWBO!K3(P%GeAnicBFnC=3ujP9$5JzHk=cidu(mbKb_Y)bqRT5(_S zlD@7YEsKcdYB`wYg_(F42J&f7L5Uf1iuu~bsV7Be-7qh9m_pN3!~7Lbv>TwqOYtJX zj!PG-*bnhBMyQ#mJZZ6^?v2SxTj47%;UI+p5LGifiuNgakJi^ znH1)QR|ssc%cOZ{+HT#b!avy35#8sql_BqFVIjEebbri^ZLH<;KO;s=gLk+iGeX5Y6`OEh)b#@MFDy z@Ob|FO7VXw1MAAW=)UNZPv9P5L~I<>EY4Y^I4JdKSy=;b6KzzUZ~+u#_{OE&?p%n6 z!i)r>+Um0^8C%F#c9;KfnYP@OMd`tpEH(@Z1I~wRZ%GI{Jf^uY9Pg z`>Q_Q49eld$rhYu*2u*Bs9ilDzvy`l#HXM-j(^bKBdX)>kC)*=>hlehZW|`ESC7&| z5aPV;KC&zZO-g9yhW(h^Z(<}IFt>Q4GEKAkxi?T};Qs<+|K}^~1#~^TMcu%} z>c6F1<3?u_ZYy{|8?e+F#g#^s8nQL?3 zeLn1ylhf2ep0<5yu@w7yy2Ba?SCr=!rUIdjh5o`xgg){@usIZRb0eU_7+T~dF-clb z4ru7T56b!xe}ah0@ufB37C9IH6e*uwRNbKm>DH)Beps<@iD--OegGLrH#Bx+_}wmS zeyCVa>x5GSl2F*MFY?z5CE^xG?O$1<#!-M%43t!_c)f--n6$=A2B#Q7V2Y`NOfjQn zC2XK{tM|~p$_m*z{iC(OzWN!+nxROp8sJh9=@7$`5E2^yl_5EQ48E#*gztHKq15{< zh?wVcWwn*NprCuA4NV`h*M?L8$IeacZ7O5~v*@vqEJejx@M^|Mq}P9k>;L~ad&{sayLM|^6i`4w8c9(= zK)Sm@y1To(TM#Lw8w3PtrMp`?q#Hy^loSL3o1Y2^){u8Roe4aXV7E!)A2juul5D)Nt1FKWlt zu``RF)TCFYN?eAjGQs*Yb0+&uQDT;-xWhM_mSbLRsC(RDj4OQ8Y5CkaWBn8MjLuHp zf<_UdNGkJ66X_S{DJ{J>i~Wn>CQ@fzFVqT<&HhNJISSh}$D=-$a{~V~>#N>G?6CGV zAzBF99t8np1Ej4YlpY z>^@RdBw9cFuj;B6HRwJH=}{hxS6i?vyoV=uM|7h#IC|0piNcu=ufv=|_0!`TN-Xvu zR~#q&MZNEH7#l6*AEjP+@MeU88_6U9bza4i@?mVrME~bJX8wK~V~q}MzpdqjhrUz! zHI#Rr1k~NsQ{Vt7fea19pOd=~!jkQDun&s%qi#F?-w;y_YjA4V)LiyIxF!wXt?~ibBV4rGsQ@UZlWuTG1+1LE4YdNQ^)qw zkLp6l_IyTdpWGT8HU|$lY^^8FmcV=JwO_Z#Tgl77LhxjbawhXIwdk4LTU5BYWK7#> zOtpcnN!0M{ffpD+_8K5Ze!JlgrwqkHaPx|<2ddC(U&9Hz5~uk6PY&6IN2Pa9g)5oQ zlpVjFqMUIRZ#U)^2ethQEWTZw6wQPuLgtfZSmN^nxoVOe5P3>6?C5+}6e##b))qp` z7D+;NEKx=!5@eKZ@zX;;(_k^1+WFp5w$v0R1Eh1ykH9|Zo{hh`X@8|?Eg9vT z89bhzBLV=C**jCDp1LEF+DJv5Q>SgYb|+MfSnx~^J2z92sdgKu zk4(&d534~bnJDAF(fBoa>Vu`i;ltujFg3TRbkkdAaOhX&Hn785S->Ru>#qS)3SX4G zTr2{ad$PA|+i+5*bRXnOCe#tGPRA90>FTO;IQr5vYHZGXO6yWZxtYL*$z9AYYJ&Na zrZRT_jJdHPSklLn$?V5_>_U}}GW3{inGV|%GneHrLh&^ERtx-wLYNuE#R5>bThhX!iMLkFMS3~W5WblPwte6}6hJe8wUS<)|_lI3dD-&7Wg<-#!T{c@DAuzVhT6F72j66U#e#pT17^8 z=UdmIjgnxWjl5+veH^tivx|J3w;G{AA>FkyY`x`^$9jErIivBa1xECD!to}pHl$Q? ztH>1-MDv^FOEK}vg|f*v?kQF1DC`MLv_5SJ6%yGVCTQ^QLDYa<{n42`mUuiiGhE}q z{X36tWAt5ick5*A6%NPyLcaw;YfisLgh4m@g9WtWyYmPLej?4Pb?$b)F2+0SN%bmn z8n$sO91S%jO&OSqntuLYuJ4tu3w+b&MbYS&VcV!`#~DF!LAJ??Wa7a6(h+SYQ@6-) zfb-?H{e^uLHmF4$ zY;UhdzRbp~iljm7k5GibYl=Q#dwi=%Y1DeQSt9WI@h9C(;Y6TYKGranz`M_Q{v@5( z2K)V;Q;N-UoA-K_G`d?rXF@4Fe4#v88w}J+GfL)C=@~Td&~1Kk3#;z z?35iNVP`rwTl_Pl)!`uIm@^@`>IsZenC$ zRDAm;^`*jSzh!8T#4u)YX;9LKM#iMi@a$Sx&-2c4s`Tb?V-e~=<&XQ6K!=!JnK8#k zQo#pA8#)Z{&SgI+VAbC>&MHuR!B~bSTLhCAeQ~GdjCu_S1D5Z^ExAp#ByP;=u$nj* z-r^h38qEcWVlrwQcn zvHkp=yA_`26-;;Ep{_hT+>Ea7q^_)jR=v16cGHTGOGOyL<}jfSzjVh>&p8m_P@T#KPzmr-3o0sXNW_a-)(I>xD^Y zaM0*9SM_TcbiRPvDDfVAn+!^rJ6IB;2bo8}S@Hhi{)AI^mNpDbDBfja>^TgIK{G18*VsudB`|db^*z zlNHFS*?*@3X=$zNOChTniPP2%*9f5VyzX|Yr4!C^Tn4#+Z?C=#Sbe_!KDHB5rotnJa!cjEl{1T7ZXwSFbm2`>BH&7GXNyv4i6; zTPjFD4m&$Mj*EX`5osYV^x-2Bxihh?;Hk`;Vv|p=*iq8T+c1-EWMT`I&hOlebrgGO ztd^{yPuK>9NlcypCub)rh9FZ~=BL(V5^!h?lwo;eiHLVgU39^aHpXNOF-JXlBe1|JRdAgja|G8EJS>uh$^ad-q$eoUm zbxNX8?1Gv^piv_Tl%VxsRB|ZIp@!5k-Bl8s4H%f5;5{6lAV? zk#oR0-2dT^7@yOEh$aYG9*mRc5Ka20k-HJt^|8q+6lEm(f?c1PS+(CZji(5tW*i&E z$7BVfXD$biCHqp`vSE4#HF7-wBwTaA!id5NnRj8*UCMvKMGmPMj};l%4eBqFQ(wI;OKY*rZJQ(0jN{a$obem?^cHWlLm(7$-bgg1i)c(iRO2r9Ip z%Z7V4;_qd1+iy)jJEwW7#T8OKZw26GrB-%TJ!J1tTv&lv1G|jadcXRHg6Srw#vit~ zfZsxIq3U-7n)ycb6*&@!HBi%2d<2O8XbB2{=>KrITmCp1Y`+1~e~5agE$jV3pxQCa zb%_?99|CQ^uPof>fc3WjE% z!jj%>ozRqpK1P7H1x&=L3%Fz3oHiUUDv<^&#^b2N@Pfi;ty7f_AKN)=#t`gS~!i~_4RBpC(7 z?kc0~mHk*C88uN>jMJ46`=mM5KV1B#8cEX5T7;;3|G{r9ik&L<2oZzU0L7cUNUEe1 z!vH5u^p23kn4WtBO=Nfd)k0mXk|}_!Z9^ztHa28Z8B3lMew}`pWX7#EnqPsZHw&A_ z;3dPm(eI{AoomXE*`zoj&_#tzjbZgbW^U;bb~*H*F(`k_@E2F#DNcbYE0RCNO{n7WWjJS@GR~nYW^FYO!=ozme)9=6|)c80osAIqj*kz zf~KdRn4L!SGa?|+Ki>XY%zIa$UPQc@pvFE^fpi>=h=5qY;zBh_&-8Ey8rD3%YJI|S zDbD@Z$_YU?%lyhE=v(-E*r(L&Kn<*jh@;>Br7bpNM%u^+$+i zgc~5Mn`c8dGvMdz+Jg%Q^J!}@gC$I#E=Fpuk>PFoaRPR?xMKlJP(tERCF{MJ5+;yv zByEmT_B_WAe#9s5NA&n`~$}iG-X+FJoe;4V|YZ5(3v|i z{lho1Ia>e^X&|<~sINGIa-Y`{Pm^1h@P*=ME9s}QGf_Y4Hp=37t@f88y?JiQ2} zM*s5;-w~#1;>znXU^@Tuh~rUuJ_B^ztYz8aopD3Dn@b^ts`1$IKl5i+qY@UKryQmyi z2fvchE%6_(G8RW%Ls5*^W)5AH1SMS_OMIw_LsG>wj-ojM7(pf7Yss;m5XzJyi|8?y z(p?N?GeJ*9Iu}X}Ri_9{MMq;>*8|zrj}I(d-eGGzPjub}EK`o?7an$@i<(bSKC!he zoQWwR-3Ti$-7(T9a1&qS%UfQ$$*p)q&tXxA>Z7U>p!~BeV9Mc#Bxa~vDn6Ldi#KMr z*%#FIZ`*3UvMcgEH60GZQR~V1l&(F08#n)%YAVSGx3+WQQ&&Ih;oFJ4ShAe9j$(%$ z8VC({dXi~uKeyj8f2O&YboQKcfY5Pat{8Ls3EP_nHS03l2@pSY24mf6>IUxEga4S( zl^@m1ooqXlKLjvgdv`YMY72{{z>RM!M8$V8tdYcZckkGRxZtTEqhY`9x6h4b0iKi= zZ7d)~O-*}NHyv_IS1QFV=-G~$qvMV7wI4PepWhkeupmu82<6?tv|IJyjS9JA%B>>4 zqu&4enP)CbRqAaVZq*Gg<)RG6B8xA2#5+WyyRUSNH^7)4=zcNAFpH7--n?gvOKVbH z;PoXF2D;xYQQ*kPTyBL35`ChRGJqh_EHQ`7!d&f2CRON0YX~p!we+cDe%-7(AV@HN z@_%gl84$uB530NyGL>iQjI8gB_PWsUNoD{`25=P_0uA2@eLgfjc2#uBCX$+?t3qF( zm+DPGn%R(cW%kQlOeJKqX#o=JKaTLPuqXCo77{)^IPfbk!8u^-GWu5a_pS18@lF9{oT7Pwd#Sa)+e8pf{n8@~R$MH1|^I;bh)5y*~-j6Qk-mh}U?4 z*?*ggp#*0Lp%Q1B3MY{;U!Y4p%)VVHn)b5BMQTA=nY^!R@oCd-)1W6Os9G}Nt+hl@|_ ziWYxVeCR-WYpZnBUAi(dnlotS1aap{13IZy>`zKWSZ}HdQ68jW zZJ7OzER`_#co3;Qt@^Z4J$su1EU#W~Xlow8Vl~oaCUl^3^U8=#4p?goYN3D0TCWG& zB;h}U5mog5I7RpOifJhbMwMCic;k3khRnT-D}dn{#2Wz$WN=VS+;BE@46vGNCsGFR)>yRh8F&FHPpcKk%|TYZ)OQsfycLx}3{^)!-j7 zYlG%OxIwC}zN|!xH4w8p2wcZ4!`|og0a6eVgP`Yn1D!7blda{1ehO~9-iaV!_DMr` z8&vy{I>Txo;>}TZ*{!EB6FQxVe`+7k)_GzW)at&q6G|@9jS;QMl=15FnDvqxa)WCE zU-!hkB(0tILGANi9_YxviG5_pVA(44YzOg1Mj4D7z=n&8SzDr$$?)`*y%UkfxDb@S z<0KFyU0qzD7@06nrxH4WhD?**9h6h%H;0vaLCBpz=c9NUghaEb4Y1WQ)A7@HF7GilfEeiclfdHXW zIvxZsYlmXVPP8VxBi|%Yr>zl`K%%==XhAr{tY9)assZ zODop(D)%2t46dvBEW@MW#1Jy!enqanMf~M`mz-rYF1B>5VY??b2sn~_G_~ip2eI4P zaoJ|;!AEQw@oD0X*30O%8ul*3Yfrwu0tUv(cEp({nakHng(uC!<`a{VA0PN_CO0N} zc0_x7#t>F5ys5}QgFkQXb2Je12JWw_;&W=9URg-|?#7eEufPJU}|Dt6=Q#RBw&E5ltZ21TrG5`X-V3K)pSee%KvO`pjs2k%z!?|u~%!YYnj8K9G z3~0_DXT@~-Y}#1x@ncPjPDa=yj9+?ByponsP^tezJSQ*6BQWze7g;@+JUdC5m9mcz7XeCw zFBV4ELsEMq@KMAfw4P^yInRx- zQ*&Hm8ECSL0h>F%uSAnggC@HNTeW}Zjun2t6Tv6FW;`05E)AvtOZ{@ie~hDums5bL zn;~s}H#BeZqD-byfKVwKFa^jkw&UJ_1I|&fKw4F+{&pl;w)c_<@EQo) zSwqv7#`>dxx%(kf1^%`zcL0l`Q5)nZ0%qG^ufbQClAxK9ed)iN0&v2bQ*h3IIn$=Q zuChhXG8lE#UO;B5c>Z z4PhFrIBx{WT@f-k*Sl{Rz)8a$lFL3EASZ@cKrFVXB!qcHIU zZSa?Xos12(0tU#=W-5K(I|JkzEY!M;v8m=Who5fr-&YGYVU<7si2=o!$1Ei;BcNv3XD(m$-tHBI_3d{i1Xhf~>pc#P8VRh5p)`ret@D2_OX0_zdc@S6q zP`!Ky$ImJ8pmVA9hya}eG?!PMvRi%;5U_$3<%)u`Yowe}ViDrq8HX0i z`A+Q;Ql>Kp;G~(Sz)6!Hou5E)f3Nz37o;pZ+78vcS6qy3fJcB93BNPyN0rJ{eLCxR zD2UEpi9;O((b4AQy9`+a?gz;q?uT-tpgzR?@H7y43=wS7&na}?f;?-${b*JC%l#-_ zp9DHc4#53r1}SNB+WTV`r!Ou|0%Gb5HviSMh4W$~`X7>0unl512nHCYJHXW#B+^_m zf7g1fw8|XRbf57@ZzcJT9I}sCZoUXq>w9>1uXrUIEf`$UwiH8=onJuUYLFN>=!XYc zj4?PNmWW|JshuL+!)9+?Fdrp=0qtrf*jBhYuvd{RBV;kye31t03ewB6xf@nC-v(v# zL#S-#JtIaL&Mp9$U9Iv_2agem+6G?2lHBJ>6-~;zT9Z~-v15S!VY2`5wTzpbmHtcp>+*~iq+{r0asJi2%Y2_2dI%Fmx(gev|G=6M3^Anl%*{e=8>S1#)~|9VIN`0XIzZonOpBz$;d zVcuI_rHz$Uk^<~IMQyFM6|~?XJZ~9c9jO|hT_OMrO{+b46M(RB31X<8NA_5xwWnkp z09*?4WmINeg|f z7qqPM;vyG^^XX?x%dQ6|#77zq7!xZbDw4&r`MlqYs!x~`;!i+5&1HWUuvxsj@Upw1 z7kQZz6Y|&*y$4xqei=na7CeCgi^e6WAhCsT=cl^pe zb#Em6VWU>1{E?96gL_-x(&)3z8We9kxAHr@7Poz7CF4{Kg$H-?F-GP3_mC4unn|W! zf287|wX1I$aovmuDmOGsa3r6TmL8)8q&&$V?Suzi3ZWy(S7htEUSQ=AH+H6_B&dV-vQ4Z7^m+6(J)X(p9B;Em4orTfpIm`&n(5M_x z>s}sjA|+ zTpCn>ET&Zs0xY3RqoUBFw*-PzSexRhv4QfrEP@P81eDLkv2I$Ai9z)Y@+*-hb;#{_ zjL1N#p=Y%Fr_?B0{W|Bx4J{n<1Ja-n9a9Jdk~p9G+Nog2jsid`q9I5HJAhOG*yH*| zo3asK^P%1X*tae?gbWnVK)%6$vDo3fTyp+vFdL+?aPuv;(U=Ym8mz93vZHRL_hBDXNQL_!i~DOM+s$2Mcv>gQhZH@ z^yvkM-l?yYksWfo+#YR+e+wK|vags_H_?uqNZ7X3%=j2Qs|am(r19(9PG(+ju37Sg zjd6*b7b^E)6KL2lUmFZsZQO0^yPL$CR)#tG6a{|gSweRTrYSvYmG zSZfPafubyT&NLQyGcDQh)NCB%xqBnUg;AL9?8SR-MV_aoGzmCr?{YkQrYHlajmf1t zPaJqY{D}C$-P#`KpDUw6ET^nJ zOjqwM&I~I}OAO90*v~?vwUv;g;LjZoJk@2$7#C_@&RfYI%z&!S zer!BG3{2H81|YA@HV?SvU*47H50_gSy}3>s0^x_C>d=m}NWJofb+XB(xiMxb0hX!* zzz-Sstxyt&?~ajygqH-|#vsPt6_Oyz(^wvDs^_nQ&2OXB*4R(k(vcqvD7aJv1d_iI z4-e9LjghrVQ{@O!is7M-M*%ePz`emL-M9$HBCOfl{hE?PIrV*IQOql_v;Z+LlZl*1 zz`(9rJbhN|hI9QKHHU7W+{HJx&L*s|;CM)mNuAF51=-1G`I2QP=`G$1(vzVi4@kyz zDL!XPdm8~u#vG1EpP+T33PPJ%a|W%Y%RJQZnBUVU1vCaUf?oxfQFo=t1+tJ;=E|^H zE?8^n8Y>H}ErL!GylxJh$3DCGb&!*bKFYkw*hgOed8TA%-}$@&{&8li>5>o2VK>8S zU%8P5x|DLeF_EiqxPSir?55CzX}4djZE?YOIQrHu!u!b#V{67#jq&!H)()p#6f8= zoI$4%=925qVKs4O&aea9vxHWbmE{lXsEd^P5fB_E7slUE>1L4Z$J<%61s zNc@?b{&$(+B`I#_Y>xA7zc9`hT{1q~X?kmZ6NG;TIvhj6LRu|_ZrBURSE`bKfkZ|yiU1J)J4tyFv!3CrkMeXJjy!lKt~tvkM% zga4HBm46qPi_?FJxwKtHdZ183M$v^M{+fK-d!!UxuDi>G)73P%O@rPUSQ#9ew11Ku z+DupxgV~+yD{}pl7SBEsByOF~K3#h$p63L|0k|1K{_}k{ey?N#)@dfRNJw&p;N(d} zHdJK~I?^9YBGT=E;-~UA+~_+*I1stE$!mf`a96xU9C%O<#g%x^Vnev;Hvn2G+mo4O zq%NjkxFH&~z8YjX`yJUInw7I}!_V+AEe#{N*zNuXYBu5-?il}Hp`gf-LZ-RaaTj{3FVFXe5t3^NQ zeT0N3uVzn70f^rRbw64b zt9NM3ZNIKNqBJ3?o?xyfYJ?gW^KVp`CN6(FuLCAusV|=d%U(eMn7KzjjPgJZDR z?g*Ln>$~D@ytvnQLjwC&_YpNpgmc$_X%b-Xe`GSzQ^jUj@Q62gKh{6_tZlnhw5nW4 zP*O#f6J0Y*JQ(K8TWN~Hj`AB2!QnVmxDSJqOhYYcdVlJIFvvU}7x&(Ev9&P(NF&ga zmi|z9v)pfcj02LTZ)<^M>4rG`nl2rZH?BML{yxHcI{eVUKmU%OwK@G-9>*4xD+g`$Bd)vmdH^uDt)} zy68w_m*s$2qZr##UCdHoObgh7&r3GI|tKk62A^lRnK3`Pw8``$X zUio6bY-<077tUi4Qq2d}QVARst(li5u^L|=R6LCT#Yb;>J+x!E+m zf^kiTJ$q>`_|5Fljot*^=xDGT{nUap%@IVLxR4RT64vFl7Zb)K$^%SG%E_Umsu9yz zz3qbbCpWUuJYpKSq{DfPD6F5CdPJtU3=#(qz7?pq?6C!06N&x3muZMKARod2WxD=k zMLYpVY+a%Lr`D<9PNu)`XE197%8ZA#%SsosM&-GuzN258)y`l;gT%&dZ)n?g_4@|a z&bt1`Xa+*ZV2c4torDqaO&+z_nQO=S(e9-5w$^gjSUekO+4LOWtg1Zy6Vii^8r=3C z;L~cb_ZzfzFCcZ$wv7hQ;mOTGW3mrqcjZarkjiv|v<5n%1f)&HSRv^~kqCx&V;mHW z-hSA7wL*eoD)L%UV|CH(nKJ&=}mQ4S|q`O+@T+0dC6jQG){n zLK4S|r>`#e^UwJ$cB&NA9%q>!r14|c|jSOLlZ)h>wZ zvX}k$o&MP*GFK$;Ka}Q*vC5mjnnz2-9KDj7_MAU(^%qr0t zUEaN~Z|~Ap4!l6PfO^@PxACf7-yJjg1tNPAXjrvKUV(Wo>>DOPBb>$rWy$ntCn6l1 zBFiQ(HXBry2ob^t*jdd$PB9;#JjeUF4vA?kZzsSrE$wmTOQ_W1#H#BBU?{oi&)>eSAmHg=Ir zonL_i0t*`?m9EcvtfF0>FxVM8P;vXczXW=@YsV8n8_g?E3%LF-t*)+omaAnI>y}Z{ zt6F_=OV%o;Nm2^ z&V7ezb1Kj2RY$nV;KB_7RJsIy~qMJJdfpkaGBFCmsZQe!Q#5bnALJOD{r= z4BVWllCiVUuBAI1F0Sa)yVurfK2^;x-osNm994)nF;l{rF;MAC!&luP(6o_~wJStx zl}f9POF#{Siy>cWVdYcZTKahryR)FDS&3z190vFHq+CG`HLm_8KaNo!dC9# zAT=!D&Dkb+%+fO0y`$w3c*)#EDC2^W!paOMO|iR2iOwMsgPm%1SJI}j*N{bRcmI9ntQ7OQA6}+LJ{UUDcCJNCj!$*Z^w-8iAZ6b(rx2mf$ z(|hyYmqAI7CR0`;&6k*0itdk(ml8j(v7ZwJ{Xh-p{(vLn99y@2E;MwInxFhOQu7|< zNCmuY-P5+4!hZHnI#ASO_9I`Qx_i`I%t@oX?k-InIxj{c#Tddu`V}WF{MFAtV6aGF z5`Zf9iyT7B#lYeJyUO`r&-t=}U4%qMK^LMT+8uwUylSkuWL8UpF+}v}YCFOhN(#r5 z!(;dcH5HSj8iAf3{*+<4T=JWYY8E)74tEr;(~K4>c!~-vCAk}Ozp%H@>pj5?l_17? z7LVV1{|r-Z$`~)FLoN0dN7J*=;cDTGdlYX-8@T$#s|1TE!)HN}GapbQSTB_)+cz3S zX^@KDT#_bn`Le6wwlr&>MX=TN+=QImv*Cte8-&3y8 zQaC0hIKD^fxQw~Ci45HED72hTN_QioGaKJT`ECz)z6Fa8Z{S4gQr#_Fuh~~cxuN%M z6?d-fpoihZRAZMjN0ng_I&xs5*IUQEg|j2$#dY=iF*yz6I1F{Y9bOMUii(<#*M1@% zPk)vEjW&X*Ml)W_rRzlZ^?Les?N6&p_%CFQMOo+YSAyK~5y!K6V3j zD2DI{D#R|}tLZg220sU>tN$Ee-7Cl$`y=~>T&QOhN)6OwfMI%5yE*FO*T8f6YX}Dm zzq`jNDeIpHKCQ+1C=5&CM=444Zj>k-Q?2ZjHGBK3x_MWc{CFQi$GV87x?q z($b6JHi+ZIBy`ixO=;@YYOe-G4jJUk?f^EVV!wwjoYU|h85pRn zCu{cDP_60x92nxU)8`t&6aJC5HdnS;nP(21+0E$gpEVp+&=PoT6VMmfrKlA3r+oU{ zIq_Q?#xt@B-m#R~m;5oy*h2^0^3zlG{|>?WUw;Rea^HWyz)AfU+$P-kw;8tov#WYJ zT1_RlPqW51LN+2jw*I%wfuiyt92g!bbCwr6nHyWABq32zxC}D_NU7Lo%vhw$=srX$ z0~rbft5Bh{4mNzlJ$a(iZEB~Kw`WF%1ry?4&^3GiiD=sV6Vc=oOI{1L zR!~G!F+|s#Hu>i@?u=^sGAq3s3BfgGyg+Ale4mNftA9fUAWXlcJO*(UPn-^+T*58U z{XN-X6fR;mJDYW{6S4)s16Jgk*ez4zSE#*si|yCneSb&nbgAEx<8i=u`H`963#)c& zPw#w!Y`$9mPp$9&df2cQ)-~oIBm9tW5}JwCEn+e%;QbPMf6}lBXoN|}asYhH0!qrB zjKjtquG+(QNu%3Xkcb^WirK(ITAdo4b&d8iZ`o-531hyPq(=KLG{L9#rtmXC)rRPl1f4Sqk=Uwt9!F(Y>H9-tm`p3pCv{lxzh{dhi;Q4-)2 zzcx|)dr4yYHD@!j=kjKobL{Cy1J-lFza=H45rA*9Q^y0DAj2aYF!!P;r6er-F=Bjv zw)49eGK4U#@Qp=8piQSp89(qi-pveujhpTIaNM(?K#AJbOy{vYKZyB#gP7o`AGevw=u%2(pDP?1ln?Lf@m8#%yC05 zO=X{cm`MQ2wuzJWhO%v7duvAnk>c*h+6_y-x2CPsLFRqJd?}d!q5`2D+RVDOC-Z+~ z+vE{*Vgr+a=_}~k5T9wVf@U$IS3l*;=JPKHEGx}B$*zG*9OSwMFg7`ipBsD6A&%vc z=MT3u$3TTU&ez||0ej?qXW3JuJY1zkF^uV5FgSn=^~!o#vsMRpbCU%AGOHL3x=fRQ zuOiusofnTjpG=Y=rFzTg44Ynxmy93-Bekzm5Lw6XGKviMLmcVZRpWrq=Ee?|<)Q zqwswC)6JdFMu;!Af6$uxu#tmGa1HTUzEb;+2YbqLl)3VqAhd|~&IeOD23xeo!#@of z;dMU~akap80D7PjJ5@pQ%lde-(t>UfWeR1hm&UA4?wMHd8e-7gFNsJa@2f6S)b3RR z`8HLCUzxvbLrGYu_uY$45kL}_`=VF#P8L~oYk(wrQ>LXnW|D}AO-p~80g%Z>3h#b< zwNSOmo^bd>vUd;gO95Egt1S1+Vu(YrI}RXt*Z-6{Z(Dr*6jL^nUVjxt6z*^kO;Y*Y zI6qEy`@jfg*Pp|*TXh3$xKJr5clC4rBdh(-8bhmi3Xm?Wgs4blzVTDWnHdT|a()m5 z2|$f}bU+@^#?ts=J!G_x=QJ}dfsrIPIvS>S=@@YHWePNP^Q$rN8ILbuX5PAuDWCAH z@nGBR7h_8VOD=LmcPAD>Y?)JH`G?gQ7AL}c<-c7NI}nlBoLTVhW3$}_IZe?KvLHez zQXOE@51yG808BcQ(y;RYL!L>U&siRZm7asJ(oZlxo>TS&q@kyh*uq}E0^K3-^4Ybj@jGx( zWeF1(3~+cb`vI&ps50ygmPvr8=y@b*RRwH5jzfx10yUxhS3hD{7Yah%{$c}RpRy~M z-C7Fa>&*+hbfHHzo;_mZ<1kUWAzU!67i#hS4Q_ej-2i$u>P@VeSmG%Z!7*8I<@b@D>gw!tT+ z6O2MYO|${(i*E7Y7?==o8OD`v_l!3!z$hGr>WikBe!+%lx>*@$bUW;B>N=m?hCELI zmJp<#wX%EFT%7Q~2$PS1etAdpqnCy(_j7Y_WgHDY2=0JJRwL^`9w)m$E*p7+iej{vW&9D5Q&HnevUFXXJL*}B{zoav!G+InUpR(YU&zb3eGkj`kmtO{ARZX5J zLx#^=jab&dVa%3qRbU57J2EhgwJ8 z-Z83E(^>Pa>W8$#VqM}d6jPUO1@;TG_XICWSyD1oOnGbp$(DrEv_>Lh0IBDEKGr<3 zHMp1YUE5GTZ_qj}x_6$v9lUF{z;uEQk?SalYm3u1<%NChowR1YDvoEQYXy z>mgR_ziEd@5nphl`jlX#o6vp6NhSL`>NYSxKmD1x?p?eMg9J`YBf0%|*D}*Z=lD-c z0yz;`=U!i&u~%8~2S)&Q{pj~>F$|=RC9t*|c`uy!0&!gWa7%<4P>DOKwK+upVWb25 zk288D(78x`6?vd6W)du7Kq3tnWyzeuo8Nb%44m;mrxb`Zbx$;r+rwA0vlb|x*ETqI z!ojh*qku7qH7uf+6=V3ps;V3cnEp7v=TX4de;+m)j58U8((^6?u|RsBE);h@T}x*eCp}u#cX`>_>m*56skDHQT83t%a#aXbF(6zplifE4XO<15~u87kxG z9x@W#Cc8guQ{wrmZNSde73B~t5h%Z#QH^;Th#Q9z;UF!%JFsX7UquqFKdx?HeV`WK z)=^u6V;HYaCxtl@vq|h2rcSCQK)jZ5` z`HnTxW4<08cc`4(0*vAwTj?)rFBFzvrs38&u^zPr?LedMlU#UH{kcvl9)=yG>j(t$ zwk_S22i+t!n1Fo~SO@e(1_F@L^C!$&LL@oY2w@nnUS=@;(bWDeGr(l;Wep8`sG~f}xQY1xyn;oK?4kLmx^2H%|&8e#Oc+ z5BROVb&4U+v%%uVvDoK>o1$B%$*x7>NgP9vKd+=Y0e`K1QKQVG*Nr1s(a+&NKquvYjv|AcZ}grXO%YDG$% z75_URIjMG=VvI(~I&r)n|9))t)aH@@{vdOEG$yL2sQ%Lf=3CT#b%h8#B^n4AKi7gE zP<;JiXfA;@idjzLogpb*kbJj*AZKVzO8xeZGG0a@TiIazH4P+&qn}yoO!St#hNuut zbmv`hatfv_1*)X#gVBEEEwR^=pP!KviHhSofBqTLfl?4v=iarUAws(U!JkV++^YDF zQWf5gF)1;R@i#WJnQu4Zto#zZofdd}ICMlx&td}A%6@-CPTtb}SWh^|oQIC=^SGI< zuG-G`8yKJB1*xcMSjTOsH@J~BktmJgw*(Gc-&^>q{F5N+Ixe&Na2kExnpl`x0QDsQ z%AA2V{%9If7ntd_?5?}OqV;% zfB91UeW1kZ=>K2Fee9UHF(05Nm}I2umb4T}@-u;D0u-hUue;&A6N$NmHs5$tBe#}A zT8$JDEB&z6Z@lO`ov19D9|Q_7@xB$D2kwN3^zg;qIgdn;={mfW!J0d37A z`$6sW&U#^h>vPWc0sQir` zwY$1?<2|w2%n84$CIjm3#w$=ca~)DF7cPCDe06-0Zhw}_ZZHkw&cm$W7&&}iMBsdB zuYT9ST1VFyMW{&WSw?6TsdG=V<)*vPr0~yddE1m)-;$G4p~{$EY|;)22EY7^hQ)4yr^M8 z`PqJvwaUI(@BTCLD;zs_hUfa>a zu6-%EbeOir)#voh@^YF-na#uq5xI-uK`iQ9cs0;C#zKwb^IjP{*ffu#@|_6tO|^pe z7M>npme*96`N`b#_>B^7GGAQJTiZJdSZpYu%+8K6TqCF|2TD%zcg3Y17QFrUp~v!7 zWDmzeuZM6$JYD@o%jmcREr{!_>5!5;Up2kVRG%Xzfd56Bj5e%P`!GyXzQ|PMU`vi^ zn$I~32svpv_F(&AtRE-jJ;8LKWt_QQ&*&`jD6icq$OsmTX8zJe3}vw`=9T1IFl zeX)Vhr=A~#XjxZ3VP;s%2ySwq`4`K}S6m{G-v>)!toSi>-4dl?p?QAm$!=UeQ$|k|I*T;g?K?w6 zEDqtpydCyGx4bypsj|hHoM!K!VJu;k;2V%@F>r__-%6T#V|gdJ3SLe|p5_jHob*tb ztGq6}xaGYRYB*Z+_ItMh<=(-=nJza(+xi=6qw${GEyJ(zZ2D9G}oxQC5ydoiPG1$Fx2o~{HI z_KP0hWplSz>62#zpSI(KRfeTB1~;N(igk*&B|GNJo4lX7I(kIDmRsad(ZQ||jltUBc z@j8R(ai%oUqkytVIuNazcz7BJg08V2ZOcEB}G=T0ZK{(kPK)R|zRVi}@JTYTw?$q$LyU*76;U{|rMxc3*8E*+qsplE-q1-L7GbzW$J;pOwiQ z85wp!E&xl$Av5`LrZOe5()ATF0_?JwZfl$6X-W?~jO1n=sq3)#@NigsI1y(QyDAN! zNgko{KwGbF0zi|Tb@1qcXp;3dy~VKjaM`54@!^2TH_yfWcBdVhmQ2BsDuHQ<;$zy} z{HiwR!8(|aPhfsQNG08<{2dh+8gbp~e(MKbT%e}9Ih-vU*N^ak znivMhN-T%eR6r6414K|0IuXJE5v$O)JO&4n1YQ^TS|e{>!{Y_!t6=o^Vq!@1W8Q#{1 zD+mB~flF;Z+ z)#&TpGXHA-5YHNqRQUi^8ZSC|hOtclbXGm%Ck^cJw+8i!E|<3mf1#*vCg|}cw+PU9 z&-^Q%qu#$ebV%E{2`z7xYGCj&J(mpYzfjcMzt$G911Em1f#vNzQh)ct`N}Srdw@Nr zwam3)XFvj&sJd%rtC0kFpF%#gKHMU7gM7!<9AM?8Moa)cW5=Edu)GzA*r<&@k4Cyn zck8lqg0~2%s(j}HjI&lB7k`BDf!i|`9H4so>dRYZSX&wK&qZO_*DPCHGcjfZtzzzj zXKr`c?)vEi7I0AzE`qDLeZa?=pOYnpVX3%p>E7#Oxw3)wXwwhK`c~>F$4%P0GPCyu z1YXAZutje%Gi4(U+*f(JDA!YY{&Iqf4eVfU;xksjM$mc_@-7`|xL742)%@_FSQxqhPWLn~iA66|Mx@o8AelU6F{!Vo7t#=+1 z=M)_soTY+Yarx<-F;{jL5F(U4$3nb4hrGlqj2cr6SMMJ*(}*Oq)oin08mSWeWi7Px zq0deGW#P#&aGkx^iHPyN{!*4ec_CSDE!?Cpc}w=d-~0of3=Kk@OB*wEof&nlB1ehN zBV9HR0a0v=KjNq*WO!Toa|45Vn^D~or|C3zHP-v|pUbB0saZF^!C(c5d^HnK?5v>a zsunMH^wStyjXbO}lI03~LGkz!fGq6Y=1Ul(^nwpKj&S~@1eMkoP>7MH`{K}R@qbG% z|2Me@7pyz{XUr#LJN5$s6Ie0-<{|Cxva;2G2-=%;NwY|+PG!FnYPw7K=Bc_UA5ds( z*F2nRgseBK82G109|1`ZY`r={cK}Ekfu-D9(4(h|2=?9ElLp_t*e)xbxa>EM1!F>? z72qv3{guc-L~?!X{xDUO|HVWtnV*G0Q_Zgf&qVJzN-|;_S|Ic^W>}v+7w)U z;Ky~jT%3Wb3!wObRTtsp7sZ9^2ld^h9c%swoZiApJ3Ghvkj@Qx%3q#1O^N-h6q4mG z(4(wBya|7qSsTB-zk$CxwbB_19rMc$wA~liS+37wN@lJv>3hN#DgjxIYaHTr>#RNE z=ApxWViQLo_{fc;g&`pr{~#gW#=u@?XXs@P!UHdJ;Q3GBoqg0}3Xl-05E7zH_YV?+ z{z(*nd4(28SNsHEGB+AzbYIg+vX(`pjbIOPF2%XEBp`vv+Vv{cS_o7i&|3yY)+z$S zp9ewlSn-=IQs!#Cw^;cWP@h=VtFdnl#V^zG$}Nay+{rMLCC%hlqa0^=T!$3~M+`dgKZIAAvj!JowJ%_>`dV61rS+(RcjSQIy|$)QVdB8~k{ zn8KYNq?JOy4*+yJ1?g(&8k!32cJw24^#+I*cWzEq|D%XExV@(|DF?`#iJ$ULh@fy_5J?+>rZ z@MKYWVRRMzd=cfB{IK*O2@h+8KcKcx^cf{#*8RHsLMllSt3fN12XZO*TIC+ZQrgg| z)#;mu3m(Yjd(+V)wTQdwE>O5%Uy72%QP0qN0{L(F_*LF8Wl@#ArPWH-byv%4&;g#$)E$nTk7enrHj(<(RArUr8U3R96u-q!=m_0ZKmNt{pH| zvZh^UF?pK=jFp|`q($)21xr3wzD`C`D)d`B&EH=Ntb_^LZjS*Dq=c^S7_s#Y-i7Ix zO|XAN3m)2z1^BVluiTS!!}`XsfA{1zn@C`x!wQ!XG`6AEWgNw__25ra?g3?X?CzA8 zYdt4Q^^Afx4HMCfw8jn^x3r~QMT|90N)_1I)k7&@n}$2S(5At_zIt}PLo!1=v&=*M z9(G!!l}4nO87c_nsQ7HFNO$VWDQ5UfLV}fqcC3HkVNk?`F^8Pay^Ec@@OK%%F0gH& zH_Ys`7Hq^r1YR6})_OUxZ5Ww1g*JZCBY7l$n*%hsZouJ+JN${v7H1)iW+txPes}|AWI87+47(VL=JD ziy6PSCkuGnf6d2$V9$@;vmS_pa;69DR+9_?d5t(6UjS0#y;p`V-6yB|x>{d1z&fBg zHHcWJ=FGfV`Xc+qS?`nY9E73Xc!J}SJFl}O;D(}NN`@P8B(*kzxTrZ!6yr&CaMOb} zz|*c_ZnAYstZOMlQJ8gzh8ON6`slvD)c99}=I8H=#xUL`!b(9jVL& zlpj4aYJx@EO_I>lj?^GMszWPDRu;jV4M+!FWaS+RP)7E8P1wFS2Ettc{4LFBY<-~d zeefK&*eUxK5e~~0tJH1#0xJmhO(FMg#7zDYReq1CKxbI5&k}k|K#G(n8tS)Q>Ic=r!*Yx_}q^Qiqqh&{#=ga5Ly}Po$=ZKOhuK0CA zvY|L-_^e&!X-4$HwcbkZ;3FzOs~{Nu8Ui@l>jxsCyHq*%p5xl8(Qq;$tsj%Dz#sdj zjEf_EIVieKa<}drBa^tpPIQ56lw6qC{SPJ#jpTd;!YV0U{XWC`N02c2N1Q4E#HsN; znnK2cSOK9h^whMqh&8%{&U*D;5&-Y$)xRx8C8jbhlSJ>EJGZfc zh<^&;%HIkA{4SD6k!nxV?1@lHR{7bYF|e@vs{Huc!)sc)NNer8>d!H3T07U7`9!l% zD=y{pJwRsjQ6BO`Lx@b~#Fz0-3iNg||D09z*x{uHL}uG(LI#H6uo1w(V?$qdn2*wx zhJay5n!m|ETnTOA4OUY^J1{Vtq{!EPHIF!uR1j5P!ey2q4OrF?F*EsChpI4F(Feq* z>fHL0{h_4M$?xRwN7Df z8f-6&S_rPJdG+(Y>?i+qu>);B{OKpvKksbON3=)vYn>fSRx{!;yT+fck$f}d0*RjT z^R0jhkqbhYu}l4#V~Yg5=GjV$*M*xK5;9yzMOBFsKlou|;^kKRvLw%Az9W&5!6Rlp z4sZJ_SMgeQL2DB=f?OwG(*%}Q*o+rExZRn`AE4XKZNh>JZnwz9i?CD>cnxm1X~rI9 zAMAE-oiISR`wk9kT{d*P=h$w5eg~dW2lP9fpx;TRe{Y9=OhQQiN#e^f3#UwnI1izs z=c`_92PBxtFQq2Tk-Qn%xvtkh-`4{jv+8N@IbpFXD|EY^Wmn*xeN)uI?cR^QO;R#v z>#L0ZnLphQ(*lW{gzWiP-(A=T@$L4>E)zIyovWWVtgD1#n14&JdmG=^J&G-J>{*35 zwABqgrV;j)pDLuDfm|fAmpJT05NjY_&g}I#Ed`d62ZN>{;X4GKMcHl3*2h)v3xxRI zv$I2>CEB{s{%>(~a7?W`qb#YyS&vd=!@?rZ7%ap}&7j_W9ytPE|HbDetB5aTva`RW zRG;2Tptht>G0JF>A^nD^Gpl+>Q`#Plx10V2i!xjUuO|=b{1fI%qf^zY<@jkVgfy2O z2Rp4z4o1&SsnXE%o_f>9Ip>AHmH-YPvq5Gwjjl_CGkzdr1WFyie<3ma{OO}1n;1+- zN-&7(_C?=D7q06&f#OrGB$SC_7g(J{H!!XjWQl>J7A7ea`|z3jNgqPyi&II zbEh#^Bvmdvo6=)AfSG9I*-$Qm@tcQi2NBP#mjZ95N}jmj7z@Gq<;D?(-ue*Xe^`0< zF(SOoLtCa+1!i?0ywR9UvaSvWA2FFM?JAFK6bo<7Fp(kJ>oNU@s@F%HmHl1hoy`mg&Ozv=oU7hD$;K>Ie%ya@5|?eI4{)=tvw&$iIxRN}W!J6{*kt{eEYcNc+Ao8&b6xi16Ujl_Ju z6a}6A3!gUDZ=ZJKSxJ+zf?E*=NqyzDGd<-Af=al0@phN0H?U!gDt=|LIry!#u3jg? z`qO_t`e4|ymC_H(%tm7$AJh)sUMKD$5P+sF-1QipDD!l?ZVIlCp+{(79!i2=0mE4{ zeaQDAf10EDSpJ)EeM3T|=V*&6Ug8HbGq{bI1%{<{Bd*25hhHjZ?}BQnh)H6PH(w7W z;;BhP0Cn2X#>4_p-*)BU5iCu>h^H1TcU+?CbvVVE$-}wV4N`xn)(c!k-G?{U4Z z!r%W4MChM!(Wh>;yitH2p{sjrcrN#uc6M!)SX~S3Q!bPZuZunkm{||Ju`)E&d9wqr z?_m>+OQ`hA^Q_E?E@@Gy=8WuQRz%KhEh&TMmi+Z_`(^?|1YsP36_e1rF6*O&Ls8v< z`4CWlHI7J2QtakZ^-p6wbdJ9KxKN$(tN_B@pgZ~xCH{Sb+o+ZRN|TetNuAyQ0yRp#Pp33zmDuFXV%N);U*+GZ zO|MHamXfj(-2R3s!5_i3*bn%#A>@qYnv)XIJCt2q0m~8<)pUB!hs^!q!7q4j^zphbH7^;@eY%h~PlEWHmFP4xB7Mj%aUnNtjstOR! z^ujEEzG3}>&AMw$2P^F1R{Kk!edhfu|0j5Ui)jB9zDLc&nmVkHJ<{QAGb_JYeDgF9 zJKpA3nw%~()=Gpg#a;FJ+aW8xUzS5)#y<3bh=>85;w`bZXVpy}FHGD$%Jm^*pRnNQ zP1-b6l<=jgA8yzMI{q?Cr+DbnlLp@!Qew;H7y>$umt4!xv1L{*+3xqqK=LeV5+Q#e z&x-p=DZYYopoCra!BCiV{ss^!*R>?gLGr8-`o+HW>B$J=;wlyc#Aes%W?9E##bGX*3~Nyixbihlf=-(~h`sN^2WJPrls_nqoLWX zJ~ferV-MUOU@aWoEWqpK2eg#PL^>EPC1-S{-S8TVM#lL5AuT%U?Vhhcy%YfL2mX=g z2t@m7T>eA*VgAjAn9l6K=MN^1#=?)EKx_y)lBe3)C?bm5)nR|w5E#w-;|&u6R)52O zDrorL<(c*0YR0H7a-T9EkSLML)yK0$k0GC&@>FDVJ~Ksuc6uOpqiXqn|Iz!7Sc4G(6|vr zL7d3+hCB7I15MN#MTUcJp6lwsmO2ITQJuLw$%jHT2AM6rr)~(}vGINJGB3>mP!7Y; zXf!gzVn^I)7$I=MG>md6d0-!v1z9Dvdje*)fR&+~Z3)Ix0xElK4&S>C$@)QKPnq*$3un0KfV=m7WMaih7zxpvoz7@joCY^3Q z%sgD7I$?b3GHYR3u184Zmf`P^p@}ZT(w+~n@MQ&C_t?bJ9}3$X*rfcn@Hu?zoH_JL zR|#wRg&M8k8eK=klapE&$=g|pE>J$`)&fS2CH%V$g-_E45y3z=n<_K9i$S~YSt8kN z{yhm!NWobE&>PC4j7}|ayUUc%?7KrR4!Cw$ed|WiLbpsatXd~VR3U<-RUB7;A})5B z|4wnMLW9CMul+;#!?TVJks+$s@XS}Zj?K;{8AhjSahH*&pMX&2LlDYL&$PmYx!!%} z8S&_&!*3iC&RC+ZmWhyr^O4koPp>^+C>$9f z*?c-&!s8gN?XVK6Fpwj_{r1YeWKVzY`&QTV`;w>mg{58e1G|nkBARJ9$|TSHX0Gi* zUrMsa2!+NuDFQZ61`e(jT~o91_f;m9JwaXwl2mgE#ry-`uF!VF0`&k1Z=_&$6E zS)8zy=*LW@k^d&)?lG@qV_6J(=SN+KuEv8< z(6;9AuQxV0r!}Gf^oVk}r@#=#UiH-L_{MCWmpu3*%mFELda<%7om0yzyJfW>tFHt) z%^`7$M^36MqvKCH^bJkXc+7!# zasc`v1E3#M>kqdnJlt*Fr`h6OOP13>viv7PGA}YfC1TC^>8P~U4OAjtm6Q2i#HBYx-|`gCl645$=^x!>-B$fJ`S~Uz#Jio<|?u5wVbaZ)knI+i#jx4o7-w|fL9nX ztbS0lH|cXZxKgHUewNT_lU*eCjW2?YB3me}pA7O`vxci5^`cN(MA08T5uL^vumVhW0u#4mo z7KylyY7tI#Hylh60W)f+Wdb7y$} zq^wGacKrGiS7C!s0r!o6!Cz;Tm5J#;Qk9XBEqvh)AcXX_2Y63vzKcKHzRAUOvzF^A zATXO;pML3zh+ZGd7y7uX=)5g4=N)c#aam!96d2{&rY;;SWdbWJfY$n26k}tOmA&|` zxlYWzm0*@Nwt2T!xtlKgz>{{_Bzjc54awIkxgMZ_Z5@@LD&>r9YfR%*hsZ4JBhPrk zYbo>n7L(SdZuM>Mgh_WM#aa>QA zsJMP%i*CMJ>Z$ETBNVGxghAe-_En#{;QsOxqpiHB3_3v|#}7}w~@H+NWOCKdR%S$bx)%UL3k zyMO-TU+3R;-M5i{caNfi2BlGf06VLHC8-LVZ}W5xi++83AmL+f>;1JzqWfJ&REuu6 zNe9y0BbGq|7+D;U)lXpr-{oML3ux-uM!~LA?|9EBbt|zT5{ruY!`oqdyoc7PYek=3 z54CSHg)?E&QEStDpWDeSmBGm}QY%lkYX2fJt*MZj&+1R>z+2`Am$EW|Q8fBpi7M!q zX!G^?*nD>E6IM<&yzXugbyU%RBcOk?U^ab4J^SE?Y*2GDDU!M;$2S>&t2q$$U?&T_ zX^v;L$KJ_eDE?e<$(|W(U3Rw3pFzkhXkof5;1ng=UssI4#wO9E(;JY_lJDwgeAy!Y zJ-h9195|-c;?vHmqQs4@a;I=HtM?|2Gp+)A{XXXqLl1#pp6#B|S)+EDQy5VF_vHkD zNx*Wg9`9?}qgjy_p87^<>1P(kVBcJW*>A-7Y2E&I;z{()3zsGI;cL1ln|v?KZlyj> z;99xHe$5OFTDmB_44|dZdm)q!5O*L@>Q9j^7B%Yt+8CBt^jd!aEj24z?uVeIup&F) zuA7H=u44to+<@XxFALHK(l|kpz0w8qSkkmU*Lz9~#T|u!b@MZf zcuSC*0Vodt@>qhnqw}N-E`G`gzA(Sm&%)SqSdlHw_VXij#=KWQ<4fu59OVDI6lTuR zPE&&wX65yae`N8DOjo-=u&Kwa2LnX6xrLISo5yZ_K(as&w2l$?ffBofFiK{CUARBw zeF#yoi+^w-HTNJ`M>y2HAL(^WG$|`Kej-Eq+D{VWy$G@UeSh(J2u@Y&;oJE)8kly zoB)<#<6g4;fvR*tbROPLywWUqGfflu!CvTELNdf)b3bL^->H>9aEg-xsU<0he2%K$ z^ev%`SgjpmkXlj}7z;9D#V{(Nj9AveF81Tn4CpHNROeQ}2LdLERwpbMtzv;Q4cf0= z-Db&qxyJk#o0^593#+b!l9s;1_eC1)xp3wK&jlzkAT&Pk;Uo)>Dwt7>(|R0=MMpYgsKEe4 z;VrcT8rU=f*5~7wvp;7cu53uwc6Ip#@1+P9R>=kILL6RzY((HSCgVf~e4Vz86DJ`f z`R`>FW|Y6?`v_Zu=6Tl7?l8D1BoBgG86+M7DG<-4;wtP#_0j$+oaxk&8qAt)VoYxY zaR$OOX%pEbdqrT?@;Q`|Rwf}ENaIq({-n_k3rK-Nut7Z+oW!`6k@um8;|coo?BfaNg^!iK=={Vs zj1hQ-Q-sXo9Hx2?=aP&Ay(?T~(A>Qu)4@`HPCYn&;PZyM9Dp{Wv?sUTCy10*i9}$D zt)W2M?K)_^HRzu+-LiO-g^6<^V=0?veD;YaK|cBkY3nwf z%85+PdS66m$(fXWsH_k0Qo>%JKZD)pC+1Dyd1i%Dr7q;gS_$My(3a=w>tf!^-tTWY z(D+t}8?F`>##>jZSS12~=UsXn!`|?oG|^K4+qDojyv35*sqgzI?E0NKRtvPj$4RXFL0( zzRj%;>&G#0@R6nRHcih0+HQ?npbvl@&st=9rn@VNvK_7T$9pmhExEW%Uj#}OSb@mO z%vJ58W!Q$;t6`1gzbQ<3Va*?i4fx~A27ozYSeoly(L8eNqoxb^%U~8zn6Pq6l=(n$ z$185&Hn(5gX6yi8Tx?i2j10(zDcLbd(t9kY*$8lpU_neq3&)It`ACqI{Tc1~Ab zd-=Ot24(lU9pv+n`Nag@CB53cdx}kE|4aW(3=PD9!S{JdT=d2pc=S!F1p+5p0zvx+O~bC(p6s|GdYDeGlx~Hd2*Et>`s(B7<8(ASk9+1 zsbP9Qe&6P1kVeNU2#FtH#_W)8t^3ezzIu1Udnt#tgUphF3pbtxKH1f+8|)>O4eUqv zqxzh)YviE|f2tC-Y1-UKq#O9TBmQ*Gq1z#T>c^;g)dPsk72peCuNP!48x4M(1hWYZ zZrIF}nmY}enYv0vpRIF|y?TKMSoBpxpw(U>XBC;kc6h0xN2_+@kTy0Os5L2qWne8g z;{~4U3GXyS_4^<2jOAC!x8Wg453qwA3UAI5{DEgc6~};kJq*t!It?Ib`~<5>Gu+{X5Tm3^q&hz(mj;q|TTUt!a6Sg6zO^SH(g8vjx67Q0>Sn%pxbZ@_DGu_$ zPb&RJ7kjUXcfY=*Tkgf?YyY?PFz^4@2oB7eWN2jufaE`OA994o@a-5V#2eNKX2pO; z5I{tygzZd)PkFSgfmqWT5qRJcrT|yBnA=r*fZbL<>hyj9TA+K6@;N+! zM`hXqc$5fdXMjq+2iljV)19<^+nirAoeyj z0dO5u0x*4p*{Hc+$+bKTKI0)%z+b04hJ6l+SeVKtGnNRD%MAZ2NBPuaDVXNEe;rU; zF+;{l8I?IFa2Od&hOZrYhOsHgP_R6)ZdZRuhK2eTYl=AW_{<3QIm|F<9-}9Zc$93g zBX2}TS8GSg%!z&Xkk4T4i9?=p50p!{tDh9okxsL!IfW+uG*!$Eo=2&58+pyyuK?CL+x(e#hZBs0Fp42$f8=O!0jE|bzH$6V^=BEKUV z!~C|w`f*k(LL5>>L#vL-!%b6#WiC3RgVgxA&^M!cB=I!&k6(L6RK|$*>d5=07v87I zUt_8;cnHRKfw>De-U{~tlRKYQ$N5i9La--Lc~q{=H~<>)DlMS=mQ`N0e2)_P>A}47 zj;f2iKjkxcd!z#Kg86CU(44t{3$Of_S-61bzzkRVRgi}Bp1GZ5*EkP0*bZVrM*zAh=3I$ zN8<_5E2f>qvlMMqph#N?-Sd&33U6w5h?pfK;{xS1(kFCPm&gRP4s6m+q^6Ep^nO%W zXshF=s-mMqiLbly022dxMPo~$n%D2cthGR|SQ%J(=>1)y)lv!W>=``4CV|J~1$Dk& zWyW_+Z0L36J`9!SL1+9C3f8&0et{3xE5=5?w*+5Fi4*^yz6oBL9d1hETT1&c#J{|e zu75pgq)jO+Nz8DxpwVe>Zc+E;q03;9Lw4tW=E&#!gUnWj06b6Glf|c!69kyIUao#7 zPoG!o3tM6{-RQSQYf~m~SK&$R=l@`hK&{d=KG~!%!D+2v>>BuBGor|90`-QGNP(ln z!$&o#;~jZC$Om_Y^N1<#_38{|A6iJ`efKWv^^>;M^(TE*9MRPP!!ZL))O?*fC113(UQA}zx8jfs)awvi~&5v|{d z?h~i`X7GDAVaJb1opbEq_CBmwrQVg^4g! zAm;you`G=F|7b#g@eh)|1we2oZrFxH5L{Z3uk=nHv4lq*;(WO>6AAW1!UoV6Gx<>G z@ozp)&w8G{4f9039^mtQgzOd89ZJ)_ zy9RNP9$x*de|sqZnJrqc#%!qml0qqheC#q?6ll7~pD*;($!5Fv3k~_LndFpHWKO|B zb3NA`py`H`ACFtsY>#*X9)17DL3avb0)@FUsi547YGa->mgosMkgOo{el!sD=x(C2 zH^!v)M$!8|v2b`i$A#!=*lK{Drv4@}13cRH(4!48kd)kF+-}|MHBUP{qk#@t!9@Nd zSYb)frH&5-D}0PYKwD(8OR7+>SMi5}6>fM+>n_kf)-q%(aLCr~PdC8_3Wc4g?Qt`4 zgQJO`o`V3_Dqy6vy)jYe0{pMh1+R!J;{FkxO^faD z*HrcIFe3@Yb6hH@Nx!<|n(v}>jGBS)<^dOY^Fm#3E(mWfmc)Y|9A*_$`555Msq9@` zJ+A38(brvK9f&EI#g6q^G`KR(A4{S1$gSw$qj)X!?;<*;%+H5L*j&d-6QX~`$XP1; zMfQ2k)`OVkSxZPCC?;;^5%Db}Q7p}V0A#t8(9K#~;Al~A7+_5n`n{Y=8==%S&6VpM z7LmWHY4c*G3ZX>?L7A{itg|L-J?LZJst#oWrOkbXm^@|+=^V>o1>5`V*aH}k$5MQI zdJScKev+u;C~yw(I`m?B47OXIaiQ(=|ydbke-48r6C^wy`{IfP8#RC!{? zUvJfWKXhazjH20-y1!b{O2;nah-#gW;b6H+iyg+qZ}(dLPEt z%;Bqo1MEyLd=9C>Vw+ar=X!i)ITYXTF7(fTg6@R4(lR6=)({VM*!V?Ne{ur zf5JIC8{a*}B@IxxO%E_L%S8h602^$0vm6a#g9)`h<}XP0;2SC_F-6!V<_7-lr`_*f z#Oh=EesC0!QptZa@x&9+*N64Qe0|^^C0#6%tail1N9l*+0FG&#c+Q`AXpLJ)zppZw zS@=P8OCX#NBoU)25x2j3K!Me#Ndp3oCU+HU;7lIf@x9F4t?F(AQ||<6AT6=(2X?{t z3Qrvldw}_M5>9(~K;m;14gM!Z00;wP!j46j4Y|-nv>b+^65#w3In~qde(KF&nx~L^H zD|5&>Q`-Z6d-s&7Lwe92M7T1kxNeBNA~=Vm^+!`!+`G=3T{;=P=n|M_!bV1%7ES$X zp#;U37bx%zl>7Z~nN0cdsh9CA(8xA>I;1TcB|(3L)9Thnt$M#O_KQpXy$ogI9(v7k zU71cme&tDcq-KQ^H9H=lLI_^_^$B9z@K=ER`knP_z}qGnz(_)y%OBxg%`jQTeail8 z8IiR*zi0ohO>QZIQAC;>*wjzRXs1Aj;#86Bmbjn50hQ61$nZ_JuzWMWFBN_-H@pkQCS};vB*MJj`h3xX8*oI?XHp0d zY`{rxS?Tp|K8{3lfIrEd349| z#GWliHwyDI^3+FO2+g(MsoaVhuTHS%$8>xk)`+8L`oK+RANnxr*dAeAq%hN8IP%xB4}xbGuN2ZRUxf zefvu&BQPe{kI_cQ$j_l~d%O92fcpx-%WRDkLm7b%YGjjmqzh9oFSI}y>kk+pIp)>z zGW?hO%)j+)z?ZV^sM0k}_0fl;p>!1Rd8=U1bemuEp@r>>&Nw|_a4sCkq9H_=Nl~Fd zm@3W{OeLJ|WV4&%NKHyj(d)f==(C$2-UOP9)mF;iu zX-F?9tmzw&EZz#uhHw38`;_#S1^b>nzo{cIqI_^(j@31*@-KbWrjLpqk8!VQm0$_8 zoVB+g;}_z)c24w9`N%({na^Mv6BFD!>yX&$>Ua+RKYG{S>7>xYSXq3M4v~&i3^cCL z0w{|jd9V;HfU;Oxk>lwBz$eLUWPPifcrOeL&56L!9P0}J8&vmxGHR?LYG-K%8=&u8 zx4@V&5H9G28e+Kb-m=?mL}g3M(CJ3T5Kh+HO|gR-s$+-sx)>P+O&8O^ zbTMZipx7o#BY#g9NsrG?MwqrKU%!nz7boeWJTMkK$*FvGe^I-eDD;zJ+VPjd@p?qD zpGW~@U-RWW;%c^oYAuh2QjI`dFG@c0cWVBP8ntcsT@*LSe9|f8%U-;ndd8`Sx?(|`5*uuGUC$yD#VvcEXEnMPrxYj95WYQhRJO3XK_;Qo^_YH;= znNCbb+Yn)z+SZ+7W|TJ~JSs)-Yes-~Mzbz^c~j)~&Oyog3@Iovmn=8NywjtO{N1&dnL=0gEI(PTf2{ zMmJBDnSa5?x7ynyUjGf^!+)uH)q~tUoiNtjE;wO9@J0Y;glFmK$>e;o_YcP2qVfgs zM%=UaKv_;%7!Uz(#DZ2CyW?Z6=+?@PQTeAoYg|g zMnG z%a7YNyQD9ttAn!eUeouAOo$Wr#J^$CmXi45O72ixv{AWN_^uH;;0NKXT}m?<_ddbi z2;`5i=|&XKx zTu{-T!&mZFxbi|!`F`?TJnkEy-g$bi9QLU<=m(?vPnZi+exWi6y24k7pYUHOUZbX< zOkIfu<|7K0=6JtK+{koPViXi*lrKmLAkgnfea2vYVuB8Mzuo#%xw&0Dxl>bb-$R0t z*lZd8kl41O0Ew+L98*R17$UJ92)9VxBVU}@G5$zl7^9p?C{ThDh|Dfff^behj@}{V z%2TZj!?8(U;Mg88VHHO~IJVby$e|;mTq@Tz2YntBzPhH6*RmI*s!kvCXjn`~J6TrU z-av_o*&wx-yoqsakmN4JV_Uffcx?0N%QWB7%Y>M}!ucX$f~c&>s2T2ph0|`_>uUYt(Sw9Rc4ucwoj$d*$6Tf@4n1P>4^~&`+|HTe8;$Ib{`Zm&M$(Ptz|6MG3lLNtV#4}=$xSs7OWWB%|`xsnq6P&>%_y}F121rz12w|k0& zkolulb{!J|{{rEt=20*ygJ}YDQO%sP^qoOEsjcVfw@iHfUe>1FxVirlEg-U>w@YRn z8n8dd%{cax$O{6|0Yu@u{$eIny87*GDoMQ|73F%6dN691H;%6Vlg9k^_!-%BlIV}? zu;_qVDLD^A|DzM<_=HD8g^bIO(Nn&yyV=e-6j&I^P9&oP6cF<(@r1+$1J6+clJYG?#2y>mS0*tQ`gr+`+JvXFyl4;Ol?&=bI{>$CxzjpZ;2s~hyPl(*|1MU$|9U*Kp@dhB3;02 zLBVX+8i-#U!~WqCJ=zxcig=>RvGGg*NM9wCfiko7AuNz6rp{}XZ=ok-uRzF}5oVAb zVBBswNM9P#n!#Sm@6RTz@JkK_0gS$_U)vPGPOc7dUpWeA z?Z*+YY~nDu#y6}9EV}hV(Z&0)=wf5A|K!Oo%9?r#_Mdh;25A7zCe^?!XN;HFi+~-- z8yLt!ruh0D%Z}R(PLN#OV@4gJD4E&m`)#wr?3zKsX z1r_J01GR32>o(yNED_W*!;HvJ7eU4d0BSH4{eY>ssc#alK+2S>554#=Th71F8I&iR zjt&>+tO_fzFH(sFo`axEqt6KJ2We}5Yb$bQyOQPnkTQjG9h!OpH-lu$fxA6tz$5o( zd~~O8BlD@bCV(_RDJ^XP%o{b*9Vvo@O6EnmT zcB4^Xo6li~-#2>H?oVt0#h<~~J=+W@QFO*&2c$Cqo+;Sq1IM?`BvG*oFPJW(34W5; zSf#a}y6N!fdAvZYj#ED?xYBoP4Z2N0qNv2zCHgb}P-o)!D#z3_DxG|J=t|qxE8S57 zh_)CgQFIxkx!7<6)Zmb4(^M|2D`CYC@NU2`>y~yKWRM2->MJ6CSi|{0Rz$cGkw60* z74c6EtZIkeZV~`bVa_{)r}>}(FzxRZd`V4-AE(8FX<%)r^ID}B7*=&KWGrIRT^icOtcxr&qyA^z$Z(&6Nmh5uJWgqfeKC&Tuj`O*u_Zihv z>YG?bo84!~6Yt&%+&Y`KjURE_!N<9!?6$Gbn-=gkp$WzrAHtmM}Jr+*6eSHXeq9vk%tSq`}25^Af#> z4ST04yYF)M$EwfXspJI+l*9)_Ff@?Pzhg%{zo-6*9RX+{8h{2`zeeJmI3o8rEAojoc6wSp9xk=}^ztv<0Og8q-TX&HR&@(1Fe+d&TPU@( zC3L;Tl-?UMLu0P4EE(Ts?wh1cr+(X=64)RPL_r_Tm$DpzC`hq+DdYH7h!Pc; z4+0fEl$pyHiC$X^Q^g+rt%~&n0WbL~Gay46P#zsyq9Mv-9gXR`pG?lLKq#>XS}lm_ zYk#G!F*r)l+6>ORa6R4LKz5?`bzAnd8rV>*96PMQSB6}Z$%n5xeV7(TG51U!^pZq9 z{M{AT6@BA?!tFJ5)<5HeW1yMUe>iJcF9|pXim+|ag^Z93_d1mPEH2)l|HW%=P!pE? z9Mn7?#rfvulG^BraIV|Dqu?1wQe!3n`@l(;V8YyeF4q9ZAo*th$Yuc}WS#^mI_S_@ z3)3$d@5h8vz?3TFY1v|M%tpiocj!9#3G((_ZIuwsyr-TA-!g`|{GPyEej>o-7k%OK zd-b;fx%v4J;FgX+DK9V++Q8o_FI2#=@Hln$7^J)`w4}p7h6<2sGS2v;GU&T#gbx#{ z1dzq64UR^BfxmpWI6<7a=%FO6)|g$iQz7BzMvGc-%b`H0zdLXl^7JP|%=1 z3RuqXW`LdRC zxr4#V#O8ovp#5~@h7TMgBhKxl;l92E`x{C3E6Wm>)fmxajOhx?C)+jrr>m=RxIZy! z^>fzxyK5+sd^qVi&PqpjJ#ITTt#v6W66O0fQ7>#TVj?w#`7_~9CA3vEB z{rTY8`obhmpnp#mX70^+7WP61eN?xgM7}CpG<(U~`tw3|i-Lx@z_6LOE$1cb8U&Sy z$>Ib2Rc~N2fM3*%t5Dkf!z;Huw=cR=Aiem<1nUkuc3nq$ast-h6D()-=278*`WS?k)$h~TH~g{cY;}Q1_K?GIWugV@ zxC<%%dD-_3$R&04?*I0stoBbt8kDGLQ_OXdsMrWgRIIy5RBUV{dX)PaVK?fRCxhqR z-c^b)&{DdYguwuNiEVLx)^rWlaT+QKYw-j`OUXI+W4#k|B5uE>omG%5KXbe$o$py- z|H}_u+O_kI5~6@GVfbNCPC@CX!AReZA0en(U9x$JGl^JDdHG^@OIZ^+@(ZmZ+CSR( z+mRWpTT$Cq5Ml=MzY#N`94a-cS54{-7XHp&xI?@Y-R2xX2P9kJ+fXe|2ZV@w8t`+e z(3jMEJ7WR%C0)Le`*V3DO?zT|&l?Lw&JzHm_n|)1O#SP&*y(Z77%v+a*%je#2^2g_ zMNT~^R&i(iB35xHLGmk45E#Y&f)$7;Z)k)58-wXKtg#G>{Ckd7yUXS<&2{13_AE`_ z}S~qrs%@FlnpKucEm;q2YkN zON^r?X__SVx$N`T2-EUh7C{+2E!Jc9Hp)PLs_b_@pMA}`&9rOVoE}rt9>3NJHH8lh z$V=tKH};kEi?sRxDdTS9mH-;i(XGB=%BT<4T;vezit2U9t63SKt=(^G9|5eZ2&suj zpCj5Z8|`l2pWV_%I3`K(bQzwL_-soV_eNx_u45Lnp9XRL<7*wkClX7-WXR6*%jUPt zOZG||sm`Qz3hFWl1$b<$L2GHZtb{^VIx+G2&5sS|>ldy{7gY~Xu$-zC{5yRIDkW%jlHXIfdw^M!qgS1&8Ya(vtbeujPI0O zI7|gDH+0|l5Txb=@N{G9ak9)N&G*`%Ae`TQMU9J#+y7CjF9Kh41M*3wEDy#fEsfe4 z=Lv7H;0EbvO>H_Oy&DUlB-%`x>(`WIkM45au-1Kum0u8ycewaq{?KATy&hHUC!#xp zllE&mTP?%41zu$i{WkA~^`#n5qc^K(-#tk?@%u5qTw1!E+R4zTyZH=l%~a)yDu(~e zU?}D+3sX-e`sA{M7vD4I89I=VVWzJ~F@h+JLHsi>))QvqRuuXb0TqTFN2CA+n+sWVYNh>LL)hrwHXme ztSr7^N4)u_$7@Nm6&WmyqZO>mOHhCg|J#ZUkl{N$U@5&*A=o|Z*e}v4`~`AV7BN$N z6NLqYTFsNVJ7(|BDncn~3Vmh!_1PGaGZG*rP0FUJt$eS1p-3QK`8y)U0H;msX!?qx z))dUXbB{i#9U4VlT}*NNa)|#j@2(= zOSZeUBB|_y{*)jigPCX;23sn`W-EWuw3``-qU#5Cn8rwOehhJma*Q7$5D#nb&V&ZU z(9XvAd@Lz^@nVNp$miQ%!F4(J1$l&&lU@KSXa5D2Q}KVJa>6fAcmOJAfP;F*%No() z$CH}cOv-p85t?e)MZIv1Ogw|ycW)v^MjX9#|CP~-w@e){r{#pgrayL#IUqDfi>kwX zVWRU+)mBt2*krm;BxoKa&#Jt)SuFTWoU+5fdWf3)die#>8uqBIc{?G&*opQ=QJb&) z=4hA`d-Gt{yRM`Jo@RTi_}2FV__2!*YHYv$lmump0joZ%%Ey4wV3M>IbFu`TRO(`M zzA!lX#M1g-8zieSFuS}gWS4hRCMjj+@ib%p0ZZiBR?50#4m~?pE>i?9H<|yM%zm)E z+YyFa1k^=uw=3nK_^+*ixnRCuP0XAb+3uI=j04TGWZSUM=pto6kensAi8J;ITrasBKK$iTdCO{k`v z^#8Z9p}=J~`nPXzt8p$%tC9|M{XR+vHb?4R#v?3Sij%1otJ`x?^taR-QtSp>5)&Q4 z)Ue_n*tYr5WhO|NKh>5h$J=r$Z>TXj0MW#B-;@Z!jAhhM0zc!pv$r2p=HIMvMmV^gieMx%vePTnS%zY25DZDmm&Cl znx?4yVtQ^uL0avD*FwNMxL)gILPm!C(<^$ZB}vOaJg1`Z46As#>P`9-!9zcA~*YZ;JI< zx(H>+*VlJ?(g_M_-_oa?6(Z1_x2V6!U@YX0YWw0mtASy>Bl{Ou67aRL@;{`K&3#8- zuawlXS5cX!txIyKY-&Z@DhYA}=f?_o@i>6;7MdJ>2f40ba;WUu@6a#bE&O6A1txT^ zt9aFf^31*0-atMqfPuvtaKvRRnHg`LjgaH_xmgliXU;kvh9!Q>GDzv1DBDc=euz{8HSxp zO$Yt^|Bth`4$G?hx<-`}>29PO=>|c%ySuv^DW$u+yQCYWq&t+9knRR)fwOP$7ti;d z^Ipf-Kagv&?{%-ea;`bY9HVu_>Y+f{i4c@L55(w$0pM+%2|@RQjq}xu^NmZBzUvao zSarQOqDY)WhHf};{gd0Blm|xNlVJ8$r_j2`R(+;kVY^(!qS;@22=0{o4E&aaAX+!1 zOdc*l;(aTYcl;w(DtTkZeB;E)V^vZ#`NlW69tF7T=$OSr#bT>isVXpX#;|O`mQq`L z#?KGg!2v0R>}_1lUq}H993T_V`&@yNm#C)q;j>c&cH>B_=iUY&qp{f#;Ot4XP65niJ$v%Mf&=TKnRu3l<-7FcL1q65I))W130+_2x$IJl&$-v{fSX+>P z#gBxj2V{Z&FXr^0K<9tb+7hqGS|wioGtlX5HJf>IefLG$FoBKMYdd_~y~6a|k!k=r zd{ZK@Zgy7zj7gp<>4Vtv2mckT_3C2_i16 zh5->5zRsPW6bShopGf*Sc4zToS2?F)F*$@YIKC{A>b|!EWO#;SKLCd}d{x%!I zHWfEJ^R`Aq@AFbEYSXgF!4urZTR5GukYZZCE@#0L?z60iXYQF)yVj`r7IlASE@XPT-i`Ky|HW#FaUZw@ixINdq0Efe6UQWIoP=C2h~!b4AwuHC zqD)>%pne0ZPKEFRoHHrA#W4BhMiNPIhw|G#B`j4_XfM=4voQm^vyHUAyA4?=Nzx%X zA+eABnjxY2Tb>o0$1j`H!KO^oU2-?v1#XP-VT8t(E7h)ck z>c^rQAopgA(v_KHjS7x~q9d+zJcC191e z;;*b%f3IUA9pW48r8b0L@^@F|{j_wfv+=rRY`k&q3_zcm&)$tFKB&>~@JWlERq_uRp0%=XG)pQ2#J6lh|*3}G-V{6sdmmN6Dx_Qr} zhrP?7eNX#+I1C77C;s=N^mHZur)Vne(x;3%7XjP^a%sW&K5(P*1(AGlvG4dlGsC|N z2)iVKwDxWQtvyKq3ke@b+;gw<2{~A(P-=t7d$}LckSFTs*d*}{2Vo+Y2~(GtAjVqK zgVW&{6?5q1a9|?R+S`o^RFPaqM?wGB_WI?iC)0~VF;L_4B z^(1AW_!&sajKM^iHPo@0ZTRDgRcfPKsRQ^!%KV+bKsCh`_p(VCK=BBdl92;id#}!r zqu{GMC{JF>!dK-d5*HK%^ymQm_2cj@*?R!5|DBIGOUqz{x9)AN0S-V-318ifBHEiUAIrCOHWM&iI3)n$-{s~h^%uo?S zo!m^=5IuDzs#FVgjG-8{B$&2_H*FG1zLswtUosJeVH*P&9P}v>o|&FYBfueT;j%Rl zQjJ3-@^Oakve~3)anaybn$NBpoxM3JRkK@_ini`E!PEo{f@%M!mKm}@%)SilStw_} zRCjv&?WJ3*{?uW5rh>MrA#xfUCu<%z?!Fo)1H@U#gYRd{xA2*yv`U{X9aC2nj5ysP z>1KnPs}M@Nq6@Wdmp-mZ{>0JQ@9vXjU3sLLKV#SMk#S# z{$HTQH9n#5=Yn5~iv^Q^LX>N$Ew6sG8voSRHeD_3q;<*NF?oLBnK34#5Dl)@Umt4L zfvEsS{f$cXgV3_;a0_xqZzH>+kq1D5oQWMpRUIu1+{)Dvsl=O^eU&Sv1cbIVh7|0} zDCIvw@<}uYV+5O^K+sXWXPTXaimx=9mk4|?JO7FeOY02zLIN4C%MMGd4?Uu&OsH)G ztP{%q(7ZyJDuE<~4&B`q++Du-ExSn~K!2`kyecb$nAklcW~SKiYm zyPt<^$=)%PaM`z)OxxjLnbtW>gur4Th#k#*?>&1fN9BpGSo5?a|gdkfFNm+ zs|=SUm2q|XA+J&$03!F7XIXrGRIS@oFYj!SJ1;N4e)0rfb&iMqugnN0hUY7J?0*iY zMDL{PlHh0eNhid~tN6nPRu^k)+piCxfvWRtZP|7|URu~aP_jRMG0#9ni~<+f10FLJ z9zslf+_GUKWpa8ring2S*IyItFF0&AKyrp6rIG{BWP(;SnyP{n<6jugY!yuL(Bl%z zfJ}uv-E$?aAs{^Zo*JnkR#IEOE{D!Dk7WWR7Hz*ghI@ltsZeF8GlEhMo`@C*&3pA6U1YPmscURBw?L>Rg?xqwu+l z_HdZ@mH|+kOM=H90BW0&@>9HT@&iC^E^x(wp2;GjHK1qWt1sF%-PCB~)l~g(HY^}U zQ&ultOH+CIrZrPR2iN7(Fw|q?=Jvx|HoQdQqzq0khu%;WfN5~Zu&b`KfsvSV)8#7b zcPQ5+?uW(s5*U!4NmlI>sc91W$0tvs+?THvpmCD=+x5i6@CT#(4~&YoGjZN7769;U z1tJi0na>!NA0KVCW^Y0qePP{>bXDMO+yx9)CLpRCIO7H$QH8T?zFHCn?qXPhg+iSW9i5M_zRpY>qC(% z#U_=Oz+4Bj#dbN~2{|-5NqoD3*)yx(dzb%~!F+WBsFla5JivntfU%@F_x}~sAQQv? z1yTmkR@%-sAQTX!zT0Zy6>ygazo8r~seSy}L?>wPj(*lY?z2rAwi%={$jwDxzRK{X zAR3fPI~v(N+|ZQT$<)!R7VV~(JP<0FXcEm)#<^%bdDCqLBTnX3D)dOo;y@`xm5|Sy z2_TEbM)3BAHz!wl@@rJT)#6fu=|Az}Prcxct68eTl9|*uTo~|sLbop8P*ZS`%zTGcj{ZM(aYA)8|9R*@F)%E>mASON}O*OY00SG7| z_LA8sHM|VD6$)|s>{84^G>SYNvzt^){TTn!z z90{dP_P`(u$7N_pQ>x?X+5+UAtR|jl+r!IT`9Zn+b=$gs4R;LD2abktSeV~|?v-|y zleZ24j&|HtLEX?N0N?Tl_#Wnbj;Jh(E!^%{EdsDnX5+& zw+vS>l~Lc7Zy$CmK54J2M)8lg*0SpDl&2-l?bDW1?O(_O%6n7&`z_~B-tB+ zv4I%yBLe(m#dCYs>L8yBpT4WX=t7HiDv(^>k^yog{&2c2QU3J=KP2_TB4%lR3xXeN z(^uH&r)ululm0{(saatN2YfEHoB$@yC^MXhI=?c(i~v9%j;r9z&pG%7YIziA4&6Mo z*;x7yJcT_Si-Y%G8}e4l!SYqHrcO|mPx=b%HjK<72(QurndR{7*Ma6l@J1zO19bVs zPPF^KO;uRuuPGZNh!6~~V+a}~-8=oD?%?4(R-}u!0QVLuv%Dh5KBvhP^VN=rEYED(9ZU$Y4@G5opC|3m5f4I7B?_n&rcY<%|ueO^A;+eNa=?d!(JaEkua zoOb{-zO3C8EO1E-Nk3!i)LpolM3Vj9A5=}m4KU-sj_l?YfeD4^5Y02j(~nnti_KXs z2Hzm!*A=XrVg0HCRSTO$ls~%`#v<240IV@Zpg0xFL)kdH04#PN=Iaz~aJL=Ol)poI zq_|;EIukgxm?o8CXKJctd`KWJ4No6?NbUE5-DE{QPK^<}6M|4%LW0wS;6rg=L^u1d z;aI-;C{Rp>SG$IHkr^l-qOhHgea6t*hTFJo;Q8Q{SCsjj0K4|1%*$^-iGF@5e4FPb zhsL#6%gDVy(@3z-)?B2WYQmM z{A5MomYBTa3)~Vp+ev?(XyCE^p~k2ChH-%Mc6m_Z`5eYI2FC=Q!+*_<&&2R_nfC&~ zvjpI3t>mZTL4OCN$fd&UJN?} zTMEcriSP{`rCipYN^@gKL9D-rI$YHwb&>!;?nfUZh@bv@CLz;e2f~u8!Hy&DMhH8w z6cmy7zgis0{sJT<6SLJ8mA&HjfEa@_e^rHpI;`@pCLVsKvRhq&O7;-Q*A0;BJ zOkrIEiC9mta-vZ{?Fi^xyuSrJGci2h!T-r1QV>e4F9|1RAixvsuDd|BWW<-6Q8Lis zS{-lp74Y%4{xCHys-?G#m4(Z1i(|mckIts?ELjriTG&6cfU5uy^>QU&tGmAv z|0>ZX$d<|TvtOj7gwf)vjJ6L&U(HNF7&?SIVl5yNB-17YO>I#Z3f0b!Ns)l4V$Ty8 zkxiaWT_R#+DU?{)*!3p~O94p2IvcY(lD5Pr1E3d@Sb%}k?=iQ~dV~XE4`cN1qym(M zbrjNU!9jO`=Qzm>n@_y+u~~k++>*deHh7ty6K$rI`FewTxrLxR;pY@yLdoRcFD_JgCLylaB+sUkDBaW1;^0{(ZhY|Ca_30Ccu? zX6k=p#(y>GazEY0NMkE;EmSl&-8nMa&PwOpo)zhUjVK$ zI{;2xzqH&0zYqqLzF*#Xl8CNjHZK3+2Qi%YD+2gI>dxpvHY85{2Tf3x(A&T7;HO(J zDBk}sj$Z%>x^sz`WF1=D#Mg*JS|0vFWdz82}q6) zSm_uQN-l3#oLM`Xe9XPQ*@!@N)e7$u7r_7^R+Hr9{;6PU#HR#m7LB9Kg3cz43BG!r z00nfkN9`3TR))J}1`7ZLN^a5jf3Q@@c5MBdr9v_m8ANhOQ9}msI!&Cno@a0ZyiRJX z8j@gvCnS#+ahE!!+J0X@f~zC8iZ5l7(s>w=r+MXph@BP*jOf?_1EQPS`2`{t_g!B?qB}8Gr>i1tzS?P@sQw|xtn-#k8*Q9 zo;`+wRQPt)D{TJ6w5z>#G?0M z6C<`)fem&AHT9on6}?>%(}3h<=`(E-#ez@5u>i5$Yd5JR#1<#WGNqj9nvXG1kUWOa zHGNtb5;S{=YB^D=#0+<`1j<%OKn@Ecuq60Nx<~5G3>7&6PLaII4`%O%m{@|1Sae6W zX+WGJ<;J3Ppe12t_mgW08(0!tl3)tg2_GLRv8QFc!oEI&g&_ly5iJt5q6RdYp{1>T z`6RX7a&D~lzl*kw55(F4sfgxpRxnqeZ75*^`Z54KG`t^(u?68qM<)XD$TR<%p^q%? zm~QulOz&3!|MJATrL@(T2int#|9Z<|{1bzg2Spe90fKx4$>jouP9y*iEnwiw3w`Tt z(^xJ(OV^h6wXv}#!mQ`o?eR|gyk+0hzF6v)Pi9@ML^zZna*niip><9VzyQUJ9=367 zF?M3Z@I3|PloBJRUn}J9)TtrWKSWB=dgU4po%+;YNN#2C>(1NcibZ@&w3z!OXs*O~ z%9Q<-`1nM%m+*ifhm*rL&5GQA$lICTARHS3|BDJXx;I9A{gy zx+CZ$5Qdb#+-rEOxyXg!!Re>jt{D7YyVOLmz-ssLnk;zhW>fcDo#y6g!$Iq35&$9Q z{+d*riShaF3Lxg5>^`wp!42-=d248CNs2ng*XDMBqMz9kn?kBm_3+%nRmq+KnKZ zG`~@xMQPOd7Xe6Fv3yu9Tmfn5F8IPj9Z?twfVG{hfJ`M-q+Mjz#esl&)EXN`-1&Wm ze8@mT!l|1l4UIJ(DeO|!GS)Db(f8LCt=80{#1|2Ukj%gY-WP3~cu*r5=y<;fZadHB z*OO17@@4DNMsTFgbVkga;nATsb0LBEBqW+o*Wu0F>9@7KVXtHF*Nugg&B}$}3E0Sy zT>W4itYuAkH#Og3bzB2^0IXIWK7x)i+$Z06uESK$p7QW*vb|zJgx8FJKb8NWn7_$9 zL;F30HWwv-Ko~&d1E=R}_f=3Y_0(f3I2_TfFyh3?!~n4G@wL)i4yAu7SpfL<%KuC6=jdY6;L=++#O~i*Tsxd z0>iv0Xfz&cmqP|{d~##@p5fpMlg#GA$9=yMvFEszklU9>S0ZsR6k>WaE|gan;FQDV zVrh-lCgxF_z^rXZHKc8--i7wC!T0l&u=ZUY^ctQR)QWb>5wSft63w)OTb1O9k=~nV zhssC5JV}}!J7*^X_z-E00M>RRN%p3!u80Z?U=p`10h73$GDj(}YMgz~0DygBzhZ@3 z{@Tu6_|6UI7l5vPPTLfK9sCM1PyXKiV|=<~dd292NInBQU>}0)mbGX00zob=aQ)D^ z-K#krkl4fGG+_T}&*pI4Ctqd4jt`Gw64av~|9cV-d)jD~tQgmRX$ll}@Mbf;H<3@) zdYZ(vXp2JjJS|~sRKbgKh8IWjv&AbfL!3fnU%tktZySbpUG}qkjN(QYg`wUuAty6N zp8@7F|Ex3&fyL+&b9`*-C$1ux7)zSZo3*7V@p-u+B!Hke z#ufJaVKPeyP$BZXsGM&AfI&WLdnUSD5RR-2jbiEyA06U+Bmqz&uv?_zVJQ4mZ*>hI zSd3bok4JYMQo1=Df>`sG=g)bopH+D=c>Tjx@o1`w8$9ST`f|m7;#~9(Wtm{_d?v^| z5&HX?{CrD{`G@O-yuKjbCR7Pl;H|;us|BN2?YXQf?&{0K2x&#M(xf{t)0L3}&r^M; zUy;0&rkp)~L_#2bd`)||r3#zEDzVDSEm9X85@kEFwbw##BoSQgnKIt>_!9-gWEIplnU5gc19>Gvm&*<4)joctpH0I9v~rR4S1jtR6}fC7>ScASkbs;EBp`EyuZnsvQXKV0f?SHfzNP_eK?YN( zlJbfk1Ck@wfFU^01{i`@TI3l;^uAup8!XY^_yMsm?PGv0=yU88NVFHkSLTN&kpu9R zX=sJ(F7oE7z%SSmE~f*$wV3!HXC(NZ{3cg+eXSq})?XuRnHc}%$V&gWR-H2eLrwuI zPMSXp$3*8+oeeIP^yePqg${}dXD~+6Nbh?N_k$74I}Op}jn*0w0Uj&?A@bU0rUL(@ z!&?ieu#Gc=vFk!Ag=T+wPw<#Lfi7*H%*BeeQ@C8M89#iy4wQ-9I z7Y}{xn76sJCN6<-1M5?cTjkQ4j;)6oG(h1#1zuSUs5drIB+l7!?18}ffUpJt&M$oT zM_2>!--_L2?*hUafRhSWSp@*+8@pO=^0@k;z)WR+@$>_PHCiN}g*8Bey^48&VDI1) z9kUznmOCi*>aRJ1nHZn0kN*b->pLRVGqCVo^86FoE2 z9S3n-EzY0f?2Y8Sej=6M;(a=k!#)wi;|MqQI+@bQr~q`9vQi$wk3cDQsU6SRPx&TWBn<=iDownKKkeOS67Zr^2WUxHI)b6 zK(e8?ynxTVVwUZR!Eb{rDp!lC3?FR|36B$=ots*fv!{$gZPqaA{}tSgiSapn3YZ(u z7s+-2psyaP7hUy4NdwX_nFZ8h7LVAt6=ND*>*LM**8vTaLH63oAEw5+h`;hP5|FNL zk)EtNB0!WtD?&d}OGiq*4LjV4Tpwv!V1qHq4iix{F361|kaAK9w_djdhKz?OhWZ5E ze6!(l=}ShyKxU<{FWGA{$9KrS-%)6UffP4*=_Ahq4jwO};>+SEO&)`Z z40{}3SGKS}mKzZ36-P$t*hYROM65`D7VT-XUWA+?5`~DG&y9c}DTo4*2#ED=j1th_ zfv@F{P|C>CXb6Zk=$%OV#o~mJ*pP9{CP$+MTO52Lr^?PRq(Fz(9L%Sqn$H{424bKf z9F+t6No$U#0=XDV4myzJ25~6zoa7cBHz1L%O+u&|P;1*!4P?2M)Fz`|VZc;m)8#JP zI4ev*!r|!2#BfPS`a>M9^b7NxqKatJ) zDIP7>7Lrs;XeAi_#0I5}HRN$d`X9TFdH_ij%hYVOQ-^j=a8nbEVZ^*2+)q(Z9JnWX zNISraY=$g%}M zl z>$r?aJg)!#gf4u+x=ajO6@QHaWny}|RRXeL63=Sk%S1rJTg4Y5!`Vh~5==pZdh` zVN*DGV8iuNWoH8GgIhE!sHZWbB7mM2Z-zN!oYymhKSY2Bt9^vbONR$TlV@J%*?_Uc z76GygDGJW(^RIuQgR%=D0>tW9&10oLysZzgqSJJsv4(y({slwQbbcNxvE@nSZh((F z@>M^mxAWhf&`Lk0=@(d0zPUnkoMvtW5^aIN=(|ScQ#GKsZO`g~-tP3!{PNQadiIL~ zT7sX$UBac$7vYx$EFcLQ**@HVxeXU80SOv@K!WDS)4|#$Mw0{LJXHB>bCK!ka`|7q z4N5O8cXW(=vde?ZCZ9y8#p0sVxvnlv_BGHb+)^GrgFHV1BGf9>w*W!bH#{JLQ>d6Z zPO@$Dq()swS}#*eX9OVn2saudiVOlGZ5)dcwuOj*N|?iLIm#V{JW&w5ySt-VMBaAq zt)0YMlbYz1Y`fP&w;$DzCNl-ajf%o=%6cTOGpmcc^BF2o&?0SyTb(H#bcq!T)E1AU z)XDM#jmRySBuk+2B~zhb^Y$d}D4FZYL3|$scko=@Z6LmnCP3|@X?T7f6XPO-zLv}X zfoHs}Lb~*K%%L=9mQ|UhavotNVUh$vQJ2l-1+eaTStJ2`j9IY=Z zShYE<)}-1WEBl&sfs%rVEslKQfs9QK#5|#MB#P2vn{7~6c?4i0DxaWLd zuR_EKgD=g&ZML2AvKmTptQy%9zQJW6Oh-C{dqeIp0#5Os=W~OtT{k3@FDh{&^^}Vx z7e;u0B{CMQOjFZmgqt3^e35W}Lb0;=kDV6beuNy02SuakamQf%dhxY7%`b4=S?diD zAXfE~?_RTm2@*m>Tv@m5?^MC0w-fI2(Uj+qkV}0cgeXg&Pb2@y)!Al<3B`|u1T~SJ z)*x$u0ddcDFmr257w$_)Nqaz7B4q$U%D>WSRkWRi1tw0|1Vw-+Wg=pLK)R}@-|y6w zUp$D;f7YRlliK6+I@&2xQnVha$$J#2%vO7~i0&lP-&v=yc@1vaWMkT~i>mf` zDUy`|*fKh?8@n*Dn^+$n?02^bKB8`QJYAO7SH)0TfjtKAH4tc)1cyL-0SyiP zf-yN(9hmF?{)Yqn226Zi2V*P8e@uTmP)YCHC}3Hl1+HqEo6uX2zZ4SGHOD-rWjpj&2fh2@vX7yGc0ep(`E>) z59sgN4y~FSHIpO~r4=8Y139x&%072pD<`BGS>Mm}cKOe{t1gKy7X`@kHFY@eEy_k( zv!`kA+RUGw?xbFi?(Y7ui$mx!%3>WaUk;jYzP_}2WWGDhjRJ?VJX=kr@s5C--N5D( z<(C;7$M(Zp-IjVEdFEJ1XgV_948e)&JqEWSb;GGjxE-5$ZhYwP#?`c(wCc6Da{9?X z-PZR7ssk?!m>kmouOB<>Q3T+@NW91}&p+gV-*oWrgIr%B<#-aK{eS(KXvE}$0{yY? z6Z9skPa?uQs4~2VzL&5;9kyS;tlHvMSv|PS=9@l-)9XA9{Ju#GV>Phj>0$m)1CRAQ z@Sq*f(ahM`$&vo)zbDfvAw%17Q6yG)ncd*g&-d8U1^v^5omeUs3l(hl&)|@I&qA^x z12q(U(s!9ze5Ast0X(mkrsC%~Yn ztZ4UYx^cTdOk$^ayyH85F!VZ0YJ7aKt$b{1zP~s+pPs%i4FYdpYU8}|{P?&KRaE3k zz^m)k)_S+P)mUna__#SadN@0L7xeMxgM<6k!r}C0f019iH|JOQ)-o?Tw_?*~0zCO?)g07Aqv52@aZ zn+@EYUU#?VM~^M-x9b(xchl>lH#b{bv;8082$s`%JU8x!vmWpEhNpw>N`pQwFF$x+ zxc#C+_;?lnW7$^MtM%6D<}T^(@ypHn{qzNW)-eI~?7R4*)oes|-J{HhtLfSOe&5X2 zTlbssqQ}Se+x7cxc8bxp>OKPRj+Ty_+sEaghxC)p;@_T1ynDr4H^p1UdGp(O^Ldeb znNfR@ToyN;Q)ipS#=Y{T+j%PVq6MLOGvlJ=x{(?Lyk_T~6I+|v4cwke8ozJe(nm@# zwx#2*@5We0-ktvzYu;*_Jl@`z#P)Rg@bTvI;&7Cp(uP2GC@4GXe)DvACBZ zhv|Kw;v@KYyM2ZF3hFgV2ckHcHxc7O1y(Xz%Lgi#ZbvRUIG>(`Z+LWUdD(MJ^4Dw2 z_PKcN*IB7*2OJDy4b^k8*0e5gEtV#sOYLYao3G%E_aIS*>a5lM>|`<-VM5=c_A+ad zhnl+JhT*@0g5s}UXT-GEqZq94!rj#gT{=xZLBI58S>0H62^zN&;jcE%t@8hAPmI^wynK_W8Hy7nK^TDZkh}CH5eAki9O~(4s8qlPt|%hoOrdGqv=2PlME_v zUJ6F*5bEpxs6fO_lb5p+n6Kh)TNk3d-h$!c~ultyHq z9&01VKlR(=t^!Y!|M$~jy+23LxZjFE*6AfT#bA*ik)n=Bb&?@a+c@@)3T+9D5-?8Y z8X8rl=eVxPV<^%ol$pU+=Dw!*_txN*)+`!*D$a}hP%YD;F{sIXmSh- z-^UQqsW8mQ;e38rh-nXwOe0cipND!}NW8vBMqmVn`}O7`buX9bJ6D^S^~3ieBj?x- zPwmJMCdeCOJfj;mA7cp}6Jo8mFvx(M?Bh}#EdQdmO}RPt)}a-ZGDh8$QaT(CCCC}W zJe;e4poN8keX^JxHDJmDeV;bjxr|wO(E@cJPkFIfNN4?Xi+~~VnFsK^X@)^Y4l|kH zqJX%g1ap)Gvm$%>DTFA4o4??6LmzW`@jOt!8ssr4(_T2}UDX1s$2j%`n1nLU(>Q5B2BKkqWAK%Ls4D3!8k-mu|g8mD1=5>!6K= z1C+U{f3c}g#V}!8BI1X{@UK-}x>-?C)m%DqF-37Pg*j+^dJhi)RW28cEukQ2=vNXQ z3w{3vH25W_6AB7OCENK%ZwibsPV2rR7!q1Bq8l&?N?Gg_lasMX!m~bAIOP@$R5;~m z!4ze|6y~kbv#HM3AbV4reRap#=Q1*$FVv2P0&Ikc+kSFjoW;Qhgs312+5Et zww!7PJeaU0;md<#I%7pja?lxTEVttInW?+`zNnH6Q&HAvk&GW3c+9JV+nZH?vm&on zS1%R}hf6h#EC4((0~IN#e$wAQOPI$QC<5&mFh`wORM#SUT4}D_t^t9XQVsL9H zyrL;QOLniwaa@l{HPK681w=L-y(gd%aT!ftu@vc?RTHG5%Y^q2GeuT3bFs+2W?`et zq{$>9{sk+J>sG6}-;87}ojMI)v}9BO0|Ke7k~K{8;d$4Yu!ltRky zD;SOlVM4jZXkcg?NHWApXXZ`9R2P|WE#a4dL8%cbC&Fc{iOb*q;=3R()4F=s^R z#3udTk-|M7dTOw$NC|onLjYLXWD#qEq7tEV9m9Erz4YGTR6}%^&KEFUbc&}&@?h21 zY}hU-sRV*&v(djfjYf}&Zu|ydN`8!qkGjOCmP5A`_jy3-FC$N5q?L?m@MuiU%^6e2 zlL>(4K4QXlu)V&8%0)GB%_je#_PiOnOZbRU+J$;>IyDr$}wHV^Uy)i2(*f zL8ycXg|VhKe_P_+P@l|(%z@87I@(p33>t~o~oHZ+S`WYw%Q+ZUS zD;+g%jo(Xhpn{xbffSbP+@70^PfdyMlME=3bG{~r_f?dB&tQ&{=2w+#{W-VR=9WlA zLkA|Kha9c@E0Z+;sVB<>wRg0;uVy4fW4NYsM|?S#OoOuAZ~a7{XyivhS{H}E1|7~~0o z(f4XuU;;B9jFV-e?P`6vME4484|B+k%1A1CpQ0~xMy6z~7t2*^4g;lo3FmMiAtYG^ z^h!)f8ERQ8M5fzN5VPj$Nt44Lcp!VF``(`p&qbwpN5&MX$X8y_tio`VVN+g?WMD(9 zANNYtz6mi$Pmt|EG58t+Y)@;r(CV|36BhY7=w@cdk@@zPN^*-h@cw1{=4PxmJ2XG! z)wWu=y2G~9B(gx|zsKHJJD!jH>uv1&bq)J20`dK3YDejnNK#`QO=%kO$@ zseVAMjl-1M<@bHE`Apu4nxI>mFicP6aJlaqxgnvEHa*V_iEKF$&rBD3HH~9wcGDmh z@Zzvh7h+PD&{6eB;O-Rg{y)XRiT2%bG%(rPvUBQ@1P_K5si8}&eFHBI3)Q1o>Y(7u z1F$BfO9;#QU~uN6Qc{0KI+**VY2wDvR(se3mEypo)&7i1F?koeP0s7E&n}=>t?2Mn z4s_`Bgo)8G{M3ZV9Q7r&4Lq6$`ZyX|ZN+xq=s*On-Fn{~b%h9I?{sl?vhzE38xfnu zPq^%TDphH0NN`0oP4-Vy=J`RG(r@q8BG4C`2$)CRH~Qe4>npxK4DQoTgH{+Pk;5E=&CF8!^ZRf_KK4m6nlzy@<=3TRtzy#-~(G>pc(7cjOXly2y@ z>pO9+k$L=&8F^JIkiCmnega33m1~FSUa_;PEY)eY`;=wM9_Ubxe7>r~m?kUMh+eTt zLM%J2ifB?l;K-T@I!iZcC{Eh_c?T^qQs)*MoY7`JXOmu$xYc`_ga721p!guq?>DQ( z6>1w$vO0YayiI4R-RW^$@+12WiX+;r1$^bvtAIsVod9O0WdCZ=!q>}C`CV^q|HMGR8w=FdhX<2 znxXzM&h3=V?9Ac6yU5xqtLLDBUp79kNyXC5nUAvduwq3StM*d-zzEk#1Zab@+=v$+ z<%Z4)b-K(Q=|dU(Z601gJvHUvJ+#i-0~ORVp*vE9V)*3(bIM<+*b)xFu%?}%hioX@ zqghWVZ2*JKFMcF?VzG*tsu%Qw;z2j9JrU_zp-#xaH(LHL;_YU?t|=I;Ff=W^Auo0PE=ff7%V9#BkDsRsyH-}uw+epLSkR<>nuGzu z8YnIFtwU1`JG^XYGDMvxtTjvvTHp>A!&j@J9jdL&OSqr;^Ad6=%#A5;Od|CvySivP zvp&nNeV)!<8j_z2fdJK`6CkeQ3#pXV)A*TxG9k|mXS!+@G!OOY-Doewr4p~tu|=zo zSP$w!V~6$L)4M74@YQ#%QG@l+C;>Rt+7Y&vR9M3-qcN^i1XtT)C~5mhFwx|<2<<8N z*(dfek)K$lvU+d}`7P7)z9MficLQA#+{Hqcvhb=sm^u;(ZH&5erL5-1*7#>%k{&K_ z+wfiZ>Fj%1`JG4Rcu(d#3b?EyPgG8E0WDS>r{ck}Oc%j&2KbC{fifA$)m~r^e?)ZtW zd^;13stW8-TlUZb6m>xB1WqM6pXVWBakHHeD1^LZT|nLP?pf;Y!reTz-JrXI{Gm)y zoT%;0YC{Ry>U?d?gJXO0@{Lt9a;wxVdE|Mf!mDMdt>jU>Z{4> zMC~l~BV%_WV|%Lddl#hf>jjQj7eD8ij36qm#>AS%#`ZMk8{4Ml%%!QU$F1d-K^nJl6# zj5S1dVcTB59ncmW%{UDh99~}*)A1f1N1zw1?k??{*?a*kEjyYF?bzhCH7%^JfyG z$#o?6DCuT!ES}cT)EDCV)J>@a?gcKGy}@wP!|Fy~kG!mTtv%AF&AYU!|xbfcAG-!3@xsolv(-!u1~2;lO5q4xi3ar*Gm0R+DGg zp}AEmmf=napFl^%cS(wnHtAvD^ZMm8bGvviKzn#Wo(xJ{8tChRvukylM9W!0zcg;A zG_Gf_(AYIK7qn|#K-*?0CGV4d*(YUeHUE)eLg`X8jdk#+f_@p4h5xF+(?vnQE^enT zuIKlERnVBBG_+M>Z5;yRltN~tCt(3Ko`>D58ciZB65Q=~X z<4n(&E1RilIdkvQ&Dj3W9KL2gd(4J6xxy7oj*B!_xqRm)DppQR|q4Jkg zQpSO&ZvyRj@48|r&=Yw=GI_+U96+Cc+Tx>5HpK%NWpkNjQEJl@?==Gr9#tYZ|5R{$ zPFg);{-dMdY0G@ql}CY2{w{>>op`vT#-;6_3hed<0~BQo6=gBj(=oIKB0=Z(z&+5c z%+SToQgz@_r`Md+fC(UebkUj$Ta+nPlu6si`S#WfEr1sXD8;bYDv_oub7v}dui@CP zs}?lbFLooDlZ$N)NHw<@3Q(0TRF%cpPsbRgj09$XXBnt>DhBXJQhT|I0@#7=u1}R= zA9!Ui^$i4*QG<-ht~3exlR~EW4W03_=99g_UweZLTHd`t!~=VqBL=~~^H+T3f@ zw(E9s?UjruK#dzog{{n#s?4OF;e7iA$f>Q@=qw8&Pbb$+zA#F?FghJG|0c3dNjrm6 zn+cj={N*hBgKJOK%HhwOJhvJWF`J zmAQdrktdmh0UssVj?p}kSu3@f(D;T{G3j@$&P|p1u?-4hPbydho$bm1o5v$!h_h>z z3=@Gq1%X~2PL+!TFhTCDBD$aV9cZprBSx*=iSc5}U(7h5bDjhyC`l?Pi8hc1{}vlR zAfvc%^=Uf`#(REr1#o`SF5?6xJuP}$C$~Y9CIK52&^maoQ}2Z%We9GRep47ssIK1x z6mLkIqKggYTPejFti=3lt73MvkI09V6~Q<}_Wn^4be>8~UP?7{sZJ;SPH5wwQ$3tS zuZuUjM9C&a$->ITF)0?gVBqA|^Bh3SaW0IPTHQ*T>za@nhBp{wD*z{S7-<@!nAVHH zRSOI>m0G>3RO^Nx(^_hAM%9qwZuD6>TunS&Fs03Xua^GybiO&=B+#wkUE<4;(OJ@YJi8h1IJWin;JsWuU)Fv2y}5Us)Oh9E zyIg7WvGw74Ys2L_nLOE^XFQypdHrz2jh(qY9b0!aywbD}C@^QdBG}@!PrVy3KVmK_ zLiEy5x&9TEwxFBUKHc~%E%wDg?D7)##f!|D|0#Az@_)q+pX0~>i6A~jkK3llEx?J8 zpiWd{nq8NcNYkW)2Mf>~-tK~{xn3eJDa&i7HdaPUf@ge}+$JUVen>uES^d>KvIZ`g zo~|s1P6l1q=NyNR)~c2HMgxh_d|~&8x{_TlbkX`MwrR#$0*4H5Xu5u$FC$q;5>qNN zCO+);x2?FJ0m4PC@P5DMTX$a?YbQp)Ei8M*UiyMNJC;7f`}22b^YtRvk213By4k`H7w8Z~PPW#)sa$^uE< z3-H?Tw#-h9t!-9f{ejcpq9X&RtI6;+RYWDZG*tQ{oemyGiH_UP*Pjx0U{*_hJp#Rj z4*fs%=l@46_6b~-puX_`aQ2pARc_zj@J6J&7u`q-i*BStT3WiKyHmQmyQCYWOJdQ2 zD4hZl(ny!Q_rksZXP@&v=lSq_;JVf|8TXi@$8W+t=87%U!u8EF{T*;39I`_}z+_cA zRcL0IIAQv7GPy8O1N?3H8;x+sY#ZXDN)OUWD(q*q zyW__8z-u4~SZPCDeDN5S4>Uh0tcY>D(C_RQ7iZI@m^LE#w)-$i+|w=bMMrDZSC;0V z^^B+v@HkzS7*~>E!k!gI>x~gdMt-P@27D#^Iw# z*62NpW!t+{8|`0|sS6F`VOssETt4AuP0ah0sVfZ!v1V(%PCBXI>IX=fTg=)>#~!jd ztz?U0IX?=&i|8Gq#~_naaaoya3F{2Inj#ll?dO;fT5vfLe7X6PcT!PDpT}(Lcs`^kQY1hQQ=6EC2+~eqNm#dCI2cn zp`yFh%cuO_01HmQDWqht4d*);AYs3FMFQ|2$Uw7vr^DvFEsmSyXw&=gzWf=Wlq#}3X|<;VLMa`KFw)Y0av{evgKnw)T67Tojp|^3B8xt z%zRL;uK(tGgy$-q(@tGN77y8Et0S6_l5eL!MoiP505iW9c9kEH=t1FpuIolY2Nj`+ z$v9HfKL*5xC)!Si66j)%yerqi9xKz+(EHa9j^6NLE-Cd4t!FPSo$Oy}$S_Y8OQzgz z3=ZlBlQ>y!zWT9BcNCx5hKW_)W{6!}(pxqiC=-+V{&l8IOjW#A;p-?ySASW|^ZeKs zVDvz$2@vK~MjKB4xIF#Bol4X$#!jFBMUtP^ z@$rlyoDQ3@#diFm+!QF(suDXl6Vi3=;b z>C+te&$B*d!z5JMn@m^T#mH@B?#r4NIgIQxpi8VOfsaXKXL-O@vjp}FCuWH+es-|( z8)#$Ju#<$F`r$^W`7U*v6;?c5`C z8)>&uZcokgAFvQkTd3dCXi7WXe|bZ;V+JJqSl+i8iOHZ8WwtIo{rarW!rnipcl{(X ze?Q#-z!pURW9~Wd{t@sJ(*OR3(*L#DJe9~l%_dpK|2VO4W z$tdofs`z{wU-&a|1(l#=#*K1nzi1yu96?A>sl8%Hg%z4@PTRTZCUFn~{AQpl z14+6($!!rF71l@nwWn8i+^ZGhdVzMq{eOKo^nYm=Pj82&=qlJR$>4Xg);)@T@uhZ4 zvPld$?2V5!scHLUSWo^Al4@dPGZvpba(|y-D*3sF@oUal%~{E<>uE%7lw^bF=~&*R zQj$n&)}hsd6CW3O;!o2Xr2)7Gj`e$;b6h3O z3fmbwRgsfvxR2f39xa{yGP_DOZS>JOxO8cWtDHML-`ntOdbCIQo%H$Fy3=jt&3wj( z^og36vCfb7P50f04?=t-qiGknSN$T3cVm`L?KOJ-Ywu$+=Rsm^GR( zlUn#jfA^;7>$kG=QYo`il?A1kQKh0^c|==^Ma!PaQ6TkfDZCvOI%73YBK2J9lw*tv z#h{9_d}e?DWLdxa(GevT&Yi~oJTh+c<+!u{`EigKq<0cJkea zFcH6%$sUogmTx1M4o3!h_l_yLT6}s2{U~BO^@gz+Z}CA>$K}@qyD%*vd?(%+C}ChqAwCWEg?f zjdcYijG5`SyLw-Z$(E(_ZsWu1aVwAC>yGP^V0?6_W>YSsd(jS;-FZf0UH;2B%~`*K z_HW|UCt_jc*X}jNI72~tHEi=+vAY|i0`6L`C-$+m2y6IQXRyy!4fD5X)LxS~M+Hov zO*in)HDC_GMrl;D)nRhN&0+6O@Wlk6Uk}8SNM2^XE8N4^(u2xf5e1bS4|axSa)w26 z%h0omFloo(mBNV2sr(kFnjQpxPAxa0;;peO_zQBomL7@o*bjDIj1H&#)tR={6kai1 zGdy_Bg}nzaXQGOhOr$xz+!0#k%1JzY5@0MM3T_D80@|$z_?0Zqq!ed{iBSS`VJtYC}kK)D&eRZQ`%YnzF|Tq zCtOQBOGd5?zDO7I;R(Xy9n>twcd|%u^t80NH4$>3BU-1e*gGF7=r&c8rix@LW!{vJ z?5lj>5VxE3=y(^rBWKyAOrgUXTwbW^O!cxUGg>1ak=SAwLG^?Bhw^DtIEJqolpGeF z#GHM5#U3y$uiC^#EQ0$BRi$~w__QO7V^4xZ^4?x(Tzq_qr~nWk`i+glzbaXRXyCFLaQVLwZyAUg?y}TrlK8kWQn*}R6hBl zs@PYfLS#FD^TS|zgC!d zA}^-fkxrOE=M!1-{!Qnj5jvq1_~k}>y}+Rf6R_N~pexf_h^elX-^(G4ZR6bW(;_}U zZXVP7Djq)#P%R08IZ|H;-(P4cnK+K+Gp4O-Q6sI>Kb|vuW&EmUNIC$;lk@mQBVF8vk#vrqSD)zd=?}|d>JKmcLu%0u72y9& znrR2;WYKOLjD!2+W}Qmv?0ni6G6@<9C8|j{A7tKzW#?a94m1e78mp$z8jyNlHnD*; zPl)~NjNm4fwc)hA1RwPJYLGTuKf7F!%Ri_)09k?kf?%9M|9*zSpibu=jTot~^uBZ% zrg-&XJwCCgLH?AgToDqb?}5>lK;c7vkSy>a8C`j=>tjDvbZKVbmvXvS9u-K4ZXYu% z+}(Kv5m{Ty<9U2?)tN9?_!1~MrJn`A9xZj0_n^?9P}R-YR~wEf+F(+Sts)$#k_J;+ z*P=4KbaEYKYNQQf6?M@;!K^2P1Xn#!Wp(!qGa}~Q9J4k&6MNU4Zgnz##pK9n=l=b$ zR40JMOC3q9k@3~Nb+mEc+{(RG&XiS7gf}6&rZrOEjL!S;3tl?YLxS^Y{BwD&i)XCw zX%O>zP%s;zyu0=%zPy^vsGEnYiNOxGF8T|+uOdIZ$fO>sUo{aNz+5Lpi#R6*V?0}w z9`AZAr1+7`m5!$5lU*{!0d)10KiQLXR3?m_5)9{B^h||EYWE9tGxB`X`_@4H3_MJo zDL6z;7noi=Jq50hh41uJ+3}xs3cN*$E*3CG4(9RMMwSp6q^A=^uazfU-KycBM0IhN z4Lm7)6%g*)iMpj9>XcZZiwL_qZb^eUNeCx4yO|!}lva>!N>9-Dp2vp>N$jfl)c`+0 z$12V@qWm&Az;0RwQ`qqhg)u3sc-wy{rHJ7Ah^d|7sAf;ulf4N?Wka1PQ9aH+Bmj#1 zS-e_ZdO;$O%vV@77HB&JYpZ_)F9r*M!!2rgj8r3T_+CRo?w!M{KH%uCY#Ii znCB`y$ba0_rL%n<-#!tdQ$THji&*=02RpcGvk82)n4ZMFEmvJej0fdPJOo}J#Hfvl z{W!(Y)CbscaSy8?2Fi6#N_ufOS;vS+4%4mZIpw~Gt&2V}sb$d{tB@6X(Rsi<0J|wq zTbB5K8vCr0@?C{~7dIFSOi2 zu%P~z{7?=WmAEZx+}IK;peJLx+GB87n)F(wY|M^H1(p$0e{>|lX}Da}is#0!(qK5M zvB5=j*PMV0*B-JqS#PSRuw~Y&R`vO{@*mA?=k6cRr+JXe;oK9!pHw2yz0&pUDQHe z=zmC!dA9C1*7o0dE+;MgM=2$958EmkVkS!Ok7=rbyVQjLIN{G?t9v~(9iVx=sO9+& zWxrBj9q}JaZ8gOLu|p-;^8bx>}YtP2|&r+-ZHYf&qKTSM&9 zVNUPgzWRHkpz43?r12ppC0=NGYxW;O#b*D*GOFm+`ODZRm8!f9%%8TO>frs`)1d3s z1cIUdwNG~-D#i11n25s9{m&$)*I;0-R>Uw!Jjib)E_JqJ-~ELS2!x`-$X zkKe!dJ}#m}`ne12*Q$vZ>CGi3w+7SYn0FzZ@z8$Vem`e?=6cbP>Ozr=HV{W*?u*@c_U@tf&%stAf4(sQ?XSO(4ga4Hwoiw-KL^`y zV->)iJ@3~wks5g{_u=r0vif0K3NhG9zM(YB4ogh6h`{hj*L@$X5d&>cwv3p&9}74Z zeX4FGa>@vxu-08z)an0uEubCvJ-m5fLi_6pMd*r~b3VF>&U(oOvub}sD4kS~Mx=tc zP=iaZO{Ut*bI>>F*|yD4p<=ip1o@HpWbvP~Ep=#SAv}ZnfV1?y;ICr&Q%*mR+aH#i z>eU70?`nh=uBA@K1HCX9LMba%D8GryB*Y+Rpo#xdp*%B^IpqrXx`9~|f!>@p4PVng z2^@1}{;$_W{V&Jdr`umra#k9aq#BBkUl^~77Tj6ZSA6P0UwVNqGZxH4w=pnquUmO& z`*tdID}H)t=QPawEq>IEuOz-*d&}3T2vcnJ2sj!IzYF`vi=#P$$D8Z*&84dz-v{;w z+lyb2O?#2f$J0Z55&wr@kN3xOk0w{Y*1B(w9}g=89uY{C9r`Pu|4i{mEHnT#vtp&)Mc#*y_VAW35J( z=J9Ih!{JsQ+3(*4Lqm6YeqML|o2L&Lk6u^TOQj+?J|6cMpG%!v$^AX=?uOFVd@eSZ z&iuV2A0c5wdEI_D$3KtH;@w#de9vai>K~7nhD3xl{ChlZ86Q0!f9EMZru*IBZtWr6 zBG1oF8r=_t5r_!?KHoOz@w+uRe&iMEez-|f+TpBzyu2?UUwyniz415b^u9~(N@HE^ zx~bg(=2H&i^89XY&9sGR&%^wE0cBPm@4lV??C&8H@xPo@+M3K8jS^ND_OD>Roj$pF z(_g4mE#hz0wfiP(GH=o6X1Zka%@Xj*??!F!div6hk^_9<2?HTn}?qt{PS+H5zPf&#v`n`@Ip zB6qHUy}+ z=fZKYmaGRP29AhMg|P0d2ZBtQXv-hmN{H+xlA)*;0>rW;S zmbaNM;_Y!UcPu`1{l%Jp^`I}~BBn@45=MXlBU2Ol&eP*mPZ$VF#pVAcD6w+5>kuyv*XG%?)W&P)u{IDtNt19_U>K`DJ7w{qVDhO#EXv{ks5T|1nD5*2Z$9!`O( z!OE;>BU~&?by*k}ck-C&%MKdNY|jh%;qJEBAL6_!B4(UC*V&X4r=Bb(j_h% z&`ojy&*ca25nW~T%TVg(8D9-I;I;Rrb!5U-W9eGoWrH06#2mzdHw2>EXIsQ^EaQEn-1Fq9!4INk#5Leg$TM8cVX+8N zf2<|%itr+Mhp&a#sN12BIoT!%t^d_LTQH6tmD0%f;B9YW&&X@@nqDL$XzZ+LZ<;vymW8cAlWFq*j8SHPn8CsCvYLpzi6SIMsErEg0%J;+Y z00EXd*sLLc3L9VK;;%7Xo%34VT?4F!kY?b;+SjpE7zCTMx+e}QdP7IFj-LnI1a$}2 zU!0Ds4JiqIVo$@LV047RGzn|ftoK@%65ZUQs&W^O-y553|=Y`Rsv2tXQi-wROQ zFsbj9In>T~@iAHuzfmy8drK9!K`;F9pn=gGHF9hi1ZRT@4VoEjm#V10Q~BBm%L|`p zTIJRLyoNh{A4Yxgd$NlC>(C{#VQVv%GXp+sVs0!B{*3OAs7o`#b39s_jSkp0R@ zG-b^PO2U9g0wOH^9vvQO(%BL`>|_)#iKu(vNCx*YLJ18${Dou2>!bLZj%NkyMhzG`+5CggDr3V|C=O)$;C5_?O-8_4W4x*Fe>cpf17Z#7XtEm>ORf<3MO=)U%Nx z2L>MEMJ>z^c7@TuXY3Kzdb^&;Kb7d{Q>tC>h;UZl01I@MLO$K_?N|6s;`Ysf8|v$L zWu9cxVwcsou~%;)JV|K!P*aAs*80Z90TBaK=xGgB=0vW;FP^}VjkoKr=Q!_4wd5J8 zeQl(DNh<5hU!Ns!6Q(!#sF}~c6vp(2aHE^bcFeyKW<<0Pa^w7fo*x4wg9L)@pJYfF zAB3{SU`gfG)!KN4{4t8=CgLmzvc*@wc`KA%es+n`=yg`606%uoDYA;QHYIv;fk1k$(>FopuRb7a<4Lz~7cw^+e~7eij89?<`C4D$|S*3%23ReTl%F9B0xZ zVos6_?+h&NWw+O1wskhL1crupUua@pKBH8$kmXcL;0MuN>h+`+0iLjg*335e*Fk1A zEC4X5`xxU;=uv5+hUtbtsaS7t+uCa)a1paouvt&~2d8kLC_;sBZ1^0K1pIjz^7|0* zHEA#}WMZetL(Dy%j0M|}kH{G<6&_kJt?M|owiTj^Z5bgdl(M@<+{y>tLN>uLMlX(b zFQBvT#c+g<|*<*Y<8zP{qA}-3OWjw;m!!|Lc;vLlb_BJS7=t}XnUg|5I8o4^XaBa-&U@YUt+xd{ zp#`icLi;Ax3jRk$y`iOd8<`Ps)j?(rm&xfOt!mUo8$Yh*39Ei1M$%AGB;}+Ve#i1r%Mtb~@Ov)UM30S_B-J54# zr=Q0I2YcBqqrr{&j&S#90QT`Zn6!fsq~$bGS*Uq8kXOboH)_LSS>wxJs%`~fw(LGyrx-wxP8 z8GkJkY6lqGx7rJY=kPvT`Vs^YI00M-wH8nmE1`Z^)sPDjyofEJRycoji zgau9_niY_7pI@JTHVs;QGVemZ8X0*iC z9qT!KIh7BdiyTCMpu8hdu;_Nwp6AGp|AxeuTNfnrWY-S-gk`<(ABq-lb+3y89ndVY zQ8OUlL@r{YKB2EZ!Q~jRIesTl)aN#0?ojNPQEa1fPJsrEx7HoTbJR-=RP-M>Yb4i* ztBC*2TZ#7m>oML1w)NV6fVqAwB_*x@^1@iOs4-c)v3C5d8w3E`2=Z;Bk62K9?c9nH zUx_Ul-`$r{$b%0vo>S&3BVSn5;{tu(k4+ZZ_s@f5``C})#-UviEVdFXhSqq)L6|y$ zBjpg&Zaaa}coQj_iK#&P4>Eb|r=yEuV%>Pt7EnU~>e*0XA1HN~Ah!hhZ?cb5K1CP? zKmzB-K+`M(&NdiYPBXgV2fCEm&)1m{1*8yUais+#4h`qo_+>W8Xr8b{g($XB##MD` z=p72}G75FjbVp#Fkqc_nd=OTJ^#M9*H0GbYjerB%>}pFc))DKyJ*y2$&JA+kk)ay$ z3osdW#qo8#GIVS2Kec0*jQOnlBHqED(%pMq1Jg}Y*+-`Jg<~Iar?-QED=9g4Bv2Wb z(V;5Ue}4+@3Zl(gXT8sP1kG?}?r@NuP;b?8--A<;Bfg+j9~1xG3^9y$1ZXk0naJ7s z?rG^~ODjTnDaNWk`JLK$XImU`R9o735dK%r45ksmbV`Lrh#&RY4DM};GT7xv z0NL49(^ThfKEuyP3NoJM9X*S8mFS6H6lTe;ndliWmT4aKAhlhYan3|(z8mR=%R{`b zcZvy6t>)i*-=6U;j+?nCqlyMcUf`oRMO?VnLTfC0K zxXHR!^z*;cAxsGI2hKi1@=1fE2hM>)@{RCo6MkF+ezeE1HOAx@^Y0~zT+pmvhM-Sa zQoKbL+C0?Ew#WGPq2O%jud@%9v!+e{(N-l7a;qTS0^$ysC{ne#IyaVWQ|~MCj7!mri_=T#dE+|9p-NL$aYg10a*IrsTk~IU zARb9<&FX@%gYeA4Q5~U86L%65l}9FJr@aV(&AW5%BG2j8Z7Szw zMYs)574NtXYAYw^P5gX9AqV>~b^F6GG44Duu^@T z-ljv9ypLR@+f1%6_p|xrFvdb?VERkec17ffDcoOHxST~ksjcBh(o!Gi+w;R8n#2w%)8FX!Az&Zu}q^!>P z0A=}ZZ$#TDvsRw+0#E$26xvkZ>U+l^tk8Bb;yVuJ+%2!a?AU~G@s0q?5;46|2t@{r zfLva4ZNW_i%KA10AH5_?K>qnh(MBE*1*GFgh7E4X2&`nTtke_FNxX~kK?ake-#`d> znHS_%&R;_#S7INg)%!eN32H0LDtWH97xHH*3?JH!GZ#Rlw+qB?>BCV*-*4NFSRMQfouE&Gh#t=oW!RB;}1^E`2PH$Z!U72eYTB|yTIR{DqsI{E?q6&Dbd85vX@g__97ivG~V`6nqaNU0D=q01Mt z%HlRYB1Ce=JxIv)np+7AdT79$C-o+L>rqF?@vll;6V zzZqTK7V_m9JW_T?$?Z>05~BY?R2I<4m9?2jFOMzMSDszaFMQ=2f+S@7XHfx-hx5$h zMy1gSQqsf_&DjFNc!49X*Zp(Icl#i~YPOKw6Y$8Vvw*pa$UeetiB&z3m^DmyqPR1`Uly3ignq%t~ckCKpL>kIOdXZvQoMjrqzLBEgMtH z*mr@8I%rjj>4SoFgj7Ox)S4=~RYFY{+<<9jG{yzUlH@JP^4Sl8k-ywUFJfHdS~(^< z1YHoodb^uF2Wd3#6CO5Ngn9VuhZBx z!zn3^m6hs7w=q)3y}XJ&tzTP(37M~KkF@+?l1Nw~Zi6eti`=OM)uml0MnWO|ijOWs zmH!pJwn{zmnFW1>uc%&iXe{0MBuSF|1Ihe-IWW;cN271uh8kFy%t(ogi*4V}aC$`(<|%ZeIfviWq=$OX)|NB@2K$85T( z^86<{K1s*w%juaVXr^d_XRPGeqpZUp+!27$=>J-#1D2Pq42QA{jNi^b1y8%q=L$K{ z04H6>0B0Qvbbv9?_~7pyYo^jo>!vV?mw|lyN1>!-?Cs|C4HD6IP#9Ir}Ed%0;mc zlnx#tNvCuhCLswS?&W?H3X}znH*dJS=!|Mcft^P`nd3^HMnA=8<&Mmm`=d z{20I_;o2MRr!_scgjTi-ENWLL>M4*c|nCrxy2(^?dP;Mzgg1q{VZR$SOm6=QHi-Z!B-^9k%84 zdBj=V#u;6sZqU%Qmeu<>~a$7?p;#siB(j>-zsLYtPDYW*ihJl#Zf9y?=( zzn02sw>2xQ5NUo4Rz36h{tZowW*R=}dPei?pu!4XZGNzd$O(ip_1@9!Y8v${MI)xA z-ih`hwZN3yJjwHQehC`rWr;jN%xPsUeD7F_k51_2T)ILV^Ez!DJSx{x@&Ksk*2}xcTRnR9;&)sPCH50p$ zlenWe7iwitzB1|apglurYmG2D7>20JFWjs1AheXJ!5#W4Tk+GDpFQtVnQ{tHQ?kKX z14@B?{;-4OMWqcZ@FBaE?eOav;4CY}KsPbi&~`eu2tOE{ifMcHAG(7!ay$&(=FS$M z>X=-27k#`Da0_d#h;UU0ZGOE(vft!It=$C47aDvDX3R)MO-KM|iOKcd@xm_^+RrOv z`#ioW&#rVz)t+d;rC7Fi#Z|h^HFhI*A@wfyPgM9wbCk$mi%X`UsY`gRGD8&p#o z`cyTlq}E-_;Za-6pVLjT(wn7tjelDUU$w&3Pq+H?%26IQ^l8=bpVbVjPYwZ#MHtRd zedGwYj2gRPSv2Gy6|jM)W=ovX7HNyy_v zQk}JQn~*mv4$IAfJ~pLE1kNA=FSzJwMc{?VEf-+{e2r^I(-^V`Hq1a8#-at-DDc2V zUb9sLDZ6s`y*T3{glMk9Ms<1osCSle*kh#1*J1!`?AQ24$s8+vetT>Or9wK)?yo3s zD^gXWXtOwxk?IRyg?oS0I(iR)02Vd|tZbRU1a-=4_iM&q-^te`VrTX7)QvFB5^R04 zj5w!nf<=p+L5zBFKVWvu9r+5V`Ezh;PQgDbe>kxTYf_cFRkD2K=KbO-q+3Ic!3_77s#mRm9_d z8D_Mzrvif3h_VVoHPA>2Y2rzy0qkI2QK>oe7tlz`piIf24E{)*vzFEZdjIH=3L!u| zrFkF5NWCB-qIHT3IV%Z!U{)p^Nl}&NPC;dc?H$fc4Yl>bGj1)}vN+_@OF_W;96qTd z4QtlvR7DT{`OSthEJ?eSz_68|pdHkaQ0Sm~A>+9Wg#t@H$-jn6s(QO*RCuGjJ_tS; zE1@V^n%fNRNFuUNYMOe>5$c-?qGG`Y2J7!BCerH>VsXhsq(DPvG< z6C~HljZ(iErOs-A*XMJC;(}8hime1DhB$eKIGa%-kF9y&8W16fbwILo%?CmD|1brb z<#OxW{{yL-&Ho^kTnA*Jgf_p#U&Fxu6;%OE6~LtcWDIcvPdyk=Q(YU@S{-O7LaJmy zNB~V-b(FfY0e-duxYNMITeV|83)n%0D(vD<%zTTKwA^dyG$pt807vKa>RITuu3(Pt)}TxEG_TpVWN3HHKoH z;vWg+KN2Lhis~r!o&@~^TGf#~O{Zq)Jv9H3K3=*eSdcwLuwmCKd7jeW;crcUBrs9N z$y3HLw-bdZoC9}108j}fSkQ^~M*;)DR4~MPGIDN%I?ClwIO-M0O5*iZ&&knHnEgFiqug^^5Kf=kXuR`}nDwEdml z%4fs>DuFzU8cnV>NIlI!{n< zsDkugi?&aTrwM_6uYrJbo1Ou(;;tr=)b(N^xU5wJ4brrzqInDy#XCmemsF|@aK5i2 z#nuuFo&aT{x#z9G;LAdKU=;wLhbl%u{i)YO$7k?Vo|RXv?lCX8S5#i zf~Q*kTqN`qTI6TvJjQ9jure(GMl(UECIA{`>P;{0!Rk>a=F1V5C=-^1 z%fA*~O9bvW^4{C~1M$^D7^tRHzW_Y{b=1H!iX3jlJ!gU5uk_EELXIn)>MJu;%)h}p za6FP@gaj+r_{K$wHZ~SGAS2On3+w}9Clj=q#f**e2OD}9Q_2jE#I7Je;(*gXV5h<$ zwK~XM701`;9EDMf78TH=FfC9l^lS06BKpDZ{6KYnMhLFF0~L~#Mo8J& zW@Y#UTXzIWbr)krA%zoNhljrjw27=M6|w#17D_P=n2;-X(Li?9hz!rIa(uC-qB{j$ z9hECr(?C|&h^(lplW^~i1+ZJ%bYx_Zig|JP-UuIaO(5C)lp*%RGklE-InCj|$gZoQ z*U>M5wdmj6^b!VmEn;MJ{6Lc}WPSzPEO{=JasXXcZ|IXNxqW%MvGWB%fCvB>$|H<~ z>0*TGg|{y)j@&@)9RS#5Ei+3&l`lY*2P@#O>S^HCRC!AnK2`ivc9BCdUh;&N4q&!z+ddQfZunl=h6K))KU0->b)~YzYcr%M z%YqZ)uT4Oq?^5$m)Sm}&jK$6hHilMkqAn+2{SXRFw6>RA?_|gI%K*@cYlRNDP9q$B z^d`7Bbc^jktE2)zC)1H~o7$kS2R*x1LB4uWET$>IjpvIbQ(VcPCmM~hE| zL_ttc3Mo|8W>C{+pz=yo^)cD}@?J6q6eN3&Vz-K>UQX#AmlQPL~eKmT-Fc3 zmIAXU*n-R!g3KnlrA2YJ_TFgblKvB5jor${4my^%#A|_=ez2kHOaK}bEw7%o&iyTQ zt|7j8M;r!^S~)vl0eT2jeTy_J_z<&O9X`mzi;f1;u#MyY3^da2iYB#E21Wr`q(^|Z zSvqGM9q{RlPQ^Feq@0mcmRNCED4?0&@apPY(hmEw|ci}9Oa7+W>a>JT_xq9;FDou4a&pG(rrq>dgpglnqk%@epo zCUmum92xxX>m{Cp6URSGtp+Ry*6x8qtRi;@v-`zJok4qAacY0;VUk?IP)O1^xGxbZ zH~#_1lJ5k`^>URZ%9SPQE~v+K49M>DULUQVf)}qyLjn8~B#fnu%jgiN@tO3C>p{xu zYjAZ+>mhL{m%K7>$nzi~b7||@I=SD);pd~m8l_XpJ4vJ>G5se?=N5fWA*~L!>cdU2 z;0p^8`_85oNT!|+)J~WkwXn3tbm`_CeFStzg>Ng4w5Xn(Z`MI z&LDca(<&nNHf~n;OUk?~RzZTV&kuGYMl)0`i%+kDy>OL}-uDqygk$0NaLZ zy#%UEvb$?}pceA|yyiyl@0um6PY{5fMhZpi6Q~x+cj_FJ#*JDTW~q6ACQ?|748J86 zS>~VJ--m-8wvHql_k9?e5me&1Bn*X*ltB`BQsu7Dq#jnL?J{3Hc!mKKdemXM5pUKwJU#HHiKMoW2Tg zchAR|%M%${zXrf&XS}-yS%JvjLT)wP>ClKMow!+@Y4PF=aJW7&``9PU$SnF^*(kfg zknP6tb_}A-V2P^s6(liFRsCf=pH1F`7pTz3wFKhD#bD*vCYY8{!KWBu@XQ(lfSwP_ z@5*o);NJv&DxY?~8duf@T^;BCoLWRN(e%YzTHmye0{(3hQcx#zh)O4-=-&lmT!;*4 zzzZsqp_xdu_;l^#goIHLiM z>ir{z*uElngp8+=0ULNh7P2&o$qpcWe)e>UwMoBFxo+cp9|(mNb_5445nI(F84SQ0 zNQw!1LvVPm#D-eLQ@QHffR?Ki4mmBObXF-)o$IpuODzRW?$=F<40;*)Vz!mnO0jDCkUKh|toDnEjC$O)a7L={y_HnyhT~-{GYAjyE zF^(K>@FxWts7FN!qD27$|EMrF)rht!{_9MXi0z>5S<)_nTD08J2DJDW|Aiw4T`bY> zeOg`rdzC|sB$Sd8Xwp#93FG{12*nI?4V z=WC)9`o8yDr>EaeKbP_VqXBnWdE}2fSr{Tc4~Lsk{+*BKKOZ{IhJIg8cm4Wyca=vb zBKWZ1iE;nydV7fV@%rMlRB5s-ja9f4m;&he{rmj))nuN~i|^mE{O@nC$?b)GZnYjB zHu`~YxVXL;!}u2W6WO_bC$x+6K)k=(Zk`A2uGwgf>)bZYpm%eJ_UYFdlvMiPrA@ zp%Wp01xmn*$i*1<{RX0rEATOkz!>W1T z=Qo=&H*xwKh;yTzi}B>7fW_tzema536%D?n`~1R&KW0oR&5)a)=_b_GET0=Y>sO?u z)R~)~5~!I@3F0TkU=>_!wL^Mf3p4%fP@w7?eO5GEaOBxQdBM@)T6{5Px_R5~;?;c7 zaj5ksHO1>Dqu_`m$@xwB*76U=w)x^st#ZzpsqPpOI}FCJTt0f;6ZdIKcg>_$e!nh; z244OabBIzWdT$#4fkGSGhc4tK1CtKiKm$j&Fq{aIR~=2XVH?p%{V+fypqqC(Y)1K z)=WHPs$m+L#cN41U5Xr&YcPWZ$x;fWe{cJ6-P0zq_OL+q=j()ADeD$gfNzzd0KO{g zf4{8bf0@~N`ao*`a@x!-&!T zG#CC&6CYdX92?`20dt}B>!QggE|$glax70NE{Htp!o1B(m^_npbKUUNr?Fj6cr6d( z@h-2@Ke^xFq!sz1??xoFqRI-=B6y1&+0rf?Ypb=@h3a{+ul~^ zZ7;3V?%z#qn{}nHb-Q|bk(h1PlM8)Lo9w^(|2TUKpt`zcT@(m}pdq-sYl6E=@Zj$5 z!QI`1TX5IlZows3aCd^c>z#}Id!KXa-nXmXsY+2W^UWUZJ({uT@u9s!p-**rna&11 zZhGXVu=nEH^V`*iZToUyi>qq8We|o*!|-y)(I!C|hV!3posn6!g!f_?CN{y(t9ED2 z9Cxmjd~r6V(*zGX3yT-)F40jTZEoQkH&<&;wrnrYjZZiGS8i3U-gC#*@!qZn4mEB| ztxwLUwM!H>>@KMy9;W>d9xEQJmMXF@-Zl#r9Sn$`tL{d@+Vh2O^Fj;{ z#oEW?!=5E7Ub_k2B}sgj=l$Be*($cXO1SPR0ovQyDz1l0UQ5NC4kc0Pn$NZjOwX6xF%xB_-YaznHSxLR8OKO_UBe_Wp@_udgtA)I8{$5wynpX-gg{Fb)u-dGa+7-*boKYJpomD-O2vOVlO-=n zP7aF@kT-#VQNuf`__pXtZ+1mo}+9-!iy+QNhLh;(}Qa_aT0>xD&-(6U6bofZo_Bc~P2Zh`9zER+` zU9+$>bw!e(xK7arK69Tpi8r}71tU;f7eJWW=lNYGcrX4(PDO)&usB&>CZa;EB$L zvjjBT{#(~V{Wmq z_{T47;fh_^HTYZ*)M!y94wysl<{5GS#MI^qCNM49aAX_qEHZDDk9)@h0Kt zlMjp8$J`ufA@v=Gm?J%<>B_@m;on}T$`2pbl8qjR)}fGh=ipDp4pov0D~e5^k;qL? zi!miz@nV}SeyOj@SCUhxubPclH!h^Jt@r0*>dKghpG8XT5()T_*l!$f!_1N7=AbDS zGQrVeI9vq4<(&pohME zls!8qa#gNe%e4FK6RAQ?Jis^A?rsjiTS3u%?F9f-e4Z>E1T{LB`syk=Pc4}sBo{>;{845G>leLx zz9f%Z(7g1i6Z17oT`jf55d9sOc}fa>u!&zU(@;By2V@yl@VpZJd;4J$+x%~0xh!IS zS4>+^AU!D54?ctRfW?o3ndxtUUYywR$g$vs}HriVQNrh%R8{z9vMGkRu(qB@$t20HjXiRP4Fx5=&sO))x~lm{dG zH|UEY!dtS~j~(Qy1>$K#W0mX_#vhYVm%(6irp9_~q5lG` zm&M{Nu}_+Y4H^1TUNN1KXA??92@g^=QJuC=p{^wPO!8c{v)hfzi!$3|o5LlZrBmYq_T>PV1}T#`kIy zvX!LNt_S%qxX#U?_5wn9DeQ&um3a^H@(w!WyGDF~-7bH}XmjQV+=Z}r#a z|9YmCCBR_+dsuT9V;&Y}omLJ>ba@O=bKNg-x(ie|3Zh?iew6zT<-h!QwgAP4etDNd z1U?a@aaQzUOaQlJ7=By8$BvA&k**{W^?1e_a{ElWUzp#6=Qj%#-#mnLYCKI{_S0r+ z7eb`pM&l$4)%(==IPz^lE8OE9Yv#ZcKB!*XLJN6(W-1Elp}+<9C5mNJSc30>+sVx4 zahuA^q{f$y6yM5cO`v+269UT8iaXB|=P;8W#a=%f)F2^hra?gkjO9EEEpndsS=)sv zSfdR`WJ0?xG>h`si#&?UG?qg8e1nFiEI!i5i6VFp9W^+eUV;~sUH0uLoW;&p3in(p zT0anT2;bl9?+88gVU0q3AZ6NraJ&KC@UOb`eDluQvxB%NMdgz~tzpP>a+uttmjl4k+< zb}GAh$nb-d91+`oP6rCAMR~v%3pC**l4Wu)iWVRTa6gq@n?zs(tN0F099%C3^&PsnV^IaM+S?1c^NmdB z*x4|<#<=PUR8RagUGe*rTTMUn1VeF&v(wQDXZF-C#@G0Wj~6JCaC1|E-Lt!0mIOG9 z194_l3}3AnOPe6^^>i^=R6&&-^zShIA`5nFt|9dR!!-J5CX83`&H7#<>NC7Ul^ZS)h$|9b z*J+jAbR1Reck6q(KW^LFHhAU+`CmP((t13&wzzRzE~h^@Js;VgJ$SS&e}Cjw_10R+ zct3ko9-2I!;aIyE>CGWfkT3MG+kSrtRa`+oVFP*+}dgWgQdd^~kjps55-|d?F*|0b+#WYQaRa0b-gw z8KFcqB57=Slfbgy-^ghU(}T*M5s0;U3yg!+ z%N`i~>akk#^d9vDrxgR~bFqQ}@*5frmbyBEG2u_@xv>z8DVcm(Z}T2>e+@@uII#V> z=DFEi+Z`TTp1FW^~2kT_2XjEu9Gc+uCC9r>^Ny?8pi?_0rC) zsbT4~am!?=9XSc4ORH5GlmW>m)>cY z6_S~e-kgF2YW{@ z7qE^oI}MI?zu2T1HQ}dSXI4NtTXiT~^&MsS4t1EP zYDu;#h+?Kv58;QdD@EdL#y0_87mQtww^HsWs0;=u8PYmg$Ma;P#*! zZ2}3M?wTkT*P^wDJSp;AZg-~hW=fig`|jEOJ0%2+0=cO=KCGu?c+r{YR5w8>tnB6x zF40~3jf@XY7-yaD3d55>bsZ7oV|_X#78G^S67G4|5q=x@K>*_u2JRaL1MuI?=1MhF zQVG@t5={NJx&aCwApKjppo?=6=w(uP#Z^z3NU8lfn3QKuPsLCXKOt3xbcrLy{#4_i zzoi|wowLTR&nNObR)lE!*x_8i_7T%E_h-(>vN|j{6~g2ev3TGO7)H}ib~~6MNBBbN zeP6_|k!hcb-}y~@|M9Ra@(&8@>P)Trs6-1Jj2WUDa40;A`=)Brpm1?+PvT7%9zWzd z;+l5Vo5tP3q6jW98WloSp=e4xeL`fzak;HupO9L3q7ZS!g1f#2xj=*wazh3aVC*6x zpatq-8`DZbg>@0VB}OOWiec*K=RT~Ov^c!X17CN;5NKR!qk)|g;uA5_JS~&*%J~}N z8zM3w3!>=G<8>dI+0IO;c}n~aqt!bwllPvAdehIu=o}?b2ws?0(2Lk4%6IPMjLGy7 z9v}zK;LFXu_C_!5LRYr&{rDF-Yz%J6xNJ&XfXj0sA<){mmzX7QtzJHU3ks0b7F|z0 z$+{1#ed{Ql&}%ynw_!+Q7K|DSiD#FnY9H3~C#IswWF%#roE^Wy8!HfpA9~uJe;V^t zd8&_&f85wcjw|mo>277iK&k>MZ?r?3RSOl_6%+n36!B312){T2nwum-LRln+mq1!u zenuE^Ojh>_v4u>G>Q$Xw$x&|n)X*(<8=4%e8;v7(yZ^B~8(HaOMd$t@eVqboXddmL z)_xHsa;8}GalLdx>^*j-6VwNh7QgYPZw*tbgA27ZniJ|VPM>2w55=_RUxyJ_?ORkT zF6vPg_n(=mAbq*^(GC|cM{<*tPRx=P8Sb*Aty2ZaLJq3#S5qRFw)0BXrTcEQa2!EH zKq9oD@NZEGb^e6jlWU&dL(FN5qG?B(gfgP`iLds=4Gw|*N*U3W;?)?jhnU|6L(_UF zo$!@I2@%hc7?z+tcy~r9M<0j_(wyCS&6Ye0Q^(+}T9R4){lM68#6zorwT+kCjV5oY%}Va3lPXQg zkm%IGh1GO=aBJ#Ey~y0fGbTG?A6;u0A_~iBYqUgK&Pu9&RF_Fz+-oQ~9Wbl0>eTO* zaA!%8@I8O@xYaafps;4SW^`RBSdDF^9p%NjJ2K{Ay)Z52OlGZc-?(1y2*;sp0RG>CGDtD{?dqOSup(q>@?nL4oha~n}{kg3Jp8q+aR^tCX zp*GgY{_liZkzyj$ZOk6<|2Cm^UXa`+3|e;eyOYv1_5!E<-e4cw+xN6ZH^gyc#KRq7 zQhZ^*YBEEMlDz*ogET@xQ?LEMo>0s7TG!P7tFED+;55C~wKsh+3FWB^3(+Rz`n95D zP=<_r-d|uo-~3EY!s6MRycx$GYm1EHds;eq%lEr=&Nw(U`&i208$N!`^z|w8r{^b= z=U@|$)aU2LTbAJtj5KCCXr->~WX!Z`k9GpY@P6rL*|3&F4jx5sHIqADs?vtL$x4^e zNkd18g_EEKX0dM1Y936Jpod5FD9g4L-h!ILGowuWAtv#RJ~n<^tLV6IZ_rT5}xU84&RyY^rM?Sq#GN;!FuYXBk)7oXA@3qjd$$Mb5M zmr-quy=yZCWoi!R!n}S2TTd3Jkx*4P0c-v8p>R(=PsV|ah zM|n*4I3>L)hf8Y8-W$z3CO&1;7LMBS#?{sMemZ=#UJ`%S66g|^9okPwI*^=h>X8|Vc@z#>gub|qY1w4EFm~#cg^iqsBPosQ zc*j#1dD^;!$fjSu{8G~=tapQ~!x6CBi(MnZ{7eX~`2Oeys!k5LXA;Ute_xgqKL5qDsEk-FN(lnyR@PQ_w|UnY#T3e;O-a-vV4OhSF<_CDB8 zg>c071O!67<=~}6iz?E7-|Wl^j8k6K5GY=V%&IYhi+Yc(FTz}2a&IuCHb*Ba3Fc!mBOm~Eqo!6cQ!oN z2LM;$-e&!BrTRT9<_VPhtDN@anjZeco88i1GYZMS$=1b{1m_XWmjtfTm7=qbdlCk# zETHUS6B;pqs)sM1Se=L(%0QxJ%K)FIi-(I6U5-42A7iWF*H~0eY$alSsOpgOhVf&T(5B^>nUB1jDXNdeuQGeX_dFEq5h@^MEwpGgy> zk8Zfp_#=|8^}EhgjFCUH^90|TnH%r%{a zT3cdN;+l$WZ7MB#+Oj^Ao>tZy&V%Sq7O2k_KrV`kQUUx1%eJCw5Y3>MkNN*9|FVeF ziVb1{hP#aJjwP(8WUQ)qMuC#I*8)}9XN?d63TZ?=sCLMi0IxyJZsd=iTRf?5tzQ~; z7~x#K$wCHlP6_Gqk21nd+2lsJFs(d%!Q_W>78!DO zpP68Thq@i5ZsZr}BsV`I)>Z~m;)oSR$g8sDHXfHITQ46uW!a8HrS{kK`3ZgyE~oP; z+6Nv_A7xFSyjO?mT7G%XLL;1d>XP?dwyB$(i_MV9(t{C>mtExvrL-gg7h&7tEO4hg z8m~7s)%bk@PC)P!M(H8Qgz`o6MoFsS*r{5Bd7e(sRHN|)hLC{3bBvU{bkxHPS|sR)u6P_UIe(d;YST+ZD6|7=k`(F|tR2kW2G(F|Ma|5rpyUfX;6SC)& zS2`y$FNNx9Ra{RQNYN}gWr&!o&W!i%CP`G8neiu%#!1I51cDnh2Z_2en^=BEQ)5+G zuLD1kt}0~?&`R)U*EOXy|0XZOebqB&kTHxZyV+7`li}pJjJ3dSk2TQu-p>Sm%H;DW zM5<@llqjJB@4#PuL^YWLQ;YpH1fZ&|LWQ!_2Xvzy|C_n;?5|b(yOupR2YS#+gZayw z-lwxKtrmF60mY35@Ta*xgB$VX_oG#9(mf! z;LXc(-A*@?w!ME>t>cM#?H>MTaeDduZrJFhZU`>L#O;@j++*JaiyaB zVyF70a=G0D7oN4@eR$#BPWQgiS?fd6!}Psb`|?xsGvi!q8OBA+tPF54=*E8OUwC)k z^upTJ?>LBA`tY!Qq~F^7bYJVT(vDEA%{|)C@|ErVaFX=P{)TnaLxbmI+ttQ%^^3)o z?aa*6&*|QS?3V^jkJaywON*I<%Rz1)LN?;b|< z@3!??hGQR27xtX48eN}kPRyEQwL5N@UoNg*R+F3;gNNI(jxSsv9BZ_Ao?gn$TArKP zFFg;G2;8Pm_KgtRezm-w;Z=an@Y+iMp5C=*so3vMr=}thn(}Ir|@b-#R)! zF5|9Tx~DxYbu`~Pv9w}(EIRWbk&u_f?G~350#agi*GdU%I~p}x{Ytt%MlNS zK_Mc09u!VPR*3oZd0&d~Q>p*55~p^_y-TtA@g{X6Ixqne_IxUxaoX2cD)ONFoj(D_ zXMDVh3zle2AwTbKUz7waKQo}I;y2egHZ=8}Zt9}Ly zWv?@8UW~_o`GhDnfWv(){I7$v6egl8&wXTSKJh=(!6Mlm$F4~_2#xnFpa98N%;?eEb4_r)}AKV|wFrnLj2En`+-0OKgz+>98 zjM}rDf>}RO5_EC5Vo(2Q(uYu>fC!JVo(f&yu+hmj_iJGC`?6c*e^7x=&3;d1(2n=W|HNvIfgM^zSJ1GXQ@z*E0^9WXO*G*a z8QnKnK{-FiC?}^-{m&7eS@kAzseCE!=EwQioKV`CPw`gxp*7`EGrHAq%GzjjZAC^o zeg%e;`{C-w(@j?@Xc)Q)>fl(K_@>ml)zjF&)(iwi!as5}_qtbMrq_fUtbQ352F}p( z12JCtIqaJj?2TSvDNrV?ZA3#F(HGSp%scDtKMKoqIpgI+mXfw%A`SS#MD0Crhjagy z#l$b-eH9vSaU``@b=euV6LQYr+}F<(2a(Kqw}h(elcPl&g*B-&q{un54gvv&4xb%s zy+_V_iwlra%^|vq$&r%ZXHI3y%%JXx3y;l6n6uA|##Sd=O6%1_1v)3lx{~uG#jypt z@ZnF>#&C|i3qVbT3xLZ_sgXx=-e>haEEOd>oA~QVh8ZIQ$6tx!n^87R<{n8*CDSbw zl6#4$+rV)a*KnS#n9$U1tf6A~ERz#^PvdkrW%$@G@T=w;DQstEu}B*d(K^W=nG9EQ5Kiq6O-UHD&%R%yUs&+~l3H@4Cz~R$Sf?-Z9Jk{Vkku@HG2yZ=b zPC<_m8!E_1N=gFh>-QP7QEn9c-oQJawWr)Tqy!7!h;u1y$M_f`EG!~u zkFjxaO5G_a>LDhxDLIiKMZ*AWT%}weZY2ZX5D*SEhUfzp(OcQ+BEphB2!!pF%bLX^ zB1!COL3sR9Il)UkE8idxPP0$Vn=Al8hc$crG|uIwqnj&3O71t02kz?z zPLAcvS2a@-m5IXWO>JA(PDCaynI@DM zVWSC8tFHBN$BE)Cn+`@Q&xe)B2`ENFS^ol9{!bu&-;%1%NLRoinkQm$$}5$!%oULU zxo|F|no%g4Ct`5YE0sF`a`timmHzHPq|98HHk`yefH_phCLAq?ToN|Us|;j7R@F$@ zoTB07-5eaerU@90^*zhl$9jrQLvX;Aa>;7HYs0&beIkm}1rrUCNE$ zPDuYtgx4iX^8d{Ww&Uv(dNhxuKjQvj?_bYK6H8)<{C{-!o0En$FGbRWc*a{Cf-nBJ zPONhytS`4j$Lbk{URQ<<$r*iBOUBY}ZxVWtY9w-(IVzHv*0CtT^6p?{hLzF($1~V2 z2IUB-9=W8J2{AS|yId0utVcmn3cwuTwt#?_6OjPPdF8hacFxE-=Cof8GQ@Lv=fkC~ zFD+?=Gz9*;gs;n-!Xc#x6k)?HC4)Q=ho@-*?r8lwK}Mia%ZrvU3J{04c_I#{Z5-cb zMCD&*q2O(%By6L(g{St84*E;#j}F>Qwzjd-Bd4p{gKNEx`bM4d`tYmghV({!^oy2F zhUoBzBbw=x^3~_C>4UMw>EY~?x_a#$HjB5FR$;-wVx<1py~FEP`Jdil1Qvn_bjB6aRJb^c9^)9A zWiji7`)sDvH;>KaB1R()3c8R@IkCF+=Fp>sa!;Brj-k$@w zjtb5)oeaTQU-d_#yZNUAv$>x!Jo8*^0`x)l=DQ@ z5=&1Pd~OQn(6GBIF#Xtdi~jZwbv= zr7RD(Qw$~&^=T&rZ2K-QlJbne)Bh;8JTeO9*D%)bWx=s4&37K~JsoVyGCaGwzua{% z7oKPK<7|oX1(e!Y4J-O+9w|@M*Inuq!cjE1Fb@%6CG0l0*9~hQh!i#&a z0{Sql+nq&#&55M`($i^o@QKapiswsobVDY-7lNJl!{kV}K~v_c_wC)qFx!t+PnSm< z*{hDmj;HmOB=3rLo|)jC?NYWMnf5Oy!HYV)C(mAImvYZe&NnxO-g(ZeZQR^V?S3cQ z_E_F+-e4;71h)#Q(Dqa{VUIdHQ>4??=Kin88 z-UN$Y=>Y%4uAp**7;=Mt%93g%PPzq&e{ zlZkFNT(bECG#e4_R|gz-9&bC^TpL|4?w^m7Fm7ZiwhhJ$`JNw>U(Rr!AJ2RL6xzRx z9C){Lw0d2Q6=v&j8@nkM;$vJE$?~lH&gHneh2vRJh?aD7D8yH}+?C})cY`m)55?as z?q}gCP>5DSh0DS3HBz{m-jus{E871NT(Xl~0hjqzx$u;T;aX;JmORj!V&2aOqj)J7ejU|eCL?B( zGep2^r%i%Kh%s0r!;YPv4Q5!bi$@uAYUaNtCvksJx}M>6y7d~{M{NhgunIl2iq{f? zmS(`huI~T4M6|sWZe2BR$da+}aeLU(d@y{lJ!>J6#|?cma!dnu|n&)^exg^-^mfG$V}&1bAKoMt7?$_70*gA`pYK1erL2(4n{CVyp9np znQ!aDIN0c+A=!%?D=xrxhh6^ZPTfckbnn#&NBPmZ#?K#(e?m{n2?Y(+ z6Z))D1%(>dh#$j`n5V&p*QcO@TA+PHnz8B(tD~}Q-0M4S!VOhp% zT?V;y|K?a@3hstZWLMP}kHt;1r8Rz+`)#L~YOK<{iqcjl@X(pxTY|Y)zlB16~-FU>?6GtT?%K_4lMI<1q}EQV{4~`gY_Diq@K7k z8DxK@QrH%`OqAk5q@qQ353kp^`L+#a1&kPT1o^97;G?I6Nt#=x|34oVUrQJ zrvb^0?E{+ZRvs<^a zTZe9Uh$F?cA81w(qHQr=D%Q`e!2ctptuwZ4s6F)N46{KZL6KYR$Cc=hx)=kLlZ9oP zmAYAxOLc#37B`%e1v_|Rj8>VoRhe074j|H835F|uNSo1Bi019f{nA+glC4%^rBe5gLP_2R2U0m3QaJ*I zf$V&lb``;44n4#KHsSt+fkv3fiNV+`S`J{lXw)R{a_a9lm+LuF>D`mY-IIOeAFusJ zj?qqV(uldOddf9Wi{6#x`zOSJNu}rWa`tcZ5~H5WlS>FxBJoS%J?R1cCH}g z3l36=X!MA75_0-c5TPQ-NBf_DSVU;#OJwXwBYsXp_>ANKa}J%gMMd+}D#Xnc!PF-g zpNsxR!V(q{HxP*I+PAt>u)|e@*(NmyX)K4~o+dT!y`qUIKOXnsFE4ki(nN9;w6Z9G zU_CVOo@ePRmnXw9w%?t%+?~g~?j^5PYtzS>6;TMtKp3CQ?YAftF5Cc?=k@INar55r(Ce4 zT)=#rlRn(`HHCD`cJytCxn-@rxMjvBn4D^C&-!;@1o!sbR#DA-QO&0fA+99%^;yI` z4s*Dj%emVoApr_HP>6&ISyf#DE-_eTSW{(afEzqYchg1Z_(76YS3EjO!+bF_LR%%I zy)%{@c?WhpZ)&iKSS)TtG%hkm({ea<8O~nhgEVWFcsOs5Q-NgYC=)gza!QvrQ(O5YBVE3&RS(@|1;c@#eNXqHm^2QvLh=dN~oBJA-O19p+*iFq@ zPtCzrAyh$II59lRhjFv^O!qXZ-3&kN z4U}@%6>^gz3S^3b>9~l7XI0!m~h1eKRP33~f zwur!{;oC;Unl*A9$dPXvR|ECc&EXdnRA=aGYDVq#{Vq?2HYQ-&B4A1b#@>Sna#DAe{RBNLl&+Peo>7@qq2v6rw9}i4hSnY2&>3veWt}j?M9`lM5T9#U{`C6 z3RGZUUejCNJ4A>CVD%{emPmfC+>W_nATYMFN7a>c*YLom{KfEN*E}k~!B9b*<|2S) z$&(RSe|s}-{bqc0_dSBmF(iELMk&*epf@0)ecVpjb3cEDMFa~ZQ;V_lGgy=dF}rZcx^NhAsc*7+sGTeq zW?J*c3aCv~5i-3$Qvv^->_Z#2fY-?X44WY|Vl6Z>`tkip&~YGE{yP?~tOBv%HK-zx zro=Zw)(EA6B5INeQR@!>O;7urJH+4d#DJcdTOygca$D!}I9pVf zJPyJ>*Ss-vyKKB8pP*v@jFOnAbY24UJD$4uA7poHBRF@9%43_7O6~GII17*4V_O1{ z5%{H5W0l2weN#ae%{FexHa7W=k*05Q(?zT8mx}jg(!c*o5Ju*wFCaLLJS4O^$eDsu zpMp}`Y_%;h)0g3*ooE&g4}YfO}fa_O9Y_xK9l<;Pz59f8AcZG;f1Y1}LW{nF1~D@5(2%592eI;U*>K?M?3p)gzTh1k0)`jkkhv2ID#);y55ec#z=p?rGarIZ^tJf{z1m{c_-NwDctpk-(Tb}4$EI0?@l7C zO5Fe`4_;WZJ(xYfG*1Em{RM)G*nWIf+rN=$?oJH9sPi~3i{S}nq9Zwa)g2Asu3q_# zM7^SS$)Y1B2*$NV#IZ#bbT|0L*MhNws@K%0$mPSv;$)fCUOQ(1dX07ue8yU&4F$4B60k9hegCZo6ck;D-~31Q{6_`G%_E>~oEW5tAncO` zF{$X`A4O_JPc*{>oJ4iez&`Dl6MFmYHmc1#OojX1gf(T@9rV&-)Y!b?; z-2N2NoXu9TL#RYUrKqaF+XeG56`3C1C=OUWHdwq!Fa!JZL+zLcvFsL?1xeEH*BNvf z*RhdeU{bJ>)g}FN3LaK0>+gEz`ag+fZHi{)PUsveXH3&vER@U0zi&cG zHII3_!LyR%MM4|ET>!b%IV{-$G7d;nE0hzi@SS-ewMd2iHM$h1wG=1zhd%rrpqjWJ z_o~*u=k$}{;oM2_^ZZ*Q@+qS{zkqhMMGi|v4$S1jf6Cq0tqXi4$dG@Ory@ABK|SP< zP@)0Kh|B$G9k|WD!|u4m?#L_}L04DfE@F-y&h7|V=yv&88ae&mPXWj%`QO*q6pLd` zT&~H|TN)g&NkkI>{f;xE+wKprIjUdIg}G5p6tnA04>~(y7s}hVm)C4LrgKQ)u}R?} zz?_ABy$Q2uGMHG|4QZ2$#-FtVC7^gv0{ZX+w!aZAaWk1@+>B&=aQFQQgfl}ca$N{N z1yYe%zq=gVJJc!+AOUb}Q2hcGDE8b!TwfU+Ul~z%(+5R{T2C2~hb)w90iau8sQRzn zg5w9k-QrUJB-<|}+cU0oyLTrbc8~2Z7_BhFwnG!! zwG-QhhQWxkH(`eM8wq@w0!jEhv5`PLn{53PA1xF=J^}h_Z1_&yTD57ogx*4B0_zvuz)0pHW!F$zh`PX8^gK&#Y$G;83Xb?_TI@B1UVh zf``k>pAfr+5Ie@yaGoH*Zho0zA$AaQfZYy})Bo^<0Gk!1s9m8RkbdJg`^`4{2j>v9 zf{z0&j6U$mtcgWOSr~!y{Y~z!4G3QG)UKj`ZfzHCZ5f@ywW4cWrK{M^^r<)cVIFrU zY#X#XqhI?Vi6$pMf_w?;43eS!N}}z_)pOUg5*_F67O(drig$-j)b8Zq+BY;Xw=)U> zI;Ilp2dEF^KRE2_)bEZbg%BA^h3eu~>yP!_x4+wl*ZeBhI4=A?rY+|gN$>|NlCQU~ zpN3r3R6?6}3flO$1NSGW@`Lo5YttE*H_RqW`YIR28pacg`g+q`aUWSZH3T`Xz`rw- zm8UNb;WoX(&(hLw1w9oGtG+W?e>jj`X4HH+L>;q7AQ)Wf;%n}O?qK>Us4xT>&NHH| z-sos}yBSI~uzI}JpyHhc^Jx=ZrCWB{>)~@%_UTcL1~*tDAuE=)F>;S{*r&g*MLua` zBZ6s1+v&qXXD#A^9WVg5tpF=2$S#j+nt+$2K*lJt{3M-3Fij}P``KBF+o)b;mp$^q z!g1I69w#L`kbbj|f&W7tZ;Qwi9r%4Pt#8qL>wGG+^49*XfoXHJJNqJayrA^Q>uQPC zGtN!)*})uX~W-w31-7D zg$(J;x1N~{j%>rH6&|C_;$)GN#@-}IRDyMk)Y!nB zTg(RaeAJGiYO6Yf#i1KuGKV3)yF9YU03l$&}HY{vosR0ES;9mcNJM zuq&Q6K7^pZT)8+TQ)JmZxH-WVmlKD{T6Ll<>{|C#crcn&5Onty8Xc)xs@OB8Z(Y~D zG<1q=6@}F>m+P?aBde=a_?goKWL5g%H-uMxu z%eb%kAYC?2M}4Is8J-K~FafSFOF%kC<)T~#tbfJ=UYL&DC2|{BV3;lJfn1?TyA22kkR%l#91!fK7fQOV-EvgN8!OoGLgw(Q zVFCh5d3IIC14zBbm7ia&kg#zS03<~&BW49mcZN@#91J|ktG5>LnyE^lUDYsao|rbW z!bY-D*R=G~Plp5Ex*r$B13u$R>RTj>9RwIcA(NdpKkQx&x5AlS6DKG~fORt{Fsm6p z*kYrSqEql2DbiJFH(&{){kY(KMLE9V?yIO^i_P)5GA@&)N@T^Pv#-+l*O2^-;)DPK zQ0&kY1(IZOju(2!0Upxq+ng#}m(fPw!ToBw)l`03n0+D3nYLO$Am6_NAy*|xT-69` zo`f`_XK?k9Z*XjizN1;ZC~x|60P{|d3^L*rPI^!<;z|ns7Jt&L`*$7VfL|9y`<5ocUZv4KvvNIJK`tRyDY=d|X0K*bb*~29 z9>9#B5W)z1ZZJrJs4iXC$rpPiI;PqRp^0wu_E7a z^GK8<4tT>GeE&4tHwUanWsBb&dB9awkV61208e5BNGnrMk5yaO7%Yo+9{GrE7%z1; zOJg0d{s%5U^%+`pwD>xvG0s>GG^hHMKmEU@?n^#AbY`oxR_JR+P@6^ZaU=s=XeN-# ztVqbRv||Uzx(P>pSkZJ{$$LFo~_k(304chZwkr&(1Vj47R_Ht|l| zUUuqlDYrE~T*?Fj)pLPy#cHr61FR;R6zOTOR&m1nI3W1dOZQ~A64a$%fctj1cB~><3|^kU+jg64X!NzxrOcw} z)=49WJR52!$Ih7FqqyZOPx48&9Ywim0X{U{1Z?$FKU;Kr6qnagzb zt2*C-e4dsC(Ch(dTKdt0XjX|gf5elO5&-^Gu)Z&@N)EKR2(b7$0RipkUV(#k z@}*j8HJXb@jrB|!J>5}E)@utD5f)r_pgny8_8THE^pf?8hwI0N-GB#Tjor7&~741hSXuN0re>tDrLv=*yR|3C&@A&OwQm=I$nR(%CYr~y*D z;QG}-8nR&3KpgQ6qQ5GWR3kY`{rS4e|BcIh2G`DIKP>5GSXfAV95jC zsSKDBQsb{FE%i9n?@TUNeXUIR2Q#G`|0luAzXT^i1m%XuWcau$jv^@TS^#phPKF}9 z6_q5@ZZ2QS12E3S>wp;UMG%S5FyqWf_8aFaIEo;a|4fx%xs z)fFTK_F*ZyE+4bw`&Nh5p5`X^tG5=927rx2M2AW;lXdF_70GJU)&Y#mAA`lpiK8$W z^i+9rO#B(D{6QirGqdCBk4q=Ijco6XxH?+E`|cSrk72qKv-G(-NY?@aK!y@hs!DiX zfZYVI+PCGur^UPB;yezNpy6d*T(0DtN}8~h?|L6XXWC%)URrB)6xZh&? z{_utmKOrW$|C|bY|J#X8v+{va^Eh-k11!l>oDe#o5nkc<7bpJ1*?x9HvbJ&0#LNdg zz8eTHrNa)u4H+MLix*Sg2zp&ZHdaak^=!)|IX0+7|Ky<#jU0N5miNVpKlKoN{(~a} zsYVsY@-jcFCzjj|sktmz&NmP*Qy^Y?`%W#QH0zXyGMA~2o%xsH29&FgJOI}f$i?{F zs~}H`o*BCVQ`JcXv;7sRQcnVV2oV~5(R_P-z#8|4tNj=XzI2X<6$r}qiKH_}*>*tE z)o+l*e0SIddD+ywh+435d&POd&11);H(VRXfsQo1ga9jep7SzxvOCw1S%@G>e<1*p zD#1Hh1rCB!nUx;b7_OWQv|{9x%_$#Q8%8lQo`~DLw1+Sm; z4sSn24ZE)oQrCYO!(w9;!%Di@vXkZCp`4Eqecclb#S)5EFyoa!dOi#nKX)T`LHp}C zskh)bcO{q@sNH@aoR+7`~;0nG8AZuoa*AP?&j zffz}_d5w{!n4mq=?aT>ah^T)yoqPE*0@rULEdBSwma;y0Wjl=3sy zld~`n=3w>g!t&Q(hYcrDZ{#|PPP|wke~JU90P<%viAw5UAQ%s3BM8X!gj7ESzPYyFu;0=+R~8)drp9ydfleaP~RJDNdPH^FZU(NTWnLt$*=|I z)|qy5HRch>f-!VJ7Cir()CW600!H?bTxs0=A=e1KL4zaU?xuGk>d^)L9B#Zwml7OZ zEtDd6{L}W-#IZYUSftOYivrmi>ojJRGf=kH-~ZFf?&uhKJld=oJ)<>RNm7fKviA&? zVn<6I{-)K&qh(n3tpw*)iCl|I)pd7agg_vqe=) zH|F+@q=$XH%m3l*t)u!{mbBp$T!TBo0t9!L5Zv9}CAho0yK8WF_XKx>1&847?%#%- zGk5O1Ywow!{FAIz``Oji)lK!Udso-9*T}bgTOQpCjh9v{Rfuo0pjlJse3z0X3BC&B z)ZNIpr1IoaDD!me%i25W=E?H(+{o8{T24p%%kvlcbVsX-1ULrl#abg@+i5u>zB?tA z`n5X!5e<4=M=j0)&HA;o`gi?YOD;ANjIax-H5A|p3N3mZ^?pwlyQaaCGnW_*`c4Ml zRclypPgt*kIME|oZ0lOD(01Sc%l9q4g>pTKn1LN*l;N(>fY!Ch8NeFVS8HJC2L-}u zea{Fxe_gJB6~Ol_Cs98()5LEu>z-PDR1=m;8$FV}3Pw3b_oBU2T3-Kg_o47Kq-6P1 z*VX!bPtAH>yiHhKY)ke1Ii-+Ck<(0ndCuKGa^pBa4tVe0NZF%%f6LczHl$P;%}WQ+ zqw=U)s2%2Yb@z01)V-J%l*U@=)p2)r&0}@!^?15>5!SiP&9&s=j(a^i1#*Yr0tD!duMr-_PF!u z^td(YW>3>$y>|W}ynS$PG5o1HQ2BvK>&e#Xfr{erBwC1H+Y<+FQz!iQc7n{v6Zv$G4QF zExsrqWvQGi73Ei6C{vf0617I8aC|ux-y~@>oQCC{> z>*uc&J86~ttnXW@ln(Xm3aR_8&dnTjo;oa%`#?YQTwng44Gq9=}l@c~c4< zI03l?4u@E~QIk>y1uC6BF3-QP;_Dk3of$ta)}yy~S}y!LE97~|I48Ow8hwwV$Q7<2 zh?h7|!EH{uKmM})(zIW=*N?CvpZG%4AeSg+$cBpIpq|(-EJ>ND-Ygv&bV!CU!+Aip z`pM8C#j0*SdDvhk`I=J_Kf>H1D-ZH2{WKY2aNyU0J-PAxxeG1A7CUiLYhD>ZHZ`T6 zE!AII=UR!^89RI>@I-~DuUoQUzqE9(Od;JnW0rrsNj~g3i!ouE6KPs3RCi&7%42It zGCO!t@yN;Y#{(sJ|RJDNfV_6%`2>*n&|BaLI=c#7F2 zW3X<;DP-b+D(V_K5^ce;OOh6d{KoHXG~3jIy4?qn?KAU`u@s{oZ9JKTwon0v0uv9(U@R>^e zQ|s8i+~dZARpem5{>nrLG`{6eoXB-b7=5a`P`L3seq6DmH8e}srxDA zA0s7#Kl^5E;j{p1GjIf!Ba4RX{iV4myFRUK1;S;FYX}SpZ3uIrM2OI{_3#Ps4Ql^bAm|Sx{NK!($`094BEI|qL&;oHRUx{udWV*tz@Fe; zdq@%ewt>b(5-|z~%M74hD!&w$>I@kcRr^^>HE-=MMC`w&hhH|-t(4qd>(|~^yw_;U z-UGu`bmxml6+maheLJ7^4TGAQnQG9AIqMbeK%xt{WNc=e7~ zy?4pxn>WO(c|m+GLqq$mj)rd5CG!0o!(u73%4osm5=zUxQWe8O`4o3|jXB#6DIe6W zX53P^jy9u+X~Rlq#h#NZNKNs|1*GkY4@MbXEq=AEZWegG#Z{4p@5`q^$mmD;Pho@j z{|Ot~wua{a7ChbpYklm@QlFOrd!Jap~gVjQHUP|xqlQuuSEgeZb9D_>aAQJ0#q z1Bbcol1Hy6oYtj$)Qy=X{wu2POnKIT{6Mk@$*9d+6NOvUv&(z0Ut&GwiXNh?%8!z( zP1#$Jji<^X%|em@%iUKOsOG|?Tqe4u+90!*@lL;hEgrioJKdK}_fJr5v+wCv88h%L zyxrSNnyM&KAYw(M?Jf6nAVxO11?XuxQ2FtHFJtn* zg;4tcO8#Vv4fs3xbCCkE&2cUhv!;x^p$Q2E$IV4a-B5iZaagv*a(sL>mI_wK6Ow9r zI6U1WTEYm4jSVc?{i4x}+r9bLtktV+)x*PW&r?-X%e>QbZh6`3>GJyA>-piHXJNRr z{o%&T?ZTt+W#jC^(XEM%<@Wye`ut+;^k}xUv>zd>5uq`v^RUCB6YpiXfBAXs+WK+l zd1_{(@#*^dxOt*GsZ_7Nq!Q2aq*Lqp^z`6!fB*3YP^NV8>bN@#VX-m@U+Q|1wlPS* zDJtuvlUu99&CbaSNccR{K5WEa)p}lj;%F{tYID8WzFT;4>wJ2~eDXN6r|dj&^x}Hb zZ*y~NcJv>4Us<;ngyd;DPJ_Of@emRzu(v?GM?sEwEPaUnbG6f|x{`?ODuuV|x0`7$|_HB;jLRWWm> z#PzFU#!QLlSH;eT5=J#<+Va#{C7N8thshETi2{`lEu4gzJtfPl3uUbgv>Z1N zj;;;a&ANv>_a>sk<(Vks^E+s*RXoep_W6?z?L*)meC`yG;py4-e1_%O*5>T)dcE3y zcd)!aI-@Nusg1~uWEqJWOtu}&jN}>VEG@t;tbrmsiu6K|llhTFfXS{$M1WKHQAmJQ zcquqMB~J9Y+ZRSb? z!WS~F1?{KDh%((g6k{)x#qB=qf&n`5N-xPu0PXiAQ>wOHvYpEB_@9Tx|?jamC}F!g>M zGt50O=mrbRGF=K31oNxZF|p<7y*zH-e*09Dz^|rVv#D}Z&PhQr#iDy zpjje+Wq9}p7%%RoM#e}TIO)UZ29Fg?H8fWT7K`UimHS?+g!sGeB82!tsM4EQ8TTK2 zn&v9*dZ_+T2ig9y^0eK@P(hYYdrpb9N}}2@b6I2_T2;xJa}#Fz>>21=Va?*)A%}so zYoosR>DZy#DvpW}8cB_r`to;%-b5EiKTJEZ2$#}6upt*taqM&%Hpe!!OuFCurR-@( zaI)LqhIFtLrw3U4v8dw2MHw;bQsnms)NM#*`X3rEg0S{?YQv%Z{k}}C9HH_#8#`uC z0Py$$8Ut)9dGi!w)?J3eyqHRJ#{hxfU{?sN9|Yk$;7Km-CO_5J*5T(KvX*PW;Gu`Dg>8?8>$G59s;L zBicqs8P^3)p-8xL$iE!|>+p)c9|QNK-wJc8rTJ#3PZ;iqo~_s?{U1P_DD{##=hI-+ZpgrL|i8t*Qe*lvR3&zYhz(j;Ut zVhj#xj>^MDowfAAfH6NqRL^9MT`rXZqWp_clL^VawxJI6UqxM{%2aLi%}NCAvt#X& zmSbu-7>i8{iM308S1dG5`39Ou2AbIhnudlMLb|f@P^6sfD49(qpe5c76V*= z{FMSZjHflB}DBW{-!D>5R=k&8xzPq-AGW+GQ+W9Z?U|v_liCpTWc=h5-e=iWZ?2NOO$z zA>^<7aRZSkdqXD30(V1VOprGu4# zAEO1u`f9*ZD8Uj;ek!&E5iv|$Fz%Ytm!_{f$&3b2thX!Q87O!Jr{*sCT7EE~k?(E$ z*gF}^jEtPZcLU$cn$2ro2#GPm9Mi)Zu)} zr;AqZLAaN_TIck@C+8yUnZ2rm6@YJf;^*lHsD9ecrH&J$R27^NS1{HV8Iu_dnuU3D z8X-C0v}m)c@GgWn*k*{=aT|oU9i=Rvp2vvAHq;hz%`OUk?5^So6;d-x_P{U-Tfh0C54O6wrUol%wwsHcf6Ju^n8Ry9 z-Hc3lfSbRzF_^T;K~b6e9x}rSGukp~K}Qs3EUxjh?cBkcydi~|zMbl%48?a*`gKtT zv~{fvYULjOmZbTJbfLFZS?c^iNz9wtqcRLaw3$6m50nj?@nM(Sz)fu35&ZO2QC`3B1b*d}_L!hO{zc`mKX-S9Xk1%d9mtgeP@Q z2CV!mZ=MqjIpkaU@fL)0yR88@jw$VHShZ`AgZji-h?q%N_*1#oR$3!sew&;fFK(Fq zQ@@`r7kN^wn_;wDezekD+afFw{>qmq1FX*-J9}Gt?g3Zpvi-aM1acQ;2p44vi-25V zkgc|5G5UDDR-(+UaLL5`*VuZg1>Fw#I!CA>`})-*3zU8flnIkHvNzZP&;gUO*Ai1L z{tIy{N7tKXTOZ~F7T%(erU|!70Ym=P(6=kG{L(|EUBG>?cMSXo(9-USTQgmKY*`sR_An6PVO!%Y*|3Y^SeGWa zCIAgZ0pir|-*KwK&#*+Ur2?jvi6vHNGg(&Fy)YdifipEsv|W6H>{L?5s1ET zGoBA2u~-bcodT*|44S+3q-BgSo3CPnP+MH&>!qZoQ=RUDB|Nfn8J?_sgCLwn!K5RsARCtU{3c~M<$8w$60L&tWt z+*8=#uk`C$JtG_MrzVHS8t=ts9KMsTQ#lcZF9r$ftBDUBMp~obcuz zQo{T2Ee$8TEa@^RE~@frmkzyiEW9OIhuhi)TF-m6Dh!2W zuxx1~>n#Q&Jcr-uYT>tm=V)N^81MKXNqVF-r}O3HlSmO|J`V4xtdYuIfB*4KHZdLy zhG&4m$dlG59y?m@<6d8Etib%yiJ6by(p}y(n4IYsv4|WTHxS*QWQ*?r&+Np>p@qd24GK;=xQ)exlcMdG=Is?f>$GNi>N6+n>criR4gy8Ao@X1*V4vfo84cf;JBqSIRdg3q6KcTDjL^LT#0vEn~DK7&zo7E*V2r?K<0uI6;t8>u>8qzMJL!?!cFtz;>Q1iwVXEwgk*DCxWT> zAQdS70DlYEd`K7w;lBIbFTt5#gaAo~{IHY;@qm;DDwOOFO1?UHpUeHj^t7NJos%qC zJPsNrr(2A&xTN+HMTt#&EO@@Gqv{W_={D)_RD~3no@poqcEYkB_Rwzk=6kBBs!4UI z=(2caKYby+FOs@|p}I2HQ*nul&2r7^S-=*9*ib$m>;%k_3odl)qINPYq5eA&h;0WLW&~UBhxT}9S_p%+9Gtje3m=e-5@AD zC{|IF3f;YX$kw_v+i!I+;DHR;(bo>bcdt(b=Lv>KBUXKL?-+RjwnpLt`z$S z(;FTR zp}zR|>XC)WeP>Mkx9NYS(5?#{2A_*x0!D1Sf*g?2$VY$87)(XMzN_HDvlxOF$RGRV4AiklMs*g5f zzrz@wsXhfnmHbpVuZ_kIlqxrsp}%L$lUA@9%#N2iQ2@_FR+LAkvo}?o3Yo|v^vDSn zdY|v&T1Wwt(4aIJ?X8=bXG#jZF+f#;?_5oiYeiqi#nbF^XIcZCC1=GqW2o$1>=U#EHQQ&AybHwq_3u z&??p4tuG;h`!h<0fu+_AGvW{`DA~KSh!Ii0~n5Ak8e!_h`&#> z_mWC^Z%+tsk>VamvMK%rQV1FZ1}BjrZ@UgQAb?*DUMj_=2mxXY_DAacCKYAk?wHF<3zAY9WObib83$B}(M_FnNBEKCQIw`eC5Co_gkHrG)11DP4r`^eac# z77TxK-51sLO|7!HD$e>ES0U3~E~JZLzXsG`GgXQ1;(c_kkOc4S+3o2mws*k5Y7EC> zOU*R>zSPd4dm1k;wgI{PcwO2*zIdeT;nw7Fv%Pp!c;vNm%(Lv}(RS}QY&AJLXp8Z4q6A{ zv9)N&5nu!)`W%d|-!p5_ckMYl|CrBnCyt!L796vd?M|M&Co`=3NJEP5DVX%B9U)BA z1qY%T`@s?--a(ax;c>+KLBi&k$=^=drzx3EXHo{0A*B&(!{TciQP9qrLvN?X(Sv8r z?dDOxFU=ECQ?D;E0SUP*rxsATR_{2kfP}izfL|Q7)W0HctK8o*Z_C0!od3Ek`ooiu zn34!TKZTNrIKQb#IL&dEHQ+fdlb)$p7Te|Ls7W;q5>? zRo!ku1l1qlL3;5MJ26X+>FEW}BSI#kHdiz?qa6=L3Jm3?5skdrp=}f?YE2lRg~q-4 zYNY1JmDLyL^m3GyZuM2`u%>&O*~w#86aiu0)yX-%#p|X1)?%|G?dg7a6DWa-zQ{zK zq2HaI+_DIiKs~t%4YK&V1S+mq&EUC*9Ny}7RQo!+blg?**O%$qr{`~t(vFZf>(*IL zX9Fy0%l!9*ZqDxfxR%w++!)8^H7g&y9I-p&3e$=oR+cIc=Z;26ccPw+ zlRD5@D>J4fIt=D#r_!9c9UeY?jj9W~Ap5*C#<5IHE4`I9Zx(-y*SJNW|ND69GMXE0 z$1CIM7&Y_YFz#jSrStgLmgBzblYAxutzgR0OMCp#BT4(UR?A`ia7XcQtD}q6{&}V5 zb6Sh9neu5t#gBS383%X{d-zJt?vi2qU2M%(A;sC<9mx%|sV38NzbvPe=bO?cuPH>y zdX9adRNJM8Hh!Pu-ht+?O&4%4ks$+?!CK`{Jk&$EdwabP8$0;3)@vNSUysH~SB;)T zPZo=>N}yDAQ$yJ)&E?O0(6r>wV$eM0t91l_D!A+Qek%@Ep*N?{QwmQFT~(#Gq(~&9 zhjJhxb|`7zP%NQID5pu-N`CrG{6P%KpSu$+MI7#(7K2;;u16=|;o(E{sBZ){*dw&Z zN2(rky;E$R@mNUKyb-HZY-l&du1pw&U%F(1Rj4~!LH^tlDjeO&T2+jK;eANxdMw$& z)y9Ll8^P^~!A}%=+M`IMh+;@^*2hl4bN#{jjP70-hZ0D+HvR$0l4N>^s^`0GdYvTe z-$L0C?F7?sGqK>T&&ylAgQ4dO8tWP(o}u|JD)fJ|RZo}MbkI&? zi(LuD_raJ=GJvrfl&2qogxU3@Nj6a7WgRkI=ic#{WC62Oe$t>6MSQJYO(GpnZ|iva--ax z$4Dz|B^C&ag(cm|`vIX9Bt^ZoZWok>5c<%zMU}B>I%CwLyIWMr(o_1>y)L^zZ>hAY zjj{I3E#mpA@%oR(E*^TJ7$+Q*g;eNrkea_zWJ;rz4GRvo)>4sdg6II%amY01HrJ;c`A( z^CGniQrqcmk;qpoZS&sxO(a}W)>C9v`GxTWo3xy$`CV6@5}}0JkFTHmMx{PBIk-<} z2|><(*qubylp|5fEr9)EFPe~G!7r+w8-b?bBMS?iAqR~FJ{tR%9Xh_l1zRI~NGe7d z+wu2GMKXcD-zdJ#N)a{`zV8FslJHGa<%^wm&{=b`w<3z@{m;}4t(WvRBsZS33)qic zlgw=RG1Th!^0dF4OMnPd3oz~Do6_VSFF#67*_|gp$81Km@~^oM4lyj^FCGW_9GJfr zIG<|Wcrr^PJ=hs0`>T~i0YNpPvY%C3L4$I%O4-T(y12=(Ae#x^&%@$-A zA6!3vXty|4^rB&Zz<*uMT57D6gMf0WdgT9UHT%!M4&K(Z|Mye3BOyhfrGPc%6=8J63|J2R+wz8{fMlFaSy7TNkgPuQ7XJUoxf(y|5|L*=ISoq-Vh+g}R%GuDZ z_U?`zzdB{VX7(ZWi@c6#_9A!ror83ZmACrI~?tsx6zi@{r$xkCp^PE zLSsSmC!La7Q-|su;;}i(9Jm+7wUnVd51u?;b0k4*o#I(>`gchdsUOyJ9#grmFLtOx zSd9z2<+WRntGt45gm=2EvmDk$8W*4VYeew1-VatcMH>vtYs9Z@w?5e&kC3K$+%?KS zwo}bZWkl_NX6Ts8k{lC%X_VHwNE>gTlIkefYn)25W^`CWsiUnLwnNpfSs|V8qm}Mm z-8-O|=Q$3QBRM%fyo~<5wd3U$^d!nv=oIXg{^GT46}mi4e=Is4l+-4I@S-gtcpa^QvCkzBZVcd&N zyf_35oD;0XB7$x*PFDOx6hjV3`pe^&Nw6hcVrqfh6e6?7z@fJ8o zUel+3%&t^;l1-xUtBs<;~P?l@nzI2?BR>SOH*vaXa9>qneHU-wv zu=zxZHU+Ynv2%F+(zQqm`&qyyAW*?3M%o*_{RdWfkS)dM; zK_1NcWyV2ZK(4kwf0g<;qJqzuD5w99V492!x$?Dlgk<++6Nq>4zL6BX!sh%cf004K ze&0O`Pv9cb-;+~8-jw6}jRX@O7~K;Fk~@wKKT*W{+lJ&c+1w}U(FX|$0a67P_&-z?f(;bJ$uZ6T=QZs zttjd^G&6&H36j_4R>f>g7UqUe=Cfr1n*K@tXJ$$+D2Py84eo#_YqlYsB`}h8%icMg z*&x|~o$@a+7ulasf^w|Y^vIB$DEmB#RPpba@QgqtBXu%;CpmV=g1`Qd!F1y6i){TY z^_7bZEEU&^iwf5+5?lWtO{-iWedxF*z-LBhZCik?4(^`@{)!l-`UxD%w$ZY~oQMY^ z+cfGOHXS&YR8xbcGBZZpkO;4N49XB4D}{e>db*X_V66EPXY4A}hN<(7P)PC%(LV6N zVL61U`0m!#?~8yUF@bp@nD0?M^~|)C!?G`vo1AYHC`jOW9a$VI$&f)%!*pczja(xM z^+K!>yl@zUsBDgeqlx$t=EIECa$>lNcwilL%rmP0-uM?x2cSnSG{=GQN+LKANPE&> zlApnR_r?=V#k}k;fNBUzdiY@Zr%$u#wpi}6gEH&B_La{C70*HnUfugGKZ5?skO9g% z*f{+oT`w^i%U&i;SYI`7>j@-5v3P2al$T_oPj*|y}5 z&WnWKYMl=S-~qMHfjl+iGEtz`x#FK%=akI;n*U1v<#PD9TIa9qBtX5w!T+jt4hPga z$3o!m^9lgCzT@6%ovU-d<*ZGglXm?7*DL(rw&-s((g|-{bi{zyEjnoa4Rb0MG&DwZ zC;`~tTl5;FKp27~;{LlkBnwVaa7ugA!{JUqq2&FdGq3gQ5klBDBd`Dm_XQo{tx z>J=`GF_Wq#g-*`R)xoKmpLb=)7=2DHxj$@nUs(3cY|iX<=;DF^mXcdU+Mr0cFRi{A zc(u2QPKkRIF7t_A#+9}97j-n036HJ0Q~C|mkuQ#pX@1U9#z~J{&o3>ii3i#GOMaPA zuN(Ado)ln%&i#e~y*Y38 zi3imX?eesH$Mmrz5VlYr*VK#YmfjX}86R?%5lNJ9zDEUlfs{0sj^#`cIa5-CWNooW zS@hCek2K^vsh+4m%UL1uJq!IOfTRT0m#M_Dv`~SELX(Gr+VC3Q1W}ai@Ev1^!8Ca^ z#+N`8kM|QEZhQvLh*n}Cpyz1aV?Pmx31INw-H%LEHPNYjjRHFUisqBLOv%;NRolEYW#?ugbfzbQaFKD=mnMZP*CE>Qr<=y$ zKxL{Z#ubA)%>Hf?EZRXv_p1F=4eZ*?$Ag7%w-zbj+f1a#`F)T|^=Bn2_z3P)e<5D_ zQ#kl)I3h5?EOu;k&#-UpYCrMP5F6xSc#|-1Rw`AngcYbhz;KoUdYAZtr993ZSjquI zCd`o7iooB``RjZO{-TA?iuta1?(;D{NSO*YwpJiHbcCZmIu4p zYVZAF?l&f;U!VS`-g|0aibi&cCJ8wi%AEMHG=(tJVp(L-I>pb}b*JQ5lDqCY=Gg%x zMyg?;V6LM9hBjZTXFAwgV(gMmHWIz%Xjs6)Oy$$kBm5WanGTT+n2H;aO>+S_Z2g28 z4ON&pu=3j7%N^cs;V7iZ;gv|f7wd>prz45fq2f43R58ykXA6K9msl6E=;f7AK`%KX zD@g73BWiU(O(ams)&~h8N(_!-mI$1b#C1a+h0u)oP&2129DaW0sxv|!U`MFn9oIZG zXZQ8G52n!#$j6hXetdwT?`oN%8qv?geE!f7@Bio<>U-l zOcJRc&FqSUECl%%IzX;TJQ#IxMrQ-lt|ZC@MQqa}q1tLPH6pYRoORMX?b})VCTd4g zcvo8`qvOhGY^BX!D3iK1b4OBo*G;VKTbnn%GbA*yF-EZYeK)c4w^|HuN-ivEpqW?v z1yh%n)<_PM`M5_?zYrY&COXt2f1SKIOHC%op+tsClK1^>I}~WDF6B@Q_O<4(V1BP2 zGHv~|P1FWap%6&hhnl=JL(wop(OPiff23_z+KKkHR9~y9zSc+#oEt}zz2KndAB1~U zuQECDt5L2Tpa{hC_*6YY`)-W~no@dA*}L*1$m!wv%e~XOom7G7Jk>@09DNYZomB9E zGw}rfL<3u?XqiJAj_k6!@-*73ERgenGZ}^ExyT3*N1uKU20FU<(D^k|9->n#81Bhc zXc{_SVDyi-E-Zgh`!(Li5Sj&wC#$`QwZr<9ZaA$dxHYW{svpgkk8Uo?*Q|BsG zkQ5$mLNAl!vK}k7GHot|&{tb4xu%i>Hp~=4oid@4g4C}Y=2TRg zI?O5N_@P{OeNoeBLliVEoYNK5Rt^5SEe}LUI1)j4xsu?*z+0D*eumzu+vB~oLQUD} zK$bBLwVE?OfCKXii(xlV0$6k%UkOISTmB{(`Fqg?rEE0+zc0FP+@J|*n({=V*a0U% zm0-{xPb>l)Xe3vJBp3Wb$GGcNSry7*x?t&~r z>v?B)HhCvo`Riiqz_9Ul^2Bj}HfYQI`QCp1>BP(OS=Pa;`Tk**W+kbV=hwKyvONo^ z)@;)9PJzqA!a+HH=0UIhli{6Gr{(jOO|zOr+Gu5I;mW;*qO@@TNDEt^+S|8USr*nRz7vVmf!M4IH%)y1R5a(heiDCV@vak z5qxE9#+Bodw&r8gzV@Z3dC+kPFki*Olj+Fmjk*eYm{Dj*ER#>k&u{=f5z7=ZdA%-y z1mqV3y=Y|M1FFoln@>;4HOu(qS}{XV;gEGodyp`jK@rR;lu*CKnH>>X&)i|*?vs09 z&yns(L^5Xds`Y|;8{L{Hj600=$Oo+^o|b5ZDZG-eip*<5{DBk(LW$=}cuGBUdi1@^ z)yZ76Xefp4y|5^m{-GiI$ERKG+_Dl*WCilQ{zPzCFlZg|Y-47>^Ysp^lDX*NTh>B~ zQ`t_awUUGFRU~okf-sJkLrm}ancItuz9$(%>#}o$)2s>F9ag8>>N)dv0`PRUGXQ={ ztRVFUe+sb;R~RqBV}AcBxgU!jc~WGlU+kivKnub5m--gZ^dPgV`DI9 z9SAE?w{N3l!#VKeoqX1+x8`t&6dLDo01VS})o=LFa%&uN!X;stVEwQVh$oclje^`S zweMk9py%F|n}bgbp}WC^E_*gIImWyBbYP<&Dsz6zFDFo2=$b}tO%RnA)<-Av(fu)5 z1FE(Flao(ck>?5;}TtdgppWgC{bg}g8nKq~AB$$HIydGrg3W^8B zs|Uh{mY)98fe~BxbCJkhj;M(Fi74|Z)w%facN4+Vj`vN>F^C=VNQnJoH@WQh1_6&I zb1D`1boIWwcJXjLLm?#qmH-}lvZ_V{U%A2e6ttw;o2?;rce-CiHaZaOtBPs`9i}?( z?Yy@)+ncvMnKp&H`)up>5)sP};dmMtzwQ?a^Sd%pQ7Qi%QhJqf4~0mTAMOhCdor27 z3elsp6EfEN9Z4lWJedqc0k*=m8!;;U3{?D*&aoOrdOs13A0d;K?WllQuOEoNm~uI_ zxGOB~S=Xfku_((o7YULxeI${}u|E-uS5BAp!!aJQA=jJDp##riI+ui7jgX2TKb$i7 zGuHS|ea^Uo&-dP0hb94)=J@iIMc=2f9L~rVKmj0_ay}LDr}5PCi)J=aR6hWplp?Pr88Faj|TY zi$mQ!0mG`s)Z&Yat@+xI2$nZyjGvkOx#Gbs^xY9h68DB0Z+sNl+(FA^goi@!WcOd4 zu>49}VKT1EDhAam`}XP^8AZm=%x=qgM8~(T{Y{qf`eqS9)G`@?RK|&r(bs0G|e{QB!pWFnw4#Y;#c-x@&1Z zLjOb@1IMxQny@}9WA!*A?J~wAB9peAqjll>l69fdL zaANBHd@x;>sK9VI0grrT?Vzm_-Vs0**bxB6XGY*t$Hu8_Waj2>z|ASt zfC%4hcFEhU?g3bdjxYxzd4r=OHAJA48@$OJUo|+w8T1qMD6b7AB378cv)Om&3^eKg z;t0p*?hIy-_wC*P5ZX%!@a?&V_@N2pHqae#9$Gktm;lt-b$_`v5vm zADmn0VJZwR%V{)sHnKdaKQ<9bGGC3_k#hnIwz+2=mI|#u+!_GXSbwhDtSx zKy})qBK9kxtYLYtOyC!{OJ?nRNh(WU%sT8W)_hG28*QT9+Me09X*S=Vxo(Vr$UN9P zJRKH)nma7AG`>1>N@>kX+gkKI-2TBjWkq{+GNoAix@6yTWNA7o$En89owdwojC{sh z;(UBMn9+VchdEaJnNkb&*I$gB|M#DS|Kdh}y)Fns4=3oAZ(y}(LrC#!KKQp%A3|`( z2&nPGq=L=!iZ2d0IO`ipEI-SZ2X0RU(aj1LSsn!M*nNvrxsQB(7B#OmpYq1<4_L{; z*-Ms@Ogcko$M@ff7;$lSuyd$ZAUy)8j{6^eAOCM>FMoK?Gt{k?M1cE(7;`T=yB_KZgX-0~Zghh+ zLbx_QKlEtFQyZ38`WmofU0pq&6Q@n-58OYAtI{&sUh2|e_Ky74C6#V_n(FNE@VFRH zYkifYy$y&{L8u!wE?7u#gK6!asU3qqNyNP-jXJtjOx`uS` z^m4ksTU$%QA0I#Qx_7@gx_<_^8!S6cUWN-7Tdigh9=ukb+&y20mq)Mq3;jAgxTDe} zy`GQmhi__ExgVXCH#V-P3Tb(k4=ug!uNFsdf>tw@mfhPu=Yjz8iecItK&MfYdPWdv!+C;Nx1=RL1g_lNW3_FC4}ikGX)mrjqn z+ta&c)mj&cUFpsho+ZzR&Q(bSr_L2F{D%kGv%4baN!ig0*-=s9`f=g35?vw|f~kH?Mihl`VwwTD{Po2Z-ThsGE8 zP8YZ5@qYd+q>y0i;N;-opx_TkqlD|ANJoV0zQMQ{-Gq9idTznDlt>$d z>q7<-u`QSbsWTT(m6zx8y-8UV45ON~J5pc)6*1z;wlcS%#F>s~^Zj@DJTaUv1Q?FI zv$ArJ7!K~mf%83uoxXT>`GF*ixWsiXq9z+OB9_MnytZD_@Sur{PO~N)<@#9Y7{DFB zk_~a@5mFAx!dG}cd0wrDsPO@f$YeB?2*mkxN2b*M{1E)(k4!Nmf7nbIXiMfFrpc0R zsXE--V({YXBGxytzsEGA1=67Q8##2H!dZth8V@c#>y49!+rAm+#d-QZ+9tUk&x4gg z|I)9RuCfmRyP<_XUMAUx6PM|nfo|}6f;eXG9IH%!@b!$*x^e@I2P+($e#Kq}7`&$) z#Fb!Zytn3+=}eH21YJLBWY9l^7%gCLHP{DUkZ=HANRf(`cr|BOMr8Y%Bt<<*f?5$h zFW)rG+u?rMf;r_^bs`0a5XT-QzxFuy6g6^Lzecg{JUTL02eGq#Qc-W%H$sym?37gk zL1R`^-wgeB%{|OJ#m*$te+)^fR#9Z%bQ~}jYcJI$v)yBvAVe?12S=YbOS3^I1Kk}-3Krj|rK>m_Q?AGw}SDlq*RR7WS`yLrktZVe6 zM@N+}BrG5y9K1j1@&^T$0_%0EDl(o&=py(j{f6t$eHUGglU!Wp#@_~2BYh}#uxb#) zvFh*bS{Z787yS&z;1+P$C~78|onUB&9kTe0IqRq72fMMbg>Da*IPa`y`D5Uto7ERz)>hg6Tv5GM|3pnQLVhVYRK~ zq20xR!!tr{f=wk(hE!k0P_I*@*7?KVOg7mDzIdPYo`TA|(&Z!>je-ov{STJL1cxdt z%=OByX}?OY*`!FLbHb%7*FG~i5Pc8VoXbc1wsvnscdV}$Dw%r22*2E*Qcqy`AXk%K&Ax!=gB)JZq1ob8H)2Gs zM7aOCCVT%A3Erpc$rAI!v6CB1hSkV?GC-GOb60Nk6aj0k;ln7Kbl!>l!z5UzvT{CF zShs`TzI>L|tQCsJq%G=RXse5J)wJsexNb_sCsJGu#6*%t+ybX9-wEp}2bw|!jI2?xxU{`uol8P^efc)g2;#GNE&g0AWF{()@N!h&2z9+@%9QhjMBAKITSsAG%jtmTbVgpoS;om0nU4@K_>K z_aZwDWO6N6Qn345tD@jHf8PIEFmh+C{~cI0!XhL|-9$(wF0XcIrbIf&_ACE^IF2_! zXjNEJ5*&pola7BbSuG=abUxXFj`wxC)WDOz>PG)7&78GB-=4ZbUk&wIpTo3(6V?&l zIqRl^Yte1i7V_kEuAp4Nu0y{5QC_qnclO1UDGg@`@7Y<)^FD@uS{c1IZ*evp>ubg8 zK}ehzj4=1R{r@O?%b>ivrc0P4xVyW%y9U?b?he5TZV}uixVyVM1b270;O+rJm~)Z) zexI3dYM!dE3jXxfyZ36@`*^Re8uom%F7tekh+IvEJ;a=bmL_(NKp~Q)P4|A6AIqVR zW23&?U97FhF52$+fz<;ttjMRCP#|`i z62Ht5xqK*}VXCO3Cz*yTni8$W4CqGLV!-)3_`r;uh=`eY9UE$;pjKL|>o0KYFL0!2 zg>|nUbEAF=yA?jyHK@YbmtzdZr>{j&j)fzI0)iH-+ltyekJ2okNy;yiNvb{Y^TtI1 zj&?u&Sk%vPoQdzpu3>Tn(dX4SjDU7RlWyy$=6T0vc_Li;k_ETxauS(e!3D;2g+i$v zsazH&4Z`P!$$G$d)&~UivN}0#r!wc-lKGzL-vnefdK<}Q_SF0BiX%FOi>h-Y|DJ8}izMH&GmDP({tKa-I-zdm~y&=Z{!hP8@>pR$0 z^E07m)_h}$XL{BImeQ(s0M-^FiY<~`x+!)`r8UL=^j9@OXG7`3YUIYsrC%#*gtZ3l z0x#~uv6rkLE=OK5g=Dwa2qyb@%X}gGPD(OFgr6Db=OnF3Zq0<%Pf1Jhwe^v;hb7{y zMM_wKg$e$_$q*K`6)v5Ppk&vAP><4c;{m9*ZBB=mv z))NDT(T_BP@Cg*@LA}=Rt@A8&pUv!)iarN&DM4jJv|g_(ozX55fOgeZ>KMH$pe`DE z&syTwTq3nE(VW7jJr{6XgHo58y2nlw3OOdzA~Q4m&Ov~G&*OU|CFIjxw%hG`rSGW! zl331?So_K5g5f;DZCtZ9bki5WgX4tI05x*rD^byPXQL>~szl6E2&(5ECK1%W+V;fv~uZtPg$Vm8raULy5_-q@WVA@uTdnX{y6;9d@O2 zJi)j11HMd&Adb}QGtPAE9YAqo-DTM`9TiM?0cv@2P|%W$K5WE<=7}!m1mD814$y76 zH9dll`dJ={A$+X!G2#+H%nsV+LVLfwCvS-H2Oxo9}Pdz#>!g zdRk+U6Zb4b<0VN$a%-rVTM@0$Zek(D(BP-N@knn8?4O3wfA>#pVCOe5j(#2{_8K70 zz5(e*Dfl4xopceG+xuLxbtG#uOZ`nCJnCx#uZ}e!m-rxIPNDp8UAnw^!7vDGy^y8$ zwKbNQ!mTOsz+caSYJ`c6wB#mtO)J4`+`yG70REywk&{^sdKz3w>YzJQZN#ymq|b^n zz_!djjU)pzGM$)!{uD2C7b$d9Fac~_Yc4Fe07_ni@B8jsioXl{eRs`G zK1ZM6%(=4-%6Cdg4nrbK1-l->bWjk-S6~jo-?oo@d z2iU$+AHI{phZ4nyI-u}~6NDhl%E)}{LQC`mzucaH3kziW=pWOOppzBl28}&#>=e!e zZg~L6qm>6P?K~k=nSkd8vR9>H=ml_W0pTzVt)dqVp+TerXm3>Tv&764pNc(#=P69wdr~4c9CQ|wl#qH3ZVGO8&5$Bc6^B>|10tOcKEBFKS30~EaKZX$$IK6pQE3cS0vwP`S%Cvm7a_3LfCJJZ z+3PjIoeTz)OrF+~`0qYEwGPgXoM5J-jk{WrXMZe36!!wc z($ZsS4IRJUs1q%CuH%Lc>^=-qMD(>`D9sRuer8Jv^N@{nhl?pe79YyKS&mb_6VZl= zG1&LxQ)@8!bwN6B@lfvZU45*}^nHX~e+ItPJ;yn`)R%pv%Cyr9nR%-HazWGr1|n0u z8}V9VIh?a%-5$YEo^Vee^0lK_wdE_1vMHbRSct6(4 z?TE9bRC+0xkT)z=`$cglWw{BMFLk?QvVP-sn0cAu*vLJck^S8=r>;Jai&jP-bOhnq zTj-2xh*y^?h19*>cWr{!>gR_At@>ohj&1x35HRx^8{Uwd>?HLRErjw|F-YYenwG+a z$3kR7C6)H$-YSh6%^n~3D;ebF$1qLf(sye`*J!>oo|R+?QmVvop2brBXo_K7C2#$n zVg4gRuo6)pzC?S%sFJ}+UJOQQ0B#z@p#wXy#3?>UK-Iz{caC6WnI?mJ1HDvo;~SQf z$4BmpTSqY;e64D2c7iJ6IWR7eE{1+gr1dv^C1S~hGR*S~0%+xD$lj{--08S>%s~{0 zPo_gw)h`3SH?c@t$j< zKar`=N}ZQDdfy}}(=GJ1|b3~pdR*F$N2&?sE*%}PRBcg{vVm;%Uq z^$l16d2*xh7$A48Yp>Rk zX?)M~j|pC1TSa-036uJ_^-iGaq!CifcIF|)Z-MC+TNbh>ohP#7PN>l5+fZ0Mj|vQV z-!zt}3wfaLxh))ER}(==pJ?fcM=hhvS*)oG{Ym48EhpF8G3@^=qBn-On*(P9Ti|9J zHo?vyBh<7Cn$^E0r7=f5aSmP1ZBf0im@2AU)1}-F=LyDz!~Qmuv4AG&tdwEJMhCsW z#F}DcQK6{RgHFvE=0le=m+fVHuB2nx{OsHg;Akn(EI--zkYG6_VFj_TT3baU%8&f6 ztRiQ~47PyFBJ2YNW`3)vD%6o-%qqyJ)YDrvYRNiJe%z+Qr@8*S%;aF-?_9GMj+xfr5CTtD%V7PyinwpO;=TxK)0r59S71*!lZ_y{~s3B&oi~H2E z!n*abf>^8hGy>vNqU+^h51`|GKbp-R=_d^3UXU)^hMKywKcHOw(Zy}VREd_Q=GC=$ z4r^JoNhb=38JdDVLF_0Imrr^Y046~G8+CGy9AYLczt8+t9%1%ZUXD7+4#Pz%p~Xb{lN}8suJ$c1}Z;p zzHCC`JyZy~+)fx`t0_pIW6JP^N^!m{Li~MLAiA7Mh`bO+Cg_LyA;Xga1?nz~cnkp~ zWrs1craL1=0?tQG1NimyDkg}`c=b9AsV;WWgQ~TG+^WGNUOr~`{gOH|(%4NT`8O|< z9yQoDa*ovN;u0f7iN@pV%V>nGK2L`AKS?4Cjr`7blknk*GS1uBbf;iA=vAVTq9qi% zXj+g~ueJg^A*F%y1zLzy(2kdzrJ?(2H4mx6=u##wRBxos43O{o~ zP<^q63=nlxoJFn%rJN32z74%GrYlLk3sbM>75M0eqO|?(57?aZ>H6>^Z zWw`t1d!*mb&>ERH5##7-4TnD>tOx_stuO-*6pApCRV}~5o5RWrI&Z3TAob{Tz=_^a zsfNjUvc1<}g`gVXai|51;PiAOKPyFxWzO*qvKDOil0&Tpb7B}$LxPOB=Oz`4LQ)eI zY@(Avv7=G{;I^`sS#kul<5?|X3Jf8TUD%dYzdFv&qt^ns1?kL@k7c_)!Xvg2*wTWO zhhLRZID)tG4LN=j&l`Yrt7=|#`=h_hb`75%Ce#&!?G%+8nGFaBO<7#Iyy>=~(Uj)1 zB*ZYpuE#8x0(Q?Et@40LEm(CD#N~SkY>Dy!l1DTv+ky&AV*CVOHC;bf?>B);)<(g9 z{Iyey;~}6O^9^r%oUH+9Mzh-3IYQa<=CEpA(yE7|g*_K2%UKuDw#@GSt`>BHA{L&C zT`(fIGT5;J2nTouUyYP&n02RNcIjhvVHMG2x`vsBGdjR_@>EsEfpk5Vr3t4~LC`tp zJ)IZ44Ubj+z{7|5{h;;}y$>e)GE0_E~`nd%9E*O$$|C6q^6xS< zY^ue8lAtU=4Nn;g#=JnpAWakyfZyrv!PSioz+j@K!$;SFk$5pZ4|hupxCENuuNz>h zFJndzRe&E=fRhE(xCPqsV;0y)xj|_Dz|hpB*AwplUetvePCvYkvqy!rQ&q(r2@gjn z@Y&Rg{@W+|F6z9KbR)c^AS0rrpyKx>e(@H?Wg^@GDDWpGYIIbdzG20h^`9*05Dt<; zGQP{KpTQouPgK&O zm)Mb)7+E6*V2P$+?Ka;GWTqyhzJZa6Dt)%F%xlp+$+5eMju+(itb)eE!L zXyhEml(H!4%ydGXa8^r?85YH|JszJHyyywt5Fdl1ZyWqDG)Y=R zOh!X2m%JJ&EVsLh_+rP9@)2!VoKhc!Mjs_T0eqxRJX>xdSOb`CvG#Ej9rs4!6Lusl z)y`L051-qCjB;*TR-RzXJnY3MYAwy5D=bJm7<1@hJ3#rBX)2PD;tmD+XeQYUz@ynk z$n*l|rGj*rsv*G=bw)&(1%T&qs<~)DT5={Vkz7b~Vr&J_PF)< zykbxRG((>_Zq4ijo}gaeuLK3H*gy4rc3(|;IJ~>t0v-VxJb7$h1J6tozj(Jl++6tc zdO6+i?oWLA{p|2Ga$Wrd&ae9d^g-XKI43--?c7*>{^AqIW+^U7SB7NayWrQg^k-2w z7bTsP#>!~fcVm;X`{bmb|6q6cJ?$*;hbX{1-i&``M2B{cu$NrGa~BX2nk6=_si=>2 zv)uGWi41y%ljq!}bHq-h{uCGn85ox!Nn^scX=Ad5<@}I$ZPY+xFUo|q(zZfqR;)_V zGXdW;hF&=Tp1U5mcPWrC_FGQ$P{#-C$}sho!NZ5&HJb+OcY0i$zka4h;pjaEIE+Df zNamcW)NSV@t4xim(7?zH$!ej*VKA9}P@HdzLl9o7QVuSmUN1rZpw#w(Bs9l8OyoxF z@-B~*9HZ1!R;r=wdrI_d*>?hxrRP+!b#HFO4Nz=LI)8G(0#EuIf!81Y-*3bIzn=DU zy?!#8W}vVmh1s{Ev3!oa>%&)l92JJijLY{S_JcZZxvP2R3^Fk;F|{<;^(7U|q3ThZ ziAo#L{$cj^(nas3WY}!{%fsbrm zRptx#7rlxHZ1*VX(1{z zP`_R}?Ux~)}bbyRXB^<3e4e@rKO?zdyZ(z2cqz5e|3 z$n{F!-G=KQuhNG#@t1||Yxbr;n^)dnj)X9;7jB0p+SlT}br@e(pIJNF50KoZ-+ZjR zWBjahYq#%Kdmfeg!e~&rANCY9#wmL?(=j!6<=FmWfBvyrR=@L}{mJ^}WufnMZ!5<> zu+@9r+rU0S;pGcoot94~(Q-#UEmHX~Qax*s`krX{Np}3RBAuJ%uHfu2k@kZLUFK8? zZ(T`bd1$!$)+8;yPQX20IstAv0iM?V=ThTe*NjL)rq$eck&6PuX|6&~l1h7Zcx|^s zK1hUPoV6p@b653^%PzyVlWRzwb`zbaJ;$Hz+mD|T_k|dP_Wg)j3!ucuLP^!w%cxcQ zf-xx7yo~}rU}%&%6mQefd6zVqi^tRUmq8bAZQhgQ57-#69Y>ZB--e=;wcQMg5Sog*8I$vXUwUu+65G;cS#>g zYhbWc;bN|nD?JSLE943KLakzOv5h>eRX zSkZFALkMnj&sOFW;b&bGH<04zs{V5Ce`m6VtrBia>29c9pT~`_fu z?KdRU10KXlA1yO`%l40(v>#0i?uMhn)PFtCrbx9Iu*oR4@m)y~e>0#rhV$TpY);;7 z_$A(Tv!(KvYTO3fC=(&J6Y?-qL-OXgLoM0QVMb|%@N*PhE(;r^09l#k>TR_!|e4blQD;fG3-lJu_7$FaN^&FFr6qQoq9;gmLLQP?cK}!qoY2FnDt#m9oekK-Wu8i_YCQN$ahQZd`)_Z3pq;r?Yd=(pb{4 z(up-ME2blNE4xQ7*-Vb@qk1onBnjkmk~T?K8i$D3N3)Qg0Y7dvoSi%@l!2Q7?5A!1 zDoj$9tzXI^ZzGukpwF)p-NqC2rSWO7xGzfK2tVPb@XMkwP#2?cE}zZjRqRaG{-{>f zyEvEtFOi_g(I@MV`Kb*rsgn>G%AG>zFsh8j-^<#SZB5bbk%?SwjZws~Xo+k! zbC@-I1iH}zCTYW3v9w+$y%#Cm(RUbgho#&Y(vxM8-U>dB;^U?S&FxSvy{osGUG*^% zp>JE%;o=WkRAO5q!5JJ;N*I%k1Jzmm2d8BMk*km8KOFIr*7lmEwU`=f>5_lU3m#_i zi(-xbSWWPtiAp3(nb2nzC=u!)pc=gBDo8XkIPm?KCTlf)1v$I2KiV4-;{{qp%W8r* zTByXb5(#}UX1QYAeRwb^E9&2t371f!Dc;nWnrfv9vnz{Ly0d~Ul}m5riV5Ow=NcJn z;kwD9+}V(jF07@@y7cCf@Z(;pqbBy@(R61ph@EIdSRF1=UQHnrA)t+HPut~+Q2@%u zD-Yi6ROidVQGp?5vYxg5Ilho(faBK>mZU@oms8L;@tTp!Vocy{zZT1-^&~-4q4d%g z^^uxfn#t zc$P^cr3=B9C3K{O!-OJ-d9Va%+AT{me1@(m*fX580BMx*ZvS9iH*4hul*oilONa{i z@$Nk6%mt9*ek5nj6e?c%kBly@B=jwMx!^??K|EX2TnbL@E(t#|4bm_%EK=9>2??*7 z4ZlT8h&y`QGey!e2_VD|Hgz!(2_V@4M>@I^hj~@C4v}f!+1Xb1k|B3l^tioK8*njX5R8PU2Ox)q8mgOK38t!`r)HqI{((od zbPy7uZ&=w4`1VhfZ(<@c77rlqEHi^qrvy=6i+h9*k!b4xCQ{!4sqrgN^I;{xdR5Cx zh~O2RoNWpWjlB+E?&=^T&&F=2I2_REi4r> zP(qz=Vj?pR4j_J6rUs*vSojC)?axd}{#DG?BUKhl8v@navmxCJG^)u;4N!saRw~2sBFT{|F%ya&1YK{dcLw zPI$i@X^Qiziw($}2=qmn;L_wU8L%Gw8edXgiHXFRszzY3UReD%LQ!-J$x?-XY8ON5eMl{4qXS{J^iENHT6J4GQ=%f?-_;Nr{bSRJsT691Al{iJ8a33wpPD4_#afQVg73 z?*}OW%pWuMQ7iUAiR4eG3oAWR_lJMBo@RD~0B*0HFJ60Jsy9A={_>!4f4w<;b#>I! zx?}P3Wc0K{^Z2y=p!#{~^s~<&$uIYRSiOLIEl-zHJDE>mt*84d8y$py-0n?keSUrV zf<2=8$C>|N=InUp@(U9oQAVko$EEvI=FZEDdlTPkXXnk5ld#b9o%5o=pBLGKsu$ZQ z&yx|QhhwD&Ey;(4+}~P~7?qM376fV?glZWss<)o1x2~%34zowMMRO}kj9W#Iw?&Vq zMUOYaW5>c{E5hO)kJO3JmkbsbO3wBgl1=Q+H;WUzOrCfZhEur+BY z%{CdzF8U=GX$g(7Xs=c1K}bhtLjrdLTLLH>>%c*EjgR{*vmm4`YxN;*%)51K`^YtY8Rog?li>{ zv&FUgTIhMPr7Q$66v^3^+$!ZYK-7nDaZa+qz*kOTxEFF^Kr<$eD*uM?QHG z(T!4q=M{!;IzyqrUO!}@RY6*5ayA9QQF8pgHp0n6 z$y>G6nd@vUGBzM2MjR^JkLA7fywZGcpcS=N=o$p|wwDu2Jl75xXM3=2 zHpUUHqLi^b2?l+y)9!5wUA>gQGMo%Ghfr9)D|LMYpet(|)go*fOg5towX+*{P2;3v zm2=_a962ADA9$?{uDA%kA25n1<{1+d5bz{o>frQe1mfLZ+&kKKhB7gXhMI2bx%_x; zD71pwl_%(Uf;ltF1AM_wtYj_wBsQ@?Zq{hZxqXc;g5$U1J_xSfi;be~NvvWzhE!jB1ZyM{KGbcw5_pRF;(V%(v$d zOsH6E*Q;EE;qAs(n>WwWST*Bs$#Bi=_wgB2{XB3H74{qYKd5og@~22>YRda>>K`ewzIHK-F%Mr@{?k^s%miX0s%O0HvEiqwv`Ve% zdvb_I&2M)87byo{M0MmC{c4H&El3oc)ik0}KcdliqPhixQYCZO!%r!6xFnXtf`(~hXQ7GtE3Pp1;297TJaIpHS}jF{VF;}-On z^?)51aPUUtvH-s}1a~s)wi-o44e$`brd3T%*0eRorC$g`GN~@t0L5aOnp2mUR1`lM zldtKAE>FdB%ln8LMrIA9BO96po{ZU{69hs^swTmo>%`Mx{AG-~M5LmE`;S+fkDqp% za9bG*m0ZUJH7Y7E8(JKEX!*#Qx~@sCApsAKY_~3uajtlR5}|?;>AuWq1{EsJZ|x2X z8fAR#n;9w}KpWE1b>|oJl%B@KB1DQI$-H*=xNvX2QklPLzzz5)+)ROv6?NnMsTGs_ zdu>ydbMVdgJO)O30$_fN!z;tXPew&4qZJSz%dilR$r@l zE}&S{bZ}ozk5EpJ5neUcNI#l-6mFPsYq~F&KOO&C#5YLv41u(r{9_m#QqW-Zga#{0|ibx1C*a?8yswcZ$ine;06K*H}~;% zP~^lNuL8={g>A-o-_)34L|>fBJp`;=`8M`yTA(K3-~UAQZxp*)Na=iU@V*r1`_@@Q z)xE7P$r<%UB{> zr(86|ZL`qb`4Yj+gcWm=;#{sXvCyR2MP`z%E>&tp{T*wwALD%F%N6OoSi_PIo5AO3 zPi$5(l?P5#*dl6lGSFe$uvk>qg^^(SX6ge}p~JaCy1$}PU+DMB8BD^Kn&j)XiCaSzMrl9$_)C8SV_DQH9N}LKfF-bb_v)E{rBRmjDB>)0w4(zVIkJT0NV#+ z($@Mb%35|$b~1%Z*olQiRr*XOfRQZVMXQE<*2=q6Klxl{Mu>l)i^?8XFDuiKG@M$a z_nZ0ZGlF#W6DCKLX|jDtvSz5mN5I>4`{g{K+Uc_s^OZ$xRu=x1ZTvv+)DmW*%lCIp z^?n5$x7!|`e1-V0RQB^fZWJB^y=7{A_0?ryPB|X{j+!P_>I?aW5!wSYi1xBL)MbH# zbcrR$V#Uq$ibS=^5m-+x8j8B7S-!vi^64>^W+eg^lx2dScX0z8ld}>*fjo%+d|;Lz zH(62d@*m+pik5EQ-2N|N@Y>4%C9}L&_#efOVF6i*bO*aK2Uw4VQqg)iD#Rn6IBaGF zsumP`-nD5t#G06>#I5w?#I9OWKC`PeleAKkGSFb_&DIzt)jInC-kCg+Eja1Ke6{F& z4AYPK;81#G?8EZ0Kza^q>R@M{j&2@vd@n?J+dHUu?tAO%I7~05+ z8r_6~B~uE0FBaTLyFbbNi*Rzr3+HGQ$#)Si0)$$pP~UIAVWN$QI3`Cfr1r5%8ThJ4 zf1F_C8v6RQczx2Ejb-%v!W$?xBF}_tzLBCymt>fEmaU7)@p)u3Cz51Zgp3|Rl#2ch z+R2Q|teFyCJ-m!3d<zx$5~3LQ<&((JKT>9Exir$bqxd`jb^Q|C2U22&s)%ldj##& zWy>+vj3u?P>v02>QYXFc9zpZfVXl49q0+92tYfyK7`21c1Kj%(IN$$4eTuHmB}eU0 z3NBZ@huKOEPOU~r)Osf&|Db(|caBw6_YStNR8ah5qHv&Bo>~b}o)&L3cG)*-^a7%9 z)uqH&GkXcbx~pKy&N}!^5Oqg$%o&mX*-Y71mxE0E@GN1e^f_M)kHuDcViNQQ)4nam z%R!)>Od!`bMUXsR*2~wS;D5}vuMAwe{YPHJkR>|MIv*n8BiM7dij$-_r^;Mo0Eu%K zla!sSdn`uQ1{nI7^ld5VJuKy7CW*Pn>S&rVCGJQB;1~ydu6lhj*%MnYy%qZYYl-j*aGwTO%^`X;blL|2yDT4H)o~{>Wx;vaZx6XC`T)%EJw^Un$iXAMn|S zy#F9{j(2L*IFRgESJ?Ma-w*IDV{5x&0f0)vK!|N2;`^$wXM z&#{FFUi$~!qNWQF{mTVj<(~lK!Z`JJwi~XG>6RngQ)pSK-;9)%Kj;tsf#(TZF5|v| z1AcJY#@CpHu<4_KnPUt1IiXlvWE~2pHha{ig0_CCH&q*qObBq0P%Ofvv%fnQLDh9MT zzq4|pIp(wuNWo2zQH;{AYj=S^vE|VpQEx9Q_l6J>kum&%&hWp~xKPeJ32z{+2>&u0 zrA*1|Q-q`Tqc=QdTKgyXLBQ|1nP*ooO@OCVsO{tUz&p|sl27PADkJT$Lp1)}Y zf4??|3SrGGJVpfrGYt7}cg+6#t;E;+W`A!b&i&_A1C6yc(^?po=D7D26h2_Bd{04V zB}Fu2D;k6O-&_y!GsFTPeqZXX5(yueLSdz05|I276QL~26E1D$#KGGFYD&D z4XYMMYN^h8`+EhXU)H_|J~`ETHLZO1 z@H+cEI6lE?AZTNM*5U2x_2>MH-k)-Tm%}xKm#e)M*ZZfQQ}bFOKK5T%7k9s1J6|4m z`uci;R>q!hdjQ_xRHmSpE63K>&UKsk7cPy>9Xy;ZFWxa0T0V|neC~ID5A{T~ zq7$t|5uFR#sIMS>S+(yt$xJ)p%&fc*asap*I|S1@y_}rfegRzrGEX^)HZopLKBwu< zr5Ut&Iknw=Ipf?Q^zED=wBKz!y}KXy63%5mQ7d@>QI$Br{CO>X%|=lF<G}30^5y09%>2vKUE3fb+T&JK(DU7&PKAQ! zv#2qrl|uCgr_$ef>U~b7+j;7{PNi3Q>T^z|M|tYEPNk3eYUz_D9(HqWb)R6$m-p3A zWlQgFCDxpB&+?VD+SiB8HZniCDhHqCyZt(gDXe>17t+m`KHDx2lv>#zIWNKa~uU^oqw=T#QAsjVdRN zj4`BWFW4ZsXdj>-9>Zrrof=2^LY0$BRw33er%d(cQmdVtKR;12zLUQsO?s3!HecUY zE+-KAX9l6>N-|Z=aHfy0_ZKne^wbkQ*a66^FqhVbz!>>7AIOHWcn-O5MhWBF6zKuO zafi$R_pS~uEVwsVa%eWFKnIDR5oy-ta2&65qzfVf6em83Y2H=TZdy3$tD)LtFu_Z7aVy(?Q>gn`Q z`_?YKTDKxyvN5PxDiKS@JE9mIg%PFmM7dg8_Lf<2ZJXghwIZl22WXhdXB(Pp0*fQa z>%62b3+)CY-C$~V+P&JnTshOHs|n6u36eoRI&-GK`lv^q&Umd%qS9zeqo4C>(c(HXV8kTyp) z3@{7_!K1C~A<|HSM;_8xi!sA)j5hr<2(Kq^sVx&gZ5RQ9G#i-=p-%Brr<4>Hb0>M3gS1hbs3P$pI&8EHQd>pEHmN}zQl+ZC~E zQew?~9Eam5tWypd@+%URVdkK+rk19fIj0$CHegw&XxXjlxf&yQ6XV5IDaWcDe-WT5{CLzGp<=DXwub;3lgY- zw&#&9-ZOr)GwUqr)C5|joi}}l;HZkf%=<~+8Uer|c`M(ge>bKuw+}h8=|{`Uy69am zuTVF;gSoJRLN7p*5(DzYb%hWQJ@V6-m*t+9MT54yW+p^p>5yClca)MpNO5+&JPu_X ziJH#LpUK%ljCuISDI;Vr8NCW2T2wPuR5Pkwd^{_z*#oNvDm+Y(>+Cp#0PH?goR9Fg zdk*?s_?ZGM8I%N?D4IV0MK=}N@`g&_j%>XgBRh7DtU{8|&0JXm3eZxlEMcgbVnCod z?gP)+u%ixR0fAE`fm8mBm5s9ux+`ySmrbIa&Z|u?1P~?bMs!G(9Pg24N4xWAyt56n8*uf72vU6V zUgp|Ed^GPMK{EoeyyCHXAMWW?ojIOTbby**)}FaKpei;2Zy*-Y%_QZP3PLjiH1ZqXNsX;Nai;7Kg0kXG-{wyrhhEdufJL3Cr6qta zlR?rv8Qsk@K}$s~<*whuI*e>Z=*$P z^k7R(CQ-0c{#C_A`$yF*L1bt5&M90U&>dB*rh2d7{kC`Zj`}LU37ZJal!`tV78ePu z22jtZ{sZP4g&1%&awa7N_Ae(OpiAcs!R_W6vcFiE9|1~c8+l6Xf^$Mk?o`&jta zg)=XbWH)AC`=N1ZxFhj4-ZQbTjzQRUY-{PBAE4#>{-wRBx4YEiJFIU7<@<(GnBYUC z^SOLq``A6MEu0q8>P^35{jadCe*DzYYwmh5(dp*vouBF-(%nl+Ptw*jjK}*aE5Wma zkbA4JSgwow>=#c*7c`9bYM-qAz_S$kmC0lAEv_a5U#Cn<6UAsjfZ9&A#`o()+j{_WoN z9*Y^`v{h0R39KiP6U4#?%f42~cckX}@4C=FAmkMHy#;eMBOi3NeIqgp(DlxKuFh`A zX|4>glrX}CsmU}07*GM;-`qQRZJq3;1rf_-j7#CKGOcy3sKgQ=KHXEA+J>XD${|e( z^%&r(r{o*s*t}h&byjg_atG#9VN9w{Wz?ngaH)%o!J;G>@gldWr z0mgC!uo+uTL?SF!qj;N@8o{Ifw3&rDX#$bms$Re4;q<+?X%8oXr~@LIE*}{|H30c@ z5qiD$7WlGXWT+balWso{_n(Vj@$rN;ibhF2zycE zk_c04CjN*3r9J~s$H9co5@`;Ld@xc1NaW$_qNklrS44T1fKH`RG;w%8xZNwLe?}Lz zX1#CF?S73PSHnNUC)CZc$)R&uzTqW;>O1s1Xm&y?Mk!t{|a@ojTDJ>kc}wbhx4Z6@l8Y!G*zS31ues&no8!r z`gy|7<+|A~KbcWJ&Mbvo5=H5-1JLf)@Ao6FF2XE}u0g=m;4Z;fxn{(1&uV$+`v^$GWUeER3v0GzI1SkAQ6JYl#O{0_<=f%N{L#8T9NdZ^SrZ` zVid@)WB;e63Cmh+x#&biWc&9#0Y6gT#$CTdVul8-1L9gv)Y+GDp5YjFQ+VC~Sa ze89kU6-7|w6r`BPkuUjeCXWHM8dvgan4cnW%NZ^43gder9-$1?{_u@dY(JgtfWS}6 zkBqXQ-WM=#kl=}EX_k$9Ks7mT58MsKWD`*MekinQw-s$`k?KKkHB%P51cFRYyDby; ztBw*n&PiH9WO_AI1CJsb?I#a<7sn2EOn08rg0&?MS@)LCpEwa!DM@f-zRG1J`S2Pl z^J-Jk=^Q6y(0v@}p{>E#ZWU@;G)`6X0gBm*ye5A~1PFa>RYb5)KmpzHyxWlwsQeyl z%Y}0NNk?b=kkGhjU=)ugCI{UleT9RgG)l(QwVe_Y@M)rV$gcEiWu{35|K7rKA!QrR zhd5L0$T$k@>=$nCJ92)*IRHM~fb+E_{MRo-2=Xj$pTYCfe3{>dS-(oU9!@iLRemV} zy6>xs;0Jox75D(w>Z?Wz%QUKupePH)cv^+1j`fyju}2Sg#EhZ(4x=)4iv}U1Ntoh zg7F5!z7}Fv0R|Wakj0zyu`jTl=1|0=?Rwh+^WTl+FL5sVs}|_zf*Y9a0^q8-ea!vj zj6Ntfx#ka~7_xl_9G%Rw0gj1ZEG*Kxl8S9b9ppmo(aPwo0T98>PH&Y9lf|4ucy$|e zNMsT{A)(>Ps|MsPI!Fr=Hw}Igil?{w|NQeSQ`a2|1N=4B@Oxvn_mGTOzH=1>vNkU> zLKojlTYOi=c{0jM3_b%x3OUeJng*$(hzB;H+19$uz%}qrqOH;ST;UoyhX%rgx4%v5 z!~zi4!f&->H^by&2f9xtfdFR+!)XJs=F68-c*TTVe#0FYgyKoKULb~HnZaRH{a8f= z;>Pt&fiOd!uAg+npU`_@YH$Xd*<_n?e;*pnAp6Wo+n`Gbc=@svL&OwxD_8_ocG}3H zf)F5lI&yN#Fw3%7td?thCxnbF(>EYbwdcmoEWtlMf@$^=7r|-zAeU9tt032h*I>So$eA{x@z@0TR z>g1eZ)&AVN;^|=pz^I?@>qk?x%bg#BR!^#zqMU(qE9Z@f6x|lCsM)aK*6`Wl;MU05 z6yXO!?8~0|qvIR>tmVg7b{)x0AnF?0n>QEH|8`FJ@29P=r-lEVwoVSeTd-orEA^fPr(}z$T7M+ik;nsBPMjsN|B&6*gH>q8mEKP0?eNg`vtW%_0enr4+u0 zE3`G{zA92wSR@s&g4kdgDql{VdXL+U( z$eJ7od(Oon`aF`y@^ur95yA|W9K0CHKiu?SN4YmIaT_7sZ4A2Tk3Sz)&s1H^vT66va6UVEqFNi}dBm&bXle?b<$_t}SUkQccx{)<)_Kh< zGLzjLk6$Jd4h**U+Ij^VLl-C4wpjS=g}+< zVDzdU%ZQ7NoN6ZcqxK%?xnl_?j5T*Q4WYSn6xlN{?}}wpKO0)QdY)E#KDvHL$dgUq zd2+w^u#zr4*_lq!>wue5rdIv=5c*z8;yzTm#O6K}gKDvDosp`gjYKbCMY+2K<{kjn zqFiH``(X1(LW+t{6`jc07dzM&Gx1O2ve80D<3kh}D(953Qp0rY@vd-|5Y`8mGXGJB z?IpSkuX{nz%DNy`7)J!;$nUPDGO>L|l}dDg=glDxw�y&3g{!P0XT~n8Cn9lVXaj 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 L&%C@wwNU#nf9-_p literal 321069 zcmeEt^+R1fvo7w%p}4!dyGxPc?i(rY?(Xg`g;IRu?oJEEo#O5e8@Tj+KfC9i^B0`E ze#qKu^5n^6Rx*>7WGl-prf6$)chBL2};+-jaA(YRR& z4{=XVWBab1NYAH}ZWME>&zQVTa;jsk_{}!lYOTwBn9%!9FK3rd>%LbrK6p>!6wegP zk6naqs{Z~Fe%&vR-=o~Fr*HI~iM;$H@<0Tg?T>fa$17#OCfi+){e&cj0l3}q+pU66 z^6cl~zwVFi&%zF;8IGH0dlaCiUW%iTcCYz=*P>5A z9ly?;e(mSIW|*b=rnz1-u(2*Myb_H=eFeI=<=5ETgcv3NKoQT7AaEseMFTNfj6@Mn zmmqK?a+N@lA|XclJtRmCKC~PiL5&jgyBv*Hs&WfmVSu_AX$4|vB~yk0Cnm5>dF1FH z&;R*gMPsHQPHK%efkT1`EG-Mua{>GD0VXVDx%+ye` z5QpWqE-_leWfcn4^20o6VW@eii>vGLLNgakLFC%FrVWAv@<(|45yBKE2l3u_dJ*R- zACHP^Fa@z|Tn^%~@4^smgdD_u?*Q(QJ#kG}=!3K=pONoX0gn#q?^N&T z?`;q+N|r(b>Y4#(@IS)8zEcVoK02uBgSSynR5G($(I@<(m^jE|>8B2HQOfxHcd=~7 zFF|9eWR^17j2VgvEA%}McxR9ayh2|Nm9gSK#0-|`*c|YPSs~C0eL+;lfri|rvRYFV ze~aa_LMRpblBkSjlWR+4wWcW;E_E#}(bG8K-*d1O`l6_eC6jB5WVL=#R9d2IaKO`p z?8N@c$)Yk=NUnXSnxUw)LZ9b=x6TUTROrL&uFLAK&yp&nc<7vi{~#`V(O!Xp7oPO{ zUXz*wm9qD|R?(6|E{L5KYz@kSj5#NRnZ6|5&w(n7nWCfy{HMhA*Weq@o=nRwt%Nq1 z77wz$amYjKfcwZ6Z?Zjl@Iztf4kZr5XK6#oKljtW?gLnutcHuyFSNhWqe>oH*{cg9 z9=8Oa-u{AA{{vbetq-Xug|;e%rYDWIDxHWV>^>BW*_(>ln~T}|6|=Vrv$qkmw-d8> z5VQ9OX73#4=%urTo+8?+BAT8O+Nu(oo-*31GMe6JwAIgOdMaqEDv1#96oOd2l32Y8 zSiS04z4}`D_Kh$ zSxY+^mR~G%vnJGnCe*zq)JH8SZY?NHEhu*_sB|r;W-X`%t&+Wk!BRo&aBl|Wdo6;q zI-XTi$ClwOhxoxwqUipAX5{=-h@Zf#!xj_WshES|u*CPfP z*xMTM;p~y$Zy_*!>9K!iTEBz4V?fs&df}I(*GYn{}%_%|7dz;l7dn{3sTs* z3@Ci$er_!RRZQMpRHmI&%`ZT96{Rt@m;!&To0EQ?A2;NU8)0tQ! z&L$R1)jlIOq_>ZF>Z+E!R<$O1PJCpZhGsTB*Am&?7%x7Iia8@HWE|)ZY?IQ%lm)l` zIt?6`ZRe=FxiHisR%&fuXurv<@XxxnGt1C#<6Lk?IyZ?QR^7F4@PG~n z=4CteXQ_BHapNug*tyd4-6V-@TBpU*iW72N`vcNHnHTBB_c1HY{A59DCc!Ap@KErqg+Lq}=^k6<77gF25j|!@PUB+-hc|iD{%pXr7df48gGOz92K_g$@(Sc+NocAx zvX6UJtavcc@)pM`bZryHOZ?L)ZV0b17fNjok@4)k^!DGySSTCyGTB)OPjm0)mDj}mde1~^WA3Kjrg!5hl2EU1pQB?id z1B>DmMDP{~y+6u3i?f&Eu9Cbv8@fN+oT+o)ck;RKXIg9;u;?|spnT*;wvh*xS`i%>2NTIp9mv`BKl%G>?CgG4SH78&B8tZpk$1X7BjYi^E0v@nX2v zSq?*os#aKlDvPto$X)E7HpUK%a(S-!4>ps99eG%!p0Wv%Djd3kEyb}ZY^AZf3XnFi zRF(V@YLvPfl<_6b%v+{uf6RcLylJWn@;l`G{4sfFX~<>JSxI$BVe7-N>xVvHow1(i z;82ZuzC}=(5R2Qiu=_2s2W6^gG$f7Uw7BNeW!1MB1|xTV=qHF#!>+I7)jgLOI$tM=+`y9iH&f+ltCUJ{c?Gz(^ zvdOa4#E#M278EW*PNb%W+bbcdD;vuS%i;h6s&8yj4Wah*B<5pD>OXx&=I8&SYjBrO zYYK7aZ9(@**qrn(TfDR3baAGSW`GF6RGUAq zSM)#-?pYP>3<1Sw&A>m1G$0V;NVhN+y(&wV1(xL^s&X{0cT-LWKP3H)7Vx8x@SG+i zV1V3+37TMWXH5Q65N<%=c8=heO>KuO4CcrCDdPjYk1j#i$dJuF1#z1f+cpUBKLwEV zp3;FsbNRnh`g@3a;&g>ulwE zAH}^|ju%OPF<#tem6htg#qaHGQ+_pE&p__QPEI`&AQOA+oG0X!zbr%~rW2z|FqZs- zhggiM-}AU^sw>3yrcJLJ1iB}%^JTTTo#nLiI~seM94@Z}owpAEFkcu_GJSqQ4F?UO zzwYyNmk15%QGK}P+8fHXr}X!7YNmGQ>F1Iz;BO;qFQ(s#TT9+u6in&Y z2p;%W+bmbW&AF0;H(Gw1<)lKqx9I!B`LO)3@ltP9>Wkgz`f6XAv%2+bF>$mQuZ=TQ zTQT>EsVihW7QQBoBfv%vcL>Y7#)!0ji8diu$BBf~?GSw{D%L5Krlv-2G9~^w>HB2l z%_I&sB7Z0u>q%dzEB%QcIm-eK{{qbf=TKO)s0Tj&I?SfNVdMVp6+D3<%B|IblBO2x z2^jcwaP1siO7RI#G0RUH{+~24>_c_~Vf?K8qHv#3`xH@w{9$T&0ky}qtkATB>$EMO z`J$-j+wv@RQU6yDJl~Q5*k37P zD=-LNU`pp7&o_R!&sX_=K!rocGo0cJ;j)+*$BVfzT@ed5q&tp*dpOdVXhxP;ii|P0 zO8S(E_PtP$)V-O-Mx7QuUc(NFqcEIjip&VqrBNk39n%4A5u$6qoY+t*IiO;Ng-n=& zPJX?wZknt0Wx{)sopf}4GkMG3so`*sIVUFRr18_xyp9&)dQh%EE0riicQ6btm_83>ok;q8^>&{{j$FcFJ9HwgnX>K!lnyzb z^R2&EItN~Da{u_{uhVk+!Z|a0oT=*6lUn)~O1O$T8wMr}n^CT)8 z@NgH=i?;y%co@$?>&BJekz7ToJ)U=lV6FY`F?gnEvmsC*OUxYI2Q;dP9U%t(DPf>s z!|(a@ctQ9){|y8>J0*U5nndvjT~T#BZmx6-zr8ZP9q-q^KXjA&zFu7mzq!9XosKVT zyn%LqYz{x{t*tNvPE?6<-X@zt8~&aA9WRG9?)LsqS3zqG*r>XUO(bs3S`sJ5bs>xO zgK-frU+OSS-f9+Z4JN$z@0GzWzG+LMiiz*}_p%0S=--C|sz2hEXS!)`GW=XEd& z?`lBSY4Z7GviA9fz|t)^3GQ7hSMwNDDzF@y5^pRbuTEcarzH#O=kz68|=__f$f8iH7QE#%Ko4ZNsey?S&pxK zI23T6I>Q9Ew>ge|ek|DC`d5)dy%EK*?l=eG25>UEQG|#0tRBH|Q7840FgV8?DG9b4 zkK$??iiBAKkQFC-a&3Z&`ctp4Z;&MQamn`tGupK3`tM3HmAJuZiRRf9hIQ^d>#*Ud zH#{^pO{&H~W>xWS2$MSLxYoP9?TwaA_yL%a)|sWIjL#if<3W*RkJ_Qpjda)bjYbI6 zvMoQg=u!L{HMV4+63FX5&D5p&8dO5}@uI7@XjQNY>mrsqRAM2s@6i>zdpk0ajrF$t zn61N+K}|07i5rgcVbxK9ZNmGxm1%1y4u;DI3jj+g1{MfPL&AUIT7+~k(%7k3qT{;& zqVRdpYcKIBoD|!nc~d-fVQJTNd6X>bH)l?*!~yLg;6sEKD3o(v5%{2gXX7*fP54kd2GC(%F`4*f!9bTT(6|??h z$cnCj`lo7h=E%xHBN0tt@~$ylKB}FXn-^Znks`6C6p8r zSR69EGh?#ZLovIAE} z2x%<1(XkMc7lro`P);m(MOfma7ZMuB-yB*a3bvn z!PR^4PC?+hPqF}R=;`%$a4?j4hZ)#O>Gek{h-mUbhn_Qy;tQ#)CO8>`oJc#f`u#n$ z9Ia2b;3*(0MjHIEj-xpA#eA?%ps@Ome8HTHk$;f3UcIQ|8fW$o&zc-n<^U%}ePywC4^ow=Gt=2zdzvT8ovO3wR5IvG`Olpd`$XrX=K&IzbL8iV@6^W*!R1G&W(qS>5& zzo?K`LMoxL_A>0%QBgc=*T9yDNoOZIYXpOe8y+_{Y2Dc}%+FdB^cXmK4&;(~#Hin^ zZ9ScsKsKot^BoqS%(1F*%WL&VmPjKZh2_1X2t>>@loT|IZ*;BfhP7Qa3IqHoIm1gy zjH>W%xShJx4q*x`Ws56%mV~l(vLR-lQbD{4;axQorgs${yMn1w%nmyQJn>W+7>5wo zc8MtOJtw#(DxaRsWzaTHf1B`54M>>FlaMuiCyx)2djBh;**~G7(xDMw-dbcnOw{ui-H0z zEuoU&o~pcA4A)1d#cG}Rsyqh%$cDVtK4g`<#`E+mzSG9Qp#qw@Ud&9kEfW>bdPH#y zdH#u-nI#R^WSZk{*iO^)la?Rp;fKDYbu~8CR`ggJ@oD_j!kP_X(O0;cj7G3n$a4Dy z9xYqaQIx(UkeZ5$wqGTjzMlhxtRGBVhABJq`j=?oG*)luFxA|I(kQ;M13{a(98|Ok z_+MNfC9yW*g_MUPF^Zy)94nRu6z6l^`zfy?ZfAGbsy+GyEbEG1ZY8cn#P$>vfm15{ zEZ4TSb5LcgHU3EIZd5%Sd+Bz;@}9}UZ|Bbmg+sl#JnY{m3WmzfrYZI@Xf`ROB4Karv-C)vgBFIp401{VYfeetbl@ZDkA8sZP>M-WBTe_MrF7q8p$ob|24uG)pmp zACV)Qpy!QH9^V9^C-A-FibAyPc0TbsvaR3o$^*`6eL2j|4J! zdsAQ8+Eeaxw7onfEag$|fbGfu=2TCz5SsE$5v`%xnS9klE3E#Uy?2WOAOV+C9d*R* z^w)2tqtfatO>4LUu#W!z@LXzdajcAs!L$>ZZ<&|NHYK1s>Krt2N2ZP3m$W;Zq{E6b zlAya;Bzmq=$X04XG&{#WsrHmZ;+q<69OP;L*+>X6slqsk^!P|5OfP#_n=ek4*8#T^ zHeT-yeS&_UOEbHQm?G#N77GpNGF!U&38wUtEpuBfNKXoX#bF0^BC-C*Tzzb=@uv0G zeiU4z0z35jonpR->H?Q4{1qAk8gd6Ebw+<_PcHc^Y~*jc-M=u9L*2(&V~+94N)81P zU1$^j2%1I)C@7dZ?g^yjct1!))iVR>!Rd8I3M2%$@Z1kE`aNSB;5iQUn`pGf!$C{~xl^3yb&=nkXE|hVY`86e2?PsE-wO;- z12Q?Y#xeKMs(J=GOFystDpf?3?vY~&z3S}gK*-W{ojM=j{gL3eTkc!Fse74fAcr$KO=z4a5rf%;ZfMA-++=uRE%@d*-|`MLgqN>#uEf@lX(OyW_;k22%sqj6Mh zc=*=%1k21R6g?L;27RxDo_mPo8X6amHrL3T-RYw}pw`ijN-pHF=ryj^5)nvv#o=sDzbIpSxRrcBdV)Df@(7jAsPI37 zyOzFG2GOw7gP=8x&jM=EC#bkw%+{hW}IgUm|n>m~-6}VLrC4G#7F@t`2Rm zyKO6v6{S{Dr7ZnJ?S3DESmlD`;Z3hE7ErXFEV`9iudEF986qp4x+^ zvUvd>y(9I#);P8x-Z4~cThS(qmCUsb3`t0Zvy&;iK0FF!gc+6~H`ayNl& zi>^ubd8D;owXAI4XHHNPPjrRmCc*A$=Zf%whf;nwqqQTIr^5m zn2@6Fcur!tirA)ph6dXvn;S$H$2zEchRpRDmo=iL1};22xvrI*(m7|&<-W^zNSFxR z?C?M1!bMb+0mmZP+tI4&;xBZ0PN2Ap*lJPjQ|iYeoG<pX-u7Dgxb zl8?w~nPi&Ae-oT-UN}0F%F#`-81iUcMZq3sOSgm>CWDkaDu7n6e80!1a5$Yb<2X^7 zHeq=-=!GvF4R>kpf=f#-Nlv6&LWsFIQzd@}|6=_#GlNF2>lFq2lN*u)DORu9Rl37u zm-LSPgS50-(+y@T2|Yi|7P?=6rW%on@XU#`1VCOX@j1jS6)pV&vZLq<_1V^z{_Ryg z5iOO`Taq?-%vFO*H*VFLf?q28MIgh+GwP+F(QLaH5W#qN-bMNS!R_gKSv0$erQ}mA-Z4qqqS6IVs@n&JtTjYccE>1 ze$^kW6n(Z-4CE#(BT})NBXnQrOD%MfawKf>nCc1aDP*yl(=H6J+aL?{C9}rRUGVp- zsdannNA8Xh(dV8PxMxzPTRm_G{dB>MN`c1v;32=yU)mrx|JBR}`+@YpwUSzTk(Dl- z&=;_pZvq2;rTXg;V?A~%C+G!B`a4vN?gR96@vDoWRl6%K%#iJpvO^YHZz_@Lv5w(^J{Z;`F^^n|6y#8s#*Y*C=z4^=OmrFYHuq9t(<4 zSk3J&AR3gTRb2buUuE`-NQeBQ?nmIcu+;)T6w1^cgWTbV?LNPY4!$th;?zQX9YcT- zcX>DbR22giyl@Xa9ZDTB2kmhMlDBZVBK(vlgPbSA7FKHbouBSXmE&+gXKuJk|Ao9d;J1DCg>FeltE*<3Ie_E>dAI!Ksp zWNToKT7|Lr5F-BoULCjDYY8~`GOLNZcup}&?Lrs@>LsHy@iT%a#O4u?E#B2rukwN7FH{#{Z5L_r7M)gK zsK|J4_b}bESW)`@sZdUB0m3P{10B8Y07*_4(I))1sShRiZHpDbD1r9i7bCuD8bhHd zwX)%62}Ci9&XEocyxF$e03S$nk5hP~jkA#< zq8T=F+63h89}11`IE&Eiz$N15G&0*S&xLpS5}`9F+2ZBwZLt+4om)66?yetx_Tlq* z3*1Ndp)(ZpK%!4{O8WA-0bd$oq8%J9dgeFX3IGr%I5+rG1emB*V>X$y(tSaS;O+FGx0zNHc3shz4A6B zORmHg!+AWoeJ1`0P3#ZyiK902vis$INS3^b4NOrEn;16Qg%v|Gf_^*-Smr6eYL-4v zx6$;aDlTKf#kYq$mnIGcaGP=?RYjBO9YA@TM5G=%GV`+uvP-!%VP=#7)B@(QBe@IQ z2j;TF`AkEmYfN&K3fd*N`pR6B?~Zt<>p3b;99GBq_Iq*sONECbQcu@Zru%3t0~4_$ zyWOH=e*bZOkYf4KC83`~A0J9^3^I)VFtuHK-9DmFPY_ro2t!;4IOZcs0tx z5GH80kLCoqNPb8eAaxkeismPxtb+r+#+8IE<+;Yw6}~aGre>AIS_s@Nd#9ap62Is$ zX*e$p<8up{Ha8Um6bRe(q!L0p^7W6>H6Km}vx1mZ{gXAM&SQQw>O5-pqm}gs^KZ#f zUbNu*mClXyw8gxLwP?^+Af1eas%iuFYcB`?@|5p`c7@!rN<(7_sJqoyUg?zm5~dR# zGCgIwnqK%Ky`@uC$yQ5l$ZnRpl8Q_8Nh8lh_~0t>50lNkdnDp}lstXeP{v3!YKw?@ zV5>DG6Z=U=Uj1kA*G$4`|A`d%GIA0l+5jVArkpViyZitzmhwNY0-fSdt+kmKiSV=Lf1J*hdu=c;;j>f_@z9aXpOQg~bKr3c8RKF=m z3PMT{T>6EzI}W@!qg2o#E8BalUisXIvK&Fdl+$KkpBV-YOmpsPm5|(KRpGf4i4q`5`UUhq8EDvKh`}dj=f)9Z(C;yZTf53y{Dx?O$# z-{CA*A_>%?O`@lGwUR4y;pjh@VIpjalk{LQ3``AtX-T5dZSB0rKax)Je>O8oT#-8W z?q;ti-k$J_7+;?Z?PX0X_oG;`EM^NQ-X`#i&{TtQ;sh|1TqC_Oe)Wy~m3bTlpFV~( zK+hNKQDbhDc_agjU3cA66sF(K87DRP<+)QcomL+AhXLrtenFfrx_bE-HAW7dIIbq0 z^ge^q?!O|xdT~Z&&^%i&dU;Y1H4bISw!(+viXb$oMFSK=M?}yXbN(?0l zN~w|MD3Ap{zaMULQ(qc*-P~U+1+}QVxzFr5UomIz?l)-XmHqI%tUK-UT5ykEX=-;1 zNhcQeQ^0@D%=X-G8@$~=d}%vfU0G!GYu|M7d%oNEXw73jpYMJ{w+VU9c=kRX&gg%_ zdb{rNi3LRnkS`6W$=oReA`;ecw z$K~-ZQY{*z1&2bJRfeG5GNEu2+-OGD{=iOBzD4>BmX8B=@9JCNC42by3%>%wFA*%s zNUYm~Gm+6{kdrcDIdq<0mJzh?vJ{)sZhj%hOPBX{tFvNenONH8w#0hy{X;UDH1+$? zZN4}8t4pA{9QUqHTLLV|q}QJ*?QH7zJ-={s9LjLxs%=2L0a;9o1dhX6A^OlXp$XB$ zn}2zh?y&fv{bukZibEvJA9};@VGlOtiAecLK`lQ7bl4~z@&8yGMY~~)weJf>e?7$r zbrN`M%ZH1Yo$nT-sHWeg*nDIVwz#e>N9m$hz$Rt<@WjWD@5guD1lAO14F4J^eA&OuY+gUCVB*XIIkTT;gpENKavX^dsZL>Nw8LIo+x9u`IaqCji zx5Sad!mAA@#l{WUSvVu78cK=ugF*7}8EG=}sOq6DUIXE#mwU43ydZ$;V%_HPR7y|t zP~Yq0p7bEbpz0v|piLFunoxe_cq{X>thnk;SJ^!{)qEf)*oK>;xnu`+sJ4A02}wo`3xyM z&TrbfLtW2;R89CUDZm7en5VZ~=bM(6rE=^}gyWK5l%=v!iBqd;DmC&Tkc;qRXyWxJ z%}P$46c1*CE^UJsEt}Z27Ad&+fCBwU{PXr8)M5O>4Eayg922ucbbFf%)se@QMsp-8 zjHA;oQkcnj@gaQ%eJrkwtu5>wnqtprCX(X<1Lt^4t&j?r@(xi;3L6S7p9x7cfT@1M zBCIMM9$MC>OW#8~+&L@=P8m9IeP)8QjX?v+Nvn7kU-H2+V_5S`g{htU9ui%$-Xw9d zFz5{cJUu9Ccby3bw~48~J*68VKjLa@^_va=>RZhmLKw9i9ETKN3ms<4nXlA#^G88gAz=W-|_MshA!xMU>z;Av^aKs96#ff)zJ`82bt zDQ1SHqG2Ci-w%ZSbDJ+i_s`D1WV4AScw_m;R1sEX6_xyK7Cp!px#$s@m9=yiMC&uc zI=Z){oL?Ot(ga`gyA7BYHeCb>pVRGy-I$d^Y2B{xXRZnSyf^@L&sIk>kC)>$n{4^a zlM7{A7o~J@FTBfLABHxWogZFqJr3?`hW-3s$8*eJGSe$1xMVa)@hK}36_Pkd)@A3( zyqPm4??i#fOVXo63BEzg<*fPOOu0+pkS+OQmY}zMF_x};G1bp`$h3eX(}I;kW}+{0 z9#ANpxw|ETDXUVpCTR?XXLMy}MhrkILf(=R3Svm(wNi4)*n_~^Qo(O+3qh-?5UvZn zuCS(`xvkbnDI1@0A||urtR+xN%4EHjU9FR6J-PqRrd^CP9VjRR7kDGH~EzB_|Kc^cO}R zc!>Zc>y(sI3FI8c%}>E!>*`7x=rT7YxpAcnMELpumGcb+CvTeB9OKZkQ3p%u`H9|qi$u@de7bnvpRx^hYoy!HwL1)Bb7^X zqZ)4O?S*3K)CB-CqXsviq#2DceZ0PwGP9DCRwvq5G?&e291&8;_V5QQ*xn4TDrUu~ zwqH)*>A+Qc6ia6WOb?&_p|0{LZVL-XDJEpV*LCLFMy;<{9$ELQH=W9#G!1eKyZrr! z$ByMAi3+eg2fm}We0;RVCZLvNK5zda_FHJ{!vb!PMlOPhmj%j219qYh4`q?cx-tG> zi)9B=EhVl;-%S<~ScQ!$W_RCi6)=-=6;H$067uZ8l|2Rh*aic$KJ|@baYDmy_V|$j ztsWiq+Nbm&-{yY&EyoY?Uwz$WLKLA-1Sxd0B7~sldobLz+qu{ULV5B}r1?@aGJ_2b zoaNYd5b@?526R_q7>7AdOEb)UJ9|(m>bNc2%T35sWM+q)B-LbFiui$KrNrF4)LM_b z%J@aXJ&!%_D$N1qBh2j)O3RDE%fs;6)*PSku=GtPd41#Gk>S6dVh(eyTxNeinDMId zKl)wvfB0Sd_v07C8`P)#B(tZ--!P5yPT!kaqoVYL&e@2bhT}D2T#7AX{Tl1@a^(552*I_!MKsz zZJ&=+03yhZ^Q?oIPK@9@#Ds}Gc#35PuOcJdu$|JbHi+|ciI>XGV-O7$XC7Q?XQ=VA zhSR0x!?%z-jDe^;&xj3ICkN`Rt>)jD_3e%sj>Rw{HfGE(tYYte*K7TY=~8KU2TFq( z0U<*Nw9j>NHjm$?tgKxud(~vXr_v1O8N(z>W6-Czi8m@F#z?FhHiF;y-#rXvZq6p? z){U;+`>-vv7MXNof#sLy0)p)Ce)o|fTqmM)W|g?ZfD>Jj$1Jqc(i)VhVQA$SKk9`= ze^7?$eXm3u8nlX2kBBk(n+8RGba1&$D`A7-i^TZQxFu--dRw=Ci6d7f|dLV*vINpYFLwfA_7lH8he(C+#C(aL36xmSG6kzYkanSRh z`%N2I^%o>~8+4I#If?NGA{!k;F)mH0UjG|ZzUDy*rDuu4f~1lMdpruei~pDWPz1`m|{N{Imx~=bge|B24Ut>R>FT`=W*6yG6`ncP_@4vs$oWAhB^|ns9 z?(QCs!rXM14!FExwjbZ{7a$@!^A`|W?FPli5?P~W3(S*oc~^>`Y}FCfRSd%n_}yxW zoN&hzqIq8}!q;sGx=Ed23@AoELiS{2KJP-WuJxO z3;mN%=XXa%Xxhv-`K%0ui1vy|Z|#$gQgDEPJ1TWS|J-BW`>H?qB&i&um?=Yf!M~Mo0>kp z|3`+n{?U27n+le@Agi;wSNRE_Zmld94%??>f0S^YxWvEK*%DG7OXuQe7lo$qsh z2e8+J(#=ml5&KB3p|$cFts%g z-VTSCSB2>GlEQ6qKgcN8gjt(|&$2(N$%Z}zm6VELWW%2wt9|L9`&kN;VnsazW1PuF zCoev1Q37MDJY(?TC)bZ^P^f&uko>C64GB9x9!~L66yw}ydZlBHXR>=SDv2sAnx!gR z^qQ_)6|RpRjBdX!)#3UCH$p3QjV~tzbrsmz#OTa{kBG#kKj>Y0(jsL3htX}*e$+MK1Zy!V60tc^`UIc>L# zZRfWir7#=-=3bO3EC?bwAZWBtyCWgObwGjL;6 zhBEVQc%o`x(X(;(s%k)jMWJQM$Zc|szrdS9eQfnQsc?5Cs$4un7$2YEY*0J+%u`XX zwEt|h;yP!*HL75iwnwiF?CK*N$Lf#;E9#`tl1M<&!eFZrV@l`f z>Zy`>z}KypdX-Bf9_>p%2MfsFerEa?n8>6)<`*zbDirO()cTF1dO>@g~XQrH0->BM)0sZj=2k-a)92tjbavMRY=a6cF#OOIC&Hfmm$V zMVeW4z({|FYkb)hq7f!T553-ZfCA!2s^9I{WfIrwU@S}}ot6u17C{pA%Z;dYSz@Zn zAZMM00-#*BsQhLmpoK6D%w|m2Jp`_8BM9YOaU~hOgM&o>w45#AD>^!jV(o~_J1Z;o zMfx(h(Yzab@6GiCX4Lw;!%~0`xqeu!i*`_z-xw*-HyXO*1E-#Ov_7>%I@*N36w5AG zXi+n#9(IC#%Q)N~6M04_<0eqKAJDY8o7Ns<^ltE_6?7)VKV@>8R{?PO2T+_dGbgt2 z>7jT`^@G23Ux^7Ydfzjyq2vp-{xAjDD^3I87F>z|50LzDZTJC z<^REW>h=S`M~*1;VOZWYQ*iUs+9b(7)rx;ehc+r^cgF3V2223n{yJ5jk|z58WrQhF zV{!n-NjnO}Ltm-T*5l0$vTWQBfJ;64xcbzt}N}Y(Q{2AzuayX zOHEDb=^+UK0PKoC8pz4KoL8MHd;|a@0Dw>(-|EUzKyex1(c{sEye5XF2C3Gki4r0I z^uS76h^wiCt*?Wv=Zo$P@Yf8`$HvAJ5aJ04m}~mG%;MtJ7;5u>06G@`pAvu7Q^6SA zEdSO`D9AWIJfxu!rlrxe%;wQQ4XNoWDzqjiHW@Y3Rb(f9%E(V9X}b=&VGltF&nZT)6$3 zp8>+_%#Z&I@viFMfD6Tcw+53pD3+A?Hk=5e6*Kl z+Hndj{}yn1K6KF7*K{+buUsFx@xA{0*s#mb-uI<1q<~FLt0%CcEfbRfepn1c;}#$z z8Kr(~4RZ!+%gOwk{7geuWm!8G@h}wRtzuJR7j@x{?g;W zerN$mVI}97!N78K!2hQOE{?xGv{YB3-j_mrnQw?g)=eLqmWBqj$C%_l-W8@8YW{;{M8hJz2Sf}EGR1cWbLR%_gy&k#;?JOWalAxFJ< zlCB;Q^ivfy#DFfJLfxI7M4H1%0-e}%`KTv;?bgx~Tq3c$I>idZ{&2I94*`d8JvhWMgLb4MRP0&D_p4Oj7~$zo!?CIeNpqk?Q6dB0p=$4M(FOv+RIB|&M!>1WFs{D zdPGdPB6tXi!laG2*ZcQF?+e%I{x4Vk_ivz@x5pXbx4X5gwS2#~bHmrm_LtiQz{{D^ zTf0p+=xN~%)V*)t{r0pX{CeAbuISIS@pAD}lX@S`y7C6HduKx_JAb=6C+>P1uOV(| zFJqMP_E?T;C@aG%>GWC-Yv5MmCqI=|@)lk8_gIFuG8800bm#`Iq98^PH`D??wX?(2Kno86-<*t+orkD~PHg(lg4Nve`KAIDl)--N zDL_Xk@R`;sfN&BO4A{6>%`GtQPzu%;g^xsbh`H}jQhHts(FbV8fVq~Oe%HYg8m(rP zCKK)Ib|-=|fb(+5I`t7zgU=9E1Kg=38lThpgeSR$Wz15ohvY{))h~&K5mLXz%52hD2z@g^X=by@!H-PA{jDc8SuXu!zKwxL)q3Q_iU;LPO|=j#c_qObAK5I= zuD(>*1OO}|32{B#2`Y@VaHJ1HTb=^&#T)w7^ z-K0+w36LjsukmB95HhI&=VyiLF9ZC@io&TRH3?mLgZ2pwelxNOu1_@LdZtC+^%`?> z{pg?#NVx^Zx1@m#uLcMV(Zqm}t^AR|iZtm<7<2Tq@^T9IP=-56J;>25yj-YM3@kiR z0n?&5JxPLPbE-joRa(2<_2!0;m-rOX8f0s71Al zy*~D76D0CRSL_rAQ%0heWsNxS_gfaRsj3^O;=WUS{3)4(FuJ9di=@e|m-4QA;${=? z=Rz(I-~YqhTL;zgHI1LRySqCCcXxMpcXxLP?iSqL-Q6KL1PJaD+&%0~zR&a8Z{Moj z{cDTj-m0E6Gu`K$>FGW*{kdAoT(*bqixz#v%x`kIHo5}-!B7*6mOu#NIl%YR_Bq$Z zc7PT^c6=J}VTOqS|M<}JKH*7|%Cv1d9Jhmn5J zl4m*mYZND%RjS5>CTR#DL9eTZ+h%~QJ0iQWkC3$yZjVI13n_nJ?8Q9E-$7X^N}6y! zPgOuF@JS`yp|~u+{oy#2jl;7>WW&!2Z#tdbu>qS#9>dEfnTxa1&2!0_ zQ5Ff9V|*;0PNU;#BM2Wbx&0G=jP5~=Gc-yIn%0-kP5-TjR|}U;Bb1I8xL0puOcLfUMRFI( zefM6CB;w5yR6dMNC(8l95U0ZWmJ;Rj$&XVCjnhqHW)0Rg%>#-~c}FOED_7I6ubX!lx6+tbz`4|Oh@V9&$z4eOka3*xy%M}QJ zG+Tl>#YYM|S)>oVt59DZsmz=WjscUnI29&&{oV+95XJUQDnxE~pT`#ZHYPd)I~1BF zL36z6JL7fO*^fW5A#`7UP~?$JB}@mo?~*sGV*%8%;k(dPlC(A1b46Pficco9dg@B; zcoaq&Vdn_MkJLPYXKIh1%gFgwz5JU%!ZhzfPlI0GR`Lkclbnzq7m2uZ0GP}ni;DCq z1!{8KBCnMoqm;QzZO95Yk;~3TM9D6=D?wig;)$@w%`{^1eiSCX-oaA zs#cd%`@qS37+;j53vo zZ3t=oqN}*#`qWC~EPo{1y6n;G15=yuuwMe*Ws%lV1}!{iFcvh#4$Z7i7WxVTb;qJyHQEa`%jF0 zM(G^lo76{Q5*K~Vi32Cw5)G7rNCSysH&Pw{3&yYR*H$;X{-9b7u828_Q)7kFM=Ez2 zeX>N@zI4CF=kU$apZ4s}Sk{MriWpN*i&&7|Wt1nGJ?frPVEa-58vL9+uC>VPBA6ji zmN><)Y4 zOH7StHa%1z%U6q?@OWUz0BdHx5~ywdM9R?c$Bpv$vHB#&-uS(kNZjWaPPz7=&kSv8 z=H7I?Vjy~Ct%)0DF*|^0=9o35BM~;^zp4P+wIxF_Rz3B0uIE+bXYWkOT(c&DYn|=i z7YvoVq9YMJUE0x4SlXG*5ILjcB}GLb=% zfUg<8IC23+@vEOb7cFKiAs~8ihKNW=6T}G=P(FvtZ_Q9(GRWukw#1brDWS}VS#L7| zzvZNl?+uPPq@WcMJ2?-hHjj=H7-Iak*gH61;S1pnrT+l(Li|Ls5!5Y-DNjzw`GYn> z^KjEnA{57hAGD`G>#e9nsbkY&_7Iw=?w$W>ew15ML=ucgTq1O;mqXU^wY}KejJ7R{ zkCH_6NVGWCjnxM)NFf&=nfvVp?tYJDZYb5DWApsQ3@@?cWMQ=Mrwk9>#7?D*M3{U4 zKXgz3`twi;q7;$OV-L(*^Gs~dBj5zt{FC{2XMnMi4@kG@b%@TlzK6a5-&!KX89R>P zusSFo%yw$G{W&VUzfqhRVj5v1TU?Oibg*xOMFF_?OZ7aHJsb_rz5R?G-{Z_A*pk<6 zBF5HjsyHZOZ5~2BJo%=IquudDXwba+~aeD#L?BZ^W2U0eZn{tSVbq+o_5>w1dk@o&+~ zP}$q0NwPoP@p6tou}}ao<`#BJ4H_dABEgbH>GYf$*;IQ)HCGF86O#;wNaV_GZioy1 zGO?5FBN4_a?t#~QJIg5R3DHq{3(%l?NDfEzD8vut}~EN>1F3ULpK-t(aYtC^e? z-!4_nsyM(YxKJ|pS=JsDm2#mC=-~lkL+fm#%vU2kfy7nL3c5!aY6_3^jmRnRCTV_F zgt&gl_BbfE8BrA2%+XOPvjz=G6<9lH4-yvxZ}KLm;@vb;N9qnzk(#1?NetK_QLS^f z)DiVdzeVH}JQA0Zj#xLOTEWm_Csv@c_8j0P=OO8{@WmIaurwL+ehtY$aN+RrY#@!> zo5n$60GerBaUafQ19AV^NO-@Pd55;RJ-9%`bS9AYrCbu8#8qqH8Lskm8C&oR$5gxk z9VGZ6m4vj}tj4%q2kf!D=>&-Pg9`H%?L-2dA&Flt357wJfy$}N;noxBmgSXb8sm0V z!sNXXlzp?8&Ffxnsf|f5ijR!hc&bm{?+hoj$f=<2Shp0=B)GT(d~D{AOU@MHAx0yz zz7pnxKYg2+J&rfAu4#$SrIiGQaPX(!BEhi3B4d-K6C4tVT$1N<01aqg7zq;L5SH<= zt#e+4FTbRMXk7Z)NunYra^bZ#l&e$AiIRm3IsKWdy+%7}($0Xv=l_-+F$LTfJFY$8|iFofmZ?7&sgA=rb2^lci!c95Jgnlbk~jOk>J#T^q8BAq$S8l(SRJs z%BW03sVAS10@Dt^_;Up3gYnLo`rM3<8lP6KxRi6DU%KMJl}pH-W>Z4++hdV9CGF+` z1##No)^hSf!-6O>rUZ^Ab?Oj3rKItpI&o>$klTWcys2V@DKl@_Yf;>M!>(uzpv@{{ zDij0dc&NksYvSpz_{^~_tm*yK634-n6N#%MAK4te@3R($C91vM*}j9eJVm;pgK*iH z?9dWo+<2u;%S3j4r`lQLSzS?czn@vAIiv{zb~V|b=K(KNkLS1y_Fu7XLlz@Z*I|mn z^(~)Ao^l?Fe!&WdNqknQ!1Jv*-;#f~SHWtp{fQN~r4)%)Yx`L2I>oB=JW^oHQsVq@ zc#!O-Bi|f#HQ4_!JL+glffwfEvE{AIZJsH)rnI{%TT$)D_0qg~i(Z=q1G*xGJ0vGOoc+JZ`egoQJT=Rm=c*X}&+UYT(eQ=%ux9)QcY z;brc=H|UUj?5unV>)nm$dC#skJdR!J;-qa6k|{N9Bz@yOS#sp~CF7^m%=alG$fFr> zt7u=)#+?TX-k2|Ck!gnnDq|PsnapYKv|7eoMl-r(Y~i?5Q*lX|PqjkHYjBeDcg*l4 zjvQA_+@33QCciZ26*%>6JErg*n0f4yI#@y|;4|esE^9}I^{&Ie&D7Z7=Vd$GYC5;o zvAMTB*(Raww3t)cg93DubPJi0Ywpb%o|H>Mo6;Rx#-;V`yyj;bi5_x#tECr?MV5D|8(69XrMqlyt~{F|Rfg(O^X0<0F%&LFXcmkt=3-fb}u%# zqQoM9wq%C*I*THoS(g^;cB1LY?q8g}(L-2EPTo?LczA|*e03i5T8nc$y^*m~3Ew?# z;u;sD54xs^UhI!sS(`9AzS%Wv-uWx7q)&MIgovd^y2EP$-~%b@I3XDA*wZvyzi@I; z>Mx1>(`OKbwa0nZXAZ4%z*)BAyZgVG9Hoem<&J)I#7@59w}aXqf}DZoVJ!A9c6IH( z+CLe7+*DP~`{nUT zx}FuKn6S?hRT1HO%c#Q}SUmYOC>XZ@1l}LyX>)|Mf+(fW6}>3;!ayX!Id8| z@m=n;3qSiLc$R-k#0+j~m9i|Dv4Qp!xsc_$J~}kVxjhmWkN5b(>H?`eD;H5N@Nh#i z)JdAb@a>mFaSWr`XEL$8C8)FmPpXC`kbUPONuKCVQOMZq~e7K za(csJZiZf4q1usGna)zq>SHbo#MvF~f z=6`&SA|nxa2!K>%M+G_#4UC41-U&n`Y{ZzT6G-$&$kLR75b4U)TfHI9AG(6v&uHgz zgLbaNm@hMf2@)(;$GA!z2W}~GMZ}vPs=kNFfVHY^&5-7FS=$=L>>G%Rr*8HDp{heH z{f-s#Blw>FTeMRfn*echHt1!x9Bwz(Crs9?Sf$Z|2J)t+WT}ogqq>_A$U-&jxmZsS zD8KK9O|kU0oNrz=mm7$C=N&th_|=IvRtx*o^`0?x3=*K8y1ZB)4EiVbCi`VBQ)(jp zDdilGjn^-rRm1hNJCgpu?StRfUr+9xhpTzx{E;)Nv!nchHhOQ^RLspH{5LR6XyS_u zL@82L<`Iz=IqXB3Qg=Pa{TNGcpPw-SgRyjjcs<5(S;6$@#4^zpy!^k#OqIFNKY)CF z_m7NJu;l}M4^u1t92#5AJ^cVR3y&zwVFU3_GlKpyJYrwjKXSa-`PVh1&;1oPt;qepIS%0RQ2hlkXr6&2IaS3%puc0QooP%p?1U2YkhsU> zu%an%Xg;@rS(KwU^Si2er>KBv-Es>r`Ueimz3V;b$!YAmxvUxCW-F#!pVNKF*nSJZ z%s{9fr|`H6xcPX7wZ@je)-OBLyaLDEjeg-OWfm2<9B2CHl#>-FkS8;^ZL@zA)F*oZ z^B(&so$*IV@*4dh>U1iLtB7&^=fFcP=Lt!ast~@ru{tgUu=67yfwk&!tZwhf>-6dYOl6LJ&S-Y`E8F`IhqOR!-)rDNx%lTN1M5hI~oyJ6JA1q z?R$S($HYtIn8VMx>svrMtYrY#w`RGU6P?$D3V8GW!gg$LXQ{&8TUXOyS#V>E-zV(f z8ubyU`3<~xKFW99%1ZRx33nLvG^YePRAP-q`36x?mxJCrYZB=b&w6N#M?K#|HMi1_ zoo?}E1nx!-mjy`IZQoy_>Gt^^GjU?FVv#Vg)IO?4#1)=m{~0nq+~((QzI;1jez-k{ zlR#ng8?YXxI2FP4`J9x6YZ%aY4M{w>pa93^x+h6b(VZ%gEmrnoG&T?)hdnxAq5wH> zhy62A8l5ZOzy&!uAdUdyV5dJ~dVg_KWiJMQ`mq`^plq+@R=C3GPvD*=Mqkf|J*nf5 z&Fz>Oq`vlZJM+hT7SQeVjIkTUiP}&(?knIw7kR`_)A2?H8u(}Q}J6I(a6y3}6YSR{O9*bQ|d@pkfC)7bw>jpkOhnF;SAo`ieQH`!k&xGH{(&sqb z_7$9ukdh0VSR~OD9RWa!Z%=81_zfu!6vil^X;WmwVU;nsYm` zpBC`$FERTy(atY)wWCm=eKCSzGY)8?2)dd;4Q^OCP_NSR^|UOGLJ-)hR>NH|oyX@i zXH$d1qU8FTl0UwJyZ$r_qJf1q3x|azN<#>N=ttI5|DEqbZc4`02n&iCm{@R!`$GDa zxZ^^H0pGa8T8f$;=apGFVv*5LeGh8bJyp#UkC=}%IA-|XZ zXOGl{YA8`V(TnH+f1qd!xt_2qfm&AXHYRpO59S<2BQMx7)41gQ(byy$N-Xj>i64*T ztf05BnZGpx7rD0B{Q{XPeBmtj9s}+AiSYV?pv#(Ctpa!ITiJ=yCx+H~oP27qN>s62 z+X!>@4_@+r+iz!>Kl%U(?6R0Avq_WfSiRr}(!BA-;axrjraHH8g9cBC(evmA^)ipn zg`76p?cMDmBV%u3Das*y0MColh1TPuf{L{AYZxj9M2r`^(g+eihVRNnEg%`fRV3zM zk-4Via`&>1$5>&Cwjl{xp;1#Pt;z^z@Hn}l%23!< zsg4jFn0N=~2j~AH!`ENtc{N&^I5eGeLmT!mhptFT{|>v6XzznigaaKyb{AnoyzV19 zYBu7DF~s1m(O*&Tr`X79o~(URPE(-<6cf6D z<@_vc-^9vPP8w&T96BZxmmVbr6k#-AfR7x{ldZd!NdQ?wJsjI&Y)VEKYk~bzSsLjQ zH?G)Lde2EGX4>dlxMuOjis#CjgrY}sisceAEa4*oo|0bIM)D3Op8ZuVV(VS8bQ#$Pc)mTGr|(O@>hGB-85% z_gDW0b>67Z1tgYx!Kf3A_n$LOv8{P?W*+hVOye_=n;+k!B$4)D>s>s22Nr)|@~Zg( zn~wD+tUlRsJ1*@&yz?yBX-TO=gTx30|HKLlhIBMYp07)Zf(t}5l2GFcVhPFaA5%o5 z6(CHFpnVq^B=ogqY;ij#hS>BPHDeVlss}DaaT3!>-&RD<3i1P~_=Otr-hb?pAEl6g zK1YdnIW+vn*>Bf=-3I=Ieqt`l)yEQ|Iy^Rx#3`srU7-knCj%RZFKb0zj0_r}@u)jC zheV9M*5l5HlX4zNxF#)uh_Dr>%E@xb2?EX%$$hv0 zAY5ovJr4;56}hBGfLn?K6=7)PtS_qs4LM&8QuzQ5EF|&}tA%Quy8+`wbHy|AwTxiI z8%I_S0#-uf_6J(`AA=t?GJ+{KawCFAW!OOy3+g-dWwr9T93|A`kEHL|M3Wr)C5U6L z8a7(fgqDmP8YEE9ni@_7PmGNzJ|dRBZToi@R$7sd=sO-2uhlZVB&kDHGGR1u2yu?o z0>q5J?foW+0g>1UF__kl1Gk)61oL}P2_Tjb{u7^x5sNDQl|qsJmHKC}fG^am>xK?% zhk?fBS94gs#SIW@W=`%m{sd^F?Sza%Bl5iGz&#C!8UasF7X;%aMzW)#6BMxvF+=LI zCb2flkt_irT_1HqSj!#>nd;~Xc{rD_RzlMTR&fFKViigL8&^VDq@pkt%q?^3i82gU z2y3+p5xfn+pf=_LD#?4&PBaQqw6OSYw|}~ZMICSqNG4$|q$k`ER#r0#3~`8JBXFcJ zHQcO3`aIv;yL7w0FjMXCKKDGoeQm@Nzz6osJKMLhih~H^**gGgM^V89k4*m!F0w$Z zG->Gv!{4Sgu?xQ$IsptcM#B;aPFJ~aFkL(9Gs`$L$L0giz6SS zMDX{X%_hZyIO%Cm9bm z3@7bzT)a%q)kw74m6+$#?>w5(ihyh{d^9&XF^+6NGM2`l1!HOPfMtQpbZ8cD=*=Ge3oKy0-V7XC;7uXFVvp~CYN+JJy278ys;*YkGuJ*YvbmYe+P|!t=!>86XG^Vq?bIFylUTG`Z-%5K2;3}!Gm3U zLCQsTAtcwCMlJfNWuat-P9v;}yl=bv^S!z>mBaVZ;iDI-N0^AR zfFTqY@a0wKy4#wS_qohECc!uSufjUj2zU##%TJO{*B+vtrZkJfJbW${SkS^e*B(}@ zx0K>D@KgNMke!GEGH@RypwvRi6Ftj#rDbCef0z$T^Qvk@B1 z-x>+En8ko#H_9LVj-;S5USF(Jzq%2ars^%W$N;Qg$mUP(f?p)LB17EycgS{di~l4A z%;Jf)n6#sTsXsnO=NUk@h*D7t4x-U@vZZ701yxTS#X9M67_xYDrQDZxwU@!Jx4w;j zV87JDfjCQa`e86w8RB261Tu{m2MXnIknB682<B!A4XPzj=#Nl&BP( z0I3uq>BT{r$ijQ`dOB#*=nlKKdF*d(&nUNXDE+_$$`t+MW2}Ds(Ah{siF5;f_1n{xW0T`heE?$8AkMVkn#g z=!$Z|dW_6HJjV2e{dS~8snewjD90~w1fhLC^pJ3+_w|I#WXTH^F7v-YLDp^UqMRm*|T!&Id6Ud(C@R)f2gu$wI zjNp|{L(?zXv5ueICS%Bhr1fUWc<2#75yQrD0mWyl{}nt zwM z!yTgO3~0=de*z0#*eZ7m6xLABgsoDOFZ5Xq(W0tb2P9Jnh6t5SP0mU1A zxqi_+R{a^HNP9q0ISBVI#>;F8lpEYaMXeb-dFU*VLl!t}UUsH$|ag(;L2FL1jGByVE>A#2JRY4RFE4AQ_mT0OTIS)Z7K?r zXF-I`Bq8jPV{@Y;0Ur&yCTY-35lJ*aQ621a@_r>%X3obm!12(1m5_z!(*n^e1y!Zi z%0It0%>X{HKYbQix)4S1mUxX$!`pqq?0Ck>TQNi5_1kHb2KGh+I;b}zAK@`ikY2@F z0ELA5XbZNTvS_k-%WHpv0Aped^&_Mm=1PYs5p3_ZOX24rhCZsLo z(GNv$a2{(pWH|hc3q20WL*(pObVbj^mpd!?feJe&h9Q;*K^)4dU;yd_B_@z&0Lm=gdAwfXUp57+f=a`rdH;XKt})r9b2)T1yDy>qTAIO2sp&?a(~3@z zflhF;szCjNXy5F8j1%IgH9-PiA86p-9K@r_G6#T1jT@qA+I{=s_S&b21~`N#iCR#3 zet!#YN9jf-R`VdqKMQ?76|n%dNOq)jPDdSs8U|-1wfE)hAl>vQ_Zk2L1ljToCaN%a zL@|>Hsm7y}JK)KBJw!4#Q`dCJ*TWJ6rZC?MId!A3ONin0imjjn$*1O=_o5Ogf6saS zx>pWDKYHUq4im}-M)`col6B5eWyjpnjCy}hw_?F6|*Fe!ua6EegvK|>@E*?LJ#Dg?@O-2W!<1mbE#`>i+=o2(+v5Sw3k2+dlg zLJ;~cTsBax-i@3o9h)pytRF6bitnMcgyah#_-_Ia$b6*&+NiI7R{=QReR!dZ=x$;+ zS0PLAnlWTOjvS}`0W{~KBSWOPrkPRv<^2d6gcw(M!{SEJftEe&Gqaf2}n zz3gTPmC~r7ln8d+&pv4hvS~A5fi=Nca2vqo_#RklZHld(L#l2zRYYKmH`PvEtlFnj2<3@!Z3ATuHJqHyB(caKVjAE9uK84!%LeosSS1rSVzA!{3)|I8wgA@_ z-NDdb4wO5HdU_%N^>`&7;L5gv{{>qMKMY|M z&U*Luqrg<~xTd~*JFUO_=hfbn6dn7JKgI3A-GRZ7cTq7VWTwOsBoDMTAku4_9)&d1 zytdTu*DmoFEV%&}mxfmURa0v2} zocv$9!Q=C~!kcm?(&}!y??*3blKpfzmS09_s)4vcK&4HWs+wdYSmfD9?n+Y<{ZvGiDKj#`({cCPq~A7*{6VIt+kb}k8PRvzn^MYPM}XD=2Yes@-6c7g!%&InO*~j zL&rm>Lgzx4LRUjKLU%$BVh-T8aVD_;^*3Z31|QQJ3qLgWva+B>d-zxX|5*wk6QLTm z44`0O%Koo4*KGg2=KA5EnrqWo^s>r-)m(S|N6mHR>!!ylSxpLcl=V+Mh~Ir(AtqjJ z8ehE8m!*niWRuZI3py%_w^E*8X7>*;05#XOnl{Z;H$t4gZJjTz8`xcLu}B2KLR}~v zTOQ!VaU8---_qqR(<@@triSV^kyf+@Dus%*lS)pr0X5eI&|iDPjDBf5-xzw!%+#>JjD&GVy;Zm?cgRx_MyI-V@gr4k%N z&l!~D9c(j8{J_lECj1HkLKvYpsHeQ2RgA2Z>fMu$frRDfHpKHkBmb~23__C5nhtWlAhV<`luun_vnp%6zr8aTlT*8Fy>0m5d<2}hND!PVYjmKgWO zTe-+GM>o%_Zl!=RU>vg|Ljpj7mZ87aH6IKBanVB;n8h6m#txJ3L0l-Ni^yI(OPmE8 zyc2MZ#Jw^i8V{(r&MhQ+bHFiO-3j!q7cafrDTy^lQ> z_Wu3K{<$)%=J&LlSLx^f^70f=_OFub%?p3)OM**-q>Fu9-p{QXZN&5+4a;e^k%usj(MW z&rgh_VTVr)^Zh4>f=Wav^9WCan<`0Kz1{~8%(o~5se8evGAJ9e>+>zE+XIRpY(J7x zO3A33oFpd`%TUv`q-2O`IaMe<1!rb12iz|yI&#{`dGccSXOu)h87D1eWopjx2olnp zeCl($r0kFNCT1-5+g(&&fnu7tOG{jj(UVxj5Y?fOnJF*Vog3?YOA%g^vZqFwC{K(* z8NJ8S(}d+S@vTqE`2T#K^S3|E?g2CLei=@Xn(8HEYPw-#nnrlw0V;|bZC;u*iX=F% z@8=tt+FbY2iXci(6Oqz3-%rQDI{?NaXD=;D$MDUgbea#&-3+<#HA)MS zw1xDvM(N$<9}-`Mxw{b8-v*f@4rFwTXUIy|QcWqe!j~Mwu*Q{S_RZKR${KeOs#XKP z8LKm-<(8y6Q&VNRnpxY>EQ^893`+Xe7)2nSVd(}HEf2!v=6VWg#77(HdF|K?iJytF;2pUDWlC5?_%HTco(Z_%7!|)CRJ~*b{}Q z5HCKbn7vS@v3~VP+)9&aoa{zb3y~_2kOlGuzA=^>NxcC?o6sze{!JMBSUzH^%EhMl zW4Vyj!~`779=w^6ksUs2M|A)e=Y{L$hf2L4YmF(po?MW89=ri}Qk$c+Hs0<{;7+Na42x|v38&iiV3qw|6{YDA7VX)=(W>^* zY6}?onJ53Vpx_K<@ONw4V+r496KoPguP@a*i01r|;Va3}y1t22jT6bp4s2exbKzrK z+C6(zPVs9T~CEdeDo{$6ExKQ6{iTN-l; zmOH{Yej#@^1@0g7?ZEvAG+Wk!*QAV{$t@nZ;No^h

2u%-~lCUGB&?*{>eB2$i@5 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%0A`&8myzQ}gtJ4$n5%&(coOP0-DX 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Ð!JVI@%e{MX7=-8K6!hg$&G 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 diff --git a/sormas-api/src/main/resources/doc/SORMAS_User_Roles.xlsx b/sormas-api/src/main/resources/doc/SORMAS_User_Roles.xlsx index c473ed74db7fba4bfdfcc4988032da91aa083064..cb90708708ed13cd796c1dc36e91c771ccc56a67 100644 GIT binary patch literal 38001 zcmd421yogC7c~rmgi4B_G)Rb)AkrPuA&Q6+inNq;BP9aT-O>UI3P^_t3P_hogLHRE z{rlW|L45ih-#^CpJ>$yVYwvZ=J?GrXoNKPlqa=%hN`Qocfr0cWDnt$G#DC|(_<@+2 zsgg(BZ7?IIbn&!MeJ(%pC7 zRk&O6`opUiBPyJH`Yk)`p6BWqpZL;>cMWyF(b~9&tl*F~_(m_WvuD-qLX_w2Pj=`x z9-!o{7;tN#e{^>s`}Rhjf!`t^?=<7PRb)~XVv4s?1^(B`J>&c?EsNyZwx-!)s$adm z`7mYlI*0Zn<&}b+#J5lAM6ccV6~E^+)0JRYa^nqkl!C`wY&)@2cgx?@>spd#QIAQ| z%yUf&Y*chUOulQ5A7e($EMi~su1i_OM&;F4TeD`h$8&5`hI`S5wA|4~QbpK0k~KpDy%^p}m9e zV&$Z64pvrF`xP3CRF7S|K77d5U?b!$U2&TwA;rzQ(>>Xscjr5a9gkqnqVR?4Y4aaa zXbN{8R`m;ZdMc}a9l3og9M8{9gl6!G#=J1HY>kihf>j)f46|kK`93_yj@ZJA?j3rR zmgw1$bK@QlJb1MirO!2m9*po;vIok4{`Q5T-AuG(t~?FB(9A8FNk>cWnWjyzQyF)m zV)Ptc`uZ?C$5{zU({Q6xx+@z$R9~7BzGTw>8QZZT+3to)cE;rvtLJ(z#dqgZ(9->_bzklpPmjb$j~u+tYl$kZpr${+#CX_d_eV?PuFn8m)s9=EzF7>aDzG0DBbBD z+i);uzM;j)OWmEBiVby$yEM6Sr|73dIh!ug>+rJD;8P9H+uQTQ19#JSzbcq<8A(xT zYkRAOp->g?pmM0r+~+F*r>Qc$^JC2>aca?@i=B)|o%lgRIUkHBm zu8twjkJCoIdH+UK&X>Rimi~tX$=76FeiQZ c1TOdacR`0hCOu@3o^4EgzDs~d}t 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 C#mtug literal 45326 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 c&@g&k#JTZE-lA1>MapGMPjZ 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 Q8H|d+6d&?Wp8tCHU# getEventActionIndexList(EventCriteria criteria, event.get(Event.UUID), event.get(Event.EVENT_TITLE), event.get(Event.DISEASE), - event.get(Event.DISEASE_VARIANT), + event.get(Event.DISEASE_VARIANT_VALUE), event.get(Event.DISEASE_DETAILS), event.get(Event.EVENT_IDENTIFICATION_SOURCE), event.get(Event.START_DATE), @@ -427,7 +427,7 @@ private List getOrderList(List sortProperties, ActionQueryC expression = event.get(Event.DISEASE); break; case EventActionIndexDto.EVENT_DISEASE_VARIANT: - expression = event.get(Event.DISEASE_VARIANT); + expression = event.get(Event.DISEASE_VARIANT_VALUE); break; case EventActionIndexDto.EVENT_IDENTIFICATION_SOURCE: expression = event.get(Event.EVENT_IDENTIFICATION_SOURCE); @@ -523,7 +523,7 @@ public List getEventActionExportList(EventCriteria criteri event.get(Event.UUID), event.get(Event.EVENT_TITLE), event.get(Event.DISEASE), - event.get(Event.DISEASE_VARIANT), + event.get(Event.DISEASE_VARIANT_VALUE), event.get(Event.DISEASE_DETAILS), event.get(Event.EVENT_DESC), event.get(Event.EVENT_IDENTIFICATION_SOURCE), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/action/EventActionIndexDtoReasultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/action/EventActionIndexDtoReasultTransformer.java index c24f8fa8d8d..21f0efa3fc1 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/action/EventActionIndexDtoReasultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/action/EventActionIndexDtoReasultTransformer.java @@ -32,7 +32,7 @@ public Object transformTuple(Object[] objects, String[] strings) { (String) objects[1], (String) objects[2], (Disease) objects[3], - (DiseaseVariant) objects[4], + (String) objects[4], (String) objects[5], (EventIdentificationSource) objects[6], (Date) objects[7], diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/bagexport/BAGExportFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/bagexport/BAGExportFacadeEjb.java index c477fa6d7df..ec36a2e45e1 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/bagexport/BAGExportFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/bagexport/BAGExportFacadeEjb.java @@ -106,7 +106,7 @@ public List getCaseExportList(Collection selectedRows, person.get(Person.BIRTHDATE_DD), person.get(Person.BIRTHDATE_MM), person.get(Person.BIRTHDATE_YYYY), - person.get(Person.OCCUPATION_TYPE), + person.get(Person.OCCUPATION_TYPE_VALUE), caseJoins.getSymptoms().get(Symptoms.SYMPTOMATIC), caseJoins.getSymptoms().get(Symptoms.ONSET_DATE), @@ -264,7 +264,7 @@ public List getContactExportList(Collection selecte person.get(Person.BIRTHDATE_DD), person.get(Person.BIRTHDATE_MM), person.get(Person.BIRTHDATE_YYYY), - person.get(Person.OCCUPATION_TYPE), + person.get(Person.OCCUPATION_TYPE_VALUE), contactRoot.get(Contact.QUARANTINE), contactRoot.get(Contact.QUARANTINE_TYPE_DETAILS), caze.get(Case.CASE_ID_ISM), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/Case.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/Case.java index 09e8f58644e..c1cb44119f9 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/Case.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/Case.java @@ -29,7 +29,6 @@ import javax.persistence.CascadeType; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -42,7 +41,6 @@ import javax.persistence.TemporalType; import javax.persistence.Transient; -import de.symeda.sormas.backend.selfreport.SelfReport; import org.hibernate.annotations.Type; import de.symeda.sormas.api.Disease; @@ -70,6 +68,7 @@ import de.symeda.sormas.api.contact.FollowUpStatus; import de.symeda.sormas.api.contact.QuarantineType; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.externaldata.HasExternalData; import de.symeda.sormas.api.infrastructure.facility.FacilityType; import de.symeda.sormas.api.utils.PersonalData; @@ -81,7 +80,6 @@ import de.symeda.sormas.backend.clinicalcourse.HealthConditions; import de.symeda.sormas.backend.common.CoreAdo; import de.symeda.sormas.backend.contact.Contact; -import de.symeda.sormas.backend.disease.DiseaseVariantConverter; import de.symeda.sormas.backend.epidata.EpiData; import de.symeda.sormas.backend.event.EventParticipant; import de.symeda.sormas.backend.hospitalization.Hospitalization; @@ -92,6 +90,7 @@ import de.symeda.sormas.backend.infrastructure.region.Region; import de.symeda.sormas.backend.person.Person; import de.symeda.sormas.backend.sample.Sample; +import de.symeda.sormas.backend.selfreport.SelfReport; import de.symeda.sormas.backend.share.ExternalShareInfo; import de.symeda.sormas.backend.sormastosormas.entities.SormasToSormasShareable; import de.symeda.sormas.backend.sormastosormas.origin.SormasToSormasOriginInfo; @@ -123,7 +122,7 @@ public class Case extends CoreAdo implements IsCase, SormasToSormasShareable, Ha public static final String PERSON = "person"; public static final String PERSON_ID = "personId"; public static final String DISEASE = "disease"; - public static final String DISEASE_VARIANT = "diseaseVariant"; + public static final String DISEASE_VARIANT_VALUE = "diseaseVariantValue"; public static final String DISEASE_DETAILS = "diseaseDetails"; public static final String DISEASE_VARIANT_DETAILS = "diseaseVariantDetails"; public static final String PLAGUE_TYPE = "plagueType"; @@ -250,6 +249,7 @@ public class Case extends CoreAdo implements IsCase, SormasToSormasShareable, Ha private Person person; private String description; private Disease disease; + private String diseaseVariantValue; private DiseaseVariant diseaseVariant; private String diseaseDetails; private String diseaseVariantDetails; @@ -478,14 +478,24 @@ public void setDisease(Disease disease) { this.disease = disease; } - @Column - @Convert(converter = DiseaseVariantConverter.class) + @Column(name = "diseasevariant") + public String getDiseaseVariantValue() { + return diseaseVariantValue; + } + + public void setDiseaseVariantValue(String diseaseVariantValue) { + this.diseaseVariantValue = diseaseVariantValue; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariantValue); + } + + @Transient public DiseaseVariant getDiseaseVariant() { return diseaseVariant; } public void setDiseaseVariant(DiseaseVariant diseaseVariant) { this.diseaseVariant = diseaseVariant; + this.diseaseVariantValue = new DiseaseVariantConverter().convertToDatabaseColumn(diseaseVariant); } @Column(length = CHARACTER_LIMIT_DEFAULT) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java index 40d39ad7b70..3e59a8302cb 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseFacadeEjb.java @@ -801,7 +801,7 @@ public List getExportList( joins.getHospitalization().get(Hospitalization.ID), joins.getRoot().get(Case.HEALTH_CONDITIONS).get(HealthConditions.ID), caseRoot.get(Case.UUID), - caseRoot.get(Case.EPID_NUMBER), caseRoot.get(Case.DISEASE), caseRoot.get(Case.DISEASE_VARIANT), caseRoot.get(Case.DISEASE_DETAILS), + caseRoot.get(Case.EPID_NUMBER), caseRoot.get(Case.DISEASE), caseRoot.get(Case.DISEASE_VARIANT_VALUE), caseRoot.get(Case.DISEASE_DETAILS), caseRoot.get(Case.DISEASE_VARIANT_DETAILS), joins.getPerson().get(Person.UUID), joins.getPerson().get(Person.FIRST_NAME), joins.getPerson().get(Person.LAST_NAME), joins.getPerson().get(Person.SALUTATION), joins.getPerson().get(Person.OTHER_SALUTATION), joins.getPerson().get(Person.SEX), caseRoot.get(Case.PREGNANT), joins.getPerson().get(Person.APPROXIMATE_AGE), @@ -852,7 +852,7 @@ public List getExportList( caseQueryContext.getSubqueryExpression(CaseQueryContext.PERSON_EMAIL_SUBQUERY), caseQueryContext.getSubqueryExpression(CaseQueryContext.PERSON_OTHER_CONTACT_DETAILS_SUBQUERY), joins.getPerson().get(Person.EDUCATION_TYPE), - joins.getPerson().get(Person.EDUCATION_DETAILS), joins.getPerson().get(Person.OCCUPATION_TYPE), + joins.getPerson().get(Person.EDUCATION_DETAILS), joins.getPerson().get(Person.OCCUPATION_TYPE_VALUE), joins.getPerson().get(Person.OCCUPATION_DETAILS), joins.getPerson().get(Person.ARMED_FORCES_RELATION_TYPE), joins.getEpiData().get(EpiData.CONTACT_WITH_SOURCE_CASE_KNOWN), caseRoot.get(Case.VACCINATION_STATUS), caseRoot.get(Case.POSTPARTUM), caseRoot.get(Case.TRIMESTER), eventCountSq, diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDetailedDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDetailedDtoResultTransformer.java index 09c6e5f68f6..30160c17219 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDetailedDtoResultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDetailedDtoResultTransformer.java @@ -45,7 +45,7 @@ public CaseIndexDetailedDto transformTuple(Object[] tuple, String[] aliases) { //@formatter:off return new CaseIndexDetailedDto((Long)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (Disease)tuple[++index], - (DiseaseVariant) tuple[++index], (String)tuple[++index], (CaseClassification)tuple[++index], (InvestigationStatus)tuple[++index], + (String) tuple[++index], (String)tuple[++index], (CaseClassification)tuple[++index], (InvestigationStatus)tuple[++index], (PresentCondition)tuple[++index], (Date)tuple[++index], (Date)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (CaseOutcome) tuple[++index], diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDtoResultTransformer.java index 2340b0bec49..3463a6e0904 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDtoResultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseIndexDtoResultTransformer.java @@ -44,7 +44,7 @@ public CaseIndexDto transformTuple(Object[] tuple, String[] aliases) { //@formatter:off return new CaseIndexDto((Long)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (Disease)tuple[++index], - (DiseaseVariant) tuple[++index], (String)tuple[++index], (CaseClassification)tuple[++index], (InvestigationStatus)tuple[++index], + (String) tuple[++index], (String)tuple[++index], (CaseClassification)tuple[++index], (InvestigationStatus)tuple[++index], (PresentCondition)tuple[++index], (Date)tuple[++index], (Date)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (String)tuple[++index], (CaseOutcome) tuple[++index], diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseListCriteriaBuilder.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseListCriteriaBuilder.java index 0b23bd5dd0c..887e5494375 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseListCriteriaBuilder.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseListCriteriaBuilder.java @@ -219,7 +219,7 @@ public List> getCaseIndexSelections(From root, CaseQueryCo joins.getPerson().get(Person.FIRST_NAME), joins.getPerson().get(Person.LAST_NAME), root.get(Case.DISEASE), - root.get(Case.DISEASE_VARIANT), + root.get(Case.DISEASE_VARIANT_VALUE), root.get(Case.DISEASE_DETAILS), root.get(Case.CASE_CLASSIFICATION), root.get(Case.INVESTIGATION_STATUS), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java index f2438dcb32b..ae3fe83e966 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/caze/CaseService.java @@ -686,7 +686,7 @@ public Predicate createCriteriaFilter(CaseCrite filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Case.DISEASE), caseCriteria.getDisease())); } if (caseCriteria.getDiseaseVariant() != null) { - filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Case.DISEASE_VARIANT), caseCriteria.getDiseaseVariant())); + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Case.DISEASE_VARIANT_VALUE), caseCriteria.getDiseaseVariant().getValue())); } if (caseCriteria.getOutcome() != null) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Case.OUTCOME), caseCriteria.getOutcome())); @@ -2216,7 +2216,8 @@ public PreviousCaseDto getMostRecentPreviousCase(String personUuid, Disease dise root.get(AbstractDomainObject.UUID), root.get(Case.REPORT_DATE), root.get(Case.EXTERNAL_TOKEN), - root.get(Case.DISEASE_VARIANT), + root.get(Case.DISEASE), + root.get(Case.DISEASE_VARIANT_VALUE), symptomsJoin.get(Symptoms.ONSET_DATE)); cq.where( diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java index a37c6e0de46..f6e2fa0efa6 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactFacadeEjb.java @@ -771,7 +771,7 @@ public List getExportList( contactQueryContext.getSubqueryExpression(ContactQueryContext.PERSON_PHONE_OWNER_SUBQUERY), contactQueryContext.getSubqueryExpression(ContactQueryContext.PERSON_EMAIL_SUBQUERY), contactQueryContext.getSubqueryExpression(ContactQueryContext.PERSON_OTHER_CONTACT_DETAILS_SUBQUERY), - joins.getPerson().get(Person.OCCUPATION_TYPE), + joins.getPerson().get(Person.OCCUPATION_TYPE_VALUE), joins.getPerson().get(Person.OCCUPATION_DETAILS), joins.getPerson().get(Person.ARMED_FORCES_RELATION_TYPE), joins.getRegion().get(Region.NAME), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java index 3dfcd3377bf..b1ee6b30c7c 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/contact/ContactService.java @@ -1492,7 +1492,8 @@ public Predicate buildCriteriaFilter(ContactCriteria contactCriteria, ContactQue CriteriaBuilderHelper.and(cb, filter, cb.or(cb.exists(sharesSubQuery), cb.isNotNull(from.get(Contact.SORMAS_TO_SORMAS_ORIGIN_INFO)))); } if (contactCriteria.getDiseaseVariant() != null) { - filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(joins.getCaze().get(Case.DISEASE_VARIANT), contactCriteria.getDiseaseVariant())); + filter = CriteriaBuilderHelper + .and(cb, filter, cb.equal(joins.getCaze().get(Case.DISEASE_VARIANT_VALUE), contactCriteria.getDiseaseVariant().getValue())); } if (contactCriteria.getWithOwnership() != null) { filter = CriteriaBuilderHelper diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumFacadeEjb.java index d6b5892daa1..9de113916a6 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumFacadeEjb.java @@ -19,6 +19,7 @@ import java.lang.reflect.InvocationTargetException; import java.util.ArrayList; +import java.util.Collections; import java.util.Comparator; import java.util.Date; import java.util.EnumMap; @@ -26,6 +27,7 @@ import java.util.List; import java.util.Map; import java.util.Optional; +import java.util.Set; import java.util.concurrent.ConcurrentHashMap; import java.util.stream.Collectors; import java.util.stream.Stream; @@ -85,12 +87,13 @@ public class CustomizableEnumFacadeEjb /** * Maps a customizable enum type to all enum value strings of that type in the database. */ - private static final EnumMap> enumValues = new EnumMap<>(CustomizableEnumType.class); + private static final EnumMap>> enumValues = new EnumMap<>(CustomizableEnumType.class); /** * Maps a customizable enum type to a map with all enum values of this type as its keys and info, e.g. properties and active status, * defined for these enum values as its values. */ - private static final EnumMap> enumInfo = new EnumMap<>(CustomizableEnumType.class); + private static final EnumMap>> enumInfo = + new EnumMap<>(CustomizableEnumType.class); /** * Maps a customizable enum type (defined by its class) to a map which in turn maps all languages for which translations exist to * the possible enum values of this type, which then finally map to their translated captions. @@ -162,10 +165,12 @@ public CustomizableEnumValueDto save(CustomizableEnumValueDto dto) { CustomizableEnumValue existingEntity = service.getByUuid(dto.getUuid()); - List dataTypeValues = enumValues.get(dto.getDataType()); - if (existingEntity == null && dataTypeValues != null && dataTypeValues.contains(dto.getValue())) { - throw new ValidationRuntimeException( - I18nProperties.getValidationError(Validations.customizableEnumValueDuplicateValue, dto.getDataType().toString(), dto.getValue())); + for (Disease disease : dto.getDiseases()) { + List dataTypeValues = enumValues.get(dto.getDataType()).getOrDefault(disease, Collections.emptyList()); + if (existingEntity == null && dataTypeValues != null && dataTypeValues.contains(dto.getValue())) { + throw new ValidationRuntimeException( + I18nProperties.getValidationError(Validations.customizableEnumValueDuplicateValue, dto.getDataType().toString(), dto.getValue())); + } } CustomizableEnumValue enumValue = fillOrBuildEntity(dto, existingEntity, true); @@ -267,8 +272,8 @@ private CustomizableEnumValueIndexDto toIndexDto(CustomizableEnumValue entity) { @Lock(LockType.READ) @Override @SuppressWarnings("unchecked") - public T getEnumValue(CustomizableEnumType type, String value) { - if (!enumValues.get(type).contains(value)) { + public T getEnumValue(CustomizableEnumType type, Disease disease, String value) { + if (!enumValues.get(type).getOrDefault(disease, Collections.emptyList()).contains(value)) { throw new IllegalArgumentException(String.format("Invalid enum value %s for customizable enum type %s", value, type.toString())); } @@ -279,7 +284,7 @@ public T getEnumValue(CustomizableEnumType type, St fillLanguageCache(type, enumClass, language); } - return buildCustomizableEnum(type, value, language, enumClass); + return buildCustomizableEnum(type, disease, value, language, enumClass); } @Lock(LockType.READ) @@ -329,8 +334,8 @@ public List getEnumValues(CustomizableEnumType t diseaseValuesStream = enumValuesByDisease.get(enumClass).get(Optional.empty()).stream(); } - return diseaseValuesStream.filter(value -> getEnumInfo(type, value).isActive()) - .map(value -> buildCustomizableEnum(type, value, language, enumClass)) + return diseaseValuesStream.filter(value -> getEnumInfo(type, disease, value).isActive()) + .map(value -> buildCustomizableEnum(type, disease, value, language, enumClass)) .sorted(Comparator.comparing(CustomizableEnum::getCaption)) .collect(Collectors.toList()); } @@ -341,12 +346,17 @@ public boolean hasEnumValues(CustomizableEnumType type, Disease disease) { return !getEnumValues(type, disease).isEmpty(); } - private T buildCustomizableEnum(CustomizableEnumType type, String value, Language language, Class enumClass) { + private T buildCustomizableEnum( + CustomizableEnumType type, + Disease disease, + String value, + Language language, + Class enumClass) { try { T enumValue = enumClass.getDeclaredConstructor().newInstance(); enumValue.setValue(value); enumValue.setCaption(enumValuesByLanguage.get(enumClass).get(language).get(value)); - enumValue.setProperties(getEnumInfo(type, value).getProperties()); + enumValue.setProperties(getEnumInfo(type, disease, value).getProperties()); return enumValue; } catch (InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException e) { throw new RuntimeException(e); @@ -426,7 +436,7 @@ public void loadData() { for (CustomizableEnumType enumType : CustomizableEnumType.values()) { enumValueEntities.putIfAbsent(enumType, new ArrayList<>()); - enumValues.putIfAbsent(enumType, new ArrayList<>()); + enumValues.putIfAbsent(enumType, new HashMap<>()); } // Build list of customizable enums mapped by their enum type; other caches are built on-demand @@ -434,14 +444,26 @@ public void loadData() { CustomizableEnumType enumType = customizableEnumValue.getDataType(); enumValueEntities.get(enumType).add(customizableEnumValue); String value = customizableEnumValue.getValue(); - if (enumValues.get(enumType).contains(value)) { - throw new InvalidCustomizationException( - "Enum value " + value + " for customizable enum type " + enumType.toString() + " is not unique"); + Set diseases = customizableEnumValue.getDiseases(); + + if (diseases == null) { + diseases = Collections.singleton(null); + } + for (Disease disease : diseases) { + if (enumValues.get(enumType).getOrDefault(disease, Collections.emptyList()).contains(value)) { + throw new InvalidCustomizationException( + "Enum value " + value + " for customizable enum type " + enumType.toString() + " is not unique"); + } + + enumValues.get(enumType).putIfAbsent(disease, new ArrayList<>()); + enumValues.get(enumType).get(disease).add(value); + + enumInfo.putIfAbsent(enumType, new HashMap<>()); + enumInfo.get(enumType).putIfAbsent(disease, new HashMap<>()); + enumInfo.get(enumType) + .get(disease) + .putIfAbsent(value, new CustomizableEnumCacheInfo(customizableEnumValue.getProperties(), customizableEnumValue.isActive())); } - enumValues.get(enumType).add(value); - enumInfo.putIfAbsent(enumType, new HashMap<>()); - enumInfo.get(enumType) - .putIfAbsent(value, new CustomizableEnumCacheInfo(customizableEnumValue.getProperties(), customizableEnumValue.isActive())); } for (CustomizableEnumType enumType : CustomizableEnumType.values()) { @@ -455,8 +477,10 @@ public void loadData() { } } - private CustomizableEnumCacheInfo getEnumInfo(CustomizableEnumType type, String value) { - return enumInfo.get(type).get(value); + private CustomizableEnumCacheInfo getEnumInfo(CustomizableEnumType type, Disease disease, String value) { + return enumInfo.get(type) + .getOrDefault(disease, Collections.emptyMap()) + .getOrDefault(value, disease != null ? getEnumInfo(type, null, value) : null); } public CustomizableEnumValueDto toDto(CustomizableEnumValue source) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumValue.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumValue.java index 2097ab0e235..5149aa366bf 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumValue.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumValue.java @@ -25,6 +25,7 @@ import javax.persistence.EnumType; import javax.persistence.Enumerated; +import de.symeda.sormas.api.customizableenum.CustomizableEnumConverter; import org.hibernate.annotations.Type; import de.symeda.sormas.api.Disease; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSample.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSample.java index 26f291e0157..8248c388098 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSample.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSample.java @@ -24,7 +24,6 @@ import javax.persistence.CascadeType; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -33,6 +32,7 @@ import javax.persistence.OneToMany; import javax.persistence.Temporal; import javax.persistence.TemporalType; +import javax.persistence.Transient; import org.hibernate.annotations.Type; @@ -85,6 +85,7 @@ public class EnvironmentSample extends DeletableAdo { private Float chlorineResiduals; private Facility laboratory; private String laboratoryDetails; + private String requestedPathogenTestsValue; private Set requestedPathogenTests; private String otherRequestedPathogenTests; private Map weatherConditions; @@ -230,14 +231,24 @@ public void setLaboratoryDetails(String laboratoryDetails) { this.laboratoryDetails = laboratoryDetails; } - @Column(columnDefinition = ModelConstants.COLUMN_DEFINITION_JSON) - @Convert(converter = RequestedPathogensConverter.class) + @Column(name = "requestedpathogentests", columnDefinition = ModelConstants.COLUMN_DEFINITION_JSON) + public String getRequestedPathogenTestsValue() { + return requestedPathogenTestsValue; + } + + public void setRequestedPathogenTestsValue(String requestedPathogenTestsValue) { + this.requestedPathogenTestsValue = requestedPathogenTestsValue; + this.requestedPathogenTests = new RequestedPathogensConverter().convertToEntityAttribute(null, requestedPathogenTestsValue); + } + + @Transient public Set getRequestedPathogenTests() { return requestedPathogenTests; } public void setRequestedPathogenTests(Set requestedPathogenTests) { this.requestedPathogenTests = requestedPathogenTests; + this.requestedPathogenTestsValue = new RequestedPathogensConverter().convertToDatabaseColumn(requestedPathogenTests); } @Column(length = CHARACTER_LIMIT_TEXT) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjb.java index 7c7fa3216fd..cea8575bc38 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjb.java @@ -42,6 +42,7 @@ import javax.validation.Valid; import javax.validation.constraints.NotNull; +import de.symeda.sormas.api.disease.PathogenConverter; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -239,11 +240,14 @@ private void loadAndSetPositivePathogens(CriteriaBuilder cb, List positivePathogens = QueryHelper.getResultList(em, positivePathogensCq, null, null); positivePathogens.stream() - .collect(Collectors.groupingBy(t -> (Long) t.get(0), Collectors.mapping(t -> (Pathogen) t.get(1), Collectors.toList()))) + .collect( + Collectors.groupingBy( + t -> (Long) t.get(0), + Collectors.mapping(t -> new PathogenConverter().convertToEntityAttribute(null, (String) t.get(1)), Collectors.toList()))) .forEach( (sampleId, pathogens) -> samples.stream() .filter(s -> s.getId().equals(sampleId)) @@ -260,7 +264,7 @@ private void loadAndSetLatestTest(CriteriaBuilder cb, List sampleIdExpr = testRoot.get(PathogenTest.ENVIRONMENT_SAMPLE).get(Sample.ID); testCq.multiselect( - testRoot.get(PathogenTest.TESTED_PATHOGEN), + testRoot.get(PathogenTest.TESTED_PATHOGEN_VALUE), testRoot.get(PathogenTest.TESTED_PATHOGEN_DETAILS), testRoot.get(PathogenTest.TEST_RESULT), sampleIdExpr); @@ -274,7 +278,7 @@ private void loadAndSetLatestTest(CriteriaBuilder cb, List (Long) pathogenTest[3], Function.identity(), (t1, t2) -> t1)) .forEach((sampleId, t) -> samples.stream().filter(s -> s.getId().equals(sampleId)).findFirst().ifPresent(s -> { - s.setLatestTestedPathogen((Pathogen) t[0]); + s.setLatestTestedPathogen(new PathogenConverter().convertToEntityAttribute(null, (String) t[0])); s.setLatestTestedPathogenDetails((String) t[1]); s.setLatestPathogenTestResult((PathogenTestResultType) t[2]); })); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleService.java index 35ac3d17963..15caca625d7 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleService.java @@ -164,8 +164,10 @@ public Predicate buildCriteriaFilter(EnvironmentSampleCriteria criteria, Environ filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(joins.getLaboratory().get(Facility.UUID), criteria.getLaboratory().getUuid())); } if (criteria.getTestedPathogen() != null) { - filter = CriteriaBuilderHelper - .and(cb, filter, cb.equal(joins.getPathogenTests().get(PathogenTestDto.TESTED_PATHOGEN), criteria.getTestedPathogen())); + filter = CriteriaBuilderHelper.and( + cb, + filter, + cb.equal(joins.getPathogenTests().get(PathogenTest.TESTED_PATHOGEN_VALUE), criteria.getTestedPathogen().getValue())); } if (criteria.getReportDateFrom() != null && criteria.getReportDateTo() != null) { filter = CriteriaBuilderHelper diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/RequestedPathogensConverter.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/RequestedPathogensConverter.java index ea8fde070e8..99c2ca24c69 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/RequestedPathogensConverter.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/environment/environmentsample/RequestedPathogensConverter.java @@ -19,28 +19,27 @@ import java.util.stream.Collectors; import java.util.stream.Stream; -import javax.persistence.AttributeConverter; - +import de.symeda.sormas.api.Disease; import org.apache.commons.lang3.StringUtils; import de.symeda.sormas.api.environment.environmentsample.Pathogen; -import de.symeda.sormas.backend.disease.PathogenConverter; +import de.symeda.sormas.api.disease.PathogenConverter; -public class RequestedPathogensConverter implements AttributeConverter, String> { +public class RequestedPathogensConverter { private final PathogenConverter pathogenConverter = new PathogenConverter(); - @Override public String convertToDatabaseColumn(Set pathogens) { return pathogens != null ? String.join(",", pathogens.stream().map(pathogenConverter::convertToDatabaseColumn).collect(Collectors.toSet())) : null; } - @Override - public Set convertToEntityAttribute(String pathogensText) { + public Set convertToEntityAttribute(Disease disease, String pathogensText) { return pathogensText != null - ? Stream.of(StringUtils.split(pathogensText, ",")).map(pathogenConverter::convertToEntityAttribute).collect(Collectors.toSet()) + ? Stream.of(StringUtils.split(pathogensText, ",")) + .map((String value) -> pathogenConverter.convertToEntityAttribute(disease, value)) + .collect(Collectors.toSet()) : null; } } diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/Event.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/Event.java index 26202be9bf5..c089347ddc3 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/Event.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/Event.java @@ -24,7 +24,6 @@ import javax.persistence.CascadeType; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -37,7 +36,9 @@ import javax.persistence.OneToOne; import javax.persistence.Temporal; import javax.persistence.TemporalType; +import javax.persistence.Transient; +import de.symeda.sormas.api.event.SpecificRiskConverter; import org.hibernate.annotations.Type; import de.symeda.sormas.api.Disease; @@ -64,7 +65,7 @@ import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.backend.action.Action; import de.symeda.sormas.backend.common.CoreAdo; -import de.symeda.sormas.backend.disease.DiseaseVariantConverter; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.backend.location.Location; import de.symeda.sormas.backend.share.ExternalShareInfo; import de.symeda.sormas.backend.sormastosormas.entities.SormasToSormasShareable; @@ -88,7 +89,7 @@ public class Event extends CoreAdo implements SormasToSormasShareable, HasExtern public static final String INTERNAL_TOKEN = "internalToken"; public static final String EVENT_STATUS = "eventStatus"; public static final String RISK_LEVEL = "riskLevel"; - public static final String SPECIFIC_RISK = "specificRisk"; + public static final String SPECIFIC_RISK_VALUE = "specificRiskValue"; public static final String EVENT_INVESTIGATION_STATUS = "eventInvestigationStatus"; public static final String EVENT_INVESTIGATION_START_DATE = "eventInvestigationStartDate"; public static final String EVENT_INVESTIGATION_END_DATE = "eventInvestigationEndDate"; @@ -123,7 +124,7 @@ public class Event extends CoreAdo implements SormasToSormasShareable, HasExtern public static final String SRC_MEDIA_NAME = "srcMediaName"; public static final String SRC_MEDIA_DETAILS = "srcMediaDetails"; public static final String DISEASE = "disease"; - public static final String DISEASE_VARIANT = "diseaseVariant"; + public static final String DISEASE_VARIANT_VALUE = "diseaseVariantValue"; public static final String DISEASE_DETAILS = "diseaseDetails"; public static final String DISEASE_VARIANT_DETAILS = "diseaseVariantDetails"; public static final String RESPONSIBLE_USER = "responsibleUser"; @@ -151,6 +152,7 @@ public class Event extends CoreAdo implements SormasToSormasShareable, HasExtern private EventStatus eventStatus; private RiskLevel riskLevel; + private String specificRiskValue; private SpecificRisk specificRisk; private EventInvestigationStatus eventInvestigationStatus; private Date eventInvestigationStartDate; @@ -185,6 +187,7 @@ public class Event extends CoreAdo implements SormasToSormasShareable, HasExtern private String srcMediaName; private String srcMediaDetails; private Disease disease; + private String diseaseVariantValue; private DiseaseVariant diseaseVariant; private String diseaseDetails; private String diseaseVariantDetails; @@ -238,14 +241,24 @@ public void setRiskLevel(RiskLevel riskLevel) { this.riskLevel = riskLevel; } - @Column - @Convert(converter = SpecificRiskConverter.class) + @Column(name = "specificrisk") + public String getSpecificRiskValue() { + return specificRiskValue; + } + + public void setSpecificRiskValue(String specificRiskValue) { + this.specificRiskValue = specificRiskValue; + this.specificRisk = new SpecificRiskConverter().convertToEntityAttribute(disease, specificRiskValue); + } + + @Transient public SpecificRisk getSpecificRisk() { return specificRisk; } public void setSpecificRisk(SpecificRisk specificRisk) { this.specificRisk = specificRisk; + this.specificRiskValue = new SpecificRiskConverter().convertToDatabaseColumn(specificRisk); } @Enumerated(EnumType.STRING) @@ -548,14 +561,24 @@ public void setDisease(Disease disease) { this.disease = disease; } - @Column - @Convert(converter = DiseaseVariantConverter.class) + @Column(name = "diseasevariant") + public String getDiseaseVariantValue() { + return diseaseVariantValue; + } + + public void setDiseaseVariantValue(String diseaseVariantValue) { + this.diseaseVariantValue = diseaseVariantValue; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariantValue); + } + + @Transient public DiseaseVariant getDiseaseVariant() { return diseaseVariant; } public void setDiseaseVariant(DiseaseVariant diseaseVariant) { this.diseaseVariant = diseaseVariant; + this.diseaseVariantValue = new DiseaseVariantConverter().convertToDatabaseColumn(diseaseVariant); } @Column(length = CHARACTER_LIMIT_DEFAULT) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventFacadeEjb.java index 3d776fe4351..da748cf65c0 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventFacadeEjb.java @@ -497,11 +497,11 @@ public List getIndexList(EventCriteria eventCriteria, Integer fir event.get(Event.INTERNAL_TOKEN), event.get(Event.EVENT_STATUS), event.get(Event.RISK_LEVEL), - event.get(Event.SPECIFIC_RISK), + event.get(Event.SPECIFIC_RISK_VALUE), event.get(Event.EVENT_INVESTIGATION_STATUS), event.get(Event.EVENT_MANAGEMENT_STATUS), event.get(Event.DISEASE), - event.get(Event.DISEASE_VARIANT), + event.get(Event.DISEASE_VARIANT_VALUE), event.get(Event.DISEASE_DETAILS), event.get(Event.START_DATE), event.get(Event.END_DATE), @@ -838,12 +838,12 @@ public List getExportList(EventCriteria eventCriteria, Collectio event.get(Event.INTERNAL_TOKEN), event.get(Event.EVENT_STATUS), event.get(Event.RISK_LEVEL), - event.get(Event.SPECIFIC_RISK), + event.get(Event.SPECIFIC_RISK_VALUE), event.get(Event.EVENT_INVESTIGATION_STATUS), event.get(Event.EVENT_INVESTIGATION_START_DATE), event.get(Event.EVENT_INVESTIGATION_END_DATE), event.get(Event.DISEASE), - event.get(Event.DISEASE_VARIANT), + event.get(Event.DISEASE_VARIANT_VALUE), event.get(Event.DISEASE_DETAILS), event.get(Event.DISEASE_VARIANT_DETAILS), event.get(Event.START_DATE), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventIndexDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventIndexDtoResultTransformer.java index eace54ae644..087c1598753 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventIndexDtoResultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventIndexDtoResultTransformer.java @@ -43,8 +43,8 @@ public EventIndexDto transformTuple(Object[] tuple, String[] aliases) { //@formatter:off return new EventIndexDto( (Long) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], - (EventStatus) tuple[++index], (RiskLevel) tuple[++index], (SpecificRisk) tuple[++index], (EventInvestigationStatus) tuple[++index], - (EventManagementStatus) tuple[++index], (Disease) tuple[++index], (DiseaseVariant) tuple[++index], (String) tuple[++index], + (EventStatus) tuple[++index], (RiskLevel) tuple[++index], (String) tuple[++index], (EventInvestigationStatus) tuple[++index], + (EventManagementStatus) tuple[++index], (Disease) tuple[++index], (String) tuple[++index], (String) tuple[++index], (Date) tuple[++index], (Date) tuple[++index], (Date) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java index 89f07b85306..f394c5f72c6 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/event/EventService.java @@ -722,7 +722,7 @@ public Predicate buildCriteriaFilter(EventCriteria eventCriteria, EventQueryCont filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.DISEASE), eventCriteria.getDisease())); } if (eventCriteria.getDiseaseVariant() != null) { - filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.DISEASE_VARIANT), eventCriteria.getDiseaseVariant())); + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.DISEASE_VARIANT_VALUE), eventCriteria.getDiseaseVariant().getValue())); } if (eventCriteria.getEventStatus() != null) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.EVENT_STATUS), eventCriteria.getEventStatus())); @@ -731,7 +731,7 @@ public Predicate buildCriteriaFilter(EventCriteria eventCriteria, EventQueryCont filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.RISK_LEVEL), eventCriteria.getRiskLevel())); } if (eventCriteria.getSpecificRisk() != null) { - filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.SPECIFIC_RISK), eventCriteria.getSpecificRisk())); + filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(Event.SPECIFIC_RISK_VALUE), eventCriteria.getSpecificRisk().getValue())); } if (eventCriteria.getEventInvestigationStatus() != null) { filter = CriteriaBuilderHelper diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java index 0e084a75d4d..72a7698665f 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessage.java @@ -8,7 +8,6 @@ import javax.persistence.CascadeType; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -19,6 +18,7 @@ import javax.persistence.OneToOne; import javax.persistence.Temporal; import javax.persistence.TemporalType; +import javax.persistence.Transient; import org.hibernate.annotations.Type; import org.hibernate.annotations.TypeDef; @@ -36,7 +36,7 @@ import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.backend.caze.surveillancereport.SurveillanceReport; import de.symeda.sormas.backend.common.AbstractDomainObject; -import de.symeda.sormas.backend.disease.DiseaseVariantConverter; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.backend.externalmessage.labmessage.SampleReport; import de.symeda.sormas.backend.infrastructure.country.Country; import de.symeda.sormas.backend.infrastructure.facility.Facility; @@ -52,7 +52,7 @@ public class ExternalMessage extends AbstractDomainObject { public static final String TYPE = "type"; public static final String DISEASE = "disease"; - public static final String DISEASE_VARIANT = "diseaseVariant"; + public static final String DISEASE_VARIANT_VALUE = "diseaseVariantValue"; public static final String DISEASE_VARIANT_DETAILS = "diseaseVariantDetails"; public static final String MESSAGE_DATE_TIME = "messageDateTime"; public static final String CASE_REPORT_DATE = "caseReportDate"; @@ -87,6 +87,7 @@ public class ExternalMessage extends AbstractDomainObject { private ExternalMessageType type; private Disease disease; + private String diseaseVariantValue; private DiseaseVariant diseaseVariant; private String diseaseVariantDetails; private Date messageDateTime; @@ -149,14 +150,24 @@ public void setDisease(Disease disease) { this.disease = disease; } - @Column - @Convert(converter = DiseaseVariantConverter.class) + @Column(name = "diseasevariant") + public String getDiseaseVariantValue() { + return diseaseVariantValue; + } + + public void setDiseaseVariantValue(String diseaseVariantValue) { + this.diseaseVariantValue = diseaseVariantValue; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariantValue); + } + + @Transient public DiseaseVariant getDiseaseVariant() { return diseaseVariant; } public void setDiseaseVariant(DiseaseVariant diseaseVariant) { this.diseaseVariant = diseaseVariant; + this.diseaseVariantValue = new DiseaseVariantConverter().convertToDatabaseColumn(diseaseVariant); } @Column(length = CHARACTER_LIMIT_DEFAULT) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java index a71d7d99e0f..81fe078bf4d 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjb.java @@ -492,7 +492,7 @@ public List getIndexList( externalMessage.get(ExternalMessage.REPORTER_NAME), externalMessage.get(ExternalMessage.REPORTER_POSTAL_CODE), externalMessage.get(ExternalMessage.DISEASE), - externalMessage.get(ExternalMessage.DISEASE_VARIANT), + externalMessage.get(ExternalMessage.DISEASE_VARIANT_VALUE), externalMessage.get(ExternalMessage.PERSON_FIRST_NAME), externalMessage.get(ExternalMessage.PERSON_LAST_NAME), externalMessage.get(ExternalMessage.PERSON_BIRTH_DATE_YYYY), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageIndexDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageIndexDtoResultTransformer.java index 0c00756ee2a..58c1574b6b7 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageIndexDtoResultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageIndexDtoResultTransformer.java @@ -37,7 +37,7 @@ public ExternalMessageIndexDto transformTuple(Object[] tuple, String[] aliases) //@formatter:off return new ExternalMessageIndexDto( (String) tuple[++index], (ExternalMessageType) tuple[++index], (Date) tuple[++index], (String) tuple[++index], - (String) tuple[++index], (Disease) tuple[++index], (DiseaseVariant) tuple[++index], + (String) tuple[++index], (Disease) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (Integer) tuple[++index], (Integer) tuple[++index], (Integer) tuple[++index], (String) tuple[++index], (ExternalMessageStatus) tuple[++index], diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java index 9ce612c32cd..d40f751e5ae 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/externalmessage/ExternalMessageService.java @@ -18,7 +18,6 @@ import javax.persistence.criteria.Predicate; import javax.persistence.criteria.Root; -import de.symeda.sormas.backend.person.Person; import org.apache.commons.lang3.StringUtils; import de.symeda.sormas.api.ReferenceDto; @@ -93,7 +92,8 @@ public Predicate buildCriteriaFilter(CriteriaBuilder cb, Root l filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(labMessage.get(ExternalMessage.DISEASE), criteria.getDisease())); } if (criteria.getDiseaseVariant() != null) { - filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(labMessage.get(ExternalMessage.DISEASE_VARIANT), criteria.getDiseaseVariant())); + filter = CriteriaBuilderHelper + .and(cb, filter, cb.equal(labMessage.get(ExternalMessage.DISEASE_VARIANT_VALUE), criteria.getDiseaseVariant().getValue())); } if (criteria.getExternalMessageStatus() != null) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(labMessage.get(ExternalMessage.STATUS), criteria.getExternalMessageStatus())); diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/OccupationTypeConverter.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/OccupationTypeConverter.java deleted file mode 100644 index 26e59dfd31a..00000000000 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/OccupationTypeConverter.java +++ /dev/null @@ -1,11 +0,0 @@ -package de.symeda.sormas.backend.person; - -import de.symeda.sormas.api.person.OccupationType; -import de.symeda.sormas.backend.customizableenum.CustomizableEnumConverter; - -public class OccupationTypeConverter extends CustomizableEnumConverter { - - public OccupationTypeConverter() { - super(OccupationType.class); - } -} diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/Person.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/Person.java index eee616d80fc..c3d8fc916a2 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/Person.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/Person.java @@ -28,7 +28,6 @@ import javax.persistence.CascadeType; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -53,6 +52,7 @@ import de.symeda.sormas.api.person.EducationType; import de.symeda.sormas.api.person.IsPerson; import de.symeda.sormas.api.person.OccupationType; +import de.symeda.sormas.api.person.OccupationTypeConverter; import de.symeda.sormas.api.person.PersonContactDetailType; import de.symeda.sormas.api.person.PersonReferenceDto; import de.symeda.sormas.api.person.PresentCondition; @@ -108,7 +108,7 @@ public class Person extends AbstractDomainObject implements IsPerson, HasExterna public static final String PRESENT_CONDITION = "presentCondition"; public static final String EDUCATION_TYPE = "educationType"; public static final String EDUCATION_DETAILS = "educationDetails"; - public static final String OCCUPATION_TYPE = "occupationType"; + public static final String OCCUPATION_TYPE_VALUE = "occupationTypeValue"; public static final String OCCUPATION_DETAILS = "occupationDetails"; public static final String ARMED_FORCES_RELATION_TYPE = "armedForcesRelationType"; public static final String FATHERS_NAME = "fathersName"; @@ -188,6 +188,7 @@ public class Person extends AbstractDomainObject implements IsPerson, HasExterna private EducationType educationType; private String educationDetails; + private String occupationTypeValue; private OccupationType occupationType; private String occupationDetails; private ArmedForcesRelationType armedForcesRelationType; @@ -455,14 +456,24 @@ public void setEducationDetails(String educationDetails) { this.educationDetails = educationDetails; } - @Column - @Convert(converter = OccupationTypeConverter.class) + @Column(name = "occupationtype") + public String getOccupationTypeValue() { + return occupationTypeValue; + } + + public void setOccupationTypeValue(String occupationType) { + this.occupationTypeValue = occupationType; + this.occupationType = new OccupationTypeConverter().convertToEntityAttribute(null, occupationTypeValue); + } + + @Transient public OccupationType getOccupationType() { return occupationType; } public void setOccupationType(OccupationType occupationType) { this.occupationType = occupationType; + this.occupationTypeValue = new OccupationTypeConverter().convertToDatabaseColumn(occupationType); } public String getOccupationDetails() { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java index 2266e8b4f51..a6d939302f2 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/person/PersonFacadeEjb.java @@ -1550,7 +1550,7 @@ public List getExportList(PersonCriteria criteria, int first, i person.get(Person.EDUCATION_TYPE), person.get(Person.EDUCATION_DETAILS), - person.get(Person.OCCUPATION_TYPE), + person.get(Person.OCCUPATION_TYPE_VALUE), person.get(Person.OCCUPATION_DETAILS), person.get(Person.ARMED_FORCES_RELATION_TYPE), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java index 79aeef5cffc..a521ca962af 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/PathogenTest.java @@ -23,7 +23,6 @@ import java.util.Date; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -32,6 +31,7 @@ import javax.persistence.ManyToOne; import javax.persistence.Temporal; import javax.persistence.TemporalType; +import javax.persistence.Transient; import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.disease.DiseaseVariant; @@ -41,8 +41,8 @@ import de.symeda.sormas.api.sample.PathogenTestResultType; import de.symeda.sormas.api.sample.PathogenTestType; import de.symeda.sormas.backend.common.DeletableAdo; -import de.symeda.sormas.backend.disease.DiseaseVariantConverter; -import de.symeda.sormas.backend.disease.PathogenConverter; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; +import de.symeda.sormas.api.disease.PathogenConverter; import de.symeda.sormas.backend.environment.environmentsample.EnvironmentSample; import de.symeda.sormas.backend.infrastructure.country.Country; import de.symeda.sormas.backend.infrastructure.facility.Facility; @@ -58,9 +58,9 @@ public class PathogenTest extends DeletableAdo { public static final String SAMPLE = "sample"; public static final String ENVIRONMENT_SAMPLE = "environmentSample"; public static final String TESTED_DISEASE = "testedDisease"; - public static final String TESTED_DISEASE_VARIANT = "testedDiseaseVariant"; + public static final String TESTED_DISEASE_VARIANT_VALUE = "testedDiseaseVariantValue"; public static final String TESTED_DISEASE_VARIANT_DETAILS = "testedDiseaseVariantDetails"; - public static final String TESTED_PATHOGEN = "testedPathogen"; + public static final String TESTED_PATHOGEN_VALUE = "testedPathogenValue"; public static final String TESTED_PATHOGEN_DETAILS = "testedPathogenDetails"; public static final String TYPING_ID = "typingId"; public static final String TEST_TYPE = "testType"; @@ -95,11 +95,11 @@ public class PathogenTest extends DeletableAdo { private Sample sample; private EnvironmentSample environmentSample; private Disease testedDisease; - @Convert(converter = DiseaseVariantConverter.class) + private String testedDiseaseVariantValue; private DiseaseVariant testedDiseaseVariant; private String testedDiseaseDetails; private String testedDiseaseVariantDetails; - @Convert(converter = PathogenConverter.class) + private String testedPathogenValue; private Pathogen testedPathogen; private String testedPathogenDetails; private String typingId; @@ -181,24 +181,44 @@ public void setTestedDiseaseVariantDetails(String testedDiseaseVariantDetails) { this.testedDiseaseVariantDetails = testedDiseaseVariantDetails; } - @Column - @Convert(converter = DiseaseVariantConverter.class) + @Column(name = "testeddiseasevariant") + public String getTestedDiseaseVariantValue() { + return testedDiseaseVariantValue; + } + + public void setTestedDiseaseVariantValue(String diseaseVariantValue) { + this.testedDiseaseVariantValue = diseaseVariantValue; + this.testedDiseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(testedDisease, testedDiseaseVariantValue); + } + + @Transient public DiseaseVariant getTestedDiseaseVariant() { return testedDiseaseVariant; } public void setTestedDiseaseVariant(DiseaseVariant diseaseVariant) { this.testedDiseaseVariant = diseaseVariant; + this.testedDiseaseVariantValue = new DiseaseVariantConverter().convertToDatabaseColumn(diseaseVariant); } - @Column - @Convert(converter = PathogenConverter.class) + @Column(name = "testedpathogen") + public String getTestedPathogenValue() { + return testedPathogenValue; + } + + public void setTestedPathogenValue(String testedPathogenValue) { + this.testedPathogenValue = testedPathogenValue; + this.testedPathogen = new PathogenConverter().convertToEntityAttribute(null, testedPathogenValue); + } + + @Transient public Pathogen getTestedPathogen() { return testedPathogen; } public void setTestedPathogen(Pathogen testedPathogen) { this.testedPathogen = testedPathogen; + this.testedPathogenValue = new PathogenConverter().convertToDatabaseColumn(testedPathogen); } @Column(length = CHARACTER_LIMIT_DEFAULT) diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java index 64efb668471..458059e7624 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/sample/SampleService.java @@ -67,6 +67,7 @@ import de.symeda.sormas.api.common.progress.ProcessedEntityStatus; import de.symeda.sormas.api.contact.ContactReferenceDto; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.event.EventParticipantReferenceDto; import de.symeda.sormas.api.feature.FeatureType; import de.symeda.sormas.api.i18n.I18nProperties; @@ -1277,7 +1278,7 @@ public List getAssociatedDiseaseVariants(String sampleUuid) { } final CriteriaBuilder cb = em.getCriteriaBuilder(); - final CriteriaQuery cq = cb.createQuery(DiseaseVariant.class); + final CriteriaQuery cq = cb.createTupleQuery(); final Root from = cq.from(getElementClass()); final Join pathogenTestJoin = from.join(Sample.PATHOGENTESTS, JoinType.LEFT); @@ -1286,8 +1287,12 @@ public List getAssociatedDiseaseVariants(String sampleUuid) { filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(AbstractDomainObject.UUID), sampleUuid)); filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(pathogenTestJoin.get(DeletableAdo.DELETED), false)); cq.where(filter); - cq.select(pathogenTestJoin.get(PathogenTest.TESTED_DISEASE_VARIANT)); - return em.createQuery(cq).getResultList(); + cq.multiselect(pathogenTestJoin.get(PathogenTest.TESTED_DISEASE), pathogenTestJoin.get(PathogenTest.TESTED_DISEASE_VARIANT_VALUE)); + return em.createQuery(cq) + .getResultList() + .stream() + .map(tuple -> new DiseaseVariantConverter().convertToEntityAttribute((Disease) tuple.get(0), (String) tuple.get(1))) + .collect(Collectors.toList()); } public void cleanupOldCovidSamples() { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReport.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReport.java index 6e678199391..f09d7bbceaf 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReport.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReport.java @@ -22,7 +22,6 @@ import javax.persistence.CascadeType; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -31,6 +30,7 @@ import javax.persistence.OneToOne; import javax.persistence.Temporal; import javax.persistence.TemporalType; +import javax.persistence.Transient; import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.disease.DiseaseVariant; @@ -41,7 +41,7 @@ import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.common.CoreAdo; import de.symeda.sormas.backend.contact.Contact; -import de.symeda.sormas.backend.disease.DiseaseVariantConverter; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.backend.location.Location; import de.symeda.sormas.backend.user.User; @@ -55,7 +55,7 @@ public class SelfReport extends CoreAdo { public static final String CASE_REFERENCE = "caseReference"; public static final String DISEASE = "disease"; public static final String DISEASE_DETAILS = "diseaseDetails"; - public static final String DISEASE_VARIANT = "diseaseVariant"; + public static final String DISEASE_VARIANT_VALUE = "diseaseVariantValue"; public static final String DISEASE_VARIANT_DETAILS = "diseaseVariantDetails"; public static final String FIRST_NAME = "firstName"; public static final String LAST_NAME = "lastName"; @@ -85,6 +85,7 @@ public class SelfReport extends CoreAdo { private String caseReference; private Disease disease; private String diseaseDetails; + private String diseaseVariantValue; private DiseaseVariant diseaseVariant; private String diseaseVariantDetails; // person data @@ -161,12 +162,26 @@ public void setDiseaseDetails(String diseaseDetails) { this.diseaseDetails = diseaseDetails; } - @Column - @Convert(converter = DiseaseVariantConverter.class) + @Column(name = "diseasevariant") + public String getDiseaseVariantValue() { + return diseaseVariantValue; + } + + public void setDiseaseVariantValue(String diseaseVariantValue) { + this.diseaseVariantValue = diseaseVariantValue; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariantValue); + } + + @Transient public DiseaseVariant getDiseaseVariant() { return diseaseVariant; } + public void setDiseaseVariant(DiseaseVariant diseaseVariant) { + this.diseaseVariant = diseaseVariant; + this.diseaseVariantValue = new DiseaseVariantConverter().convertToDatabaseColumn(diseaseVariant); + } + @Column(length = CHARACTER_LIMIT_DEFAULT) public String getDiseaseVariantDetails() { return diseaseVariantDetails; @@ -176,10 +191,6 @@ public void setDiseaseVariantDetails(String diseaseVariantDetails) { this.diseaseVariantDetails = diseaseVariantDetails; } - public void setDiseaseVariant(DiseaseVariant diseaseVariant) { - this.diseaseVariant = diseaseVariant; - } - @Column(nullable = false, length = CHARACTER_LIMIT_DEFAULT) public String getFirstName() { return firstName; diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportExportDtoResultTransformer.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportExportDtoResultTransformer.java index c1fe9af4a0c..192029a2eaf 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportExportDtoResultTransformer.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportExportDtoResultTransformer.java @@ -27,7 +27,7 @@ public SelfReportExportDto transformTuple(Object[] tuple, String[] aliases) { //@formatter:off return new SelfReportExportDto( (String) tuple[++index], (SelfReportType) tuple[++index], (Date) tuple[++index], - (String) tuple[++index], (Disease) tuple[++index], (String) tuple[++index], (DiseaseVariant) tuple[++index], + (String) tuple[++index], (Disease) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (Sex) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], (String) tuple[++index], new BirthDateDto((Integer)tuple[++index], (Integer) tuple[++index], (Integer) tuple[++index]), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportFacadeEjb.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportFacadeEjb.java index dd86712e11f..95bb2a6605c 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportFacadeEjb.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportFacadeEjb.java @@ -253,7 +253,7 @@ public List getExportList(SelfReportCriteria selfReportCrit selfReport.get(SelfReport.CASE_REFERENCE), selfReport.get(SelfReport.DISEASE), selfReport.get(SelfReport.DISEASE_DETAILS), - selfReport.get(SelfReport.DISEASE_VARIANT), + selfReport.get(SelfReport.DISEASE_VARIANT_VALUE), selfReport.get(SelfReport.DISEASE_VARIANT_DETAILS), selfReport.get(SelfReport.FIRST_NAME), selfReport.get(SelfReport.LAST_NAME), diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportService.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportService.java index 19b1fe0192c..825b6674e09 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportService.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/selfreport/SelfReportService.java @@ -15,6 +15,11 @@ package de.symeda.sormas.backend.selfreport; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; + import javax.ejb.LocalBean; import javax.ejb.Stateless; import javax.persistence.Tuple; @@ -26,25 +31,19 @@ import javax.persistence.criteria.Root; import javax.persistence.criteria.Selection; -import de.symeda.sormas.api.selfreport.SelfReportListEntryDto; -import de.symeda.sormas.backend.caze.Case; -import de.symeda.sormas.backend.contact.Contact; -import de.symeda.sormas.backend.sample.Sample; -import de.symeda.sormas.backend.util.QueryHelper; import org.apache.commons.lang3.StringUtils; import de.symeda.sormas.api.EntityRelevanceStatus; import de.symeda.sormas.api.common.DeletableEntityType; import de.symeda.sormas.api.selfreport.SelfReportCriteria; +import de.symeda.sormas.api.selfreport.SelfReportListEntryDto; import de.symeda.sormas.api.utils.DataHelper; +import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.common.AbstractCoreAdoService; import de.symeda.sormas.backend.common.CriteriaBuilderHelper; +import de.symeda.sormas.backend.contact.Contact; import de.symeda.sormas.backend.location.Location; - -import java.util.ArrayList; -import java.util.Arrays; -import java.util.Collections; -import java.util.List; +import de.symeda.sormas.backend.util.QueryHelper; @Stateless @LocalBean @@ -115,7 +114,8 @@ public Predicate buildCriteriaFilter(SelfReportCriteria criteria, SelfReportQuer } if (criteria.getDiseaseVariant() != null) { - filter = CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(SelfReport.DISEASE_VARIANT), criteria.getDiseaseVariant())); + filter = + CriteriaBuilderHelper.and(cb, filter, cb.equal(from.get(SelfReport.DISEASE_VARIANT_VALUE), criteria.getDiseaseVariant().getValue())); } if (criteria.getReportDateFrom() != null && criteria.getReportDateTo() != null) { diff --git a/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntry.java b/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntry.java index 193dcdba31e..7d356fd1bea 100644 --- a/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntry.java +++ b/sormas-backend/src/main/java/de/symeda/sormas/backend/travelentry/TravelEntry.java @@ -4,7 +4,6 @@ import java.util.List; import javax.persistence.Column; -import javax.persistence.Convert; import javax.persistence.Entity; import javax.persistence.EnumType; import javax.persistence.Enumerated; @@ -13,18 +12,19 @@ import javax.persistence.ManyToOne; import javax.persistence.Temporal; import javax.persistence.TemporalType; +import javax.persistence.Transient; import org.hibernate.annotations.Type; import de.symeda.sormas.api.Disease; import de.symeda.sormas.api.contact.QuarantineType; import de.symeda.sormas.api.disease.DiseaseVariant; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; import de.symeda.sormas.api.travelentry.DeaContentEntry; import de.symeda.sormas.api.travelentry.IsTravelEntry; import de.symeda.sormas.api.utils.YesNoUnknown; import de.symeda.sormas.backend.caze.Case; import de.symeda.sormas.backend.common.CoreAdo; -import de.symeda.sormas.backend.disease.DiseaseVariantConverter; import de.symeda.sormas.backend.infrastructure.community.Community; import de.symeda.sormas.backend.infrastructure.district.District; import de.symeda.sormas.backend.infrastructure.pointofentry.PointOfEntry; @@ -46,7 +46,7 @@ public class TravelEntry extends CoreAdo implements IsTravelEntry { public static final String REPORTING_USER = "reportingUser"; public static final String DELETED = "deleted"; public static final String DISEASE = "disease"; - public static final String DISEASE_VARIANT = "diseaseVariant"; + public static final String DISEASE_VARIANT_VALUE = "diseaseVariantValue"; public static final String RESPONSIBLE_REGION = "responsibleRegion"; public static final String RESPONSIBLE_DISTRICT = "responsibleDistrict"; public static final String RESPONSIBLE_COMMUNITY = "responsibleCommunity"; @@ -69,6 +69,7 @@ public class TravelEntry extends CoreAdo implements IsTravelEntry { private boolean deleted; private Disease disease; private String diseaseDetails; + private String diseaseVariantValue; private DiseaseVariant diseaseVariant; private String diseaseVariantDetails; private Region responsibleRegion; @@ -174,14 +175,24 @@ public void setDiseaseDetails(String diseaseDetails) { this.diseaseDetails = diseaseDetails; } - @Column - @Convert(converter = DiseaseVariantConverter.class) + @Column(name = "diseasevariant") + public String getDiseaseVariantValue() { + return diseaseVariantValue; + } + + public void setDiseaseVariantValue(String diseaseVariantValue) { + this.diseaseVariantValue = diseaseVariantValue; + this.diseaseVariant = new DiseaseVariantConverter().convertToEntityAttribute(disease, diseaseVariantValue); + } + + @Transient public DiseaseVariant getDiseaseVariant() { return diseaseVariant; } public void setDiseaseVariant(DiseaseVariant diseaseVariant) { this.diseaseVariant = diseaseVariant; + this.diseaseVariantValue = new DiseaseVariantConverter().convertToDatabaseColumn(diseaseVariant); } @Column(columnDefinition = "text") diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/TestDataCreator.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/TestDataCreator.java index a45518e4ce8..b679bac8f35 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/TestDataCreator.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/TestDataCreator.java @@ -2236,7 +2236,7 @@ public DiseaseVariant createDiseaseVariant(String name, Disease disease) { beanTest.getCustomizableEnumValueService().ensurePersisted(diseaseVariant); - return beanTest.getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, name); + return beanTest.getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, disease, name); } public Pathogen createPathogen(String value, String caption) { @@ -2251,7 +2251,7 @@ public Pathogen createPathogen(String value, String caption) { beanTest.getCustomizableEnumFacade().loadData(); - return beanTest.getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.PATHOGEN, value); + return beanTest.getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.PATHOGEN, null, value); } public ExternalShareInfo createExternalShareInfo( diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/caze/CaseFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/caze/CaseFacadeEjbTest.java index 46bc12c5d7b..03015193c41 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/caze/CaseFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/caze/CaseFacadeEjbTest.java @@ -3343,7 +3343,9 @@ public void testBulkUpdateBulkEditDiseaseVariant() { DiseaseVariant diseaseVariant = creator.createDiseaseVariant("BF.1.2", Disease.CORONAVIRUS); Mockito - .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, diseaseVariant.getValue())) + .when( + MockProducer.getCustomizableEnumFacadeForConverter() + .getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, diseaseVariant.getValue())) .thenReturn(diseaseVariant); CaseBulkEditData bulkEditData = new CaseBulkEditData(); @@ -3376,7 +3378,9 @@ public void testBulkUpdateBulkEditDiseaseVariant() { public void testBulkUpdateBulkEditDiseaseClearVariantOfChangedDisease() { DiseaseVariant diseaseVariant = creator.createDiseaseVariant("BF.1.2", Disease.CORONAVIRUS); Mockito - .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, diseaseVariant.getValue())) + .when( + MockProducer.getCustomizableEnumFacadeForConverter() + .getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, diseaseVariant.getValue())) .thenReturn(diseaseVariant); CaseDataDto caze = creator.createCase(surveillanceSupervisor.toReference(), creator.createPerson().toReference(), rdcf, c -> { c.setDisease(Disease.CORONAVIRUS); diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverterTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverterTest.java new file mode 100644 index 00000000000..745c09082ef --- /dev/null +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/customizableenum/CustomizableEnumConverterTest.java @@ -0,0 +1,114 @@ +/* + * SORMAS® - Surveillance Outbreak Response Management & Analysis System + * Copyright © 2016-2024 Helmholtz-Zentrum für Infektionsforschung GmbH (HZI) + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +package de.symeda.sormas.backend.customizableenum; + +import static org.hamcrest.MatcherAssert.assertThat; +import static org.hamcrest.Matchers.is; + +import java.util.HashSet; +import java.util.List; +import java.util.Set; + +import org.junit.jupiter.api.Test; +import org.mockito.Mockito; + +import de.symeda.sormas.api.Disease; +import de.symeda.sormas.api.caze.CaseDataDto; +import de.symeda.sormas.api.customizableenum.CustomizableEnum; +import de.symeda.sormas.api.customizableenum.CustomizableEnumType; +import de.symeda.sormas.api.disease.DiseaseVariantConverter; +import de.symeda.sormas.api.user.DefaultUserRole; +import de.symeda.sormas.api.user.UserReferenceDto; +import de.symeda.sormas.backend.AbstractBeanTest; +import de.symeda.sormas.backend.MockProducer; +import de.symeda.sormas.backend.TestDataCreator; + +public class CustomizableEnumConverterTest extends AbstractBeanTest { + + private TestDataCreator.RDCF rdcf; + private UserReferenceDto reportingUser; + private CustomizableEnum bf_1_2; + private CustomizableEnum generic; + + @Override + public void init() { + super.init(); + + rdcf = creator.createRDCF(); + reportingUser = creator.createUser(rdcf, DefaultUserRole.SURVEILLANCE_OFFICER).toReference(); + + List enumValues = getCustomizableEnumValueService().getAll(); + if (enumValues.isEmpty()) { + CustomizableEnumValue bf_1_2_value = new CustomizableEnumValue(); + bf_1_2_value.setDataType(CustomizableEnumType.DISEASE_VARIANT); + bf_1_2_value.setValue("BF.1.2"); + Set diseases = new HashSet<>(); + diseases.add(Disease.CORONAVIRUS); + bf_1_2_value.setDiseases(diseases); + bf_1_2_value.setCaption("BF.1.2 variant"); + bf_1_2_value.setActive(true); + getCustomizableEnumValueService().ensurePersisted(bf_1_2_value); + + CustomizableEnumValue generic_value = new CustomizableEnumValue(); + generic_value.setDataType(CustomizableEnumType.DISEASE_VARIANT); + generic_value.setValue("GENERIC"); + generic_value.setCaption("Variant 2"); + generic_value.setActive(true); + getCustomizableEnumValueService().ensurePersisted(generic_value); + + getCustomizableEnumFacade().loadData(); + + bf_1_2 = getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, bf_1_2_value.getValue()); + generic = getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, null, generic_value.getValue()); + + Mockito + .when( + MockProducer.getCustomizableEnumFacadeForConverter() + .getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, bf_1_2_value.getValue())) + .thenReturn(bf_1_2); + Mockito + .when( + MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, null, generic_value.getValue())) + .thenReturn(generic); + } + } + + @Test + public void testConvertCovidDiseaseVariant() { + CaseDataDto caze = creator.createCase(reportingUser, creator.createPerson().toReference(), rdcf, c -> { + c.setDisease(Disease.CORONAVIRUS); + c.setDiseaseVariant(new DiseaseVariantConverter().convertToEntityAttribute(Disease.CORONAVIRUS, bf_1_2.getValue())); + }); + + assertThat(caze.getDiseaseVariant().getValue(), is(bf_1_2.getValue())); + + CaseDataDto reloadedCase = getCaseFacade().getByUuid(caze.getUuid()); + assertThat(reloadedCase.getDiseaseVariant().getValue(), is(bf_1_2.getValue())); + } + + @Test + public void testConvertGenericDiseaseVariant() { + CaseDataDto caze = creator.createCase(reportingUser, creator.createPerson().toReference(), rdcf, c -> { + c.setDisease(Disease.CORONAVIRUS); + c.setDiseaseVariant(new DiseaseVariantConverter().convertToEntityAttribute(Disease.CORONAVIRUS, generic.getValue())); + }); + + assertThat(caze.getDiseaseVariant().getValue(), is(generic.getValue())); + + CaseDataDto reloadedCase = getCaseFacade().getByUuid(caze.getUuid()); + assertThat(reloadedCase.getDiseaseVariant().getValue(), is(generic.getValue())); + } +} diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjbTest.java index c993b2d4fe2..e378f6c844a 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/environment/environmentsample/EnvironmentSampleFacadeEjbTest.java @@ -449,7 +449,8 @@ public void testGetIndexList() { reportingUser.toReference(), PathogenTestResultType.POSITIVE, null); - Mockito.when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.PATHOGEN, positivePathogen.getValue())) + Mockito + .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.PATHOGEN, null, positivePathogen.getValue())) .thenReturn(positivePathogen); Pathogen pendingPathogen = creator.createPathogen("TEST_PATHOGEN_2", "Test pathogen 2"); @@ -461,7 +462,8 @@ public void testGetIndexList() { reportingUser.toReference(), PathogenTestResultType.PENDING, null); - Mockito.when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.PATHOGEN, pendingPathogen.getValue())) + Mockito + .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.PATHOGEN, null, pendingPathogen.getValue())) .thenReturn(pendingPathogen); creator.createEnvironmentSample(environment.toReference(), reportingUser.toReference(), rdcf, lab.toReference(), null); diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjbTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjbTest.java index 3529521a1da..b4847774593 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjbTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/externalmessage/ExternalMessageFacadeEjbTest.java @@ -306,10 +306,14 @@ public void testDiseaseVariantDeterminationOnSave() { diseaseVariant2.setValue(diseaseVariantEnumValue2.getValue()); Mockito - .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, diseaseVariant.getValue())) + .when( + MockProducer.getCustomizableEnumFacadeForConverter() + .getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, diseaseVariant.getValue())) .thenReturn(diseaseVariant); Mockito - .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, diseaseVariant2.getValue())) + .when( + MockProducer.getCustomizableEnumFacadeForConverter() + .getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, diseaseVariant2.getValue())) .thenReturn(diseaseVariant2); ExternalMessageDto labMessage = diff --git a/sormas-backend/src/test/java/de/symeda/sormas/backend/selfreport/AbstractSelfReportProcessingFlowTest.java b/sormas-backend/src/test/java/de/symeda/sormas/backend/selfreport/AbstractSelfReportProcessingFlowTest.java index 208a652ef8d..a73999e8f10 100644 --- a/sormas-backend/src/test/java/de/symeda/sormas/backend/selfreport/AbstractSelfReportProcessingFlowTest.java +++ b/sormas-backend/src/test/java/de/symeda/sormas/backend/selfreport/AbstractSelfReportProcessingFlowTest.java @@ -582,7 +582,9 @@ private SelfReportDto createSelfReport(SelfReportType type) { DiseaseVariant diseaseVariant = creator.createDiseaseVariant("BF.1.2", Disease.CORONAVIRUS); Mockito - .when(MockProducer.getCustomizableEnumFacadeForConverter().getEnumValue(CustomizableEnumType.DISEASE_VARIANT, diseaseVariant.getValue())) + .when( + MockProducer.getCustomizableEnumFacadeForConverter() + .getEnumValue(CustomizableEnumType.DISEASE_VARIANT, Disease.CORONAVIRUS, diseaseVariant.getValue())) .thenReturn(diseaseVariant); selfReport.setDiseaseVariant(diseaseVariant); selfReport.setDiseaseVariantDetails("Variant Details"); diff --git a/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DataImporter.java b/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DataImporter.java index dbd5ac555d3..9a1da11079d 100644 --- a/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DataImporter.java +++ b/sormas-ui/src/main/java/de/symeda/sormas/ui/importer/DataImporter.java @@ -448,7 +448,7 @@ protected boolean executeDefaultInvoke(PropertyDescriptor pd, Object element, St if (propertyType.isAssignableFrom(OccupationType.class)) { OccupationType occupationType = null; try { - occupationType = FacadeProvider.getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.OCCUPATION_TYPE, entry); + occupationType = FacadeProvider.getCustomizableEnumFacade().getEnumValue(CustomizableEnumType.OCCUPATION_TYPE, null, entry); } catch (EJBException e) { //ignore, occupationType will remain null } From 57c2c160cc8b5dbab994e1bab3b473ec805c4698 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Thu, 19 Dec 2024 12:35:18 +0000 Subject: [PATCH 52/56] [GITFLOW]updating poms for 1.100.0 branch with snapshot versions From c52644dfc3f68f5ea64571a1df73f35545c33f85 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Thu, 19 Dec 2024 12:35:19 +0000 Subject: [PATCH 53/56] [GITFLOW]updating poms for 1.101.0-SNAPSHOT development --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 55e897ce46c..8c30a302954 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index 518447fdc15..523596b5398 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index f8b626e57bc..b79abebdcb8 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index 1abac7db51f..54b0dfea876 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 3d0a47e1f67..ddd91e81e6f 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index 9a7dbfc5f2d..b4b5ab67ecb 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 58dfb145183..dc275ddc073 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index ae6d8243402..a1026dd3168 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index 2205ebaec18..e958c99ad63 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index a76762d90f0..8f035e19788 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 2fb2a7852c6..64262b9eaa8 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 From 43f2b9a59a79ae541954bd77d271e12714666ebc Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Thu, 19 Dec 2024 12:35:20 +0000 Subject: [PATCH 54/56] [GITFLOW]updating poms for branch'release-1.100.0' with non-snapshot versions --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 55e897ce46c..8992292c9af 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index 518447fdc15..b98ed71c997 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index f8b626e57bc..6ea3f2dbd9a 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index 1abac7db51f..e9e3188ea96 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.100.0-SNAPSHOT + 1.100.0 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 3d0a47e1f67..1767451dac2 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index 9a7dbfc5f2d..ece288abcd6 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 58dfb145183..87ed14a628d 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index ae6d8243402..fd31944b50c 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index 2205ebaec18..e9dc64d836b 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index a76762d90f0..2fdf5d8faea 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 2fb2a7852c6..44916ffcac1 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 From 6e41305c2bac06b89d89051ce8bf56b3a74b1000 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Thu, 19 Dec 2024 12:42:10 +0000 Subject: [PATCH 55/56] [GITFLOW]updating develop poms to master versions to avoid merge conflicts --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 8c30a302954..8992292c9af 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index 523596b5398..b98ed71c997 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index b79abebdcb8..6ea3f2dbd9a 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index 54b0dfea876..e9e3188ea96 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.101.0-SNAPSHOT + 1.100.0 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index ddd91e81e6f..1767451dac2 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index b4b5ab67ecb..ece288abcd6 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index dc275ddc073..87ed14a628d 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index a1026dd3168..fd31944b50c 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index e958c99ad63..e9dc64d836b 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index 8f035e19788..2fdf5d8faea 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 64262b9eaa8..44916ffcac1 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.101.0-SNAPSHOT + 1.100.0 ../sormas-base 4.0.0 From 40d0f733e3c4bfd20b2608f745476efe2046eef2 Mon Sep 17 00:00:00 2001 From: SORMAS-Robot <175835661+SORMAS-Robot@users.noreply.github.com> Date: Thu, 19 Dec 2024 12:42:12 +0000 Subject: [PATCH 56/56] [GITFLOW]Updating develop poms back to pre merge state --- sormas-api/pom.xml | 2 +- sormas-app/pom.xml | 2 +- sormas-backend/pom.xml | 2 +- sormas-base/pom.xml | 2 +- sormas-cargoserver/pom.xml | 2 +- sormas-ear/pom.xml | 2 +- sormas-keycloak-service-provider/pom.xml | 2 +- sormas-rest/pom.xml | 2 +- sormas-serverlibs/pom.xml | 2 +- sormas-ui/pom.xml | 2 +- sormas-widgetset/pom.xml | 2 +- 11 files changed, 11 insertions(+), 11 deletions(-) diff --git a/sormas-api/pom.xml b/sormas-api/pom.xml index 8992292c9af..8c30a302954 100644 --- a/sormas-api/pom.xml +++ b/sormas-api/pom.xml @@ -2,7 +2,7 @@ de.symeda.sormas sormas-base - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-app/pom.xml b/sormas-app/pom.xml index b98ed71c997..523596b5398 100644 --- a/sormas-app/pom.xml +++ b/sormas-app/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-backend/pom.xml b/sormas-backend/pom.xml index 6ea3f2dbd9a..b79abebdcb8 100644 --- a/sormas-backend/pom.xml +++ b/sormas-backend/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-base/pom.xml b/sormas-base/pom.xml index e9e3188ea96..54b0dfea876 100644 --- a/sormas-base/pom.xml +++ b/sormas-base/pom.xml @@ -5,7 +5,7 @@ de.symeda.sormas sormas-base pom - 1.100.0 + 1.101.0-SNAPSHOT 3.6.3 diff --git a/sormas-cargoserver/pom.xml b/sormas-cargoserver/pom.xml index 1767451dac2..ddd91e81e6f 100644 --- a/sormas-cargoserver/pom.xml +++ b/sormas-cargoserver/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base diff --git a/sormas-ear/pom.xml b/sormas-ear/pom.xml index ece288abcd6..b4b5ab67ecb 100644 --- a/sormas-ear/pom.xml +++ b/sormas-ear/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base diff --git a/sormas-keycloak-service-provider/pom.xml b/sormas-keycloak-service-provider/pom.xml index 87ed14a628d..dc275ddc073 100644 --- a/sormas-keycloak-service-provider/pom.xml +++ b/sormas-keycloak-service-provider/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-rest/pom.xml b/sormas-rest/pom.xml index fd31944b50c..a1026dd3168 100644 --- a/sormas-rest/pom.xml +++ b/sormas-rest/pom.xml @@ -3,7 +3,7 @@ de.symeda.sormas sormas-base - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base diff --git a/sormas-serverlibs/pom.xml b/sormas-serverlibs/pom.xml index e9dc64d836b..e958c99ad63 100644 --- a/sormas-serverlibs/pom.xml +++ b/sormas-serverlibs/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-ui/pom.xml b/sormas-ui/pom.xml index 2fdf5d8faea..8f035e19788 100644 --- a/sormas-ui/pom.xml +++ b/sormas-ui/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0 diff --git a/sormas-widgetset/pom.xml b/sormas-widgetset/pom.xml index 44916ffcac1..64262b9eaa8 100644 --- a/sormas-widgetset/pom.xml +++ b/sormas-widgetset/pom.xml @@ -3,7 +3,7 @@ sormas-base de.symeda.sormas - 1.100.0 + 1.101.0-SNAPSHOT ../sormas-base 4.0.0