From a5192ada3fbb61789cae66dc9a1b18050a24c0ce Mon Sep 17 00:00:00 2001 From: Alexandre Monney Date: Thu, 7 Nov 2024 17:25:30 +0100 Subject: [PATCH] feat(orga): create attestations page --- .../components/attestations/sixth-grade.gjs | 51 ++++++++++ .../controllers/authenticated/attestations.js | 31 +++++++ orga/app/routes/authenticated/attestations.js | 19 ++++ orga/app/styles/app.scss | 1 + .../pages/authenticated/attestations.scss | 24 +++++ .../templates/authenticated/attestations.hbs | 5 + .../attestations/sixth-grade-test.gjs | 70 ++++++++++++++ .../authenticated/attestations-test.js | 93 +++++++++++++++++++ .../routes/authenticated/attestations-test.js | 89 ++++++++++++++++++ orga/translations/en.json | 8 ++ orga/translations/fr.json | 8 ++ orga/translations/nl.json | 8 ++ 12 files changed, 407 insertions(+) create mode 100644 orga/app/components/attestations/sixth-grade.gjs create mode 100644 orga/app/controllers/authenticated/attestations.js create mode 100644 orga/app/routes/authenticated/attestations.js create mode 100644 orga/app/styles/pages/authenticated/attestations.scss create mode 100644 orga/app/templates/authenticated/attestations.hbs create mode 100644 orga/tests/integration/components/attestations/sixth-grade-test.gjs create mode 100644 orga/tests/unit/controllers/authenticated/attestations-test.js create mode 100644 orga/tests/unit/routes/authenticated/attestations-test.js diff --git a/orga/app/components/attestations/sixth-grade.gjs b/orga/app/components/attestations/sixth-grade.gjs new file mode 100644 index 00000000000..88a2e7b5841 --- /dev/null +++ b/orga/app/components/attestations/sixth-grade.gjs @@ -0,0 +1,51 @@ +import PixButton from '@1024pix/pix-ui/components/pix-button'; +import PixMultiSelect from '@1024pix/pix-ui/components/pix-multi-select'; +import { on } from '@ember/modifier'; +import { action } from '@ember/object'; +import Component from '@glimmer/component'; +import { tracked } from '@glimmer/tracking'; +import { t } from 'ember-intl'; + +export default class AttestationsSixthGrade extends Component { + @tracked selectedDivisions = []; + + @action + onSubmit(event) { + event.preventDefault(); + + this.args.onSubmit(this.selectedDivisions); + } + + @action + onSelectDivision(value) { + this.selectedDivisions = value; + } + + get isDisabled() { + return !this.selectedDivisions.length; + } + + +} diff --git a/orga/app/controllers/authenticated/attestations.js b/orga/app/controllers/authenticated/attestations.js new file mode 100644 index 00000000000..91d711c6d09 --- /dev/null +++ b/orga/app/controllers/authenticated/attestations.js @@ -0,0 +1,31 @@ +import Controller from '@ember/controller'; +import { action } from '@ember/object'; +import { service } from '@ember/service'; + +export const SIXTH_GRADE_ATTESTATION_KEY = 'SIXTH_GRADE'; +export const SIXTH_GRADE_ATTESTATION_FILE_NAME = 'attestations'; + +export default class AuthenticatedAttestationsController extends Controller { + @service fileSaver; + @service session; + @service currentUser; + @service notifications; + + @action + async downloadSixthGradeAttestationsFile(selectedDivisions) { + try { + const organizationId = this.currentUser.organization.id; + const formatedDivisionsForQuery = selectedDivisions + .map((division) => `divisions[]=${encodeURIComponent(division)}`) + .join('&'); + + const url = `/api/organizations/${organizationId}/attestations/${SIXTH_GRADE_ATTESTATION_KEY}?${formatedDivisionsForQuery}`; + + const token = this.session.isAuthenticated ? this.session.data.authenticated.access_token : ''; + + await this.fileSaver.save({ url, token, fileName: SIXTH_GRADE_ATTESTATION_FILE_NAME }); + } catch (error) { + this.notifications.sendError(error.message, { autoClear: false }); + } + } +} diff --git a/orga/app/routes/authenticated/attestations.js b/orga/app/routes/authenticated/attestations.js new file mode 100644 index 00000000000..671056adbbb --- /dev/null +++ b/orga/app/routes/authenticated/attestations.js @@ -0,0 +1,19 @@ +import Route from '@ember/routing/route'; +import { service } from '@ember/service'; + +export default class AuthenticatedAttestationsRoute extends Route { + @service currentUser; + @service router; + + beforeModel() { + if (!this.currentUser.canAccessAttestationsPage) { + this.router.replaceWith('application'); + } + } + + async model() { + const divisions = await this.currentUser.organization.divisions; + const options = divisions.map(({ name }) => ({ label: name, value: name })); + return { options }; + } +} diff --git a/orga/app/styles/app.scss b/orga/app/styles/app.scss index 513f60332e0..dbc39f979e6 100644 --- a/orga/app/styles/app.scss +++ b/orga/app/styles/app.scss @@ -58,6 +58,7 @@ @import 'pages/join'; @import 'pages/join-request'; @import 'pages/authenticated'; +@import 'pages/authenticated/attestations'; @import 'pages/authenticated/campaigns/campaign'; @import 'pages/authenticated/campaigns/create-form'; @import 'pages/authenticated/campaigns/new'; diff --git a/orga/app/styles/pages/authenticated/attestations.scss b/orga/app/styles/pages/authenticated/attestations.scss new file mode 100644 index 00000000000..4114f4a6596 --- /dev/null +++ b/orga/app/styles/pages/authenticated/attestations.scss @@ -0,0 +1,24 @@ +.attestations-page { + display: flex; + flex-direction: column; + + &__title { + @extend %pix-title-m; + + margin: var(--pix-spacing-8x) 0; + } + + &__text { + @extend %pix-body-m; + + margin-bottom: var(--pix-spacing-8x); + + } + + &__action { + display: flex; + flex-wrap: wrap; + gap: var(--pix-spacing-4x); + align-items: end + } +} diff --git a/orga/app/templates/authenticated/attestations.hbs b/orga/app/templates/authenticated/attestations.hbs new file mode 100644 index 00000000000..6f78b8cd424 --- /dev/null +++ b/orga/app/templates/authenticated/attestations.hbs @@ -0,0 +1,5 @@ +{{page-title (t "pages.attestations.title")}} + +
+ +
\ No newline at end of file diff --git a/orga/tests/integration/components/attestations/sixth-grade-test.gjs b/orga/tests/integration/components/attestations/sixth-grade-test.gjs new file mode 100644 index 00000000000..ff076bdeecb --- /dev/null +++ b/orga/tests/integration/components/attestations/sixth-grade-test.gjs @@ -0,0 +1,70 @@ +import { render } from '@1024pix/ember-testing-library'; +import { click } from '@ember/test-helpers'; +import { t } from 'ember-intl/test-support'; +import SixthGrade from 'pix-orga/components/attestations/sixth-grade'; +import { module, test } from 'qunit'; +import sinon from 'sinon'; + +import setupIntlRenderingTest from '../../../helpers/setup-intl-rendering'; + +module('Integration | Component | Attestations | Sixth-grade', function (hooks) { + setupIntlRenderingTest(hooks); + + test('it should display all basics informations', async function (assert) { + // given + const onSubmit = sinon.stub(); + const divisions = []; + + // when + const screen = await render(); + // then + assert.ok(screen.getByRole('heading', { name: t('pages.attestations.title') })); + assert.ok(screen.getByText(t('pages.attestations.description'))); + assert.ok(screen.getByRole('textbox', { name: t('pages.attestations.select-label') })); + assert.ok(screen.getByPlaceholderText(t('pages.attestations.placeholder'))); + assert.ok(screen.getByRole('button', { name: t('pages.attestations.download-attestations-button') })); + }); + + test('download button is disabled if there is no selected divisions', async function (assert) { + // given + const onSubmit = sinon.stub(); + const divisions = []; + + // when + const screen = await render(); + + // then + const downloadButton = await screen.getByRole('button', { + name: t('pages.attestations.download-attestations-button'), + }); + assert.dom(downloadButton).isDisabled(); + }); + + test('it should call onSubmit action with selected divisions', async function (assert) { + // given + const onSubmit = sinon.stub(); + + const divisions = [{ label: 'division1', value: 'division1' }]; + + // when + const screen = await render(); + + const multiSelect = await screen.getByRole('textbox', { name: t('pages.attestations.select-label') }); + await click(multiSelect); + + const firstDivisionOption = await screen.findByRole('checkbox', { name: 'division1' }); + await click(firstDivisionOption); + + const downloadButton = await screen.getByRole('button', { + name: t('pages.attestations.download-attestations-button'), + }); + + // we need to get out of input choice to click on download button, so we have to click again on the multiselect to close it + await click(multiSelect); + await click(downloadButton); + + // then + sinon.assert.calledWithExactly(onSubmit, ['division1']); + assert.ok(true); + }); +}); diff --git a/orga/tests/unit/controllers/authenticated/attestations-test.js b/orga/tests/unit/controllers/authenticated/attestations-test.js new file mode 100644 index 00000000000..68c6bb87bce --- /dev/null +++ b/orga/tests/unit/controllers/authenticated/attestations-test.js @@ -0,0 +1,93 @@ +import Service from '@ember/service'; +import { + SIXTH_GRADE_ATTESTATION_FILE_NAME, + SIXTH_GRADE_ATTESTATION_KEY, +} from 'pix-orga/controllers/authenticated/attestations'; +import { module, test } from 'qunit'; +import sinon from 'sinon'; + +import setupIntlRenderingTest from '../../../helpers/setup-intl-rendering'; + +module('Unit | Controller | authenticated/attestations', function (hooks) { + setupIntlRenderingTest(hooks); + + module('#downloadSixthGradeAttestationsFile', function () { + test('should call the file-saver service with the right parameters', async function (assert) { + // given + const controller = this.owner.lookup('controller:authenticated/attestations'); + + const token = 'a token'; + const organizationId = 12345; + const selectedDivision = ['3èmea']; + + controller.session = { + isAuthenticated: true, + data: { + authenticated: { + access_token: token, + }, + }, + }; + + controller.currentUser = { + organization: { + id: organizationId, + }, + }; + + controller.fileSaver = { + save: sinon.stub(), + }; + + controller.model = { + options: [{ label: '3èmeA', value: '3èmeA' }], + }; + + // when + await controller.downloadSixthGradeAttestationsFile(selectedDivision); + + // then + assert.ok( + controller.fileSaver.save.calledWith({ + token, + url: `/api/organizations/${organizationId}/attestations/${SIXTH_GRADE_ATTESTATION_KEY}?divisions[]=${encodeURIComponent(selectedDivision)}`, + fileName: SIXTH_GRADE_ATTESTATION_FILE_NAME, + }), + ); + }); + + test('it should not call file-save service and display an error if an error occurs', async function (assert) { + // given + const controller = this.owner.lookup('controller:authenticated/attestations'); + const selectedDivision = ['3emeA']; + const organizationId = 123; + const errorMessage = 'oops'; + + controller.currentUser = { + organization: { + id: organizationId, + }, + }; + controller.fileSaver = { + save: sinon.stub(), + }; + controller.fileSaver.save.rejects(new Error(errorMessage)); + + controller.model = { + options: [{ label: '3èmeA', value: '3èmeA' }], + }; + class NotificationsStub extends Service { + sendError = errorMock; + } + this.owner.register('service:notifications', NotificationsStub); + const errorMock = sinon.stub(); + + // when + await controller.downloadSixthGradeAttestationsFile(selectedDivision); + + // then + sinon.assert.calledWith(errorMock, errorMessage, { autoClear: false }); + assert.ok(true); + }); + }); +}); diff --git a/orga/tests/unit/routes/authenticated/attestations-test.js b/orga/tests/unit/routes/authenticated/attestations-test.js new file mode 100644 index 00000000000..b27baa2dfa8 --- /dev/null +++ b/orga/tests/unit/routes/authenticated/attestations-test.js @@ -0,0 +1,89 @@ +import Service from '@ember/service'; +import { setupTest } from 'ember-qunit'; +import { module, test } from 'qunit'; +import sinon from 'sinon'; + +module('Unit | Route | authenticated/attestations', function (hooks) { + setupTest(hooks); + + module('beforeModel', function () { + test('should redirect to application when currentUser.canAccessAttestationsPage is false', function (assert) { + // given + class CurrentUserStub extends Service { + canAccessAttestationsPage = false; + } + + this.owner.register('service:current-user', CurrentUserStub); + const route = this.owner.lookup('route:authenticated.attestations'); + const replaceWithStub = sinon.stub(); + route.router.replaceWith = replaceWithStub; + + // when + route.beforeModel(); + + // then + sinon.assert.calledOnceWithExactly(replaceWithStub, 'application'); + assert.ok(true); + }); + + test('should not redirect to application when currentUser.canAccessAttestationsPage is true', function (assert) { + // given + class CurrentUserStub extends Service { + canAccessAttestationsPage = true; + } + + this.owner.register('service:current-user', CurrentUserStub); + const route = this.owner.lookup('route:authenticated.attestations'); + const replaceWithStub = sinon.stub(); + route.router.replaceWith = replaceWithStub; + + // when + route.beforeModel(); + + // then + sinon.assert.notCalled(replaceWithStub); + assert.ok(true); + }); + }); + + module('#model', function () { + test('it should return a list of options based on organization divisions', async function (assert) { + // given + const divisions = [{ name: '3èmeA' }, { name: '2ndE' }]; + class CurrentUserStub extends Service { + canAccessAttestationsPage = true; + organization = { + id: 12345, + divisions, + }; + } + + const findRecordStub = sinon.stub(); + class StoreStub extends Service { + findRecord = findRecordStub; + } + + this.owner.register('service:current-user', CurrentUserStub); + this.owner.register('service:store', StoreStub); + + const route = this.owner.lookup('route:authenticated/attestations'); + + // when + const actualOptions = await route.model(); + + // then + assert.deepEqual(actualOptions, { + options: [ + { + label: '3èmeA', + value: '3èmeA', + }, + { + label: '2ndE', + value: '2ndE', + }, + ], + }); + }); + }); +}); diff --git a/orga/translations/en.json b/orga/translations/en.json index b1090443c83..4e52d37c817 100644 --- a/orga/translations/en.json +++ b/orga/translations/en.json @@ -348,6 +348,7 @@ }, "main": { "aria-label": "Main navigation", + "attestations": "Attestations", "campaigns": "Campaigns", "certifications": "Certifications", "documentation": "Documentation", @@ -410,6 +411,13 @@ "row-title": "Competence" } }, + "attestations": { + "title": "Attestations", + "description": "Select the class for which you wish to export attestations in PDF format.", + "download-attestations-button": "Download", + "placeholder": "-- Select --", + "select-label": "Select one or more classes" + }, "campaign": { "actions": { "export-results": "Export the results (.csv)", diff --git a/orga/translations/fr.json b/orga/translations/fr.json index b0adad8c269..bd70c7a3336 100644 --- a/orga/translations/fr.json +++ b/orga/translations/fr.json @@ -354,6 +354,7 @@ }, "main": { "aria-label": "Navigation principale", + "attestations": "Attestations", "campaigns": "Campagnes", "certifications": "Certifications", "documentation": "Documentation", @@ -416,6 +417,13 @@ "row-title": "Compétence" } }, + "attestations": { + "title": "Attestations", + "description": "Sélectionnez la classe pour laquelle vous souhaitez exporter les attestations au format PDF.", + "download-attestations-button": "Télécharger", + "placeholder": "-- Sélectionner --", + "select-label": "Sélectionnez une ou plusieurs classes" + }, "campaign": { "actions": { "export-results": "Exporter les résultats (.csv)", diff --git a/orga/translations/nl.json b/orga/translations/nl.json index 463c364be1b..9465752c7c5 100644 --- a/orga/translations/nl.json +++ b/orga/translations/nl.json @@ -348,6 +348,7 @@ }, "main": { "aria-label": "Hoofdnavigatie", + "attestations": "certificaten", "campaigns": "Campagnes", "certifications": "Certificeringen", "documentation": "Documentatie", @@ -410,6 +411,13 @@ }, "title": "Resultaten voor {firstName} {lastName}" }, + "attestations": { + "title": "Certificaten", + "description": "Selecteer de klas waarvoor je de attesten in PDF-formaat wilt exporteren.", + "download-attestations-button": "Downloaden", + "placeholder": "-- Selecteer --", + "select-label": "Selecteer een of meer klassen" + }, "campaign-activity": { "actions": { "delete": "Verwijderen"