Skip to content

Commit ac6c681

Browse files
[FEATURE] Permettre de naviguer dans le module en cliquant sur les étapes de la sidebar (PIX-15397)
#10691
2 parents 964c67b + 16c41a4 commit ac6c681

File tree

7 files changed

+178
-43
lines changed

7 files changed

+178
-43
lines changed

mon-pix/app/components/module/_navbar.scss

+21-18
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
top: 0;
44
z-index: var(--modulix-z-index-above-all);
55
height: var(--module-navbar-height);
6-
padding: var(--pix-spacing-4x);
6+
padding: var(--pix-spacing-3x);
77
background: var(--pix-neutral-0);
88
box-shadow: 0 4px 20px 0 rgb(0 0 0 / 6%);
99

@@ -16,26 +16,29 @@
1616
}
1717
}
1818

19-
.module-sidebar-list-item {
20-
&__link {
21-
@extend %pix-body-l;
19+
.module-sidebar__list-item {
20+
@extend %pix-body-l;
2221

23-
display: inline-block;
24-
width: 100%;
25-
padding: var(--pix-spacing-3x) var(--pix-spacing-8x);
26-
color: var(--pix-neutral-800);
22+
display: inline-block;
23+
width: 100%;
24+
padding: var(--pix-spacing-3x) var(--pix-spacing-8x);
25+
color: var(--pix-neutral-800);
26+
cursor: pointer;
2727

28-
&.current-grain {
29-
--border-width: 5px;
28+
&.current-grain {
29+
--border-width: 5px;
3030

31-
padding-left: calc(var(--pix-spacing-8x) - var(--border-width));
32-
color: var(--pix-primary-500);
33-
border-left: var(--border-width) solid var(--pix-primary-500);
34-
}
31+
padding-left: calc(var(--pix-spacing-8x) - var(--border-width));
32+
color: var(--pix-primary-500);
33+
border-left: var(--border-width) solid var(--pix-primary-500);
34+
}
35+
36+
&:hover {
37+
color: var(--pix-primary-700);
38+
background-color: var(--pix-neutral-20);
39+
}
3540

36-
&:hover {
37-
color: var(--pix-primary-700);
38-
background-color: var(--pix-neutral-20);
39-
}
41+
&:active {
42+
background-color: var(--pix-neutral-100);
4043
}
4144
}

mon-pix/app/components/module/grain.gjs

+10-1
Original file line numberDiff line numberDiff line change
@@ -162,8 +162,17 @@ export default class ModuleGrain extends Component {
162162
await this.args.onModuleTerminate({ grainId: this.args.grain.id });
163163
}
164164

165+
get elementId() {
166+
return `grain_${this.args.grain.id}`;
167+
}
168+
165169
<template>
166-
<article class="grain {{if @hasJustAppeared 'grain--active'}}" tabindex="-1" {{didInsert this.focusAndScroll}}>
170+
<article
171+
id={{this.elementId}}
172+
class="grain {{if @hasJustAppeared 'grain--active'}}"
173+
tabindex="-1"
174+
{{didInsert this.focusAndScroll}}
175+
>
167176
<h2 class="screen-reader-only">{{@grain.title}}</h2>
168177

169178
{{#if @transition}}

mon-pix/app/components/module/navbar.gjs

+19-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
import PixButton from '@1024pix/pix-ui/components/pix-button';
22
import PixProgressGauge from '@1024pix/pix-ui/components/pix-progress-gauge';
33
import PixSidebar from '@1024pix/pix-ui/components/pix-sidebar';
4+
import { fn } from '@ember/helper';
5+
import { on } from '@ember/modifier';
46
import { action } from '@ember/object';
57
import { service } from '@ember/service';
68
import Component from '@glimmer/component';
@@ -31,12 +33,21 @@ export default class ModulixNavbar extends Component {
3133
this.sidebarOpened = false;
3234
}
3335

34-
get grainTypeTexts() {
35-
return this.args.grainsToDisplay.map((grain) => this.intl.t(`pages.modulix.grain.tag.${grain.type}`));
36+
get grainsWithIdAndTranslatedType() {
37+
return this.args.grainsToDisplay.map((grain) => ({
38+
type: this.intl.t(`pages.modulix.grain.tag.${grain.type}`),
39+
id: grain.id,
40+
}));
3641
}
3742

3843
get currentGrainIndex() {
39-
return this.grainTypeTexts.length - 1;
44+
return this.grainsWithIdAndTranslatedType.length - 1;
45+
}
46+
47+
@action
48+
onMenuItemClick(grainId) {
49+
this.closeSidebar();
50+
this.args.goToGrain(grainId);
4051
}
4152

4253
<template>
@@ -61,12 +72,14 @@ export default class ModulixNavbar extends Component {
6172
<:content>
6273
<nav>
6374
<ul>
64-
{{#each this.grainTypeTexts as |type index|}}
75+
{{#each this.grainsWithIdAndTranslatedType as |grain index|}}
6576
<li
66-
class="module-sidebar-list-item__link {{if (eq index this.currentGrainIndex) 'current-grain'}}"
77+
class="module-sidebar__list-item {{if (eq index this.currentGrainIndex) 'current-grain'}}"
6778
aria-current={{if (eq index this.currentGrainIndex) "step"}}
79+
{{on "click" (fn this.onMenuItemClick grain.id)}}
80+
role="link"
6881
>
69-
{{type}}
82+
{{grain.type}}
7083
</li>
7184
{{/each}}
7285
</ul>

mon-pix/app/components/module/passage.gjs

+7
Original file line numberDiff line numberDiff line change
@@ -196,13 +196,20 @@ export default class ModulePassage extends Component {
196196
});
197197
}
198198

199+
@action
200+
async goToGrain(grainId) {
201+
const element = document.getElementById(`grain_${grainId}`);
202+
this.modulixAutoScroll.focusAndScroll(element);
203+
}
204+
199205
<template>
200206
{{pageTitle @module.title}}
201207
<ModuleNavbar
202208
@currentStep={{this.currentPassageStep}}
203209
@totalSteps={{this.displayableGrains.length}}
204210
@module={{@module}}
205211
@grainsToDisplay={{this.grainsToDisplay}}
212+
@goToGrain={{this.goToGrain}}
206213
/>
207214

208215
<main class="module-passage">

mon-pix/tests/integration/components/module/grain_test.gjs

+2-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ module('Integration | Component | Module | Grain', function (hooks) {
1515
test('should display given grain', async function (assert) {
1616
// given
1717
const store = this.owner.lookup('service:store');
18-
const grain = store.createRecord('grain', { title: 'Grain title' });
18+
const grain = store.createRecord('grain', { id: '12345-abcdef', title: 'Grain title' });
1919
this.set('grain', grain);
2020

2121
// when
@@ -24,6 +24,7 @@ module('Integration | Component | Module | Grain', function (hooks) {
2424

2525
// then
2626
assert.ok(screen.getByRole('heading', { name: grain.title, level: 2 }));
27+
assert.dom('.grain').hasAttribute('id', 'grain_12345-abcdef');
2728
});
2829

2930
module('when grain has transition', function () {

mon-pix/tests/integration/components/module/navbar_test.gjs

+79-16
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,12 @@
1-
import { clickByName, render, within } from '@1024pix/ember-testing-library';
1+
import { clickByName, clickByText, render } from '@1024pix/ember-testing-library';
22
import { click } from '@ember/test-helpers';
33
import { t } from 'ember-intl/test-support';
44
import ModulixNavbar from 'mon-pix/components/module/navbar';
55
import { module, test } from 'qunit';
6+
import sinon from 'sinon';
67

78
import setupIntlRenderingTest from '../../../helpers/setup-intl-rendering';
8-
import { waitForDialog } from '../../../helpers/wait-for';
9+
import { waitForDialog, waitForDialogClose } from '../../../helpers/wait-for';
910

1011
module('Integration | Component | Module | Navbar', function (hooks) {
1112
setupIntlRenderingTest(hooks);
@@ -126,26 +127,88 @@ module('Integration | Component | Module | Navbar', function (hooks) {
126127
await waitForDialog();
127128

128129
// then
129-
assert.ok(screen);
130-
const list = screen.getByRole('list');
131-
assert.dom(list).exists();
132-
const items = within(list).getAllByRole('listitem');
133-
assert.strictEqual(items.length, 3);
134-
assert.strictEqual(items[0].textContent.trim(), t('pages.modulix.grain.tag.discovery'));
135-
assert.strictEqual(items[1].textContent.trim(), t('pages.modulix.grain.tag.activity'));
136-
assert.strictEqual(items[2].textContent.trim(), t('pages.modulix.grain.tag.lesson'));
137-
assert.dom(items[2]).hasAria('current', 'step');
138-
assert.dom(screen.queryByRole('listitem', { name: t('pages.modulix.grain.tag.summary') })).doesNotExist();
130+
assert.strictEqual(
131+
screen.getByRole('link', { name: 'Découverte' }).textContent.trim(),
132+
t('pages.modulix.grain.tag.discovery'),
133+
);
134+
assert.strictEqual(
135+
screen.getByRole('link', { name: 'Activité' }).textContent.trim(),
136+
t('pages.modulix.grain.tag.activity'),
137+
);
138+
assert.strictEqual(
139+
screen.getByRole('link', { name: 'Leçon' }).textContent.trim(),
140+
t('pages.modulix.grain.tag.lesson'),
141+
);
142+
assert.dom(screen.getByRole('link', { name: 'Leçon' })).hasAria('current', 'step');
143+
144+
assert.dom(screen.queryByRole('link', { name: "Récap'" })).doesNotExist();
145+
});
146+
147+
module('when user clicks on grain’s type', function () {
148+
test('should call goToGrain action on matching grain element', async function (assert) {
149+
// given
150+
const module = createModule(this.owner);
151+
const threeFirstGrains = module.grains.slice(0, -1);
152+
const goToGrainSpy = sinon.spy();
153+
154+
// when
155+
await render(
156+
<template>
157+
<ModulixNavbar
158+
@currentStep={{3}}
159+
@totalSteps={{4}}
160+
@module={{module}}
161+
@grainsToDisplay={{threeFirstGrains}}
162+
@goToGrain={{goToGrainSpy}}
163+
/>
164+
</template>,
165+
);
166+
await clickByName('Afficher les étapes du module');
167+
await waitForDialog();
168+
await clickByText('Activité');
169+
170+
// then
171+
sinon.assert.calledOnce(goToGrainSpy);
172+
sinon.assert.calledWithExactly(goToGrainSpy, '234-abc');
173+
assert.ok(true);
174+
});
175+
176+
test('should close sidebar', async function (assert) {
177+
// given
178+
const module = createModule(this.owner);
179+
const threeFirstGrains = module.grains.slice(0, -1);
180+
const goToGrainMock = sinon.mock();
181+
182+
// when
183+
const screen = await render(
184+
<template>
185+
<ModulixNavbar
186+
@currentStep={{3}}
187+
@totalSteps={{4}}
188+
@module={{module}}
189+
@grainsToDisplay={{threeFirstGrains}}
190+
@goToGrain={{goToGrainMock}}
191+
/>
192+
</template>,
193+
);
194+
await clickByName('Afficher les étapes du module');
195+
await waitForDialog();
196+
await clickByText('Activité');
197+
await waitForDialogClose();
198+
199+
// then
200+
assert.dom(screen.queryByRole('dialog', { name: module.title })).doesNotExist();
201+
});
139202
});
140203
});
141204
});
142205

143206
function createModule(owner) {
144207
const store = owner.lookup('service:store');
145-
const grain1 = store.createRecord('grain', { title: 'Grain title', type: 'discovery' });
146-
const grain2 = store.createRecord('grain', { title: 'Grain title', type: 'activity' });
147-
const grain3 = store.createRecord('grain', { title: 'Grain title', type: 'lesson' });
148-
const grain4 = store.createRecord('grain', { title: 'Grain title', type: 'summary' });
208+
const grain1 = store.createRecord('grain', { title: 'Grain title', type: 'discovery', id: '123-abc' });
209+
const grain2 = store.createRecord('grain', { title: 'Grain title', type: 'activity', id: '234-abc' });
210+
const grain3 = store.createRecord('grain', { title: 'Grain title', type: 'lesson', id: '345-abc' });
211+
const grain4 = store.createRecord('grain', { title: 'Grain title', type: 'summary', id: '456-abc' });
149212
return store.createRecord('module', {
150213
title: 'Didacticiel',
151214
grains: [grain1, grain2, grain3, grain4],

mon-pix/tests/integration/components/module/passage_test.gjs

+40-1
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,14 @@
11
import { clickByName, render } from '@1024pix/ember-testing-library';
22
// eslint-disable-next-line no-restricted-imports
3-
import { find, findAll } from '@ember/test-helpers';
3+
import { click, find, findAll } from '@ember/test-helpers';
44
import { t } from 'ember-intl/test-support';
55
import ApplicationAdapter from 'mon-pix/adapters/application';
66
import ModulePassage from 'mon-pix/components/module/passage';
77
import { module, test } from 'qunit';
88
import sinon from 'sinon';
99

1010
import setupIntlRenderingTest from '../../../helpers/setup-intl-rendering';
11+
import { waitForDialog } from '../../../helpers/wait-for';
1112

1213
module('Integration | Component | Module | Passage', function (hooks) {
1314
setupIntlRenderingTest(hooks);
@@ -1048,4 +1049,42 @@ module('Integration | Component | Module | Passage', function (hooks) {
10481049
assert.ok(true);
10491050
});
10501051
});
1052+
1053+
module('when user clicks on grain’s type in sidebar', function () {
1054+
test('should focus and scroll on matching grain element', async function (assert) {
1055+
// given
1056+
const store = this.owner.lookup('service:store');
1057+
const element = { type: 'text', isAnswerable: false, content: 'Ceci est un grain dans un test d‘intégration' };
1058+
const grain1 = store.createRecord('grain', {
1059+
title: 'Grain title',
1060+
type: 'discovery',
1061+
id: '123-abc',
1062+
components: [{ type: 'element', element }],
1063+
});
1064+
const grain2 = store.createRecord('grain', {
1065+
title: 'Grain title',
1066+
type: 'activity',
1067+
id: '234-abc',
1068+
components: [{ type: 'element', element }],
1069+
});
1070+
const module = store.createRecord('module', {
1071+
title: 'Didacticiel',
1072+
grains: [grain1, grain2],
1073+
transitionTexts: [],
1074+
});
1075+
const passage = store.createRecord('passage');
1076+
const modulixAutoScroll = this.owner.lookup('service:modulix-auto-scroll');
1077+
modulixAutoScroll.focusAndScroll = sinon.mock();
1078+
1079+
// when
1080+
const screen = await render(<template><ModulePassage @module={{module}} @passage={{passage}} /></template>);
1081+
await clickByName('Afficher les étapes du module');
1082+
await waitForDialog();
1083+
const item = screen.getByRole('link', { name: 'Découverte' });
1084+
await click(item);
1085+
1086+
// then
1087+
assert.ok(modulixAutoScroll.focusAndScroll.calledOnce);
1088+
});
1089+
});
10511090
});

0 commit comments

Comments
 (0)