Compare commits
316 Commits
abdullahwa
...
dependabot
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
125b60b969 | ||
|
|
7318fb3ef7 | ||
|
|
7233f08d3d | ||
|
|
d6d229f1c3 | ||
|
|
47b9a436a6 | ||
|
|
e556d5b74c | ||
|
|
694d95a816 | ||
|
|
e83813da8e | ||
|
|
a54a1b8c3c | ||
|
|
d3188efbcc | ||
|
|
33f737579a | ||
|
|
870263001e | ||
|
|
af50d5a6ed | ||
|
|
7fccf7794c | ||
|
|
c760bc479b | ||
|
|
d5140a6bf0 | ||
|
|
9bf5d01c41 | ||
|
|
f3334085d7 | ||
|
|
4840fff44b | ||
|
|
579bd0365b | ||
|
|
2b4a9661a5 | ||
|
|
a6e4e28e58 | ||
|
|
e6f7588ccd | ||
|
|
db29e314c3 | ||
|
|
e9121f9261 | ||
|
|
9cbc2276d6 | ||
|
|
4c8aa7c80c | ||
|
|
68926334a1 | ||
|
|
56a73eee15 | ||
|
|
bf95916063 | ||
|
|
48270c35dd | ||
|
|
33d7d669d9 | ||
|
|
a75c89cd14 | ||
|
|
06902d8ae8 | ||
|
|
e4134641e6 | ||
|
|
77fcc83efd | ||
|
|
b0505352be | ||
|
|
ddbc2124ef | ||
|
|
462e75f6a6 | ||
|
|
bc4c8c2dec | ||
|
|
ecd5164806 | ||
|
|
44d952bef7 | ||
|
|
7eddc918bb | ||
|
|
f28528e813 | ||
|
|
ab3f5fd7bc | ||
|
|
73eaf61261 | ||
|
|
db9663b664 | ||
|
|
7edac93752 | ||
|
|
d1dede568e | ||
|
|
31b02d777f | ||
|
|
67bb54a028 | ||
|
|
847d4e5ce6 | ||
|
|
b89cdb4a69 | ||
|
|
a1d0afff6c | ||
|
|
1714f285b0 | ||
|
|
03cda5326a | ||
|
|
a71152b008 | ||
|
|
d14c2a9ffd | ||
|
|
b6c29df0a0 | ||
|
|
2ce833341b | ||
|
|
ff57a6b217 | ||
|
|
dc6ee749be | ||
|
|
236fb57023 | ||
|
|
d3d2f75c12 | ||
|
|
8e9306d35a | ||
|
|
b1ee8a3713 | ||
|
|
73406fbb31 | ||
|
|
f4ae1c51ff | ||
|
|
7ef3892027 | ||
|
|
1484bc50f7 | ||
|
|
6b197aad27 | ||
|
|
1412bfe209 | ||
|
|
e8d3bd7c24 | ||
|
|
511091055b | ||
|
|
24c9437e91 | ||
|
|
fb6f110732 | ||
|
|
1656b73a31 | ||
|
|
81671ad328 | ||
|
|
4cc716b20c | ||
|
|
756fbbac83 | ||
|
|
903fe28ff6 | ||
|
|
14c662dc53 | ||
|
|
af432eab27 | ||
|
|
dde640df33 | ||
|
|
b827db800d | ||
|
|
5b7f76b43d | ||
|
|
cf4bea3604 | ||
|
|
85e6e9266d | ||
|
|
360af1f0e9 | ||
|
|
26f4a90976 | ||
|
|
0d45c78ace | ||
|
|
c18214dc41 | ||
|
|
54611c1b4d | ||
|
|
7ca4b71ff7 | ||
|
|
63a7ff83cf | ||
|
|
8ecaa018da | ||
|
|
64ca156095 | ||
|
|
c06f2c37ab | ||
|
|
d5a092b220 | ||
|
|
81b621195e | ||
|
|
226c4cc1d7 | ||
|
|
7f6a59b701 | ||
|
|
7ea0bd175b | ||
|
|
dae1d63e23 | ||
|
|
6ab0deb7b7 | ||
|
|
e5f04d92b9 | ||
|
|
f39a50e7dc | ||
|
|
72724bcafb | ||
|
|
964abbe0c3 | ||
|
|
96d20e20e6 | ||
|
|
a56fd7d0e1 | ||
|
|
679caa61f3 | ||
|
|
420060967b | ||
|
|
91d3762513 | ||
|
|
2e3ed087d1 | ||
|
|
d76d4db097 | ||
|
|
04b314d157 | ||
|
|
1db4848d1a | ||
|
|
8f294781d2 | ||
|
|
d6908abb13 | ||
|
|
96d3d0da7e | ||
|
|
14cc32fcf6 | ||
|
|
0ac127e4c9 | ||
|
|
06e5fb5a44 | ||
|
|
2235737490 | ||
|
|
fca32ae872 | ||
|
|
ae04e5b366 | ||
|
|
db3f1b9cb0 | ||
|
|
ec360bc545 | ||
|
|
6c5220b4d7 | ||
|
|
f433118a8d | ||
|
|
e798331855 | ||
|
|
adb5796ff6 | ||
|
|
9958638a86 | ||
|
|
b8e844eba6 | ||
|
|
da633ffbd9 | ||
|
|
46889c2aba | ||
|
|
3cbbb0272b | ||
|
|
911c7658f5 | ||
|
|
b54d1e467e | ||
|
|
e34d18d727 | ||
|
|
6949e5708f | ||
|
|
eef6b1efe2 | ||
|
|
9dc45e192d | ||
|
|
bd9c97c269 | ||
|
|
c70fb138f0 | ||
|
|
8823cfaa0a | ||
|
|
7865fadec2 | ||
|
|
5be1620f1d | ||
|
|
d5a6a59d07 | ||
|
|
826f1382dd | ||
|
|
5e5fdeba44 | ||
|
|
01369eb00d | ||
|
|
4bb4bb7a88 | ||
|
|
1d154f46c1 | ||
|
|
420db8133f | ||
|
|
1ffc93dc6d | ||
|
|
346e15abd4 | ||
|
|
4726c23bc3 | ||
|
|
cbbb417894 | ||
|
|
c57f28ad40 | ||
|
|
310fb84517 | ||
|
|
623f6946e5 | ||
|
|
cf124877e8 | ||
|
|
0456ad9318 | ||
|
|
7be87b0f83 | ||
|
|
cbe5b28762 | ||
|
|
4a80532b8d | ||
|
|
e505f78cfb | ||
|
|
3811f5f9d5 | ||
|
|
8a20b908c7 | ||
|
|
8a6fa937ea | ||
|
|
dafdcad2b4 | ||
|
|
cd56ffaf9d | ||
|
|
c11cb85d78 | ||
|
|
b09bcbd3ae | ||
|
|
4a925f9c11 | ||
|
|
f5b6243c61 | ||
|
|
98c670afe7 | ||
|
|
038b05ba6c | ||
|
|
020e7fb42c | ||
|
|
ead98538b9 | ||
|
|
90ef6ace5c | ||
|
|
e0196f2a2a | ||
|
|
b7befcff7e | ||
|
|
642031bf87 | ||
|
|
f778f27647 | ||
|
|
b3bce8713c | ||
|
|
dacb30c73e | ||
|
|
81a4deeec0 | ||
|
|
9a1b05a1a4 | ||
|
|
b9e1fb0d2b | ||
|
|
ebd0f8816c | ||
|
|
d749429361 | ||
|
|
19b8df35ae | ||
|
|
e468d2087b | ||
|
|
42e0ac86d7 | ||
|
|
ea5cf37fd8 | ||
|
|
e4cdec7389 | ||
|
|
8aafc6b8bd | ||
|
|
913c8e4086 | ||
|
|
c221770213 | ||
|
|
4fe40c264f | ||
|
|
c20c7677a3 | ||
|
|
2ff8c3949e | ||
|
|
4a5c43d365 | ||
|
|
4da37f369b | ||
|
|
0effb32318 | ||
|
|
6813872dd3 | ||
|
|
e337a367d1 | ||
|
|
65343470e1 | ||
|
|
e69114a839 | ||
|
|
2d63a14c2e | ||
|
|
2d1f893a40 | ||
|
|
64f92deeb1 | ||
|
|
d47433ee83 | ||
|
|
6f1159617e | ||
|
|
8cc6b8cdde | ||
|
|
048488fb25 | ||
|
|
2a58ad2477 | ||
|
|
798c51b4e7 | ||
|
|
9a83d67d78 | ||
|
|
356b183c5c | ||
|
|
d64a4e448b | ||
|
|
860b3f9952 | ||
|
|
4418c5422f | ||
|
|
372c9de1db | ||
|
|
65adaf18d4 | ||
|
|
ed77465282 | ||
|
|
f5f6747ecb | ||
|
|
fbe16483ac | ||
|
|
e4a0105042 | ||
|
|
1d19ae0e7b | ||
|
|
b9d11982e3 | ||
|
|
73590f1ccd | ||
|
|
f8d35bf45d | ||
|
|
2038bad822 | ||
|
|
6c11947397 | ||
|
|
26565cd89c | ||
|
|
3a203e8351 | ||
|
|
500e4abcb9 | ||
|
|
d78851bb5b | ||
|
|
8c9a43d02b | ||
|
|
13c7c1de89 | ||
|
|
ba44b28cec | ||
|
|
79affe0629 | ||
|
|
d0ec7e3fb2 | ||
|
|
3f8b8077a9 | ||
|
|
290f17d76d | ||
|
|
64a1149550 | ||
|
|
e907ade40a | ||
|
|
68a7bf5527 | ||
|
|
db75ea28e4 | ||
|
|
ba4bdfe6af | ||
|
|
b63508db97 | ||
|
|
82b27e59cc | ||
|
|
ec8b5c5d6e | ||
|
|
dc1e9cd2e8 | ||
|
|
a681333a08 | ||
|
|
7cbbc720d1 | ||
|
|
863a838e6e | ||
|
|
5b046e828a | ||
|
|
a8f72c5e75 | ||
|
|
bb6c678904 | ||
|
|
71c2a31531 | ||
|
|
99a44dda37 | ||
|
|
5ae86465cc | ||
|
|
6e9c105eb9 | ||
|
|
26199fa954 | ||
|
|
3a542766d7 | ||
|
|
bbe03dc46f | ||
|
|
167d51b596 | ||
|
|
7efe8f5cc3 | ||
|
|
263fe6d1a2 | ||
|
|
76f98d5bb2 | ||
|
|
e0386fe40b | ||
|
|
7d99677acd | ||
|
|
29bc2d9e17 | ||
|
|
27f3e79508 | ||
|
|
ed74bee760 | ||
|
|
c7a81fe07a | ||
|
|
d880aac569 | ||
|
|
072d608c64 | ||
|
|
9437142bc8 | ||
|
|
58c8ec5777 | ||
|
|
07357b9f10 | ||
|
|
1264b4245c | ||
|
|
e3ecee18e3 | ||
|
|
c3d96622e8 | ||
|
|
e577efbd27 | ||
|
|
df361236d0 | ||
|
|
e656f5445c | ||
|
|
f124c0d491 | ||
|
|
cc041ba348 | ||
|
|
257c9dcd7f | ||
|
|
1857b86c7e | ||
|
|
1c3610e9af | ||
|
|
796bbef10b | ||
|
|
799e57f970 | ||
|
|
cf3a91dde0 | ||
|
|
72381a783b | ||
|
|
75f56ea4bd | ||
|
|
a418ba6adb | ||
|
|
2da930f819 | ||
|
|
a2c38112fb | ||
|
|
36535d188d | ||
|
|
79f49032e3 | ||
|
|
9b3b123e45 | ||
|
|
98436b4605 | ||
|
|
7652fa46d1 | ||
|
|
2347ce88cd | ||
|
|
78e5c57bd3 | ||
|
|
108636761c | ||
|
|
5f56828bda | ||
|
|
23e522e893 | ||
|
|
9423a889ba |
9
.env
9
.env
@@ -4,7 +4,7 @@
|
|||||||
NODE_ENV='production'
|
NODE_ENV='production'
|
||||||
|
|
||||||
ACCESS_TOKEN_COOKIE_NAME=''
|
ACCESS_TOKEN_COOKIE_NAME=''
|
||||||
AI_TRANSLATIONS_URL=''
|
APP_ID='learning'
|
||||||
BASE_URL=''
|
BASE_URL=''
|
||||||
CONTACT_URL=''
|
CONTACT_URL=''
|
||||||
CREDENTIALS_BASE_URL=''
|
CREDENTIALS_BASE_URL=''
|
||||||
@@ -12,11 +12,12 @@ CREDIT_HELP_LINK_URL=''
|
|||||||
CSRF_TOKEN_API_PATH=''
|
CSRF_TOKEN_API_PATH=''
|
||||||
DISCOVERY_API_BASE_URL=''
|
DISCOVERY_API_BASE_URL=''
|
||||||
DISCUSSIONS_MFE_BASE_URL=''
|
DISCUSSIONS_MFE_BASE_URL=''
|
||||||
|
DISCOUNT_CODE_INFO_URL=''
|
||||||
ECOMMERCE_BASE_URL=''
|
ECOMMERCE_BASE_URL=''
|
||||||
ENABLE_JUMPNAV='true'
|
ENABLE_JUMPNAV='true'
|
||||||
ENABLE_NEW_SIDEBAR=''
|
|
||||||
ENABLE_NOTICES=''
|
ENABLE_NOTICES=''
|
||||||
ENTERPRISE_LEARNER_PORTAL_HOSTNAME=''
|
ENTERPRISE_LEARNER_PORTAL_HOSTNAME=''
|
||||||
|
ENTERPRISE_LEARNER_PORTAL_URL=''
|
||||||
EXAMS_BASE_URL=''
|
EXAMS_BASE_URL=''
|
||||||
FAVICON_URL=''
|
FAVICON_URL=''
|
||||||
IGNORED_ERROR_REGEX=''
|
IGNORED_ERROR_REGEX=''
|
||||||
@@ -49,3 +50,7 @@ TWITTER_HASHTAG=''
|
|||||||
TWITTER_URL=''
|
TWITTER_URL=''
|
||||||
USER_INFO_COOKIE_NAME=''
|
USER_INFO_COOKIE_NAME=''
|
||||||
OPTIMIZELY_FULL_STACK_SDK_KEY=''
|
OPTIMIZELY_FULL_STACK_SDK_KEY=''
|
||||||
|
SHOW_UNGRADED_ASSIGNMENT_PROGRESS=''
|
||||||
|
# Fallback in local style files
|
||||||
|
PARAGON_THEME_URLS={}
|
||||||
|
FEATURE_ENABLE_CHAT_V2_ENDPOINT=''
|
||||||
|
|||||||
@@ -4,19 +4,20 @@
|
|||||||
NODE_ENV='development'
|
NODE_ENV='development'
|
||||||
|
|
||||||
ACCESS_TOKEN_COOKIE_NAME='edx-jwt-cookie-header-payload'
|
ACCESS_TOKEN_COOKIE_NAME='edx-jwt-cookie-header-payload'
|
||||||
AI_TRANSLATIONS_URL='http://localhost:18760'
|
APP_ID='learning'
|
||||||
BASE_URL='http://localhost:2000'
|
BASE_URL='http://localhost:2000'
|
||||||
CONTACT_URL='http://localhost:18000/contact'
|
CONTACT_URL='http://localhost:18000/contact'
|
||||||
CREDENTIALS_BASE_URL='http://localhost:18150'
|
CREDENTIALS_BASE_URL='http://localhost:18150'
|
||||||
CREDIT_HELP_LINK_URL='https://edx.readthedocs.io/projects/edx-guide-for-students/en/latest/SFD_credit_courses.html#keep-track-of-credit-requirements'
|
CREDIT_HELP_LINK_URL='https://help.edx.org/edxlearner/s/article/Can-I-receive-college-credit-or-credit-hours-for-my-course'
|
||||||
CSRF_TOKEN_API_PATH='/csrf/api/v1/token'
|
CSRF_TOKEN_API_PATH='/csrf/api/v1/token'
|
||||||
DISCOVERY_API_BASE_URL='http://localhost:18381'
|
DISCOVERY_API_BASE_URL='http://localhost:18381'
|
||||||
DISCUSSIONS_MFE_BASE_URL='http://localhost:2002'
|
DISCUSSIONS_MFE_BASE_URL='http://localhost:2002'
|
||||||
|
DISCOUNT_CODE_INFO_URL=''
|
||||||
ECOMMERCE_BASE_URL='http://localhost:18130'
|
ECOMMERCE_BASE_URL='http://localhost:18130'
|
||||||
ENABLE_JUMPNAV='true'
|
ENABLE_JUMPNAV='true'
|
||||||
ENABLE_NEW_SIDEBAR=''
|
|
||||||
ENABLE_NOTICES=''
|
ENABLE_NOTICES=''
|
||||||
ENTERPRISE_LEARNER_PORTAL_HOSTNAME='localhost:8734'
|
ENTERPRISE_LEARNER_PORTAL_HOSTNAME='localhost:8734'
|
||||||
|
ENTERPRISE_LEARNER_PORTAL_URL='http://localhost:8734'
|
||||||
EXAMS_BASE_URL=''
|
EXAMS_BASE_URL=''
|
||||||
FAVICON_URL=https://edx-cdn.org/v3/default/favicon.ico
|
FAVICON_URL=https://edx-cdn.org/v3/default/favicon.ico
|
||||||
IGNORED_ERROR_REGEX=''
|
IGNORED_ERROR_REGEX=''
|
||||||
@@ -51,3 +52,7 @@ SESSION_COOKIE_DOMAIN='localhost'
|
|||||||
CHAT_RESPONSE_URL='http://localhost:18000/api/learning_assistant/v1/course_id'
|
CHAT_RESPONSE_URL='http://localhost:18000/api/learning_assistant/v1/course_id'
|
||||||
PRIVACY_POLICY_URL='http://localhost:18000/privacy'
|
PRIVACY_POLICY_URL='http://localhost:18000/privacy'
|
||||||
OPTIMIZELY_FULL_STACK_SDK_KEY=''
|
OPTIMIZELY_FULL_STACK_SDK_KEY=''
|
||||||
|
SHOW_UNGRADED_ASSIGNMENT_PROGRESS=''
|
||||||
|
# Fallback in local style files
|
||||||
|
PARAGON_THEME_URLS={}
|
||||||
|
FEATURE_ENABLE_CHAT_V2_ENDPOINT='false'
|
||||||
|
|||||||
10
.env.test
10
.env.test
@@ -4,19 +4,20 @@
|
|||||||
NODE_ENV='test'
|
NODE_ENV='test'
|
||||||
|
|
||||||
ACCESS_TOKEN_COOKIE_NAME='edx-jwt-cookie-header-payload'
|
ACCESS_TOKEN_COOKIE_NAME='edx-jwt-cookie-header-payload'
|
||||||
AI_TRANSLATIONS_URL='http://localhost:18760'
|
APP_ID='learning'
|
||||||
BASE_URL='http://localhost:2000'
|
BASE_URL='http://localhost:2000'
|
||||||
CONTACT_URL='http://localhost:18000/contact'
|
CONTACT_URL='http://localhost:18000/contact'
|
||||||
CREDENTIALS_BASE_URL='http://localhost:18150'
|
CREDENTIALS_BASE_URL='http://localhost:18150'
|
||||||
CREDIT_HELP_LINK_URL='https://edx.readthedocs.io/projects/edx-guide-for-students/en/latest/SFD_credit_courses.html#keep-track-of-credit-requirements'
|
CREDIT_HELP_LINK_URL='https://help.edx.org/edxlearner/s/article/Can-I-receive-college-credit-or-credit-hours-for-my-course'
|
||||||
CSRF_TOKEN_API_PATH='/csrf/api/v1/token'
|
CSRF_TOKEN_API_PATH='/csrf/api/v1/token'
|
||||||
DISCOVERY_API_BASE_URL='http://localhost:18381'
|
DISCOVERY_API_BASE_URL='http://localhost:18381'
|
||||||
DISCUSSIONS_MFE_BASE_URL='http://localhost:2002'
|
DISCUSSIONS_MFE_BASE_URL='http://localhost:2002'
|
||||||
|
DISCOUNT_CODE_INFO_URL=''
|
||||||
ECOMMERCE_BASE_URL='http://localhost:18130'
|
ECOMMERCE_BASE_URL='http://localhost:18130'
|
||||||
ENABLE_JUMPNAV='true'
|
ENABLE_JUMPNAV='true'
|
||||||
ENABLE_NEW_SIDEBAR=''
|
|
||||||
ENABLE_NOTICES=''
|
ENABLE_NOTICES=''
|
||||||
ENTERPRISE_LEARNER_PORTAL_HOSTNAME='localhost:8734'
|
ENTERPRISE_LEARNER_PORTAL_HOSTNAME='localhost:8734'
|
||||||
|
ENTERPRISE_LEARNER_PORTAL_URL='http://localhost:8734'
|
||||||
EXAMS_BASE_URL='http://localhost:18740'
|
EXAMS_BASE_URL='http://localhost:18740'
|
||||||
FAVICON_URL=https://edx-cdn.org/v3/default/favicon.ico
|
FAVICON_URL=https://edx-cdn.org/v3/default/favicon.ico
|
||||||
IGNORED_ERROR_REGEX=''
|
IGNORED_ERROR_REGEX=''
|
||||||
@@ -48,3 +49,6 @@ TWITTER_HASHTAG='myedxjourney'
|
|||||||
TWITTER_URL='https://twitter.com/edXOnline'
|
TWITTER_URL='https://twitter.com/edXOnline'
|
||||||
USER_INFO_COOKIE_NAME='edx-user-info'
|
USER_INFO_COOKIE_NAME='edx-user-info'
|
||||||
PRIVACY_POLICY_URL='http://localhost:18000/privacy'
|
PRIVACY_POLICY_URL='http://localhost:18000/privacy'
|
||||||
|
SHOW_UNGRADED_ASSIGNMENT_PROGRESS=''
|
||||||
|
ENTERPRISE_LEARNER_PORTAL_URL='http://localhost:Enterprise'
|
||||||
|
FEATURE_ENABLE_CHAT_V2_ENDPOINT='false'
|
||||||
|
|||||||
@@ -3,3 +3,5 @@ dist/
|
|||||||
packages/
|
packages/
|
||||||
node_modules/
|
node_modules/
|
||||||
jest.config.js
|
jest.config.js
|
||||||
|
env.config.jsx
|
||||||
|
example.env.config.jsx
|
||||||
|
|||||||
7
.github/dependabot.yml
vendored
Normal file
7
.github/dependabot.yml
vendored
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
version: 2
|
||||||
|
updates:
|
||||||
|
# Adding new check for github-actions
|
||||||
|
- package-ecosystem: "github-actions"
|
||||||
|
directory: "/"
|
||||||
|
schedule:
|
||||||
|
interval: "weekly"
|
||||||
18
.github/workflows/add-issue-to-btr-project.yml
vendored
Normal file
18
.github/workflows/add-issue-to-btr-project.yml
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Run the workflow that adds new tickets that are labelled "release testing"
|
||||||
|
# to the org-wide BTR project board
|
||||||
|
|
||||||
|
name: Add release testing issues to the BTR project board
|
||||||
|
|
||||||
|
on:
|
||||||
|
issues:
|
||||||
|
types: [labeled]
|
||||||
|
# This workflow is triggered when an issue is labeled with 'release testing'.
|
||||||
|
# It adds the issue to the BTR project and applies the 'needs triage' label
|
||||||
|
# if it doesn't already have it.
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
handle-release-testing:
|
||||||
|
uses: openedx/.github/.github/workflows/add-issue-to-btr-project.yml@master
|
||||||
|
secrets:
|
||||||
|
GITHUB_APP_ID: ${{ secrets.GRAPHQL_AUTH_APP_ID }}
|
||||||
|
GITHUB_APP_PRIVATE_KEY: ${{ secrets.GRAPHQL_AUTH_APP_PEM }}
|
||||||
2
.github/workflows/lockfileversion-check.yml
vendored
2
.github/workflows/lockfileversion-check.yml
vendored
@@ -10,4 +10,4 @@ on:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
version-check:
|
version-check:
|
||||||
uses: openedx/.github/.github/workflows/lockfile-check.yml@master
|
uses: openedx/.github/.github/workflows/lockfileversion-check-v3.yml@master
|
||||||
|
|||||||
25
.github/workflows/validate.yml
vendored
25
.github/workflows/validate.yml
vendored
@@ -10,14 +10,27 @@ jobs:
|
|||||||
tests:
|
tests:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v3
|
- uses: actions/checkout@v5
|
||||||
- name: Setup Nodejs Env
|
- uses: actions/setup-node@v6
|
||||||
run: echo "NODE_VER=`cat .nvmrc`" >> $GITHUB_ENV
|
|
||||||
- uses: actions/setup-node@v3
|
|
||||||
with:
|
with:
|
||||||
node-version: ${{ env.NODE_VER }}
|
node-version-file: '.nvmrc'
|
||||||
- run: make validate.ci
|
- run: make validate.ci
|
||||||
|
- name: Archive code coverage results
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: code-coverage-report
|
||||||
|
path: coverage/*.*
|
||||||
|
coverage:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: tests
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v5
|
||||||
|
- name: Download code coverage results
|
||||||
|
uses: actions/download-artifact@v5
|
||||||
|
with:
|
||||||
|
pattern: code-coverage-report
|
||||||
- name: Upload coverage
|
- name: Upload coverage
|
||||||
uses: codecov/codecov-action@v3
|
uses: codecov/codecov-action@v5
|
||||||
with:
|
with:
|
||||||
fail_ci_if_error: true
|
fail_ci_if_error: true
|
||||||
|
token: ${{ secrets.CODECOV_TOKEN }}
|
||||||
1
.husky/_/.gitignore
vendored
1
.husky/_/.gitignore
vendored
@@ -1 +0,0 @@
|
|||||||
*
|
|
||||||
@@ -1,31 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
if [ -z "$husky_skip_init" ]; then
|
|
||||||
debug () {
|
|
||||||
if [ "$HUSKY_DEBUG" = "1" ]; then
|
|
||||||
echo "husky (debug) - $1"
|
|
||||||
fi
|
|
||||||
}
|
|
||||||
|
|
||||||
readonly hook_name="$(basename "$0")"
|
|
||||||
debug "starting $hook_name..."
|
|
||||||
|
|
||||||
if [ "$HUSKY" = "0" ]; then
|
|
||||||
debug "HUSKY env variable is set to 0, skipping hook"
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
if [ -f ~/.huskyrc ]; then
|
|
||||||
debug "sourcing ~/.huskyrc"
|
|
||||||
. ~/.huskyrc
|
|
||||||
fi
|
|
||||||
|
|
||||||
export readonly husky_skip_init=1
|
|
||||||
sh -e "$0" "$@"
|
|
||||||
exitCode="$?"
|
|
||||||
|
|
||||||
if [ $exitCode != 0 ]; then
|
|
||||||
echo "husky - $hook_name hook exited with code $exitCode (error)"
|
|
||||||
fi
|
|
||||||
|
|
||||||
exit $exitCode
|
|
||||||
fi
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
. "$(dirname "$0")/_/husky.sh"
|
|
||||||
|
|
||||||
npm run lint
|
|
||||||
2
Makefile
2
Makefile
@@ -55,8 +55,10 @@ validate:
|
|||||||
make validate-no-uncommitted-package-lock-changes
|
make validate-no-uncommitted-package-lock-changes
|
||||||
npm run i18n_extract
|
npm run i18n_extract
|
||||||
npm run lint -- --max-warnings 0
|
npm run lint -- --max-warnings 0
|
||||||
|
npm run types
|
||||||
npm run test
|
npm run test
|
||||||
npm run build
|
npm run build
|
||||||
|
npm run bundlewatch
|
||||||
|
|
||||||
.PHONY: validate.ci
|
.PHONY: validate.ci
|
||||||
validate.ci:
|
validate.ci:
|
||||||
|
|||||||
123
README.rst
123
README.rst
@@ -1,77 +1,109 @@
|
|||||||
#####################
|
|
||||||
frontend-app-learning
|
frontend-app-learning
|
||||||
#####################
|
#####################
|
||||||
|
|
||||||
|codecov| |license|
|
|codecov| |license|
|
||||||
|
|
||||||
********
|
|
||||||
Purpose
|
Purpose
|
||||||
********
|
*******
|
||||||
|
|
||||||
This is the Learning MFE (micro-frontend application), which renders all
|
This is the Learning MFE (micro-frontend application), which renders all
|
||||||
learner-facing course pages (like the course outline, the progress page,
|
learner-facing course pages (like the course outline, the progress page,
|
||||||
actual course content, etc).
|
actual course content, etc).
|
||||||
|
|
||||||
Please tag **@edx/engage-squad** on any PRs or issues. Thanks.
|
|
||||||
|
|
||||||
.. |codecov| image:: https://codecov.io/gh/edx/frontend-app-learning/branch/master/graph/badge.svg?token=3z7XvuzTq3
|
.. |codecov| image:: https://codecov.io/gh/edx/frontend-app-learning/branch/master/graph/badge.svg?token=3z7XvuzTq3
|
||||||
:target: https://codecov.io/gh/edx/frontend-app-learning
|
:target: https://codecov.io/gh/edx/frontend-app-learning
|
||||||
.. |license| image:: https://img.shields.io/badge/license-AGPL-informational
|
.. |license| image:: https://img.shields.io/badge/license-AGPL-informational
|
||||||
:target: https://github.com/openedx/frontend-app-account/blob/master/LICENSE
|
:target: https://github.com/openedx/frontend-app-account/blob/master/LICENSE
|
||||||
|
|
||||||
***************
|
|
||||||
Getting Started
|
Getting Started
|
||||||
***************
|
***************
|
||||||
|
|
||||||
Prerequisites
|
Prerequisites
|
||||||
=============
|
=============
|
||||||
|
|
||||||
The `devstack`_ is currently recommended as a development environment for your
|
`Tutor`_ is currently recommended as a development environment for the Learning
|
||||||
new MFE. If you start it with ``make dev.up.lms`` that should give you
|
MFE. Most likely, it already has this MFE configured; however, you'll need to
|
||||||
everything you need as a companion to this frontend.
|
make some changes in order to run it in development mode. You can refer
|
||||||
|
to the `relevant tutor-mfe documentation`_ for details, or follow the quick
|
||||||
Note that it is also possible to use `Tutor`_ to develop an MFE. You can refer
|
guide below.
|
||||||
to the `relevant tutor-mfe documentation`_ to get started using it.
|
|
||||||
|
|
||||||
.. _Devstack: https://github.com/openedx/devstack
|
|
||||||
|
|
||||||
.. _Tutor: https://github.com/overhangio/tutor
|
.. _Tutor: https://github.com/overhangio/tutor
|
||||||
|
|
||||||
.. _relevant tutor-mfe documentation: https://github.com/overhangio/tutor-mfe#mfe-development
|
.. _relevant tutor-mfe documentation: https://github.com/overhangio/tutor-mfe#mfe-development
|
||||||
|
|
||||||
To use this application, `devstack <https://github.com/openedx/devstack>`__ must be running and you must be logged into it.
|
|
||||||
|
|
||||||
- Visit http://localhost:2000/course/course-v1:edX+DemoX+Demo_Course to view the demo course. You can replace ``course-v1:edX+DemoX+Demo_Course`` with a different course key.
|
Cloning and Setup
|
||||||
|
=================
|
||||||
|
|
||||||
Cloning and Startup
|
1. Clone your new repo:
|
||||||
===================
|
|
||||||
|
|
||||||
.. code-block::
|
.. code-block:: bash
|
||||||
|
|
||||||
1. Clone your new repo:
|
git clone https://github.com/openedx/frontend-app-learning.git
|
||||||
|
|
||||||
``git clone https://github.com/openedx/frontend-app-learning.git``
|
2. Use the version of Node specified in ``.nvmrc``.
|
||||||
|
|
||||||
2. Use node v18.x.
|
Using other major versions of node *may* work, but this is unsupported. For
|
||||||
|
convenience, this repository includes an ``.nvmrc`` file to help in setting the
|
||||||
|
correct node version via `nvm <https://github.com/nvm-sh/nvm>`_.
|
||||||
|
|
||||||
The current version of the micro-frontend build scripts support node 18.
|
3. Stop the Tutor devstack, if it's running: ``tutor dev stop``
|
||||||
Using other major versions of node *may* work, but this is unsupported. For
|
|
||||||
convenience, this repository includes an .nvmrc file to help in setting the
|
|
||||||
correct node version via `nvm <https://github.com/nvm-sh/nvm>`_.
|
|
||||||
|
|
||||||
3. Install npm dependencies:
|
4. Next, we need to tell Tutor that we're going to be running this repo in
|
||||||
|
development mode, and it should be excluded from the ``mfe`` container that
|
||||||
|
otherwise runs every MFE. Run this:
|
||||||
|
|
||||||
``cd frontend-app-learning && npm ci``
|
.. code-block:: bash
|
||||||
|
|
||||||
4. Start the dev server:
|
tutor mounts add /path/to/frontend-app-learning
|
||||||
|
|
||||||
``npm start``
|
5. Start Tutor in development mode. This command will start the LMS and Studio,
|
||||||
|
and other required MFEs like ``authn`` and ``account``, but will not start
|
||||||
|
the learning MFE, which we're going to run on the host instead of in a
|
||||||
|
container managed by Tutor. Run:
|
||||||
|
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
tutor dev start lms cms mfe
|
||||||
|
|
||||||
|
Startup
|
||||||
|
=======
|
||||||
|
|
||||||
|
1. Install npm dependencies:
|
||||||
|
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
cd frontend-app-learning && npm ci
|
||||||
|
|
||||||
|
2. Start the dev server:
|
||||||
|
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
npm run dev
|
||||||
|
|
||||||
|
Then you can access the app at http://local.openedx.io:2000/learning/
|
||||||
|
|
||||||
|
Troubleshooting
|
||||||
|
---------------
|
||||||
|
|
||||||
|
If you see an "Invalid Host header" error, then you're probably using a different domain name for your devstack such as
|
||||||
|
``local.edly.io`` or ``local.overhang.io`` (not the new recommended default, ``local.openedx.io``). In that case, run
|
||||||
|
these commands to update your devstack's domain names:
|
||||||
|
|
||||||
|
.. code-block:: bash
|
||||||
|
|
||||||
|
tutor dev stop
|
||||||
|
tutor config save --set LMS_HOST=local.openedx.io --set CMS_HOST=studio.local.openedx.io
|
||||||
|
tutor dev launch -I --skip-build
|
||||||
|
tutor dev stop learning # We will run this MFE on the host
|
||||||
|
|
||||||
Local module development
|
Local module development
|
||||||
=========================
|
=========================
|
||||||
|
|
||||||
To develop locally on modules that are installed into this app, you'll need to create a ``module.config.js``
|
To develop locally on modules that are installed into this app, you'll need to create a ``module.config.js``
|
||||||
file (which is git-ignored) that defines where to find your local modules, for instance::
|
file (which is git-ignored) that defines where to find your local modules, for instance:
|
||||||
|
|
||||||
|
.. code-block:: js
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
/*
|
/*
|
||||||
@@ -98,15 +130,21 @@ Deployment
|
|||||||
|
|
||||||
The Learning MFE is similar to all the other Open edX MFEs. Read the Open
|
The Learning MFE is similar to all the other Open edX MFEs. Read the Open
|
||||||
edX Developer Guide's section on
|
edX Developer Guide's section on
|
||||||
`MFE applications <https://edx.readthedocs.io/projects/edx-developer-docs/en/latest/developers_guide/micro_frontends_in_open_edx.html>`_.
|
`MFE applications <https://openedx.github.io/frontend-platform/>`_.
|
||||||
|
|
||||||
|
Plugins
|
||||||
|
=======
|
||||||
|
This MFE can be customized using `Frontend Plugin Framework <https://github.com/openedx/frontend-plugin-framework>`_.
|
||||||
|
|
||||||
|
The parts of this MFE that can be customized in that manner are documented `here </src/plugin-slots>`_.
|
||||||
|
|
||||||
Environment Variables
|
Environment Variables
|
||||||
======================
|
=====================
|
||||||
|
|
||||||
This MFE is configured via environment variables supplied at build time.
|
This MFE is configured via environment variables supplied at build time.
|
||||||
All micro-frontends have a shared set of required environment variables,
|
All micro-frontends have a shared set of required environment variables,
|
||||||
as documented in the Open edX Developer Guide under
|
as documented in the Open edX Developer Guide under
|
||||||
`Required Environment Variables <https://edx.readthedocs.io/projects/edx-developer-docs/en/latest/developers_guide/micro_frontends_in_open_edx.html#required-environment-variables>`_.
|
`Required Environment Variables <https://openedx.github.io/frontend-platform/>`_.
|
||||||
|
|
||||||
The learning micro-frontend also supports the following additional variables:
|
The learning micro-frontend also supports the following additional variables:
|
||||||
|
|
||||||
@@ -127,7 +165,7 @@ SOCIAL_UTM_MILESTONE_CAMPAIGN
|
|||||||
|
|
||||||
SUPPORT_URL_CALCULATOR_MATH
|
SUPPORT_URL_CALCULATOR_MATH
|
||||||
A link that explains how to use the in-course calculator. You can use the
|
A link that explains how to use the in-course calculator. You can use the
|
||||||
one in the example below, if you don't want to have your own branded version.
|
one in the example below if you don't want to have your own branded version.
|
||||||
|
|
||||||
Example: https://support.edx.org/hc/en-us/articles/360000038428-Entering-math-expressions-in-assignments-or-the-calculator
|
Example: https://support.edx.org/hc/en-us/articles/360000038428-Entering-math-expressions-in-assignments-or-the-calculator
|
||||||
|
|
||||||
@@ -140,7 +178,7 @@ SUPPORT_URL_ID_VERIFICATION
|
|||||||
|
|
||||||
SUPPORT_URL_VERIFIED_CERTIFICATE
|
SUPPORT_URL_VERIFIED_CERTIFICATE
|
||||||
A link that explains what a verified certificate is. You can use the
|
A link that explains what a verified certificate is. You can use the
|
||||||
one in the example below, if you don't want to have your own branded version.
|
one in the example below if you don't want to have your own branded version.
|
||||||
Optional.
|
Optional.
|
||||||
|
|
||||||
Example: https://support.edx.org/hc/en-us/articles/206502008-What-is-a-verified-certificate
|
Example: https://support.edx.org/hc/en-us/articles/206502008-What-is-a-verified-certificate
|
||||||
@@ -156,13 +194,13 @@ TWITTER_URL
|
|||||||
A link to your Twitter account. The Twitter social-share link won't appear
|
A link to your Twitter account. The Twitter social-share link won't appear
|
||||||
unless this is set. Optional.
|
unless this is set. Optional.
|
||||||
|
|
||||||
Example: https://twitter.com/edXOnline
|
Example: https://twitter.com/openedx
|
||||||
|
|
||||||
Getting Help
|
Getting Help
|
||||||
===========
|
============
|
||||||
|
|
||||||
If you're having trouble, we have discussion forums at
|
If you're having trouble, we have `discussion forums`_
|
||||||
https://discuss.openedx.org where you can connect with others in the community.
|
where you can connect with others in the community.
|
||||||
|
|
||||||
Our real-time conversations are on Slack. You can request a `Slack
|
Our real-time conversations are on Slack. You can request a `Slack
|
||||||
invitation`_, then join our `community Slack workspace`_. Because this is a
|
invitation`_, then join our `community Slack workspace`_. Because this is a
|
||||||
@@ -180,17 +218,18 @@ For more information about these options, see the `Getting Help`_ page.
|
|||||||
.. _community Slack workspace: https://openedx.slack.com/
|
.. _community Slack workspace: https://openedx.slack.com/
|
||||||
.. _#wg-frontend channel: https://openedx.slack.com/archives/C04BM6YC7A6
|
.. _#wg-frontend channel: https://openedx.slack.com/archives/C04BM6YC7A6
|
||||||
.. _Getting Help: https://openedx.org/community/connect
|
.. _Getting Help: https://openedx.org/community/connect
|
||||||
|
.. _discussion forums: https://discuss.openedx.org
|
||||||
|
|
||||||
Contributing
|
Contributing
|
||||||
============
|
============
|
||||||
|
|
||||||
Contributions are very welcome. Please read `How To Contribute`_ for details.
|
Contributions are very welcome. Please read `How To Contribute`_ for details.
|
||||||
|
|
||||||
.. _How To Contribute: https://openedx.org/r/how-to-contribute
|
.. _How To Contribute: https://openedx.org/r/how-to-contribute
|
||||||
|
|
||||||
This project is currently accepting all types of contributions, bug fixes,
|
This project is currently accepting all types of contributions, bug fixes,
|
||||||
security fixes, maintenance work, or new features. However, please make sure
|
security fixes, maintenance work, or new features. However, please make sure
|
||||||
to have a discussion about your new feature idea with the maintainers prior to
|
to discuss your new feature idea with the maintainers before
|
||||||
beginning development to maximize the chances of your change being accepted.
|
beginning development to maximize the chances of your change being accepted.
|
||||||
You can start a conversation by creating a new issue on this repo summarizing
|
You can start a conversation by creating a new issue on this repo summarizing
|
||||||
your idea.
|
your idea.
|
||||||
|
|||||||
@@ -12,7 +12,8 @@ metadata:
|
|||||||
icon: "Web"
|
icon: "Web"
|
||||||
annotations:
|
annotations:
|
||||||
openedx.org/arch-interest-groups: ""
|
openedx.org/arch-interest-groups: ""
|
||||||
|
openedx.org/release: "master"
|
||||||
spec:
|
spec:
|
||||||
owner: group:2u-aurora
|
owner: group:committers-frontend-app-learning
|
||||||
type: 'website'
|
type: 'website'
|
||||||
lifecycle: 'production'
|
lifecycle: 'production'
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import UnitTranslationPlugin from '@plugins/UnitTranslationPlugin';
|
import UnitTranslationPlugin from '@edx/unit-translation-selector-plugin';
|
||||||
import { PLUGIN_OPERATIONS, DIRECT_PLUGIN } from '@openedx/frontend-plugin-framework';
|
import { PLUGIN_OPERATIONS, DIRECT_PLUGIN } from '@openedx/frontend-plugin-framework';
|
||||||
|
|
||||||
// Load environment variables from .env file
|
// Load environment variables from .env file
|
||||||
|
|||||||
@@ -9,13 +9,12 @@ const config = createConfig('jest', {
|
|||||||
'src/i18n',
|
'src/i18n',
|
||||||
'src/.*\\.exp\\..*',
|
'src/.*\\.exp\\..*',
|
||||||
],
|
],
|
||||||
// see https://github.com/axios/axios/issues/5026
|
|
||||||
moduleNameMapper: {
|
moduleNameMapper: {
|
||||||
"^axios$": "axios/dist/axios.js",
|
|
||||||
// See https://stackoverflow.com/questions/72382316/jest-encountered-an-unexpected-token-react-markdown
|
// See https://stackoverflow.com/questions/72382316/jest-encountered-an-unexpected-token-react-markdown
|
||||||
'react-markdown': '<rootDir>/node_modules/react-markdown/react-markdown.min.js',
|
'react-markdown': '<rootDir>/node_modules/react-markdown/react-markdown.min.js',
|
||||||
'@src/(.*)': '<rootDir>/src/$1',
|
'@src/(.*)': '<rootDir>/src/$1',
|
||||||
'@plugins/(.*)': '<rootDir>/plugins/$1',
|
// Explicit mapping to ensure Jest resolves the module correctly
|
||||||
|
'@edx/frontend-lib-special-exams': '<rootDir>/node_modules/@edx/frontend-lib-special-exams',
|
||||||
},
|
},
|
||||||
testTimeout: 30000,
|
testTimeout: 30000,
|
||||||
globalSetup: "./global-setup.js",
|
globalSetup: "./global-setup.js",
|
||||||
@@ -27,7 +26,7 @@ const config = createConfig('jest', {
|
|||||||
|
|
||||||
config.reporters = [...(config.reporters || []), ["jest-console-group-reporter", {
|
config.reporters = [...(config.reporters || []), ["jest-console-group-reporter", {
|
||||||
// change this setting if need to see less details for each test
|
// change this setting if need to see less details for each test
|
||||||
// reportType: "summary" | "details",
|
// reportType: "summary" | "details",
|
||||||
// enable: true | false,
|
// enable: true | false,
|
||||||
afterEachTest: {
|
afterEachTest: {
|
||||||
enable: true,
|
enable: true,
|
||||||
|
|||||||
10
openedx.yaml
10
openedx.yaml
@@ -1,10 +0,0 @@
|
|||||||
# This file describes this Open edX repo, as described in OEP-2:
|
|
||||||
# https://open-edx-proposals.readthedocs.io/en/latest/oep-0002-bp-repo-metadata.html#specification
|
|
||||||
|
|
||||||
oeps: {}
|
|
||||||
owner: edx/platform-core-tnl
|
|
||||||
openedx-release:
|
|
||||||
# The openedx-release key is described in OEP-10:
|
|
||||||
# https://open-edx-proposals.readthedocs.io/en/latest/oep-0010-proc-openedx-releases.html
|
|
||||||
# The FAQ might also be helpful: https://openedx.atlassian.net/wiki/spaces/COMM/pages/1331268879/Open+edX+Release+FAQ
|
|
||||||
ref: master
|
|
||||||
20780
package-lock.json
generated
20780
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
96
package.json
96
package.json
@@ -11,13 +11,16 @@
|
|||||||
],
|
],
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "fedx-scripts webpack",
|
"build": "fedx-scripts webpack",
|
||||||
|
"bundlewatch": "bundlewatch",
|
||||||
"i18n_extract": "fedx-scripts formatjs extract",
|
"i18n_extract": "fedx-scripts formatjs extract",
|
||||||
"lint": "fedx-scripts eslint --ext .js --ext .jsx .",
|
"lint": "fedx-scripts eslint --ext .js --ext .jsx --ext .ts --ext .tsx .",
|
||||||
"lint:fix": "fedx-scripts eslint --fix --ext .js --ext .jsx .",
|
"lint:fix": "fedx-scripts eslint --fix --ext .js --ext .jsx --ext .ts --ext .tsx .",
|
||||||
"prepare": "husky install",
|
|
||||||
"snapshot": "fedx-scripts jest --updateSnapshot",
|
|
||||||
"start": "fedx-scripts webpack-dev-server --progress",
|
"start": "fedx-scripts webpack-dev-server --progress",
|
||||||
"test": "fedx-scripts jest --coverage --passWithNoTests"
|
"start:with-theme": "paragon install-theme && npm start && npm install",
|
||||||
|
"dev": "PUBLIC_PATH=/learning/ MFE_CONFIG_API_URL='http://localhost:8000/api/mfe_config/v1' fedx-scripts webpack-dev-server --progress --host apps.local.openedx.io",
|
||||||
|
"test": "NODE_ENV=test fedx-scripts jest --coverage --passWithNoTests",
|
||||||
|
"test:watch": "fedx-scripts jest --watch --passWithNoTests",
|
||||||
|
"types": "tsc --noEmit"
|
||||||
},
|
},
|
||||||
"author": "edX",
|
"author": "edX",
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
@@ -30,65 +33,66 @@
|
|||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@edx/brand": "npm:@openedx/brand-openedx@^1.2.2",
|
"@edx/brand": "npm:@openedx/brand-openedx@^1.2.2",
|
||||||
"@edx/frontend-component-footer": "^13.0.4",
|
"@edx/browserslist-config": "1.5.0",
|
||||||
"@edx/frontend-component-header": "^5.0.2",
|
"@edx/frontend-component-footer": "^14.6.0",
|
||||||
"@edx/frontend-lib-learning-assistant": "^2.0.0",
|
"@edx/frontend-component-header": "^8.0.0",
|
||||||
"@edx/frontend-lib-special-exams": "^3.0.0",
|
"@edx/frontend-lib-learning-assistant": "^2.23.1",
|
||||||
"@edx/frontend-platform": "^7.1.2",
|
"@edx/frontend-lib-special-exams": "^4.0.0",
|
||||||
"@edx/openedx-atlas": "^0.6.0",
|
"@edx/frontend-platform": "^8.4.0",
|
||||||
"@edx/react-unit-test-utils": "^2.0.0",
|
"@edx/openedx-atlas": "^0.7.0",
|
||||||
"@fortawesome/fontawesome-svg-core": "1.3.0",
|
|
||||||
"@fortawesome/free-brands-svg-icons": "5.15.4",
|
"@fortawesome/free-brands-svg-icons": "5.15.4",
|
||||||
"@fortawesome/free-regular-svg-icons": "5.15.4",
|
"@fortawesome/free-regular-svg-icons": "5.15.4",
|
||||||
"@fortawesome/free-solid-svg-icons": "5.15.4",
|
"@fortawesome/free-solid-svg-icons": "5.15.4",
|
||||||
"@fortawesome/react-fontawesome": "^0.1.4",
|
"@fortawesome/react-fontawesome": "^0.1.4",
|
||||||
"@openedx/frontend-plugin-framework": "^1.0.2",
|
"@openedx/frontend-build": "^14.6.2",
|
||||||
"@openedx/paragon": "^22.1.1",
|
"@openedx/frontend-plugin-framework": "^1.7.0",
|
||||||
|
"@openedx/paragon": "^23.4.5",
|
||||||
"@popperjs/core": "2.11.8",
|
"@popperjs/core": "2.11.8",
|
||||||
"@reduxjs/toolkit": "1.8.1",
|
"@reduxjs/toolkit": "1.9.7",
|
||||||
"classnames": "2.3.2",
|
"buffer": "^6.0.3",
|
||||||
"core-js": "3.22.2",
|
"classnames": "2.5.1",
|
||||||
"history": "5.3.0",
|
"copy-webpack-plugin": "^12.0.0",
|
||||||
"joi": "^17.11.0",
|
"joi": "^17.11.0",
|
||||||
"js-cookie": "3.0.5",
|
"js-cookie": "3.0.5",
|
||||||
|
"lodash": "^4.17.21",
|
||||||
"lodash.camelcase": "4.3.0",
|
"lodash.camelcase": "4.3.0",
|
||||||
|
"postcss-loader": "^8.1.1",
|
||||||
"prop-types": "15.8.1",
|
"prop-types": "15.8.1",
|
||||||
"query-string": "^7.1.3",
|
"query-string": "^7.1.3",
|
||||||
"react": "17.0.2",
|
"react": "^18.3.1",
|
||||||
"react-dom": "17.0.2",
|
"react-dom": "^18.3.1",
|
||||||
"react-helmet": "6.1.0",
|
"react-helmet": "6.1.0",
|
||||||
"react-redux": "7.2.9",
|
"react-redux": "7.2.9",
|
||||||
"react-router": "6.15.0",
|
"react-router": "6.15.0",
|
||||||
"react-router-dom": "6.15.0",
|
"react-router-dom": "6.15.0",
|
||||||
"react-share": "4.4.1",
|
"react-share": "4.4.1",
|
||||||
"redux": "4.1.2",
|
"redux": "4.2.1",
|
||||||
"regenerator-runtime": "0.13.11",
|
|
||||||
"reselect": "4.1.8",
|
"reselect": "4.1.8",
|
||||||
"truncate-html": "1.0.4",
|
"sass": "^1.79.3",
|
||||||
"util": "0.12.5"
|
"sass-loader": "^16.0.2",
|
||||||
|
"source-map-loader": "^5.0.0",
|
||||||
|
"truncate-html": "1.0.4"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@edx/browserslist-config": "1.2.0",
|
"@pact-foundation/pact": "^13.0.0",
|
||||||
"@edx/reactifex": "2.2.0",
|
"@testing-library/jest-dom": "^6.6.3",
|
||||||
"@openedx/frontend-build": "13.0.30",
|
"@testing-library/react": "^16.2.0",
|
||||||
"@pact-foundation/pact": "^11.0.2",
|
"@testing-library/user-event": "14.6.1",
|
||||||
"@testing-library/jest-dom": "5.16.5",
|
"axios-mock-adapter": "2.1.0",
|
||||||
"@testing-library/react": "12.1.5",
|
"bundlewatch": "^0.4.0",
|
||||||
"@testing-library/react-hooks": "^8.0.1",
|
"eslint-import-resolver-webpack": "^0.13.9",
|
||||||
"@testing-library/user-event": "13.5.0",
|
"jest": "^29.7.0",
|
||||||
"axios-mock-adapter": "1.20.0",
|
"jest-console-group-reporter": "^1.1.1",
|
||||||
"copy-webpack-plugin": "^11.0.0",
|
|
||||||
"es-check": "6.2.1",
|
|
||||||
"eslint-import-resolver-webpack": "^0.13.8",
|
|
||||||
"husky": "7.0.4",
|
|
||||||
"jest": "^26.6.3",
|
|
||||||
"jest-console-group-reporter": "^1.0.1",
|
|
||||||
"jest-when": "^3.6.0",
|
"jest-when": "^3.6.0",
|
||||||
"postcss-loader": "^8.1.1",
|
"rosie": "2.1.1"
|
||||||
"rosie": "2.1.1",
|
},
|
||||||
"sass": "^1.72.0",
|
"bundlewatch": {
|
||||||
"sass-loader": "^14.1.1",
|
"files": [
|
||||||
"source-map-loader": "^5.0.0",
|
{
|
||||||
"style-loader": "^3.3.4"
|
"path": "dist/*.js",
|
||||||
|
"maxSize": "1450kB"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"normalizeFilenames": "^.+?(\\..+?)\\.\\w+$"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,17 +0,0 @@
|
|||||||
## How to develop plugin
|
|
||||||
|
|
||||||
You can define plugin in `env.config.jsx` see `example.env.config.jsx` as example.
|
|
||||||
|
|
||||||
## Current caveat
|
|
||||||
|
|
||||||
- The way for how I deal with override method is still wonky
|
|
||||||
- The redux still require middleware to ignore the plugin's action from serializing
|
|
||||||
- I am not sure how it behave with useCallback, useMemo, ...etc
|
|
||||||
- There are still open question on how to write it properly
|
|
||||||
|
|
||||||
## Current work that should consider core part and extendable for the future plugin framework
|
|
||||||
|
|
||||||
- `usePluingsCallback` is the callback supose to be some level of equality to be using `React.useCallback`. It would try to execute the function, then any plugin that try `registerOverrideMethod`. The order of the it being run isn't the determined. There are a couple things I want to add:
|
|
||||||
- I might consider testing it with `zustand` library to make sure it is portable and not rely on `redux`. I tried to do this with provider, but it seems to run into infinite loop of trigger changed.
|
|
||||||
|
|
||||||
- `registerOverrideMethod` is working like a way to register callback that behave like a middleware. It ran the default one, then pass the result of the default one to the plugin. Any plugin that register the override can update the value. Alternatively, we can override the function completely instead applying each affect. Or we can support both. But it requires a bit more thought out architecture.
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`<UnitTranslationPlugin /> render TranslationSelection when translation is enabled and language is available 1`] = `
|
|
||||||
<TranslationSelection
|
|
||||||
availableLanguages={
|
|
||||||
Array [
|
|
||||||
"en",
|
|
||||||
]
|
|
||||||
}
|
|
||||||
courseId="courseId"
|
|
||||||
id="id"
|
|
||||||
language="en"
|
|
||||||
unitId="unitId"
|
|
||||||
/>
|
|
||||||
`;
|
|
||||||
@@ -1,90 +0,0 @@
|
|||||||
import { getConfig, camelCaseObject } from '@edx/frontend-platform';
|
|
||||||
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
|
|
||||||
import { logError } from '@edx/frontend-platform/logging';
|
|
||||||
import { stringify } from 'query-string';
|
|
||||||
|
|
||||||
export const fetchTranslationConfig = async (courseId) => {
|
|
||||||
const url = `${
|
|
||||||
getConfig().LMS_BASE_URL
|
|
||||||
}/api/translatable_xblocks/config/?course_id=${encodeURIComponent(courseId)}`;
|
|
||||||
try {
|
|
||||||
const { data } = await getAuthenticatedHttpClient().get(url);
|
|
||||||
return {
|
|
||||||
enabled: data.feature_enabled,
|
|
||||||
availableLanguages: data.available_translation_languages || [
|
|
||||||
{
|
|
||||||
code: 'en',
|
|
||||||
label: 'English',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
code: 'es',
|
|
||||||
label: 'Spanish',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
} catch (error) {
|
|
||||||
logError(`Translation plugin fail to fetch from ${url}`, error);
|
|
||||||
return {
|
|
||||||
enabled: false,
|
|
||||||
availableLanguages: [],
|
|
||||||
};
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export async function getTranslationFeedback({
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
}) {
|
|
||||||
const params = stringify({
|
|
||||||
translation_language: translationLanguage,
|
|
||||||
course_id: encodeURIComponent(courseId),
|
|
||||||
unit_id: encodeURIComponent(unitId),
|
|
||||||
user_id: userId,
|
|
||||||
});
|
|
||||||
const fetchFeedbackUrl = `${
|
|
||||||
getConfig().AI_TRANSLATIONS_URL
|
|
||||||
}/api/v1/whole-course-translation-feedback?${params}`;
|
|
||||||
try {
|
|
||||||
const { data } = await getAuthenticatedHttpClient().get(fetchFeedbackUrl);
|
|
||||||
return camelCaseObject(data);
|
|
||||||
} catch (error) {
|
|
||||||
logError(
|
|
||||||
`Translation plugin fail to fetch from ${fetchFeedbackUrl}`,
|
|
||||||
error,
|
|
||||||
);
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function createTranslationFeedback({
|
|
||||||
courseId,
|
|
||||||
feedbackValue,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
}) {
|
|
||||||
const createFeedbackUrl = `${
|
|
||||||
getConfig().AI_TRANSLATIONS_URL
|
|
||||||
}/api/v1/whole-course-translation-feedback/`;
|
|
||||||
try {
|
|
||||||
const { data } = await getAuthenticatedHttpClient().post(
|
|
||||||
createFeedbackUrl,
|
|
||||||
{
|
|
||||||
course_id: courseId,
|
|
||||||
feedback_value: feedbackValue,
|
|
||||||
translation_language: translationLanguage,
|
|
||||||
unit_id: unitId,
|
|
||||||
user_id: userId,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
return camelCaseObject(data);
|
|
||||||
} catch (error) {
|
|
||||||
logError(
|
|
||||||
`Translation plugin fail to create feedback from ${createFeedbackUrl}`,
|
|
||||||
error,
|
|
||||||
);
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,125 +0,0 @@
|
|||||||
import { camelCaseObject } from '@edx/frontend-platform';
|
|
||||||
import { logError } from '@edx/frontend-platform/logging';
|
|
||||||
import { stringify } from 'query-string';
|
|
||||||
|
|
||||||
import {
|
|
||||||
fetchTranslationConfig,
|
|
||||||
getTranslationFeedback,
|
|
||||||
createTranslationFeedback,
|
|
||||||
} from './api';
|
|
||||||
|
|
||||||
const mockGetMethod = jest.fn();
|
|
||||||
const mockPostMethod = jest.fn();
|
|
||||||
jest.mock('@edx/frontend-platform/auth', () => ({
|
|
||||||
getAuthenticatedHttpClient: () => ({
|
|
||||||
get: mockGetMethod,
|
|
||||||
post: mockPostMethod,
|
|
||||||
}),
|
|
||||||
}));
|
|
||||||
jest.mock('@edx/frontend-platform/logging', () => ({
|
|
||||||
logError: jest.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
describe('UnitTranslation api', () => {
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks();
|
|
||||||
});
|
|
||||||
describe('fetchTranslationConfig', () => {
|
|
||||||
const courseId = 'course-v1:edX+DemoX+Demo_Course';
|
|
||||||
const expectedResponse = {
|
|
||||||
feature_enabled: true,
|
|
||||||
available_translation_languages: [
|
|
||||||
{
|
|
||||||
code: 'en',
|
|
||||||
label: 'English',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
code: 'es',
|
|
||||||
label: 'Spanish',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
it('should fetch translation config', async () => {
|
|
||||||
const expectedUrl = `http://localhost:18000/api/translatable_xblocks/config/?course_id=${encodeURIComponent(
|
|
||||||
courseId,
|
|
||||||
)}`;
|
|
||||||
mockGetMethod.mockResolvedValueOnce({ data: expectedResponse });
|
|
||||||
const result = await fetchTranslationConfig(courseId);
|
|
||||||
expect(result).toEqual({
|
|
||||||
enabled: true,
|
|
||||||
availableLanguages: expectedResponse.available_translation_languages,
|
|
||||||
});
|
|
||||||
expect(mockGetMethod).toHaveBeenCalledWith(expectedUrl);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should return disabled and unavailable languages on error', async () => {
|
|
||||||
mockGetMethod.mockRejectedValueOnce(new Error('error'));
|
|
||||||
const result = await fetchTranslationConfig(courseId);
|
|
||||||
expect(result).toEqual({
|
|
||||||
enabled: false,
|
|
||||||
availableLanguages: [],
|
|
||||||
});
|
|
||||||
expect(logError).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('getTranslationFeedback', () => {
|
|
||||||
const props = {
|
|
||||||
courseId: 'course-v1:edX+DemoX+Demo_Course',
|
|
||||||
translationLanguage: 'es',
|
|
||||||
unitId: 'unit-v1:edX+DemoX+Demo_Course+type@video+block@video',
|
|
||||||
userId: 'test_user',
|
|
||||||
};
|
|
||||||
const expectedResponse = {
|
|
||||||
feedback: 'good',
|
|
||||||
};
|
|
||||||
it('should fetch translation feedback', async () => {
|
|
||||||
const params = stringify({
|
|
||||||
translation_language: props.translationLanguage,
|
|
||||||
course_id: encodeURIComponent(props.courseId),
|
|
||||||
unit_id: encodeURIComponent(props.unitId),
|
|
||||||
user_id: props.userId,
|
|
||||||
});
|
|
||||||
const expectedUrl = `http://localhost:18760/api/v1/whole-course-translation-feedback?${params}`;
|
|
||||||
mockGetMethod.mockResolvedValueOnce({ data: expectedResponse });
|
|
||||||
const result = await getTranslationFeedback(props);
|
|
||||||
expect(result).toEqual(camelCaseObject(expectedResponse));
|
|
||||||
expect(mockGetMethod).toHaveBeenCalledWith(expectedUrl);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should return empty object on error', async () => {
|
|
||||||
mockGetMethod.mockRejectedValueOnce(new Error('error'));
|
|
||||||
const result = await getTranslationFeedback(props);
|
|
||||||
expect(result).toEqual({});
|
|
||||||
expect(logError).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('createTranslationFeedback', () => {
|
|
||||||
const props = {
|
|
||||||
courseId: 'course-v1:edX+DemoX+Demo_Course',
|
|
||||||
feedbackValue: 'good',
|
|
||||||
translationLanguage: 'es',
|
|
||||||
unitId: 'unit-v1:edX+DemoX+Demo_Course+type@video+block@video',
|
|
||||||
userId: 'test_user',
|
|
||||||
};
|
|
||||||
it('should create translation feedback', async () => {
|
|
||||||
const expectedUrl = 'http://localhost:18760/api/v1/whole-course-translation-feedback/';
|
|
||||||
mockPostMethod.mockResolvedValueOnce({});
|
|
||||||
await createTranslationFeedback(props);
|
|
||||||
expect(mockPostMethod).toHaveBeenCalledWith(expectedUrl, {
|
|
||||||
course_id: props.courseId,
|
|
||||||
feedback_value: props.feedbackValue,
|
|
||||||
translation_language: props.translationLanguage,
|
|
||||||
unit_id: props.unitId,
|
|
||||||
user_id: props.userId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should log error on failure', async () => {
|
|
||||||
mockPostMethod.mockRejectedValueOnce(new Error('error'));
|
|
||||||
await createTranslationFeedback(props);
|
|
||||||
expect(logError).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,204 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`<FeedbackWidget /> render feedback widget 1`] = `
|
|
||||||
<div
|
|
||||||
className="d-none"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="sequence w-100"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="ml-4 mr-2"
|
|
||||||
>
|
|
||||||
<ActionRow>
|
|
||||||
Rate this page translation
|
|
||||||
<Spacer />
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
alt="positive-feedback"
|
|
||||||
className="m-1"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="positive-feedback-button"
|
|
||||||
onClick={[MockFunction onThumbsUpClick]}
|
|
||||||
src="ThumbUpOutline"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
<IconButton
|
|
||||||
alt="negative-feedback"
|
|
||||||
className="mr-2"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="negative-feedback-button"
|
|
||||||
onClick={[MockFunction onThumbsDownClick]}
|
|
||||||
src="ThumbDownOffAlt"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="mb-1 text-light action-row-divider"
|
|
||||||
>
|
|
||||||
|
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
alt="close-feedback"
|
|
||||||
className="ml-1 mr-2 float-right"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="close-feedback-button"
|
|
||||||
onClick={[MockFunction closeFeedbackWidget]}
|
|
||||||
src="Close"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`<FeedbackWidget /> render gratitude text 1`] = `
|
|
||||||
<div
|
|
||||||
className="d-none"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="sequence w-100"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="ml-4 mr-4"
|
|
||||||
>
|
|
||||||
<ActionRow
|
|
||||||
className="m-2 justify-content-center"
|
|
||||||
>
|
|
||||||
Thank you! Your feedback matters.
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`<FeedbackWidget /> renders hidden by default 1`] = `
|
|
||||||
<div
|
|
||||||
className="d-none"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="sequence w-100"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="ml-4 mr-2"
|
|
||||||
>
|
|
||||||
<ActionRow>
|
|
||||||
Rate this page translation
|
|
||||||
<Spacer />
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
alt="positive-feedback"
|
|
||||||
className="m-1"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="positive-feedback-button"
|
|
||||||
onClick={[MockFunction onThumbsUpClick]}
|
|
||||||
src="ThumbUpOutline"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
<IconButton
|
|
||||||
alt="negative-feedback"
|
|
||||||
className="mr-2"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="negative-feedback-button"
|
|
||||||
onClick={[MockFunction onThumbsDownClick]}
|
|
||||||
src="ThumbDownOffAlt"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="mb-1 text-light action-row-divider"
|
|
||||||
>
|
|
||||||
|
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
alt="close-feedback"
|
|
||||||
className="ml-1 mr-2 float-right"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="close-feedback-button"
|
|
||||||
onClick={[MockFunction closeFeedbackWidget]}
|
|
||||||
src="Close"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="ml-4 mr-4"
|
|
||||||
>
|
|
||||||
<ActionRow
|
|
||||||
className="m-2 justify-content-center"
|
|
||||||
>
|
|
||||||
Thank you! Your feedback matters.
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`<FeedbackWidget /> renders show when elemReady is true 1`] = `
|
|
||||||
<div
|
|
||||||
className="sequence-container d-inline-flex flex-row w-100"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="sequence w-100"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
className="ml-4 mr-2"
|
|
||||||
>
|
|
||||||
<ActionRow>
|
|
||||||
Rate this page translation
|
|
||||||
<Spacer />
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
alt="positive-feedback"
|
|
||||||
className="m-1"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="positive-feedback-button"
|
|
||||||
onClick={[MockFunction onThumbsUpClick]}
|
|
||||||
src="ThumbUpOutline"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
<IconButton
|
|
||||||
alt="negative-feedback"
|
|
||||||
className="mr-2"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="negative-feedback-button"
|
|
||||||
onClick={[MockFunction onThumbsDownClick]}
|
|
||||||
src="ThumbDownOffAlt"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="mb-1 text-light action-row-divider"
|
|
||||||
>
|
|
||||||
|
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
alt="close-feedback"
|
|
||||||
className="ml-1 mr-2 float-right"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="close-feedback-button"
|
|
||||||
onClick={[MockFunction closeFeedbackWidget]}
|
|
||||||
src="Close"
|
|
||||||
variant="secondary"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
className="ml-4 mr-4"
|
|
||||||
>
|
|
||||||
<ActionRow
|
|
||||||
className="m-2 justify-content-center"
|
|
||||||
>
|
|
||||||
Thank you! Your feedback matters.
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
@@ -1,116 +0,0 @@
|
|||||||
import React, {
|
|
||||||
useEffect, useRef, useState,
|
|
||||||
} from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
|
|
||||||
import { useIntl } from '@edx/frontend-platform/i18n';
|
|
||||||
import { ActionRow, IconButton, Icon } from '@openedx/paragon';
|
|
||||||
import { Close, ThumbUpOutline, ThumbDownOffAlt } from '@openedx/paragon/icons';
|
|
||||||
|
|
||||||
import './index.scss';
|
|
||||||
import messages from './messages';
|
|
||||||
import useFeedbackWidget from './useFeedbackWidget';
|
|
||||||
|
|
||||||
const FeedbackWidget = ({
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
}) => {
|
|
||||||
const { formatMessage } = useIntl();
|
|
||||||
const ref = useRef(null);
|
|
||||||
const [elemReady, setElemReady] = useState(false);
|
|
||||||
const {
|
|
||||||
closeFeedbackWidget,
|
|
||||||
showFeedbackWidget,
|
|
||||||
showGratitudeText,
|
|
||||||
onThumbsUpClick,
|
|
||||||
onThumbsDownClick,
|
|
||||||
} = useFeedbackWidget({
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (ref.current) {
|
|
||||||
const domNode = document.getElementById('whole-course-translation-feedback-widget');
|
|
||||||
domNode.appendChild(ref.current);
|
|
||||||
setElemReady(true);
|
|
||||||
}
|
|
||||||
}, [ref.current]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div ref={ref} className={(elemReady) ? 'sequence-container d-inline-flex flex-row w-100' : 'd-none'}>
|
|
||||||
{(showFeedbackWidget || showGratitudeText) ? (
|
|
||||||
<div className="sequence w-100">
|
|
||||||
{
|
|
||||||
showFeedbackWidget && (
|
|
||||||
<div className="ml-4 mr-2">
|
|
||||||
<ActionRow>
|
|
||||||
{formatMessage(messages.rateTranslationText)}
|
|
||||||
<ActionRow.Spacer />
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
src={ThumbUpOutline}
|
|
||||||
iconAs={Icon}
|
|
||||||
alt="positive-feedback"
|
|
||||||
onClick={onThumbsUpClick}
|
|
||||||
variant="secondary"
|
|
||||||
className="m-1"
|
|
||||||
id="positive-feedback-button"
|
|
||||||
/>
|
|
||||||
<IconButton
|
|
||||||
src={ThumbDownOffAlt}
|
|
||||||
iconAs={Icon}
|
|
||||||
alt="negative-feedback"
|
|
||||||
onClick={onThumbsDownClick}
|
|
||||||
variant="secondary"
|
|
||||||
className="mr-2"
|
|
||||||
id="negative-feedback-button"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="mb-1 text-light action-row-divider">
|
|
||||||
|
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<IconButton
|
|
||||||
src={Close}
|
|
||||||
iconAs={Icon}
|
|
||||||
alt="close-feedback"
|
|
||||||
onClick={closeFeedbackWidget}
|
|
||||||
variant="secondary"
|
|
||||||
className="ml-1 mr-2 float-right"
|
|
||||||
id="close-feedback-button"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
{
|
|
||||||
showGratitudeText && (
|
|
||||||
<div className="ml-4 mr-4">
|
|
||||||
<ActionRow className="m-2 justify-content-center">
|
|
||||||
{formatMessage(messages.gratitudeText)}
|
|
||||||
</ActionRow>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
</div>
|
|
||||||
) : null}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
FeedbackWidget.propTypes = {
|
|
||||||
courseId: PropTypes.string.isRequired,
|
|
||||||
translationLanguage: PropTypes.string.isRequired,
|
|
||||||
userId: PropTypes.string.isRequired,
|
|
||||||
unitId: PropTypes.string.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
FeedbackWidget.defaultProps = {};
|
|
||||||
|
|
||||||
export default FeedbackWidget;
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
.action-row-divider {
|
|
||||||
font-size: 31px;
|
|
||||||
font-weight: 100;
|
|
||||||
}
|
|
||||||
@@ -1,107 +0,0 @@
|
|||||||
import { useState } from 'react';
|
|
||||||
import { shallow } from '@edx/react-unit-test-utils';
|
|
||||||
|
|
||||||
import FeedbackWidget from './index';
|
|
||||||
import useFeedbackWidget from './useFeedbackWidget';
|
|
||||||
|
|
||||||
jest.mock('react', () => ({
|
|
||||||
...jest.requireActual('react'),
|
|
||||||
useState: jest.fn((value) => [value, jest.fn()]),
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon', () => jest.requireActual('@edx/react-unit-test-utils').mockComponents({
|
|
||||||
ActionRow: {
|
|
||||||
Spacer: 'Spacer',
|
|
||||||
},
|
|
||||||
IconButton: 'IconButton',
|
|
||||||
Icon: 'Icon',
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon/icons', () => ({
|
|
||||||
Close: 'Close',
|
|
||||||
ThumbUpOutline: 'ThumbUpOutline',
|
|
||||||
ThumbDownOffAlt: 'ThumbDownOffAlt',
|
|
||||||
}));
|
|
||||||
jest.mock('./useFeedbackWidget');
|
|
||||||
jest.mock('@edx/frontend-platform/i18n', () => {
|
|
||||||
const i18n = jest.requireActual('@edx/frontend-platform/i18n');
|
|
||||||
const { formatMessage } = jest.requireActual('@edx/react-unit-test-utils');
|
|
||||||
return {
|
|
||||||
...i18n,
|
|
||||||
useIntl: jest.fn(() => ({
|
|
||||||
formatMessage,
|
|
||||||
})),
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('<FeedbackWidget />', () => {
|
|
||||||
const props = {
|
|
||||||
courseId: 'course-v1:edX+DemoX+Demo_Course',
|
|
||||||
translationLanguage: 'es',
|
|
||||||
unitId:
|
|
||||||
'block-v1:edX+DemoX+Demo_Course+type@vertical+block@37b72b3915204b70acb00c55b604b563',
|
|
||||||
userId: '123',
|
|
||||||
};
|
|
||||||
|
|
||||||
const mockUseFeedbackWidget = ({ showFeedbackWidget, showGratitudeText }) => {
|
|
||||||
useFeedbackWidget.mockReturnValueOnce({
|
|
||||||
closeFeedbackWidget: jest.fn().mockName('closeFeedbackWidget'),
|
|
||||||
sendFeedback: jest.fn().mockName('sendFeedback'),
|
|
||||||
onThumbsUpClick: jest.fn().mockName('onThumbsUpClick'),
|
|
||||||
onThumbsDownClick: jest.fn().mockName('onThumbsDownClick'),
|
|
||||||
showFeedbackWidget,
|
|
||||||
showGratitudeText,
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
it('renders hidden by default', () => {
|
|
||||||
mockUseFeedbackWidget({
|
|
||||||
showFeedbackWidget: true,
|
|
||||||
showGratitudeText: true,
|
|
||||||
});
|
|
||||||
const wrapper = shallow(<FeedbackWidget {...props} />);
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
expect(wrapper.instance.findByType('div')[0].props.className).toContain(
|
|
||||||
'd-none',
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('renders show when elemReady is true', () => {
|
|
||||||
mockUseFeedbackWidget({
|
|
||||||
showFeedbackWidget: true,
|
|
||||||
showGratitudeText: true,
|
|
||||||
});
|
|
||||||
useState.mockReturnValueOnce([true, jest.fn()]);
|
|
||||||
const wrapper = shallow(<FeedbackWidget {...props} />);
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
expect(wrapper.instance.findByType('div')[0].props.className).not.toContain(
|
|
||||||
'd-none',
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('render empty when showFeedbackWidget and showGratitudeText are false', () => {
|
|
||||||
mockUseFeedbackWidget({
|
|
||||||
showFeedbackWidget: false,
|
|
||||||
showGratitudeText: false,
|
|
||||||
});
|
|
||||||
useState.mockReturnValueOnce([true, jest.fn()]);
|
|
||||||
const wrapper = shallow(<FeedbackWidget {...props} />);
|
|
||||||
expect(wrapper.instance.findByType('div')[0].children.length).toBe(0);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('render feedback widget', () => {
|
|
||||||
mockUseFeedbackWidget({
|
|
||||||
showFeedbackWidget: true,
|
|
||||||
showGratitudeText: false,
|
|
||||||
});
|
|
||||||
const wrapper = shallow(<FeedbackWidget {...props} />);
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('render gratitude text', () => {
|
|
||||||
mockUseFeedbackWidget({
|
|
||||||
showFeedbackWidget: false,
|
|
||||||
showGratitudeText: true,
|
|
||||||
});
|
|
||||||
const wrapper = shallow(<FeedbackWidget {...props} />);
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
import { defineMessages } from '@edx/frontend-platform/i18n';
|
|
||||||
|
|
||||||
const messages = defineMessages({
|
|
||||||
rateTranslationText: {
|
|
||||||
id: 'feedbackWidget.rateTranslationText',
|
|
||||||
defaultMessage: 'Rate this page translation',
|
|
||||||
description: 'Title for the feedback widget action row.',
|
|
||||||
},
|
|
||||||
gratitudeText: {
|
|
||||||
id: 'feedbackWidget.gratitudeText',
|
|
||||||
defaultMessage: 'Thank you! Your feedback matters.',
|
|
||||||
description: 'Title for secondary action row.',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export default messages;
|
|
||||||
@@ -1,82 +0,0 @@
|
|||||||
import { useCallback, useEffect, useState } from 'react';
|
|
||||||
|
|
||||||
import { createTranslationFeedback, getTranslationFeedback } from '../data/api';
|
|
||||||
|
|
||||||
const useFeedbackWidget = ({
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
}) => {
|
|
||||||
const [showFeedbackWidget, setShowFeedbackWidget] = useState(false);
|
|
||||||
const [showGratitudeText, setShowGratitudeText] = useState(false);
|
|
||||||
|
|
||||||
const closeFeedbackWidget = useCallback(() => {
|
|
||||||
setShowFeedbackWidget(false);
|
|
||||||
}, [setShowFeedbackWidget]);
|
|
||||||
|
|
||||||
const openFeedbackWidget = useCallback(() => {
|
|
||||||
setShowFeedbackWidget(true);
|
|
||||||
}, [setShowFeedbackWidget]);
|
|
||||||
|
|
||||||
useEffect(async () => {
|
|
||||||
const translationFeedback = await getTranslationFeedback({
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
});
|
|
||||||
setShowFeedbackWidget(!translationFeedback);
|
|
||||||
}, [
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
]);
|
|
||||||
|
|
||||||
const openGratitudeText = useCallback(() => {
|
|
||||||
setShowGratitudeText(true);
|
|
||||||
setTimeout(() => {
|
|
||||||
setShowGratitudeText(false);
|
|
||||||
}, 3000);
|
|
||||||
}, [setShowGratitudeText]);
|
|
||||||
|
|
||||||
const sendFeedback = useCallback(async (feedbackValue) => {
|
|
||||||
await createTranslationFeedback({
|
|
||||||
courseId,
|
|
||||||
feedbackValue,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
});
|
|
||||||
closeFeedbackWidget();
|
|
||||||
openGratitudeText();
|
|
||||||
}, [
|
|
||||||
courseId,
|
|
||||||
translationLanguage,
|
|
||||||
unitId,
|
|
||||||
userId,
|
|
||||||
closeFeedbackWidget,
|
|
||||||
openGratitudeText,
|
|
||||||
]);
|
|
||||||
|
|
||||||
const onThumbsUpClick = useCallback(() => {
|
|
||||||
sendFeedback(true);
|
|
||||||
}, [sendFeedback]);
|
|
||||||
const onThumbsDownClick = useCallback(() => {
|
|
||||||
sendFeedback(false);
|
|
||||||
}, [sendFeedback]);
|
|
||||||
|
|
||||||
return {
|
|
||||||
closeFeedbackWidget,
|
|
||||||
openFeedbackWidget,
|
|
||||||
openGratitudeText,
|
|
||||||
sendFeedback,
|
|
||||||
showFeedbackWidget,
|
|
||||||
showGratitudeText,
|
|
||||||
onThumbsUpClick,
|
|
||||||
onThumbsDownClick,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useFeedbackWidget;
|
|
||||||
@@ -1,163 +0,0 @@
|
|||||||
import { renderHook, act } from '@testing-library/react-hooks';
|
|
||||||
|
|
||||||
import useFeedbackWidget from './useFeedbackWidget';
|
|
||||||
import { createTranslationFeedback, getTranslationFeedback } from '../data/api';
|
|
||||||
|
|
||||||
jest.mock('../data/api', () => ({
|
|
||||||
createTranslationFeedback: jest.fn(),
|
|
||||||
getTranslationFeedback: jest.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
const initialProps = {
|
|
||||||
courseId: 'course-v1:edX+DemoX+Demo_Course',
|
|
||||||
translationLanguage: 'es',
|
|
||||||
unitId: 'block-v1:edX+DemoX+Demo_Course+type@vertical+block@vertical_0270f6de40fc',
|
|
||||||
userId: 3,
|
|
||||||
};
|
|
||||||
|
|
||||||
const newProps = {
|
|
||||||
courseId: 'course-v1:edX+DemoX+Demo_Course',
|
|
||||||
translationLanguage: 'fr',
|
|
||||||
unitId: 'block-v1:edX+DemoX+Demo_Course+type@vertical+block@vertical_0270f6de40fc',
|
|
||||||
userId: 3,
|
|
||||||
};
|
|
||||||
|
|
||||||
describe('useFeedbackWidget', () => {
|
|
||||||
beforeEach(async () => {
|
|
||||||
getTranslationFeedback.mockReturnValue('');
|
|
||||||
});
|
|
||||||
|
|
||||||
afterEach(() => {
|
|
||||||
jest.restoreAllMocks();
|
|
||||||
});
|
|
||||||
|
|
||||||
test('closeFeedbackWidget behavior', () => {
|
|
||||||
const { result, waitFor } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
waitFor(() => expect(result.current.showFeedbackWidget.toBe(true)));
|
|
||||||
act(() => {
|
|
||||||
result.current.closeFeedbackWidget();
|
|
||||||
});
|
|
||||||
expect(result.current.showFeedbackWidget).toBe(false);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('openFeedbackWidget behavior', () => {
|
|
||||||
const { result } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
act(() => {
|
|
||||||
result.current.closeFeedbackWidget();
|
|
||||||
});
|
|
||||||
expect(result.current.showFeedbackWidget).toBe(false);
|
|
||||||
act(() => {
|
|
||||||
result.current.openFeedbackWidget();
|
|
||||||
});
|
|
||||||
expect(result.current.showFeedbackWidget).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
test('openGratitudeText behavior', async () => {
|
|
||||||
const { result, waitFor } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
|
|
||||||
expect(result.current.showGratitudeText).toBe(false);
|
|
||||||
act(() => {
|
|
||||||
result.current.openGratitudeText();
|
|
||||||
});
|
|
||||||
expect(result.current.showGratitudeText).toBe(true);
|
|
||||||
// Wait for 3 seconds to hide the gratitude text
|
|
||||||
waitFor(() => {
|
|
||||||
expect(result.current.showGratitudeText).toBe(false);
|
|
||||||
}, { timeout: 3000 });
|
|
||||||
});
|
|
||||||
|
|
||||||
test('sendFeedback behavior', () => {
|
|
||||||
const { result, waitFor } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
const feedbackValue = true;
|
|
||||||
|
|
||||||
waitFor(() => expect(result.current.showFeedbackWidget.toBe(true)));
|
|
||||||
|
|
||||||
expect(result.current.showGratitudeText).toBe(false);
|
|
||||||
act(() => {
|
|
||||||
result.current.sendFeedback(feedbackValue);
|
|
||||||
});
|
|
||||||
|
|
||||||
waitFor(() => {
|
|
||||||
expect(result.current.showFeedbackWidget).toBe(false);
|
|
||||||
expect(result.current.showGratitudeText).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(createTranslationFeedback).toHaveBeenCalledWith({
|
|
||||||
courseId: initialProps.courseId,
|
|
||||||
feedbackValue,
|
|
||||||
translationLanguage: initialProps.translationLanguage,
|
|
||||||
unitId: initialProps.unitId,
|
|
||||||
userId: initialProps.userId,
|
|
||||||
});
|
|
||||||
|
|
||||||
// Wait for 3 seconds to hide the gratitude text
|
|
||||||
waitFor(() => {
|
|
||||||
expect(result.current.showGratitudeText).toBe(false);
|
|
||||||
}, { timeout: 3000 });
|
|
||||||
});
|
|
||||||
|
|
||||||
test('onThumbsUpClick behavior', () => {
|
|
||||||
const { result } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
|
|
||||||
act(() => {
|
|
||||||
result.current.onThumbsUpClick();
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(createTranslationFeedback).toHaveBeenCalledWith({
|
|
||||||
courseId: initialProps.courseId,
|
|
||||||
feedbackValue: true,
|
|
||||||
translationLanguage: initialProps.translationLanguage,
|
|
||||||
unitId: initialProps.unitId,
|
|
||||||
userId: initialProps.userId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test('onThumbsDownClick behavior', () => {
|
|
||||||
const { result } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
|
|
||||||
act(() => {
|
|
||||||
result.current.onThumbsDownClick();
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(createTranslationFeedback).toHaveBeenCalledWith({
|
|
||||||
courseId: initialProps.courseId,
|
|
||||||
feedbackValue: false,
|
|
||||||
translationLanguage: initialProps.translationLanguage,
|
|
||||||
unitId: initialProps.unitId,
|
|
||||||
userId: initialProps.userId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test('fetch feedback on initialization', () => {
|
|
||||||
const { waitFor } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
waitFor(() => {
|
|
||||||
expect(getTranslationFeedback).toHaveBeenCalledWith({
|
|
||||||
courseId: initialProps.courseId,
|
|
||||||
translationLanguage: initialProps.translationLanguage,
|
|
||||||
unitId: initialProps.unitId,
|
|
||||||
userId: initialProps.userId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test('fetch feedback on props update', () => {
|
|
||||||
const { rerender, waitFor } = renderHook(() => useFeedbackWidget(initialProps));
|
|
||||||
waitFor(() => {
|
|
||||||
expect(getTranslationFeedback).toHaveBeenCalledWith({
|
|
||||||
courseId: initialProps.courseId,
|
|
||||||
translationLanguage: initialProps.translationLanguage,
|
|
||||||
unitId: initialProps.unitId,
|
|
||||||
userId: initialProps.userId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
rerender(newProps);
|
|
||||||
waitFor(() => {
|
|
||||||
expect(getTranslationFeedback).toHaveBeenCalledWith({
|
|
||||||
courseId: newProps.courseId,
|
|
||||||
translationLanguage: newProps.translationLanguage,
|
|
||||||
unitId: newProps.unitId,
|
|
||||||
userId: newProps.userId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,43 +0,0 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
|
|
||||||
import { useModel } from '@src/generic/model-store';
|
|
||||||
|
|
||||||
import TranslationSelection from './translation-selection';
|
|
||||||
import { fetchTranslationConfig } from './data/api';
|
|
||||||
|
|
||||||
const UnitTranslationPlugin = ({ id, courseId, unitId }) => {
|
|
||||||
const { language } = useModel('coursewareMeta', courseId);
|
|
||||||
const [translationConfig, setTranslationConfig] = useState({
|
|
||||||
enabled: false,
|
|
||||||
availableLanguages: [],
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
fetchTranslationConfig(courseId).then(setTranslationConfig);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const { enabled, availableLanguages } = translationConfig;
|
|
||||||
|
|
||||||
if (!enabled || !language || !availableLanguages.length) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<TranslationSelection
|
|
||||||
id={id}
|
|
||||||
courseId={courseId}
|
|
||||||
language={language}
|
|
||||||
availableLanguages={availableLanguages}
|
|
||||||
unitId={unitId}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
UnitTranslationPlugin.propTypes = {
|
|
||||||
id: PropTypes.string.isRequired,
|
|
||||||
courseId: PropTypes.string.isRequired,
|
|
||||||
unitId: PropTypes.string.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default UnitTranslationPlugin;
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
import { shallow } from '@edx/react-unit-test-utils';
|
|
||||||
import { useState } from 'react';
|
|
||||||
import { useModel } from '@src/generic/model-store';
|
|
||||||
|
|
||||||
import UnitTranslationPlugin from './index';
|
|
||||||
|
|
||||||
jest.mock('@src/generic/model-store');
|
|
||||||
jest.mock('./data/api', () => ({
|
|
||||||
fetchTranslationConfig: jest.fn(),
|
|
||||||
}));
|
|
||||||
jest.mock('./translation-selection', () => 'TranslationSelection');
|
|
||||||
jest.mock('react', () => ({
|
|
||||||
...jest.requireActual('react'),
|
|
||||||
useState: jest.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
describe('<UnitTranslationPlugin />', () => {
|
|
||||||
const props = {
|
|
||||||
id: 'id',
|
|
||||||
courseId: 'courseId',
|
|
||||||
unitId: 'unitId',
|
|
||||||
};
|
|
||||||
const mockInitialState = ({ enabled = true, availableLanguages = ['en'] }) => {
|
|
||||||
useState.mockReturnValue([{ enabled, availableLanguages }, jest.fn()]);
|
|
||||||
};
|
|
||||||
it('render empty when translation is not enabled', () => {
|
|
||||||
useModel.mockReturnValue({ language: 'en' });
|
|
||||||
mockInitialState({ enabled: false });
|
|
||||||
|
|
||||||
const wrapper = shallow(<UnitTranslationPlugin {...props} />);
|
|
||||||
|
|
||||||
expect(wrapper.isEmptyRender()).toBe(true);
|
|
||||||
});
|
|
||||||
it('render empty when available languages is empty', () => {
|
|
||||||
useModel.mockReturnValue({ language: 'fr' });
|
|
||||||
mockInitialState({
|
|
||||||
availableLanguages: [],
|
|
||||||
});
|
|
||||||
|
|
||||||
const wrapper = shallow(<UnitTranslationPlugin {...props} />);
|
|
||||||
|
|
||||||
expect(wrapper.isEmptyRender()).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('render empty when course language has not been set', () => {
|
|
||||||
useModel.mockReturnValue({ language: undefined });
|
|
||||||
mockInitialState({});
|
|
||||||
|
|
||||||
const wrapper = shallow(<UnitTranslationPlugin {...props} />);
|
|
||||||
|
|
||||||
expect(wrapper.isEmptyRender()).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('render TranslationSelection when translation is enabled and language is available', () => {
|
|
||||||
useModel.mockReturnValue({ language: 'en' });
|
|
||||||
mockInitialState({});
|
|
||||||
|
|
||||||
const wrapper = shallow(<UnitTranslationPlugin {...props} />);
|
|
||||||
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,82 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
|
|
||||||
import { useIntl } from '@edx/frontend-platform/i18n';
|
|
||||||
import {
|
|
||||||
StandardModal,
|
|
||||||
ActionRow,
|
|
||||||
Button,
|
|
||||||
Icon,
|
|
||||||
ListBox,
|
|
||||||
ListBoxOption,
|
|
||||||
} from '@openedx/paragon';
|
|
||||||
import { Check } from '@openedx/paragon/icons';
|
|
||||||
|
|
||||||
import useTranslationModal from './useTranslationModal';
|
|
||||||
import messages from './messages';
|
|
||||||
|
|
||||||
import './TranslationModal.scss';
|
|
||||||
|
|
||||||
const TranslationModal = ({
|
|
||||||
isOpen,
|
|
||||||
close,
|
|
||||||
selectedLanguage,
|
|
||||||
setSelectedLanguage,
|
|
||||||
availableLanguages,
|
|
||||||
}) => {
|
|
||||||
const { formatMessage } = useIntl();
|
|
||||||
const { selectedIndex, setSelectedIndex, onSubmit } = useTranslationModal({
|
|
||||||
selectedLanguage,
|
|
||||||
setSelectedLanguage,
|
|
||||||
close,
|
|
||||||
availableLanguages,
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
|
||||||
<StandardModal
|
|
||||||
title={formatMessage(messages.languageSelectionModalTitle)}
|
|
||||||
isOpen={isOpen}
|
|
||||||
onClose={close}
|
|
||||||
footerNode={(
|
|
||||||
<ActionRow>
|
|
||||||
<ActionRow.Spacer />
|
|
||||||
<Button variant="tertiary" onClick={close}>
|
|
||||||
{formatMessage(messages.cancelButtonText)}
|
|
||||||
</Button>
|
|
||||||
<Button onClick={onSubmit}>
|
|
||||||
{formatMessage(messages.submitButtonText)}
|
|
||||||
</Button>
|
|
||||||
</ActionRow>
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<ListBox className="listbox-container">
|
|
||||||
{availableLanguages.map(({ code, label }, index) => (
|
|
||||||
<ListBoxOption
|
|
||||||
className="d-flex justify-content-between"
|
|
||||||
key={code}
|
|
||||||
selectedOptionIndex={selectedIndex}
|
|
||||||
onSelect={() => setSelectedIndex(index)}
|
|
||||||
>
|
|
||||||
{label}
|
|
||||||
{selectedIndex === index && <Icon src={Check} />}
|
|
||||||
</ListBoxOption>
|
|
||||||
))}
|
|
||||||
</ListBox>
|
|
||||||
</StandardModal>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
TranslationModal.propTypes = {
|
|
||||||
isOpen: PropTypes.bool.isRequired,
|
|
||||||
close: PropTypes.func.isRequired,
|
|
||||||
selectedLanguage: PropTypes.string.isRequired,
|
|
||||||
setSelectedLanguage: PropTypes.func.isRequired,
|
|
||||||
availableLanguages: PropTypes.arrayOf(
|
|
||||||
PropTypes.shape({
|
|
||||||
code: PropTypes.string.isRequired,
|
|
||||||
label: PropTypes.string.isRequired,
|
|
||||||
}),
|
|
||||||
).isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default TranslationModal;
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
.listbox-container {
|
|
||||||
max-height: 400px;
|
|
||||||
|
|
||||||
:last-child {
|
|
||||||
margin-bottom: 5px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,59 +0,0 @@
|
|||||||
import { shallow } from '@edx/react-unit-test-utils';
|
|
||||||
|
|
||||||
import TranslationModal from './TranslationModal';
|
|
||||||
|
|
||||||
jest.mock('./useTranslationModal', () => ({
|
|
||||||
__esModule: true,
|
|
||||||
default: () => ({
|
|
||||||
selectedIndex: 0,
|
|
||||||
setSelectedIndex: jest.fn(),
|
|
||||||
onSubmit: jest.fn().mockName('onSubmit'),
|
|
||||||
}),
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon', () => jest.requireActual('@edx/react-unit-test-utils').mockComponents({
|
|
||||||
StandardModal: 'StandardModal',
|
|
||||||
ActionRow: {
|
|
||||||
Spacer: 'Spacer',
|
|
||||||
},
|
|
||||||
Button: 'Button',
|
|
||||||
Icon: 'Icon',
|
|
||||||
ListBox: 'ListBox',
|
|
||||||
ListBoxOption: 'ListBoxOption',
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon/icons', () => ({
|
|
||||||
Check: jest.fn().mockName('icons.Check'),
|
|
||||||
}));
|
|
||||||
jest.mock('@edx/frontend-platform/i18n', () => {
|
|
||||||
const i18n = jest.requireActual('@edx/frontend-platform/i18n');
|
|
||||||
const { formatMessage } = jest.requireActual('@edx/react-unit-test-utils');
|
|
||||||
return {
|
|
||||||
...i18n,
|
|
||||||
useIntl: jest.fn(() => ({
|
|
||||||
formatMessage,
|
|
||||||
})),
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('TranslationModal', () => {
|
|
||||||
const props = {
|
|
||||||
isOpen: true,
|
|
||||||
close: jest.fn().mockName('close'),
|
|
||||||
selectedLanguage: 'en',
|
|
||||||
setSelectedLanguage: jest.fn().mockName('setSelectedLanguage'),
|
|
||||||
availableLanguages: [
|
|
||||||
{
|
|
||||||
code: 'en',
|
|
||||||
label: 'English',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
code: 'es',
|
|
||||||
label: 'Spanish',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
};
|
|
||||||
it('renders correctly', () => {
|
|
||||||
const wrapper = shallow(<TranslationModal {...props} />);
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
expect(wrapper.instance.findByType('ListBoxOption')).toHaveLength(2);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`TranslationModal renders correctly 1`] = `
|
|
||||||
<StandardModal
|
|
||||||
footerNode={
|
|
||||||
<ActionRow>
|
|
||||||
<Spacer />
|
|
||||||
<Button
|
|
||||||
onClick={[MockFunction close]}
|
|
||||||
variant="tertiary"
|
|
||||||
>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
onClick={[MockFunction onSubmit]}
|
|
||||||
>
|
|
||||||
Submit
|
|
||||||
</Button>
|
|
||||||
</ActionRow>
|
|
||||||
}
|
|
||||||
isOpen={true}
|
|
||||||
onClose={[MockFunction close]}
|
|
||||||
title="Translate this course"
|
|
||||||
>
|
|
||||||
<ListBox
|
|
||||||
className="listbox-container"
|
|
||||||
>
|
|
||||||
<ListBoxOption
|
|
||||||
className="d-flex justify-content-between"
|
|
||||||
key="en"
|
|
||||||
onSelect={[Function]}
|
|
||||||
selectedOptionIndex={0}
|
|
||||||
>
|
|
||||||
English
|
|
||||||
<Icon
|
|
||||||
src={[MockFunction icons.Check]}
|
|
||||||
/>
|
|
||||||
</ListBoxOption>
|
|
||||||
<ListBoxOption
|
|
||||||
className="d-flex justify-content-between"
|
|
||||||
key="es"
|
|
||||||
onSelect={[Function]}
|
|
||||||
selectedOptionIndex={0}
|
|
||||||
>
|
|
||||||
Spanish
|
|
||||||
</ListBoxOption>
|
|
||||||
</ListBox>
|
|
||||||
</StandardModal>
|
|
||||||
`;
|
|
||||||
@@ -1,50 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`<TranslationSelection /> renders 1`] = `
|
|
||||||
<Fragment>
|
|
||||||
<ProductTour
|
|
||||||
tours={
|
|
||||||
Array [
|
|
||||||
Object {
|
|
||||||
"abitrarily": "defined",
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
<IconButton
|
|
||||||
alt="change-language"
|
|
||||||
className="mr-2 mb-2 float-right"
|
|
||||||
iconAs="Icon"
|
|
||||||
id="translation-selection-button"
|
|
||||||
onClick={[MockFunction open]}
|
|
||||||
src="Language"
|
|
||||||
variant="primary"
|
|
||||||
/>
|
|
||||||
<TranslationModal
|
|
||||||
availableLanguages={
|
|
||||||
Array [
|
|
||||||
Object {
|
|
||||||
"code": "en",
|
|
||||||
"label": "English",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"code": "es",
|
|
||||||
"label": "Spanish",
|
|
||||||
},
|
|
||||||
]
|
|
||||||
}
|
|
||||||
close={[MockFunction close]}
|
|
||||||
courseId="course-v1:edX+DemoX+Demo_Course"
|
|
||||||
id="plugin-test-id"
|
|
||||||
isOpen={false}
|
|
||||||
selectedLanguage="en"
|
|
||||||
setSelectedLanguage={[MockFunction setSelectedLanguage]}
|
|
||||||
/>
|
|
||||||
<FeedbackWidget
|
|
||||||
courseId="course-v1:edX+DemoX+Demo_Course"
|
|
||||||
translationLanguage="en"
|
|
||||||
unitId="unit-test-id"
|
|
||||||
userId="123"
|
|
||||||
/>
|
|
||||||
</Fragment>
|
|
||||||
`;
|
|
||||||
@@ -1,100 +0,0 @@
|
|||||||
import React, { useContext, useEffect } from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
|
|
||||||
import { AppContext } from '@edx/frontend-platform/react';
|
|
||||||
import { IconButton, Icon, ProductTour } from '@openedx/paragon';
|
|
||||||
import { Language } from '@openedx/paragon/icons';
|
|
||||||
import { useDispatch } from 'react-redux';
|
|
||||||
import { stringifyUrl } from 'query-string';
|
|
||||||
|
|
||||||
import { registerOverrideMethod } from '@src/generic/plugin-store';
|
|
||||||
|
|
||||||
import TranslationModal from './TranslationModal';
|
|
||||||
import useTranslationTour from './useTranslationTour';
|
|
||||||
import useSelectLanguage from './useSelectLanguage';
|
|
||||||
import FeedbackWidget from '../feedback-widget';
|
|
||||||
|
|
||||||
const TranslationSelection = ({
|
|
||||||
id, courseId, language, availableLanguages, unitId,
|
|
||||||
}) => {
|
|
||||||
const {
|
|
||||||
authenticatedUser: { userId },
|
|
||||||
} = useContext(AppContext);
|
|
||||||
const dispatch = useDispatch();
|
|
||||||
const {
|
|
||||||
translationTour, isOpen, open, close,
|
|
||||||
} = useTranslationTour();
|
|
||||||
|
|
||||||
const { selectedLanguage, setSelectedLanguage } = useSelectLanguage({
|
|
||||||
courseId,
|
|
||||||
language,
|
|
||||||
});
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
dispatch(
|
|
||||||
registerOverrideMethod({
|
|
||||||
pluginName: id,
|
|
||||||
methodName: 'getIFrameUrl',
|
|
||||||
method: (iframeUrl) => {
|
|
||||||
const finalUrl = stringifyUrl({
|
|
||||||
url: iframeUrl,
|
|
||||||
query: {
|
|
||||||
...(language
|
|
||||||
&& selectedLanguage
|
|
||||||
&& language !== selectedLanguage && {
|
|
||||||
src_lang: language,
|
|
||||||
dest_lang: selectedLanguage,
|
|
||||||
}),
|
|
||||||
},
|
|
||||||
});
|
|
||||||
return finalUrl;
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
}, [language, selectedLanguage]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<ProductTour tours={[translationTour]} />
|
|
||||||
<IconButton
|
|
||||||
src={Language}
|
|
||||||
iconAs={Icon}
|
|
||||||
alt="change-language"
|
|
||||||
onClick={open}
|
|
||||||
variant="primary"
|
|
||||||
className="mr-2 mb-2 float-right"
|
|
||||||
id="translation-selection-button"
|
|
||||||
/>
|
|
||||||
<TranslationModal
|
|
||||||
isOpen={isOpen}
|
|
||||||
close={close}
|
|
||||||
courseId={courseId}
|
|
||||||
selectedLanguage={selectedLanguage}
|
|
||||||
setSelectedLanguage={setSelectedLanguage}
|
|
||||||
availableLanguages={availableLanguages}
|
|
||||||
id={id}
|
|
||||||
/>
|
|
||||||
<FeedbackWidget
|
|
||||||
courseId={courseId}
|
|
||||||
translationLanguage={selectedLanguage}
|
|
||||||
unitId={unitId}
|
|
||||||
userId={userId}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
TranslationSelection.propTypes = {
|
|
||||||
id: PropTypes.string.isRequired,
|
|
||||||
courseId: PropTypes.string.isRequired,
|
|
||||||
unitId: PropTypes.string.isRequired,
|
|
||||||
language: PropTypes.string.isRequired,
|
|
||||||
availableLanguages: PropTypes.arrayOf(PropTypes.shape({
|
|
||||||
code: PropTypes.string.isRequired,
|
|
||||||
label: PropTypes.string.isRequired,
|
|
||||||
})).isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
TranslationSelection.defaultProps = {};
|
|
||||||
|
|
||||||
export default TranslationSelection;
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
import { shallow } from '@edx/react-unit-test-utils';
|
|
||||||
|
|
||||||
import TranslationSelection from './index';
|
|
||||||
|
|
||||||
jest.mock('react', () => ({
|
|
||||||
...jest.requireActual('react'),
|
|
||||||
useContext: jest.fn().mockName('useContext').mockReturnValue({
|
|
||||||
authenticatedUser: {
|
|
||||||
userId: '123',
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon', () => ({
|
|
||||||
IconButton: 'IconButton',
|
|
||||||
Icon: 'Icon',
|
|
||||||
ProductTour: 'ProductTour',
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon/icons', () => ({
|
|
||||||
Language: 'Language',
|
|
||||||
}));
|
|
||||||
jest.mock('./useTranslationTour', () => () => ({
|
|
||||||
translationTour: {
|
|
||||||
abitrarily: 'defined',
|
|
||||||
},
|
|
||||||
isOpen: false,
|
|
||||||
open: jest.fn().mockName('open'),
|
|
||||||
close: jest.fn().mockName('close'),
|
|
||||||
}));
|
|
||||||
jest.mock('react-redux', () => ({
|
|
||||||
useDispatch: jest.fn().mockName('useDispatch'),
|
|
||||||
}));
|
|
||||||
jest.mock('@src/generic/plugin-store', () => ({
|
|
||||||
registerOverrideMethod: jest.fn().mockName('registerOverrideMethod'),
|
|
||||||
}));
|
|
||||||
jest.mock('./TranslationModal', () => 'TranslationModal');
|
|
||||||
jest.mock('./useSelectLanguage', () => () => ({
|
|
||||||
selectedLanguage: 'en',
|
|
||||||
setSelectedLanguage: jest.fn().mockName('setSelectedLanguage'),
|
|
||||||
}));
|
|
||||||
jest.mock('../feedback-widget', () => 'FeedbackWidget');
|
|
||||||
|
|
||||||
describe('<TranslationSelection />', () => {
|
|
||||||
const props = {
|
|
||||||
id: 'plugin-test-id',
|
|
||||||
courseId: 'course-v1:edX+DemoX+Demo_Course',
|
|
||||||
language: 'en',
|
|
||||||
availableLanguages: [
|
|
||||||
{
|
|
||||||
code: 'en',
|
|
||||||
label: 'English',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
code: 'es',
|
|
||||||
label: 'Spanish',
|
|
||||||
},
|
|
||||||
],
|
|
||||||
unitId: 'unit-test-id',
|
|
||||||
};
|
|
||||||
it('renders', () => {
|
|
||||||
const wrapper = shallow(<TranslationSelection {...props} />);
|
|
||||||
expect(wrapper.snapshot).toMatchSnapshot();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,41 +0,0 @@
|
|||||||
import { defineMessages } from '@edx/frontend-platform/i18n';
|
|
||||||
|
|
||||||
const messages = defineMessages({
|
|
||||||
translationTourModalTitle: {
|
|
||||||
id: 'translationSelection.translationTourModalTitle',
|
|
||||||
defaultMessage: 'This is a standard modal dialog',
|
|
||||||
description: 'Title for the translation modal.',
|
|
||||||
},
|
|
||||||
translationTourModalBody: {
|
|
||||||
id: 'translationSelection.translationTourModalBody',
|
|
||||||
defaultMessage: 'Now you can easily translate course content.',
|
|
||||||
description: 'Body for the translation modal.',
|
|
||||||
},
|
|
||||||
tryItButtonText: {
|
|
||||||
id: 'translationSelection.tryItButtonText',
|
|
||||||
defaultMessage: 'Try it',
|
|
||||||
description: 'Button text for the translation modal.',
|
|
||||||
},
|
|
||||||
dismissButtonText: {
|
|
||||||
id: 'translationSelection.dismissButtonText',
|
|
||||||
defaultMessage: 'Dismiss',
|
|
||||||
description: 'Button text for the translation modal.',
|
|
||||||
},
|
|
||||||
languageSelectionModalTitle: {
|
|
||||||
id: 'translationSelection.languageSelectionModalTitle',
|
|
||||||
defaultMessage: 'Translate this course',
|
|
||||||
description: 'Title for the translation modal.',
|
|
||||||
},
|
|
||||||
cancelButtonText: {
|
|
||||||
id: 'translationSelection.cancelButtonText',
|
|
||||||
defaultMessage: 'Cancel',
|
|
||||||
description: 'Button text for the translation modal.',
|
|
||||||
},
|
|
||||||
submitButtonText: {
|
|
||||||
id: 'translationSelection.submitButtonText',
|
|
||||||
defaultMessage: 'Submit',
|
|
||||||
description: 'Button text for the translation modal.',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
export default messages;
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
import { useCallback } from 'react';
|
|
||||||
import { StrictDict, useKeyedState } from '@edx/react-unit-test-utils';
|
|
||||||
import {
|
|
||||||
getLocalStorage,
|
|
||||||
setLocalStorage,
|
|
||||||
} from '@src/data/localStorage';
|
|
||||||
|
|
||||||
export const selectedLanguageKey = 'selectedLanguages';
|
|
||||||
|
|
||||||
export const stateKeys = StrictDict({
|
|
||||||
selectedLanguage: 'selectedLanguage',
|
|
||||||
});
|
|
||||||
|
|
||||||
const useSelectLanguage = ({ courseId, language }) => {
|
|
||||||
const selectedLanguageItem = getLocalStorage(selectedLanguageKey) || {};
|
|
||||||
const [selectedLanguage, updateSelectedLanguage] = useKeyedState(
|
|
||||||
stateKeys.selectedLanguage,
|
|
||||||
selectedLanguageItem[courseId] || language,
|
|
||||||
);
|
|
||||||
|
|
||||||
const setSelectedLanguage = useCallback((newSelectedLanguage) => {
|
|
||||||
setLocalStorage(selectedLanguageKey, {
|
|
||||||
...selectedLanguageItem,
|
|
||||||
[courseId]: newSelectedLanguage,
|
|
||||||
});
|
|
||||||
updateSelectedLanguage(newSelectedLanguage);
|
|
||||||
});
|
|
||||||
|
|
||||||
return {
|
|
||||||
selectedLanguage,
|
|
||||||
setSelectedLanguage,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useSelectLanguage;
|
|
||||||
@@ -1,63 +0,0 @@
|
|||||||
import { mockUseKeyedState } from '@edx/react-unit-test-utils';
|
|
||||||
import {
|
|
||||||
getLocalStorage,
|
|
||||||
setLocalStorage,
|
|
||||||
} from '@src/data/localStorage';
|
|
||||||
|
|
||||||
import useSelectLanguage, {
|
|
||||||
stateKeys,
|
|
||||||
selectedLanguageKey,
|
|
||||||
} from './useSelectLanguage';
|
|
||||||
|
|
||||||
const state = mockUseKeyedState(stateKeys);
|
|
||||||
|
|
||||||
jest.mock('react', () => ({
|
|
||||||
...jest.requireActual('react'),
|
|
||||||
useCallback: jest.fn((cb, prereqs) => (...args) => [
|
|
||||||
cb(...args),
|
|
||||||
{ cb, prereqs },
|
|
||||||
]),
|
|
||||||
}));
|
|
||||||
jest.mock('@src/data/localStorage', () => ({
|
|
||||||
getLocalStorage: jest.fn(),
|
|
||||||
setLocalStorage: jest.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
describe('useSelectLanguage', () => {
|
|
||||||
const props = {
|
|
||||||
courseId: 'test-course-id',
|
|
||||||
language: 'en',
|
|
||||||
};
|
|
||||||
const languages = [
|
|
||||||
{ code: 'en', label: 'English' },
|
|
||||||
{ code: 'es', label: 'Spanish' },
|
|
||||||
];
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks();
|
|
||||||
state.mock();
|
|
||||||
});
|
|
||||||
afterEach(() => {
|
|
||||||
state.resetVals();
|
|
||||||
});
|
|
||||||
|
|
||||||
languages.forEach(({ code, label }) => {
|
|
||||||
it(`initializes selectedLanguage to the selected language (${label})`, () => {
|
|
||||||
getLocalStorage.mockReturnValueOnce({ [props.courseId]: code });
|
|
||||||
const { selectedLanguage } = useSelectLanguage(props);
|
|
||||||
|
|
||||||
state.expectInitializedWith(stateKeys.selectedLanguage, code);
|
|
||||||
expect(selectedLanguage).toBe(code);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test('setSelectedLanguage behavior', () => {
|
|
||||||
const { setSelectedLanguage } = useSelectLanguage(props);
|
|
||||||
|
|
||||||
setSelectedLanguage('es');
|
|
||||||
state.expectSetStateCalledWith(stateKeys.selectedLanguage, 'es');
|
|
||||||
expect(setLocalStorage).toHaveBeenCalledWith(selectedLanguageKey, {
|
|
||||||
[props.courseId]: 'es',
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
import { useCallback } from 'react';
|
|
||||||
import { StrictDict, useKeyedState } from '@edx/react-unit-test-utils';
|
|
||||||
|
|
||||||
export const stateKeys = StrictDict({
|
|
||||||
selectedIndex: 'selectedIndex',
|
|
||||||
});
|
|
||||||
|
|
||||||
const useTranslationModal = ({
|
|
||||||
selectedLanguage, setSelectedLanguage, close, availableLanguages,
|
|
||||||
}) => {
|
|
||||||
const [selectedIndex, setSelectedIndex] = useKeyedState(
|
|
||||||
stateKeys.selectedIndex,
|
|
||||||
availableLanguages.findIndex((lang) => lang.code === selectedLanguage),
|
|
||||||
);
|
|
||||||
|
|
||||||
const onSubmit = useCallback(() => {
|
|
||||||
const newSelectedLanguage = availableLanguages[selectedIndex].code;
|
|
||||||
setSelectedLanguage(newSelectedLanguage);
|
|
||||||
close();
|
|
||||||
}, [selectedIndex]);
|
|
||||||
|
|
||||||
return {
|
|
||||||
selectedIndex,
|
|
||||||
setSelectedIndex,
|
|
||||||
onSubmit,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useTranslationModal;
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
import { mockUseKeyedState } from '@edx/react-unit-test-utils';
|
|
||||||
|
|
||||||
import useTranslationModal, { stateKeys } from './useTranslationModal';
|
|
||||||
|
|
||||||
const state = mockUseKeyedState(stateKeys);
|
|
||||||
|
|
||||||
jest.mock('react', () => ({
|
|
||||||
...jest.requireActual('react'),
|
|
||||||
useCallback: jest.fn((cb, prereqs) => (...args) => ([
|
|
||||||
cb(...args), { cb, prereqs },
|
|
||||||
])),
|
|
||||||
}));
|
|
||||||
|
|
||||||
describe('useTranslationModal', () => {
|
|
||||||
const props = {
|
|
||||||
selectedLanguage: 'en',
|
|
||||||
setSelectedLanguage: jest.fn(),
|
|
||||||
close: jest.fn(),
|
|
||||||
availableLanguages: [
|
|
||||||
{ code: 'en', label: 'English' },
|
|
||||||
{ code: 'es', label: 'Spanish' },
|
|
||||||
],
|
|
||||||
};
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks();
|
|
||||||
state.mock();
|
|
||||||
});
|
|
||||||
afterEach(() => {
|
|
||||||
state.resetVals();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('initializes selectedIndex to the index of the selected language', () => {
|
|
||||||
const { selectedIndex } = useTranslationModal(props);
|
|
||||||
|
|
||||||
state.expectInitializedWith(stateKeys.selectedIndex, 0);
|
|
||||||
expect(selectedIndex).toBe(0);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('onSubmit updates the selected language and closes the modal', () => {
|
|
||||||
const { onSubmit } = useTranslationModal({
|
|
||||||
...props,
|
|
||||||
selectedLanguage: 'es',
|
|
||||||
});
|
|
||||||
onSubmit();
|
|
||||||
expect(props.setSelectedLanguage).toHaveBeenCalledWith('es');
|
|
||||||
expect(props.close).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,62 +0,0 @@
|
|||||||
import { useCallback } from 'react';
|
|
||||||
|
|
||||||
import { useIntl } from '@edx/frontend-platform/i18n';
|
|
||||||
import { useToggle } from '@openedx/paragon';
|
|
||||||
import { StrictDict, useKeyedState } from '@edx/react-unit-test-utils';
|
|
||||||
|
|
||||||
import messages from './messages';
|
|
||||||
|
|
||||||
const hasSeenTranslationTourKey = 'hasSeenTranslationTour';
|
|
||||||
|
|
||||||
export const stateKeys = StrictDict({
|
|
||||||
showTranslationTour: 'showTranslationTour',
|
|
||||||
});
|
|
||||||
|
|
||||||
const useTranslationTour = () => {
|
|
||||||
const { formatMessage } = useIntl();
|
|
||||||
|
|
||||||
const [isTourEnabled, setIsTourEnabled] = useKeyedState(
|
|
||||||
stateKeys.showTranslationTour,
|
|
||||||
global.localStorage.getItem(hasSeenTranslationTourKey) !== 'true',
|
|
||||||
);
|
|
||||||
const [isOpen, open, close] = useToggle(false);
|
|
||||||
|
|
||||||
const endTour = useCallback(() => {
|
|
||||||
global.localStorage.setItem(hasSeenTranslationTourKey, 'true');
|
|
||||||
setIsTourEnabled(false);
|
|
||||||
}, [isTourEnabled, setIsTourEnabled]);
|
|
||||||
|
|
||||||
const tryIt = useCallback(() => {
|
|
||||||
endTour();
|
|
||||||
open();
|
|
||||||
}, [endTour, open]);
|
|
||||||
|
|
||||||
const translationTour = isTourEnabled
|
|
||||||
? {
|
|
||||||
tourId: 'translation',
|
|
||||||
enabled: isTourEnabled,
|
|
||||||
onDismiss: endTour,
|
|
||||||
onEnd: tryIt,
|
|
||||||
checkpoints: [
|
|
||||||
{
|
|
||||||
title: formatMessage(messages.translationTourModalTitle),
|
|
||||||
body: formatMessage(messages.translationTourModalBody),
|
|
||||||
placement: 'bottom',
|
|
||||||
target: '#translation-selection-button',
|
|
||||||
showDismissButton: true,
|
|
||||||
endButtonText: formatMessage(messages.tryItButtonText),
|
|
||||||
dismissButtonText: formatMessage(messages.dismissButtonText),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
}
|
|
||||||
: {};
|
|
||||||
|
|
||||||
return {
|
|
||||||
translationTour,
|
|
||||||
isOpen,
|
|
||||||
open,
|
|
||||||
close,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
export default useTranslationTour;
|
|
||||||
@@ -1,95 +0,0 @@
|
|||||||
import { mockUseKeyedState } from '@edx/react-unit-test-utils';
|
|
||||||
import { useToggle } from '@openedx/paragon';
|
|
||||||
|
|
||||||
import useTranslationTour, { stateKeys } from './useTranslationTour';
|
|
||||||
|
|
||||||
jest.mock('react', () => ({
|
|
||||||
...jest.requireActual('react'),
|
|
||||||
useCallback: jest.fn((cb, prereqs) => () => {
|
|
||||||
cb();
|
|
||||||
return { useCallback: { cb, prereqs } };
|
|
||||||
}),
|
|
||||||
}));
|
|
||||||
jest.mock('@openedx/paragon', () => ({
|
|
||||||
useToggle: jest.fn(),
|
|
||||||
}));
|
|
||||||
jest.mock('@edx/frontend-platform/i18n', () => {
|
|
||||||
const i18n = jest.requireActual('@edx/frontend-platform/i18n');
|
|
||||||
const { formatMessage } = jest.requireActual('@edx/react-unit-test-utils');
|
|
||||||
// this provide consistent for the test on different platform/timezone
|
|
||||||
const formatDate = jest.fn(date => new Date(date).toISOString()).mockName('useIntl.formatDate');
|
|
||||||
return {
|
|
||||||
...i18n,
|
|
||||||
useIntl: jest.fn(() => ({
|
|
||||||
formatMessage,
|
|
||||||
formatDate,
|
|
||||||
})),
|
|
||||||
defineMessages: m => m,
|
|
||||||
FormattedMessage: () => 'FormattedMessage',
|
|
||||||
};
|
|
||||||
});
|
|
||||||
jest.mock('@src/data/localStorage', () => ({
|
|
||||||
getLocalStorage: jest.fn(),
|
|
||||||
setLocalStorage: jest.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
const state = mockUseKeyedState(stateKeys);
|
|
||||||
|
|
||||||
describe('useTranslationSelection', () => {
|
|
||||||
const mockLocalStroage = {
|
|
||||||
getItem: jest.fn(),
|
|
||||||
setItem: jest.fn(),
|
|
||||||
};
|
|
||||||
|
|
||||||
const toggleOpen = jest.fn();
|
|
||||||
const toggleClose = jest.fn();
|
|
||||||
|
|
||||||
useToggle.mockReturnValue([false, toggleOpen, toggleClose]);
|
|
||||||
|
|
||||||
beforeEach(() => {
|
|
||||||
jest.clearAllMocks();
|
|
||||||
state.mock();
|
|
||||||
window.localStorage = mockLocalStroage;
|
|
||||||
});
|
|
||||||
afterEach(() => {
|
|
||||||
state.resetVals();
|
|
||||||
delete window.localStorage;
|
|
||||||
});
|
|
||||||
|
|
||||||
it('do not have translation tour if user already seen it', () => {
|
|
||||||
mockLocalStroage.getItem.mockReturnValueOnce('not seen');
|
|
||||||
const { translationTour } = useTranslationTour();
|
|
||||||
|
|
||||||
expect(translationTour.enabled).toBe(true);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('show translation tour if user has not seen it', () => {
|
|
||||||
mockLocalStroage.getItem.mockReturnValueOnce('true');
|
|
||||||
const { translationTour } = useTranslationTour();
|
|
||||||
|
|
||||||
expect(translationTour).toMatchObject({});
|
|
||||||
});
|
|
||||||
test('open and close as pass from useToggle', () => {
|
|
||||||
const { isOpen, open, close } = useTranslationTour();
|
|
||||||
expect(isOpen).toBe(false);
|
|
||||||
expect(toggleOpen).toBe(open);
|
|
||||||
expect(toggleClose).toBe(close);
|
|
||||||
});
|
|
||||||
test('end tour on dismiss button click', () => {
|
|
||||||
mockLocalStroage.getItem.mockReturnValueOnce('not seen');
|
|
||||||
const { translationTour } = useTranslationTour();
|
|
||||||
translationTour.onDismiss();
|
|
||||||
expect(mockLocalStroage.setItem).toHaveBeenCalledWith(
|
|
||||||
'hasSeenTranslationTour',
|
|
||||||
'true',
|
|
||||||
);
|
|
||||||
state.expectSetStateCalledWith(stateKeys.showTranslationTour, false);
|
|
||||||
});
|
|
||||||
test('end tour and open modal on try it button click', () => {
|
|
||||||
mockLocalStroage.getItem.mockReturnValueOnce('not seen');
|
|
||||||
const { translationTour } = useTranslationTour();
|
|
||||||
translationTour.onEnd();
|
|
||||||
state.expectSetStateCalledWith(stateKeys.showTranslationTour, false);
|
|
||||||
expect(toggleOpen).toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -9,6 +9,9 @@
|
|||||||
<% if (htmlWebpackPlugin.options.OPTIMIZELY_PROJECT_ID) { %>
|
<% if (htmlWebpackPlugin.options.OPTIMIZELY_PROJECT_ID) { %>
|
||||||
<script src="https://www.edx.org/optimizelyjs/<%= htmlWebpackPlugin.options.OPTIMIZELY_PROJECT_ID %>.js"></script>
|
<script src="https://www.edx.org/optimizelyjs/<%= htmlWebpackPlugin.options.OPTIMIZELY_PROJECT_ID %>.js"></script>
|
||||||
<% } %>
|
<% } %>
|
||||||
|
<% if (htmlWebpackPlugin.options.META_TAG_ROBOTS_CONTENT_ATTR) { %>
|
||||||
|
<meta name="robots" content="<%= htmlWebpackPlugin.options.META_TAG_ROBOTS_CONTENT_ATTR %>">
|
||||||
|
<% } %>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="root"></div>
|
<div id="root"></div>
|
||||||
|
|||||||
@@ -1,14 +1,13 @@
|
|||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
||||||
import {
|
import { FormattedMessage, FormattedDate, useIntl } from '@edx/frontend-platform/i18n';
|
||||||
FormattedMessage, FormattedDate, injectIntl, intlShape,
|
|
||||||
} from '@edx/frontend-platform/i18n';
|
|
||||||
import { Alert, Hyperlink } from '@openedx/paragon';
|
import { Alert, Hyperlink } from '@openedx/paragon';
|
||||||
import { Info } from '@openedx/paragon/icons';
|
import { Info } from '@openedx/paragon/icons';
|
||||||
|
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
|
|
||||||
const AccessExpirationAlert = ({ intl, payload }) => {
|
const AccessExpirationAlert = ({ payload }) => {
|
||||||
|
const intl = useIntl();
|
||||||
const {
|
const {
|
||||||
accessExpiration,
|
accessExpiration,
|
||||||
courseId,
|
courseId,
|
||||||
@@ -119,7 +118,6 @@ const AccessExpirationAlert = ({ intl, payload }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
AccessExpirationAlert.propTypes = {
|
AccessExpirationAlert.propTypes = {
|
||||||
intl: intlShape.isRequired,
|
|
||||||
payload: PropTypes.shape({
|
payload: PropTypes.shape({
|
||||||
accessExpiration: PropTypes.shape({
|
accessExpiration: PropTypes.shape({
|
||||||
expirationDate: PropTypes.string.isRequired,
|
expirationDate: PropTypes.string.isRequired,
|
||||||
@@ -134,4 +132,4 @@ AccessExpirationAlert.propTypes = {
|
|||||||
}).isRequired,
|
}).isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(AccessExpirationAlert);
|
export default AccessExpirationAlert;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { FormattedMessage, injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { FormattedMessage, useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { Alert, Hyperlink } from '@openedx/paragon';
|
import { Alert, Hyperlink } from '@openedx/paragon';
|
||||||
import { WarningFilled } from '@openedx/paragon/icons';
|
import { WarningFilled } from '@openedx/paragon/icons';
|
||||||
@@ -7,7 +7,8 @@ import { WarningFilled } from '@openedx/paragon/icons';
|
|||||||
import { getConfig } from '@edx/frontend-platform';
|
import { getConfig } from '@edx/frontend-platform';
|
||||||
import genericMessages from './messages';
|
import genericMessages from './messages';
|
||||||
|
|
||||||
const ActiveEnterpriseAlert = ({ intl, payload }) => {
|
const ActiveEnterpriseAlert = ({ payload }) => {
|
||||||
|
const intl = useIntl();
|
||||||
const { text, courseId } = payload;
|
const { text, courseId } = payload;
|
||||||
const changeActiveEnterprise = (
|
const changeActiveEnterprise = (
|
||||||
<Hyperlink
|
<Hyperlink
|
||||||
@@ -38,11 +39,10 @@ const ActiveEnterpriseAlert = ({ intl, payload }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
ActiveEnterpriseAlert.propTypes = {
|
ActiveEnterpriseAlert.propTypes = {
|
||||||
intl: intlShape.isRequired,
|
|
||||||
payload: PropTypes.shape({
|
payload: PropTypes.shape({
|
||||||
text: PropTypes.string,
|
text: PropTypes.string,
|
||||||
courseId: PropTypes.string,
|
courseId: PropTypes.string,
|
||||||
}).isRequired,
|
}).isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(ActiveEnterpriseAlert);
|
export default ActiveEnterpriseAlert;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { Alert, Button } from '@openedx/paragon';
|
import { Alert, Button } from '@openedx/paragon';
|
||||||
import { Info, WarningFilled } from '@openedx/paragon/icons';
|
import { Info, WarningFilled } from '@openedx/paragon/icons';
|
||||||
@@ -11,7 +11,8 @@ import { useModel } from '../../generic/model-store';
|
|||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import useEnrollClickHandler from './clickHook';
|
import useEnrollClickHandler from './clickHook';
|
||||||
|
|
||||||
const EnrollmentAlert = ({ intl, payload }) => {
|
const EnrollmentAlert = ({ payload }) => {
|
||||||
|
const intl = useIntl();
|
||||||
const {
|
const {
|
||||||
canEnroll,
|
canEnroll,
|
||||||
courseId,
|
courseId,
|
||||||
@@ -58,7 +59,6 @@ const EnrollmentAlert = ({ intl, payload }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
EnrollmentAlert.propTypes = {
|
EnrollmentAlert.propTypes = {
|
||||||
intl: intlShape.isRequired,
|
|
||||||
payload: PropTypes.shape({
|
payload: PropTypes.shape({
|
||||||
canEnroll: PropTypes.bool,
|
canEnroll: PropTypes.bool,
|
||||||
courseId: PropTypes.string,
|
courseId: PropTypes.string,
|
||||||
@@ -67,4 +67,4 @@ EnrollmentAlert.propTypes = {
|
|||||||
}).isRequired,
|
}).isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(EnrollmentAlert);
|
export default EnrollmentAlert;
|
||||||
|
|||||||
@@ -9,13 +9,12 @@ import {
|
|||||||
Icon,
|
Icon,
|
||||||
} from '@openedx/paragon';
|
} from '@openedx/paragon';
|
||||||
import { Check, ArrowForward } from '@openedx/paragon/icons';
|
import { Check, ArrowForward } from '@openedx/paragon/icons';
|
||||||
import { FormattedMessage, injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { FormattedMessage, useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import { sendActivationEmail } from '../../courseware/data';
|
import { sendActivationEmail } from '../../courseware/data';
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
|
|
||||||
const AccountActivationAlert = ({
|
const AccountActivationAlert = () => {
|
||||||
intl,
|
const intl = useIntl();
|
||||||
}) => {
|
|
||||||
const [showModal, setShowModal] = useState(false);
|
const [showModal, setShowModal] = useState(false);
|
||||||
const [showSpinner, setShowSpinner] = useState(false);
|
const [showSpinner, setShowSpinner] = useState(false);
|
||||||
const [showCheck, setShowCheck] = useState(false);
|
const [showCheck, setShowCheck] = useState(false);
|
||||||
@@ -125,8 +124,4 @@ const AccountActivationAlert = ({
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
AccountActivationAlert.propTypes = {
|
export default AccountActivationAlert;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(AccountActivationAlert);
|
|
||||||
|
|||||||
@@ -1,13 +1,14 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { getConfig } from '@edx/frontend-platform';
|
import { getConfig } from '@edx/frontend-platform';
|
||||||
import { injectIntl, intlShape, FormattedMessage } from '@edx/frontend-platform/i18n';
|
import { useIntl, FormattedMessage } from '@edx/frontend-platform/i18n';
|
||||||
import { getLoginRedirectUrl } from '@edx/frontend-platform/auth';
|
import { getLoginRedirectUrl } from '@edx/frontend-platform/auth';
|
||||||
import { Alert, Hyperlink } from '@openedx/paragon';
|
import { Alert, Hyperlink } from '@openedx/paragon';
|
||||||
import { WarningFilled } from '@openedx/paragon/icons';
|
import { WarningFilled } from '@openedx/paragon/icons';
|
||||||
|
|
||||||
import genericMessages from '../../generic/messages';
|
import genericMessages from '../../generic/messages';
|
||||||
|
|
||||||
const LogistrationAlert = ({ intl }) => {
|
const LogistrationAlert = () => {
|
||||||
|
const intl = useIntl();
|
||||||
const signIn = (
|
const signIn = (
|
||||||
<Hyperlink
|
<Hyperlink
|
||||||
style={{ textDecoration: 'underline' }}
|
style={{ textDecoration: 'underline' }}
|
||||||
@@ -43,8 +44,4 @@ const LogistrationAlert = ({ intl }) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
LogistrationAlert.propTypes = {
|
export default LogistrationAlert;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(LogistrationAlert);
|
|
||||||
|
|||||||
@@ -1,35 +0,0 @@
|
|||||||
export const DECODE_ROUTES = {
|
|
||||||
ACCESS_DENIED: '/course/:courseId/access-denied',
|
|
||||||
HOME: '/course/:courseId/home',
|
|
||||||
LIVE: '/course/:courseId/live',
|
|
||||||
DATES: '/course/:courseId/dates',
|
|
||||||
DISCUSSION: '/course/:courseId/discussion/:path/*',
|
|
||||||
PROGRESS: [
|
|
||||||
'/course/:courseId/progress/:targetUserId/',
|
|
||||||
'/course/:courseId/progress',
|
|
||||||
],
|
|
||||||
COURSE_END: '/course/:courseId/course-end',
|
|
||||||
COURSEWARE: [
|
|
||||||
'/course/:courseId/:sequenceId/:unitId',
|
|
||||||
'/course/:courseId/:sequenceId',
|
|
||||||
'/course/:courseId',
|
|
||||||
],
|
|
||||||
REDIRECT_HOME: 'home/:courseId',
|
|
||||||
REDIRECT_SURVEY: 'survey/:courseId',
|
|
||||||
};
|
|
||||||
|
|
||||||
export const ROUTES = {
|
|
||||||
UNSUBSCRIBE: '/goal-unsubscribe/:token',
|
|
||||||
REDIRECT: '/redirect/*',
|
|
||||||
DASHBOARD: 'dashboard',
|
|
||||||
ENTERPRISE_LEARNER_DASHBOARD: 'enterprise-learner-dashboard',
|
|
||||||
CONSENT: 'consent',
|
|
||||||
};
|
|
||||||
|
|
||||||
export const REDIRECT_MODES = {
|
|
||||||
DASHBOARD_REDIRECT: 'dashboard-redirect',
|
|
||||||
ENTERPRISE_LEARNER_DASHBOARD_REDIRECT: 'enterprise-learner-dashboard-redirect',
|
|
||||||
CONSENT_REDIRECT: 'consent-redirect',
|
|
||||||
HOME_REDIRECT: 'home-redirect',
|
|
||||||
SURVEY_REDIRECT: 'survey-redirect',
|
|
||||||
};
|
|
||||||
74
src/constants.ts
Normal file
74
src/constants.ts
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
export const DECODE_ROUTES = {
|
||||||
|
ACCESS_DENIED: '/course/:courseId/access-denied',
|
||||||
|
HOME: '/course/:courseId/home',
|
||||||
|
LIVE: '/course/:courseId/live',
|
||||||
|
DATES: '/course/:courseId/dates',
|
||||||
|
DISCUSSION: '/course/:courseId/discussion/:path/*',
|
||||||
|
PROGRESS: [
|
||||||
|
'/course/:courseId/progress/:targetUserId/',
|
||||||
|
'/course/:courseId/progress',
|
||||||
|
],
|
||||||
|
COURSE_END: '/course/:courseId/course-end',
|
||||||
|
COURSEWARE: [
|
||||||
|
'/course/:courseId/:sequenceId/:unitId',
|
||||||
|
'/course/:courseId/:sequenceId',
|
||||||
|
'/course/:courseId',
|
||||||
|
'/preview/course/:courseId/:sequenceId/:unitId',
|
||||||
|
'/preview/course/:courseId/:sequenceId',
|
||||||
|
],
|
||||||
|
REDIRECT_HOME: 'home/:courseId',
|
||||||
|
REDIRECT_SURVEY: 'survey/:courseId',
|
||||||
|
} as const satisfies Readonly<{ [k: string]: string | readonly string[] }>;
|
||||||
|
|
||||||
|
export const ROUTES = {
|
||||||
|
UNSUBSCRIBE: '/goal-unsubscribe/:token',
|
||||||
|
PREFERENCES_UNSUBSCRIBE: '/preferences-unsubscribe/:userToken/:updatePatch?',
|
||||||
|
REDIRECT: '/redirect/*',
|
||||||
|
DASHBOARD: 'dashboard',
|
||||||
|
ENTERPRISE_LEARNER_DASHBOARD: 'enterprise-learner-dashboard',
|
||||||
|
CONSENT: 'consent',
|
||||||
|
} as const satisfies Readonly<{ [k: string]: string }>;
|
||||||
|
|
||||||
|
export const REDIRECT_MODES = {
|
||||||
|
DASHBOARD_REDIRECT: 'dashboard-redirect',
|
||||||
|
ENTERPRISE_LEARNER_DASHBOARD_REDIRECT: 'enterprise-learner-dashboard-redirect',
|
||||||
|
CONSENT_REDIRECT: 'consent-redirect',
|
||||||
|
HOME_REDIRECT: 'home-redirect',
|
||||||
|
SURVEY_REDIRECT: 'survey-redirect',
|
||||||
|
} as const satisfies Readonly<{ [k: string]: string }>;
|
||||||
|
|
||||||
|
export const VERIFIED_MODES = [
|
||||||
|
'professional',
|
||||||
|
'verified',
|
||||||
|
'no-id-professional',
|
||||||
|
'credit',
|
||||||
|
'masters',
|
||||||
|
'executive-education',
|
||||||
|
'paid-executive-education',
|
||||||
|
'paid-bootcamp',
|
||||||
|
] as const satisfies readonly string[];
|
||||||
|
|
||||||
|
export const AUDIT_MODES = [
|
||||||
|
'audit',
|
||||||
|
'honor',
|
||||||
|
'unpaid-executive-education',
|
||||||
|
'unpaid-bootcamp',
|
||||||
|
] as const satisfies readonly string[];
|
||||||
|
|
||||||
|
// In sync with CourseMode.UPSELL_TO_VERIFIED_MODES
|
||||||
|
// https://github.com/openedx/edx-platform/blob/master/common/djangoapps/course_modes/models.py#L231
|
||||||
|
export const ALLOW_UPSELL_MODES = [
|
||||||
|
'audit',
|
||||||
|
'honor',
|
||||||
|
] as const satisfies readonly string[];
|
||||||
|
|
||||||
|
export const WIDGETS = {
|
||||||
|
DISCUSSIONS: 'DISCUSSIONS',
|
||||||
|
NOTIFICATIONS: 'NOTIFICATIONS',
|
||||||
|
} as const satisfies Readonly<{ [k: string]: string }>;
|
||||||
|
|
||||||
|
export const LOADING = 'loading';
|
||||||
|
export const LOADED = 'loaded';
|
||||||
|
export const FAILED = 'failed';
|
||||||
|
export const DENIED = 'denied';
|
||||||
|
export type StatusValue = typeof LOADING | typeof LOADED | typeof FAILED | typeof DENIED;
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
import React, { useMemo } from 'react';
|
import React, { useMemo } from 'react';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import { Tabs, Tab } from '@openedx/paragon';
|
import { Tabs, Tab } from '@openedx/paragon';
|
||||||
|
|
||||||
import { useParams } from 'react-router';
|
import { useParams } from 'react-router';
|
||||||
@@ -13,7 +13,8 @@ const filterTypes = ['text', 'video', 'sequence'];
|
|||||||
const filterOther = 'other';
|
const filterOther = 'other';
|
||||||
const validFilters = [filterAll, ...filterTypes, filterOther];
|
const validFilters = [filterAll, ...filterTypes, filterOther];
|
||||||
|
|
||||||
export const CoursewareSearchResultsFilter = ({ intl }) => {
|
export const CoursewareSearchResultsFilter = () => {
|
||||||
|
const intl = useIntl();
|
||||||
const { courseId } = useParams();
|
const { courseId } = useParams();
|
||||||
const lastSearch = useModel('contentSearchResults', courseId);
|
const lastSearch = useModel('contentSearchResults', courseId);
|
||||||
const { filter: filterKeyword, setFilter } = useCoursewareSearchParams();
|
const { filter: filterKeyword, setFilter } = useCoursewareSearchParams();
|
||||||
@@ -73,8 +74,4 @@ export const CoursewareSearchResultsFilter = ({ intl }) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
CoursewareSearchResultsFilter.propTypes = {
|
export default CoursewareSearchResultsFilter;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(CoursewareSearchResultsFilter);
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import React, { useEffect } from 'react';
|
import React, { useEffect, useRef } from 'react';
|
||||||
import { useParams } from 'react-router';
|
import { useParams } from 'react-router';
|
||||||
import { useDispatch } from 'react-redux';
|
import { useDispatch } from 'react-redux';
|
||||||
import { sendTrackingLogEvent } from '@edx/frontend-platform/analytics';
|
import { sendTrackingLogEvent } from '@edx/frontend-platform/analytics';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import {
|
import {
|
||||||
Alert, Button, Icon, Spinner,
|
Alert, Button, Icon, Spinner,
|
||||||
} from '@openedx/paragon';
|
} from '@openedx/paragon';
|
||||||
@@ -18,7 +18,8 @@ import CoursewareSearchResultsFilterContainer from './CoursewareResultsFilter';
|
|||||||
import { updateModel, useModel } from '../../generic/model-store';
|
import { updateModel, useModel } from '../../generic/model-store';
|
||||||
import { searchCourseContent } from '../data/thunks';
|
import { searchCourseContent } from '../data/thunks';
|
||||||
|
|
||||||
const CoursewareSearch = ({ intl, ...sectionProps }) => {
|
const CoursewareSearch = ({ ...sectionProps }) => {
|
||||||
|
const { formatMessage } = useIntl();
|
||||||
const { courseId } = useParams();
|
const { courseId } = useParams();
|
||||||
const { query: searchKeyword, setQuery, clearSearchParams } = useCoursewareSearchParams();
|
const { query: searchKeyword, setQuery, clearSearchParams } = useCoursewareSearchParams();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
@@ -29,6 +30,7 @@ const CoursewareSearch = ({ intl, ...sectionProps }) => {
|
|||||||
errors,
|
errors,
|
||||||
total,
|
total,
|
||||||
} = useModel('contentSearchResults', courseId);
|
} = useModel('contentSearchResults', courseId);
|
||||||
|
const dialogRef = useRef();
|
||||||
|
|
||||||
useLockScroll();
|
useLockScroll();
|
||||||
|
|
||||||
@@ -44,7 +46,8 @@ const CoursewareSearch = ({ intl, ...sectionProps }) => {
|
|||||||
searchKeyword: '',
|
searchKeyword: '',
|
||||||
results: [],
|
results: [],
|
||||||
errors: undefined,
|
errors: undefined,
|
||||||
loading: false,
|
loading:
|
||||||
|
false,
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
};
|
};
|
||||||
@@ -66,20 +69,46 @@ const CoursewareSearch = ({ intl, ...sectionProps }) => {
|
|||||||
setQuery(value);
|
setQuery(value);
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
handleSubmit(searchKeyword);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const handleOnChange = (value) => {
|
const handleOnChange = (value) => {
|
||||||
if (value === searchKeyword) { return; }
|
if (value === searchKeyword) { return; }
|
||||||
if (!value) { clearSearch(); }
|
if (!value) { clearSearch(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleSearchCloseClick = () => {
|
const close = () => {
|
||||||
clearSearch();
|
clearSearch();
|
||||||
dispatch(setShowSearch(false));
|
dispatch(setShowSearch(false));
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handlePopState = () => close();
|
||||||
|
|
||||||
|
const handleBackdropClick = function (event) {
|
||||||
|
if (event.target === dialogRef.current) {
|
||||||
|
dialogRef.current.close();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// We need this to keep the dialog reference when unmounting.
|
||||||
|
const dialog = dialogRef.current;
|
||||||
|
|
||||||
|
// Open the dialog as a modal on render to confine focus within it.
|
||||||
|
dialogRef.current.showModal();
|
||||||
|
|
||||||
|
if (searchKeyword) {
|
||||||
|
handleSubmit(searchKeyword); // In case it's opened with a search link, we run the search.
|
||||||
|
}
|
||||||
|
|
||||||
|
const controller = new AbortController();
|
||||||
|
const { signal } = controller;
|
||||||
|
|
||||||
|
window.addEventListener('popstate', handlePopState, { signal });
|
||||||
|
dialog.addEventListener('click', handleBackdropClick, { signal });
|
||||||
|
|
||||||
|
return () => controller.abort(); // Removes event listeners.
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleSearchClose = () => close();
|
||||||
|
|
||||||
let status = 'idle';
|
let status = 'idle';
|
||||||
if (loading) {
|
if (loading) {
|
||||||
status = 'loading';
|
status = 'loading';
|
||||||
@@ -90,59 +119,58 @@ const CoursewareSearch = ({ intl, ...sectionProps }) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<section className="courseware-search" style={{ '--modal-top-position': top }} data-testid="courseware-search-section" {...sectionProps}>
|
<dialog ref={dialogRef} className="courseware-search" style={{ '--modal-top-position': top }} data-testid="courseware-search-dialog" onClose={handleSearchClose} {...sectionProps}>
|
||||||
<div className="courseware-search__close">
|
|
||||||
<Button
|
|
||||||
variant="tertiary"
|
|
||||||
className="p-1"
|
|
||||||
aria-label={intl.formatMessage(messages.searchCloseAction)}
|
|
||||||
onClick={handleSearchCloseClick}
|
|
||||||
data-testid="courseware-search-close-button"
|
|
||||||
><Icon src={Close} />
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
<div className="courseware-search__outer-content">
|
<div className="courseware-search__outer-content">
|
||||||
<div className="courseware-search__content">
|
<div className="courseware-search__content" data-testid="courseware-search-content">
|
||||||
<h1 className="h2">{intl.formatMessage(messages.searchModuleTitle)}</h1>
|
<div className="courseware-search__form">
|
||||||
<CoursewareSearchForm
|
<h1 className="h2">{formatMessage(messages.searchModuleTitle)}</h1>
|
||||||
searchTerm={searchKeyword}
|
<CoursewareSearchForm
|
||||||
onSubmit={handleSubmit}
|
searchTerm={searchKeyword}
|
||||||
onChange={handleOnChange}
|
onSubmit={handleSubmit}
|
||||||
placeholder={intl.formatMessage(messages.searchBarPlaceholderText)}
|
onChange={handleOnChange}
|
||||||
/>
|
placeholder={formatMessage(messages.searchBarPlaceholderText)}
|
||||||
{status === 'loading' ? (
|
/>
|
||||||
<div className="courseware-search__spinner" data-testid="courseware-search-spinner">
|
<div className="courseware-search__close">
|
||||||
<Spinner animation="border" variant="light" screenReaderText={intl.formatMessage(messages.loading)} />
|
<Button
|
||||||
|
variant="tertiary"
|
||||||
|
className="p-1"
|
||||||
|
aria-label={formatMessage(messages.searchCloseAction)}
|
||||||
|
onClick={() => dialogRef.current.close()}
|
||||||
|
data-testid="courseware-search-close-button"
|
||||||
|
><Icon src={Close} />
|
||||||
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
) : null}
|
</div>
|
||||||
{status === 'error' && (
|
<div className="courseware-search__results" aria-live="polite" data-testid="courseware-search-results">
|
||||||
<Alert className="mt-4" variant="danger" data-testid="courseware-search-error">
|
{status === 'loading' ? (
|
||||||
{intl.formatMessage(messages.searchResultsError)}
|
<div className="courseware-search__spinner" data-testid="courseware-search-spinner">
|
||||||
</Alert>
|
<Spinner animation="border" variant="light" screenReaderText={formatMessage(messages.loading)} />
|
||||||
)}
|
</div>
|
||||||
{status === 'results' ? (
|
) : null}
|
||||||
<>
|
{status === 'error' && (
|
||||||
{total > 0 ? (
|
<Alert className="mt-4" variant="danger" data-testid="courseware-search-error">
|
||||||
<div
|
{formatMessage(messages.searchResultsError)}
|
||||||
className="courseware-search__results-summary"
|
</Alert>
|
||||||
aria-live="polite"
|
)}
|
||||||
aria-relevant="all"
|
{status === 'results' ? (
|
||||||
aria-atomic="true"
|
<>
|
||||||
data-testid="courseware-search-summary"
|
{total > 0 ? (
|
||||||
>{intl.formatMessage(messages.searchResultsLabel, { total, keyword: lastSearchKeyword })}
|
<div
|
||||||
</div>
|
className="courseware-search__results-summary"
|
||||||
) : null}
|
aria-relevant="all"
|
||||||
<CoursewareSearchResultsFilterContainer />
|
aria-atomic="true"
|
||||||
</>
|
data-testid="courseware-search-summary"
|
||||||
) : null}
|
>{formatMessage(messages.searchResultsLabel, { total, keyword: lastSearchKeyword })}
|
||||||
|
</div>
|
||||||
|
) : null}
|
||||||
|
<CoursewareSearchResultsFilterContainer />
|
||||||
|
</>
|
||||||
|
) : null}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</dialog>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
CoursewareSearch.propTypes = {
|
export default CoursewareSearch;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(CoursewareSearch);
|
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import {
|
|||||||
screen,
|
screen,
|
||||||
waitFor,
|
waitFor,
|
||||||
fireEvent,
|
fireEvent,
|
||||||
|
within,
|
||||||
} from '../../setupTest';
|
} from '../../setupTest';
|
||||||
import { CoursewareSearch } from './index';
|
import { CoursewareSearch } from './index';
|
||||||
import { useElementBoundingBox, useLockScroll, useCoursewareSearchParams } from './hooks';
|
import { useElementBoundingBox, useLockScroll, useCoursewareSearchParams } from './hooks';
|
||||||
@@ -19,6 +20,7 @@ import { updateModel, useModel } from '../../generic/model-store';
|
|||||||
|
|
||||||
jest.mock('./hooks');
|
jest.mock('./hooks');
|
||||||
jest.mock('../../generic/model-store', () => ({
|
jest.mock('../../generic/model-store', () => ({
|
||||||
|
...jest.requireActual('../../generic/model-store'),
|
||||||
updateModel: jest.fn(),
|
updateModel: jest.fn(),
|
||||||
useModel: jest.fn(),
|
useModel: jest.fn(),
|
||||||
}));
|
}));
|
||||||
@@ -56,7 +58,7 @@ const defaultProps = {
|
|||||||
total: 0,
|
total: 0,
|
||||||
};
|
};
|
||||||
|
|
||||||
const coursewareSearch = {
|
const defaultSearchParams = {
|
||||||
query: '',
|
query: '',
|
||||||
filter: '',
|
filter: '',
|
||||||
setQuery: jest.fn(),
|
setQuery: jest.fn(),
|
||||||
@@ -96,14 +98,20 @@ const mockModels = ((props = defaultProps) => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const mockSearchParams = ((props = coursewareSearch) => {
|
const mockSearchParams = ((params) => {
|
||||||
|
const props = { ...defaultSearchParams, ...params };
|
||||||
useCoursewareSearchParams.mockReturnValue(props);
|
useCoursewareSearchParams.mockReturnValue(props);
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('CoursewareSearch', () => {
|
describe('CoursewareSearch', () => {
|
||||||
beforeAll(initializeMockApp);
|
beforeAll(() => initializeMockApp());
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
|
mockModels();
|
||||||
|
mockSearchParams();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
jest.clearAllMocks();
|
jest.clearAllMocks();
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -113,27 +121,22 @@ describe('CoursewareSearch', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should use useElementBoundingBox() and useLockScroll() hooks', () => {
|
it('should use useElementBoundingBox() and useLockScroll() hooks', () => {
|
||||||
mockModels();
|
|
||||||
mockSearchParams();
|
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
expect(useElementBoundingBox).toBeCalledTimes(1);
|
expect(useElementBoundingBox).toHaveBeenCalledTimes(1);
|
||||||
expect(useLockScroll).toBeCalledTimes(1);
|
expect(useLockScroll).toHaveBeenCalledTimes(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should have a "--modal-top-position" CSS variable matching the CourseTabsNavigation top position', () => {
|
it('should have a "--modal-top-position" CSS variable matching the CourseTabsNavigation top position', () => {
|
||||||
mockModels();
|
|
||||||
mockSearchParams();
|
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
const section = screen.getByTestId('courseware-search-section');
|
const section = screen.getByTestId('courseware-search-dialog');
|
||||||
expect(section.style.getPropertyValue('--modal-top-position')).toBe(`${tabsTopPosition}px`);
|
expect(section.style.getPropertyValue('--modal-top-position')).toBe(`${tabsTopPosition}px`);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when clicking on the "Close" button', () => {
|
describe('when clicking on the "Close" button', () => {
|
||||||
it('should dispatch setShowSearch(false)', async () => {
|
it('should close the dialog', async () => {
|
||||||
mockModels();
|
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
await waitFor(() => {
|
await waitFor(() => {
|
||||||
@@ -141,7 +144,8 @@ describe('CoursewareSearch', () => {
|
|||||||
fireEvent.click(close);
|
fireEvent.click(close);
|
||||||
});
|
});
|
||||||
|
|
||||||
expect(setShowSearch).toBeCalledWith(false);
|
expect(HTMLDialogElement.prototype.close).toHaveBeenCalled();
|
||||||
|
expect(setShowSearch).toHaveBeenCalledWith(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -149,29 +153,24 @@ describe('CoursewareSearch', () => {
|
|||||||
it('should use "--modal-top-position: 0" if nce element is not present', () => {
|
it('should use "--modal-top-position: 0" if nce element is not present', () => {
|
||||||
useElementBoundingBox.mockImplementation(() => undefined);
|
useElementBoundingBox.mockImplementation(() => undefined);
|
||||||
|
|
||||||
mockModels();
|
|
||||||
mockSearchParams();
|
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
const section = screen.getByTestId('courseware-search-section');
|
const section = screen.getByTestId('courseware-search-dialog');
|
||||||
expect(section.style.getPropertyValue('--modal-top-position')).toBe('0');
|
expect(section.style.getPropertyValue('--modal-top-position')).toBe('0');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when passing extra props', () => {
|
describe('when passing extra props', () => {
|
||||||
it('should pass on extra props to section element', () => {
|
it('should pass on extra props to section element', () => {
|
||||||
mockModels();
|
|
||||||
mockSearchParams();
|
|
||||||
renderComponent({ foo: 'bar' });
|
renderComponent({ foo: 'bar' });
|
||||||
|
|
||||||
const section = screen.getByTestId('courseware-search-section');
|
const section = screen.getByTestId('courseware-search-dialog');
|
||||||
expect(section).toHaveAttribute('foo', 'bar');
|
expect(section).toHaveAttribute('foo', 'bar');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when submitting an empty search', () => {
|
describe('when submitting an empty search', () => {
|
||||||
it('should clear the search by dispatch updateModel', async () => {
|
it('should clear the search by dispatch updateModel', async () => {
|
||||||
mockModels();
|
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
await waitFor(() => {
|
await waitFor(() => {
|
||||||
@@ -203,7 +202,6 @@ describe('CoursewareSearch', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('should call searchCourseContent', async () => {
|
it('should call searchCourseContent', async () => {
|
||||||
mockModels();
|
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
const searchKeyword = 'course';
|
const searchKeyword = 'course';
|
||||||
@@ -246,19 +244,23 @@ describe('CoursewareSearch', () => {
|
|||||||
expect(screen.queryByTestId('courseware-search-summary')).not.toBeInTheDocument();
|
expect(screen.queryByTestId('courseware-search-summary')).not.toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should show a summary for the results', () => {
|
it('should show a summary for the results within a container with aria-live="polite"', () => {
|
||||||
mockModels({
|
mockModels({
|
||||||
searchKeyword: 'fubar',
|
searchKeyword: 'fubar',
|
||||||
total: 1,
|
total: 1,
|
||||||
});
|
});
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
|
||||||
expect(screen.queryByTestId('courseware-search-summary').textContent).toBe('Results for "fubar":');
|
const results = screen.queryByTestId('courseware-search-results');
|
||||||
|
|
||||||
|
expect(results).toHaveAttribute('aria-live', 'polite');
|
||||||
|
expect(within(results).queryByTestId('courseware-search-summary').textContent).toBe('Results for "fubar":');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when clearing the search input', () => {
|
describe('when clearing the search input', () => {
|
||||||
it('should clear the search by dispatch updateModel', async () => {
|
it('should clear the search by dispatch updateModel', async () => {
|
||||||
|
mockSearchParams({ query: 'fubar' });
|
||||||
mockModels({
|
mockModels({
|
||||||
searchKeyword: 'fubar',
|
searchKeyword: 'fubar',
|
||||||
total: 2,
|
total: 2,
|
||||||
|
|||||||
@@ -1,15 +1,14 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
|
|
||||||
const CoursewareSearchEmpty = ({ intl }) => (
|
const CoursewareSearchEmpty = () => {
|
||||||
<div className="courseware-search-results">
|
const intl = useIntl();
|
||||||
<p className="courseware-search-results__empty" data-testid="no-results">{intl.formatMessage(messages.searchResultsNone)}</p>
|
return (
|
||||||
</div>
|
<div className="courseware-search-results">
|
||||||
);
|
<p className="courseware-search-results__empty" data-testid="no-results">{intl.formatMessage(messages.searchResultsNone)}</p>
|
||||||
|
</div>
|
||||||
CoursewareSearchEmpty.propTypes = {
|
);
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(CoursewareSearchEmpty);
|
export default CoursewareSearchEmpty;
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import {
|
|||||||
screen,
|
screen,
|
||||||
} from '../../setupTest';
|
} from '../../setupTest';
|
||||||
import CoursewareSearchEmpty from './CoursewareSearchEmpty';
|
import CoursewareSearchEmpty from './CoursewareSearchEmpty';
|
||||||
|
import messages from './messages';
|
||||||
|
|
||||||
function renderComponent() {
|
function renderComponent() {
|
||||||
const { container } = render(<CoursewareSearchEmpty />);
|
const { container } = render(<CoursewareSearchEmpty />);
|
||||||
@@ -16,9 +17,12 @@ describe('CoursewareSearchEmpty', () => {
|
|||||||
initializeMockApp();
|
initializeMockApp();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should match the snapshot', () => {
|
it('render empty results text and corresponding classes', () => {
|
||||||
renderComponent();
|
renderComponent();
|
||||||
|
const emptyText = screen.getByText(messages.searchResultsNone.defaultMessage);
|
||||||
expect(screen.getByTestId('no-results')).toMatchSnapshot();
|
expect(emptyText).toBeInTheDocument();
|
||||||
|
expect(emptyText).toHaveClass('courseware-search-results__empty');
|
||||||
|
expect(emptyText).toHaveAttribute('data-testid', 'no-results');
|
||||||
|
expect(emptyText.parentElement).toHaveClass('courseware-search-results');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,43 +1,44 @@
|
|||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { SearchField } from '@openedx/paragon';
|
import { SearchField } from '@openedx/paragon';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
|
|
||||||
const CoursewareSearchForm = ({
|
const CoursewareSearchForm = ({
|
||||||
intl,
|
|
||||||
searchTerm,
|
searchTerm,
|
||||||
onSubmit,
|
onSubmit,
|
||||||
onChange,
|
onChange,
|
||||||
placeholder,
|
placeholder,
|
||||||
}) => (
|
}) => {
|
||||||
<SearchField.Advanced
|
const { formatMessage } = useIntl();
|
||||||
value={searchTerm}
|
|
||||||
onSubmit={onSubmit}
|
return (
|
||||||
onChange={onChange}
|
<SearchField.Advanced
|
||||||
submitButtonLocation="external"
|
value={searchTerm}
|
||||||
className="courseware-search-form"
|
onSubmit={onSubmit}
|
||||||
screenReaderText={{
|
onChange={onChange}
|
||||||
label: intl.formatMessage(messages.searchSubmitLabel),
|
|
||||||
clearButton: intl.formatMessage(messages.searchClearAction),
|
|
||||||
submitButton: null, // Remove the sr-only label in the button.
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div className="pgn__searchfield_wrapper" data-testid="courseware-search-form">
|
|
||||||
<SearchField.Label />
|
|
||||||
<SearchField.Input placeholder={placeholder} autoFocus />
|
|
||||||
<SearchField.ClearButton />
|
|
||||||
</div>
|
|
||||||
<SearchField.SubmitButton
|
|
||||||
buttonText={intl.formatMessage(messages.searchSubmitLabel)}
|
|
||||||
submitButtonLocation="external"
|
submitButtonLocation="external"
|
||||||
data-testid="courseware-search-form-submit"
|
className="courseware-search-form"
|
||||||
/>
|
screenReaderText={{
|
||||||
</SearchField.Advanced>
|
label: formatMessage(messages.searchSubmitLabel),
|
||||||
);
|
clearButton: formatMessage(messages.searchClearAction),
|
||||||
|
submitButton: null, // Remove the sr-only label in the button.
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<div className="pgn__searchfield_wrapper" data-testid="courseware-search-form">
|
||||||
|
<SearchField.Label />
|
||||||
|
<SearchField.Input placeholder={placeholder} autoFocus />
|
||||||
|
<SearchField.ClearButton />
|
||||||
|
</div>
|
||||||
|
<SearchField.SubmitButton
|
||||||
|
buttonText={formatMessage(messages.searchSubmitLabel)}
|
||||||
|
submitButtonLocation="external"
|
||||||
|
data-testid="courseware-search-form-submit"
|
||||||
|
/>
|
||||||
|
</SearchField.Advanced>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
CoursewareSearchForm.propTypes = {
|
CoursewareSearchForm.propTypes = {
|
||||||
intl: intlShape.isRequired,
|
|
||||||
searchTerm: PropTypes.string,
|
searchTerm: PropTypes.string,
|
||||||
onSubmit: PropTypes.func,
|
onSubmit: PropTypes.func,
|
||||||
onChange: PropTypes.func,
|
onChange: PropTypes.func,
|
||||||
@@ -51,4 +52,4 @@ CoursewareSearchForm.defaultProps = {
|
|||||||
placeholder: undefined,
|
placeholder: undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(CoursewareSearchForm);
|
export default CoursewareSearchForm;
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import {
|
|||||||
import CoursewareSearchResults from './CoursewareSearchResults';
|
import CoursewareSearchResults from './CoursewareSearchResults';
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import searchResultsFactory from './test-data/search-results-factory';
|
import searchResultsFactory from './test-data/search-results-factory';
|
||||||
|
import * as mock from './test-data/mocked-response.json';
|
||||||
|
|
||||||
jest.mock('react-redux');
|
jest.mock('react-redux');
|
||||||
|
|
||||||
@@ -34,8 +35,53 @@ describe('CoursewareSearchResults', () => {
|
|||||||
renderComponent({ results });
|
renderComponent({ results });
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should match the snapshot', () => {
|
it('should render complete list', () => {
|
||||||
expect(screen.getByTestId('search-results')).toMatchSnapshot();
|
const courses = screen.getAllByRole('link');
|
||||||
|
expect(courses.length).toBe(mock.results.length);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render correct link for internal course', () => {
|
||||||
|
const courses = screen.getAllByRole('link');
|
||||||
|
const firstCourse = courses[0];
|
||||||
|
const firstCourseTitle = firstCourse.querySelector('.courseware-search-results__title span');
|
||||||
|
expect(firstCourseTitle.innerHTML).toEqual(mock.results[0].data.content.display_name);
|
||||||
|
expect(firstCourse.href).toContain(mock.results[0].data.url);
|
||||||
|
expect(firstCourse).not.toHaveAttribute('target', '_blank');
|
||||||
|
expect(firstCourse).not.toHaveAttribute('rel', 'nofollow');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render correct link if is External url course', () => {
|
||||||
|
const courses = screen.getAllByRole('link');
|
||||||
|
const externalCourse = courses[courses.length - 1];
|
||||||
|
const externalCourseTitle = externalCourse.querySelector('.courseware-search-results__title span');
|
||||||
|
expect(externalCourseTitle.innerHTML).toEqual(mock.results[mock.results.length - 1].data.content.display_name);
|
||||||
|
expect(externalCourse.href).toContain(mock.results[mock.results.length - 1].data.url);
|
||||||
|
expect(externalCourse).toHaveAttribute('target', '_blank');
|
||||||
|
expect(externalCourse).toHaveAttribute('rel', 'nofollow');
|
||||||
|
const icon = externalCourse.querySelector('svg');
|
||||||
|
expect(icon).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render location breadcrumbs', () => {
|
||||||
|
const breadcrumbs = screen.getAllByText(mock.results[0].data.location[0]);
|
||||||
|
expect(breadcrumbs.length).toBeGreaterThan(0);
|
||||||
|
const firstBreadcrumb = breadcrumbs[0].closest('li');
|
||||||
|
expect(firstBreadcrumb).toBeInTheDocument();
|
||||||
|
expect(firstBreadcrumb.querySelector('div').textContent).toBe(mock.results[0].data.location[0]);
|
||||||
|
expect(firstBreadcrumb.nextSibling.querySelector('div').textContent).toBe(mock.results[0].data.location[1]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('when results are provided with content hits', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
const { results } = searchResultsFactory('Passing');
|
||||||
|
renderComponent({ results });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should render content hits', () => {
|
||||||
|
const contentHits = screen.getByText('1');
|
||||||
|
expect(contentHits).toBeInTheDocument();
|
||||||
|
expect(contentHits.tagName).toBe('EM');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,15 +1,14 @@
|
|||||||
import React, { useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import { Button, Icon } from '@openedx/paragon';
|
import { Button } from '@openedx/paragon';
|
||||||
import { Search } from '@openedx/paragon/icons';
|
import { ManageSearch } from '@openedx/paragon/icons';
|
||||||
import { useDispatch } from 'react-redux';
|
import { useDispatch } from 'react-redux';
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import { useCoursewareSearchFeatureFlag, useCoursewareSearchParams } from './hooks';
|
import { useCoursewareSearchFeatureFlag, useCoursewareSearchParams } from './hooks';
|
||||||
import { setShowSearch } from '../data/slice';
|
import { setShowSearch } from '../data/slice';
|
||||||
|
|
||||||
const CoursewareSearchToggle = ({
|
const CoursewareSearchToggle = () => {
|
||||||
intl,
|
const intl = useIntl();
|
||||||
}) => {
|
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const enabled = useCoursewareSearchFeatureFlag();
|
const enabled = useCoursewareSearchFeatureFlag();
|
||||||
const { query } = useCoursewareSearchParams();
|
const { query } = useCoursewareSearchParams();
|
||||||
@@ -25,23 +24,20 @@ const CoursewareSearchToggle = ({
|
|||||||
if (!enabled) { return null; }
|
if (!enabled) { return null; }
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="courseware-searc-toggle">
|
<div className="courseware-search-toggle">
|
||||||
<Button
|
<Button
|
||||||
variant="tertiary"
|
variant="outline-primary"
|
||||||
size="sm"
|
size="sm"
|
||||||
className="p-1 mt-2 mr-2 rounded-lg"
|
className="p-1 mt-2 mr-2"
|
||||||
aria-label={intl.formatMessage(messages.searchOpenAction)}
|
aria-label={intl.formatMessage(messages.searchOpenAction)}
|
||||||
onClick={handleSearchOpenClick}
|
onClick={handleSearchOpenClick}
|
||||||
data-testid="courseware-search-open-button"
|
data-testid="courseware-search-open-button"
|
||||||
|
iconAfter={ManageSearch}
|
||||||
>
|
>
|
||||||
<Icon src={Search} />
|
{intl.formatMessage(messages.contentSearchButton)}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
CoursewareSearchToggle.propTypes = {
|
export default CoursewareSearchToggle;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(CoursewareSearchToggle);
|
|
||||||
|
|||||||
@@ -1,10 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`CoursewareSearchEmpty should match the snapshot 1`] = `
|
|
||||||
<p
|
|
||||||
class="courseware-search-results__empty"
|
|
||||||
data-testid="no-results"
|
|
||||||
>
|
|
||||||
No results found.
|
|
||||||
</p>
|
|
||||||
`;
|
|
||||||
@@ -1,1238 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`CoursewareSearchResults when list of results is provided should match the snapshot 1`] = `
|
|
||||||
<div
|
|
||||||
class="courseware-search-results"
|
|
||||||
data-testid="search-results"
|
|
||||||
>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@sequential+block@edx_introduction"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M10 4H2v16h20V6H12l-2-2z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Demo Course Overview
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Introduction
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Demo Course Overview
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@5e009378f0b64585baa0a14b155974b9"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Passing a Course
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
1
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
About Exams and Certificates
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
edX Exams
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Passing a Course
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@vertical+block@c7e98fd39a6944edb6b286c32e1150ff"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M10 4H2v16h20V6H12l-2-2z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Passing a Course
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
About Exams and Certificates
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
edX Exams
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Passing a Course
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@0d759dee4f9d459c8956136dbde55f02"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Text Input
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Question Styles
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Text input
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@c554538a57664fac80783b99d9d6da7c"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Pointing on a Picture
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Question Styles
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Pointing on a Picture
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@45d46192272c4f6db6b63586520bbdf4"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Getting Answers
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
About Exams and Certificates
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
edX Exams
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Getting Answers
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@video+block@0b9e39477cf34507a7a48f74be381fdd"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M17 10.5V6H3v12h14v-4.5l4 4v-11l-4 4Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Welcome!
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
30
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Introduction
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Demo Course Overview
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Introduction: Video and Sequences
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@a0effb954cca4759994f1ac9e9434bf4"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Multiple Choice Questions
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Question Styles
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Multiple Choice Questions
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@75f9562c77bc4858b61f907bb810d974"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Numerical Input
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Question Styles
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Numerical Input
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@video+block@636541acbae448d98ab484b028c9a7f6"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M17 10.5V6H3v12h14v-4.5l4 4v-11l-4 4Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Connecting a Circuit and a Circuit Diagram
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
3
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Lesson 1 - Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Video Presentation Styles
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@python_grader"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
CAPA
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 2: Get Interactive
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Labs and Demos
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Code Grader
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@9cee77a606ea4c1aa5440e0ea5d0f618"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Interactive Questions
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Lesson 1 - Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Interactive Questions
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@030e35c4756a4ddc8d40b95fbbfff4d4"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Blank HTML Page
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
6
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Introduction
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Demo Course Overview
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Introduction: Video and Sequences
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@html_49b4494da2f7"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Discussion Forums
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
5
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 3: Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Lesson 3 - Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Discussion Forums
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@f4a39219742149f781a1dda6f43a623c"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Overall Grade
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
7
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
About Exams and Certificates
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
edX Exams
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Overall Grade Performance
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@87fa6792d79f4862be098e5169e93339"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Blank HTML Page
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
3
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 3: Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Lesson 3 - Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Find Your Study Buddy
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@6018785795994726950614ce7d0f38c5"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Find Your Study Buddy
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
3
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 3: Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Find Your Study Buddy
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Homework - Find Your Study Buddy
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@f9f3a25e7bab46e583fd1fbbd7a2f6a0"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
Be Social
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
4
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 3: Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Lesson 3 - Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Be Social
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@8293139743f34377817d537b69911530"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
EdX Exams
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
4
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
About Exams and Certificates
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
edX Exams
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
EdX Exams
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@9d5104b502f24ee89c3d2f4ce9d347cf"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M2.5 4v3h5v12h3V7h5V4h-13zm19 5h-9v3h3v7h3v-7h3V9z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
When Are Your Exams?
|
|
||||||
</span>
|
|
||||||
<em>
|
|
||||||
2
|
|
||||||
</em>
|
|
||||||
</div>
|
|
||||||
<ul
|
|
||||||
class="courseware-search-results__breadcrumbs"
|
|
||||||
>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Example Week 1: Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
Lesson 1 - Getting Started
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<div>
|
|
||||||
When Are Your Exams?
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
<a
|
|
||||||
class="courseware-search-results__item"
|
|
||||||
href="https://www.edx.org"
|
|
||||||
rel="nofollow"
|
|
||||||
target="_blank"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__icon"
|
|
||||||
>
|
|
||||||
<span
|
|
||||||
class="pgn__icon"
|
|
||||||
>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
fill="none"
|
|
||||||
focusable="false"
|
|
||||||
height="24"
|
|
||||||
role="img"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
width="24"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
d="M3 3v18h18V3H3Zm11 14H7v-2h7v2Zm3-4H7v-2h10v2Zm0-4H7V7h10v2Z"
|
|
||||||
fill="currentColor"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__info"
|
|
||||||
>
|
|
||||||
<div
|
|
||||||
class="courseware-search-results__title"
|
|
||||||
>
|
|
||||||
<span>
|
|
||||||
External Course Link Test
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
@@ -1,306 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`mapSearchResponse when the response is correct should match snapshot 1`] = `
|
|
||||||
Object {
|
|
||||||
"filters": Array [
|
|
||||||
Object {
|
|
||||||
"count": 7,
|
|
||||||
"key": "capa",
|
|
||||||
"label": "CAPA",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"count": 2,
|
|
||||||
"key": "sequence",
|
|
||||||
"label": "Sequence",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"count": 9,
|
|
||||||
"key": "text",
|
|
||||||
"label": "Text",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"count": 1,
|
|
||||||
"key": "unknown",
|
|
||||||
"label": "Unknown",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"count": 2,
|
|
||||||
"key": "video",
|
|
||||||
"label": "Video",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"maxScore": 3.4545178,
|
|
||||||
"ms": 5,
|
|
||||||
"results": Array [
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@sequential+block@edx_introduction",
|
|
||||||
"location": Array [
|
|
||||||
"Introduction",
|
|
||||||
"Demo Course Overview",
|
|
||||||
],
|
|
||||||
"score": 3.4545178,
|
|
||||||
"title": "Demo Course Overview",
|
|
||||||
"type": "sequence",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@sequential+block@edx_introduction",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@5e009378f0b64585baa0a14b155974b9",
|
|
||||||
"location": Array [
|
|
||||||
"About Exams and Certificates",
|
|
||||||
"edX Exams",
|
|
||||||
"Passing a Course",
|
|
||||||
],
|
|
||||||
"score": 3.4545178,
|
|
||||||
"title": "Passing a Course",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@5e009378f0b64585baa0a14b155974b9",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@vertical+block@c7e98fd39a6944edb6b286c32e1150ff",
|
|
||||||
"location": Array [
|
|
||||||
"About Exams and Certificates",
|
|
||||||
"edX Exams",
|
|
||||||
"Passing a Course",
|
|
||||||
],
|
|
||||||
"score": 3.4545178,
|
|
||||||
"title": "Passing a Course",
|
|
||||||
"type": "sequence",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@vertical+block@c7e98fd39a6944edb6b286c32e1150ff",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@0d759dee4f9d459c8956136dbde55f02",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Homework - Question Styles",
|
|
||||||
"Text input",
|
|
||||||
],
|
|
||||||
"score": 1.5874016,
|
|
||||||
"title": "Text Input",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@0d759dee4f9d459c8956136dbde55f02",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@c554538a57664fac80783b99d9d6da7c",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Homework - Question Styles",
|
|
||||||
"Pointing on a Picture",
|
|
||||||
],
|
|
||||||
"score": 1.5499392,
|
|
||||||
"title": "Pointing on a Picture",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@c554538a57664fac80783b99d9d6da7c",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@45d46192272c4f6db6b63586520bbdf4",
|
|
||||||
"location": Array [
|
|
||||||
"About Exams and Certificates",
|
|
||||||
"edX Exams",
|
|
||||||
"Getting Answers",
|
|
||||||
],
|
|
||||||
"score": 1.5003732,
|
|
||||||
"title": "Getting Answers",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@45d46192272c4f6db6b63586520bbdf4",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@video+block@0b9e39477cf34507a7a48f74be381fdd",
|
|
||||||
"location": Array [
|
|
||||||
"Introduction",
|
|
||||||
"Demo Course Overview",
|
|
||||||
"Introduction: Video and Sequences",
|
|
||||||
],
|
|
||||||
"score": 1.4792063,
|
|
||||||
"title": "Welcome!",
|
|
||||||
"type": "video",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@video+block@0b9e39477cf34507a7a48f74be381fdd",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@a0effb954cca4759994f1ac9e9434bf4",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Homework - Question Styles",
|
|
||||||
"Multiple Choice Questions",
|
|
||||||
],
|
|
||||||
"score": 1.4341705,
|
|
||||||
"title": "Multiple Choice Questions",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@a0effb954cca4759994f1ac9e9434bf4",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@75f9562c77bc4858b61f907bb810d974",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Homework - Question Styles",
|
|
||||||
"Numerical Input",
|
|
||||||
],
|
|
||||||
"score": 1.2987298,
|
|
||||||
"title": "Numerical Input",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@75f9562c77bc4858b61f907bb810d974",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@video+block@636541acbae448d98ab484b028c9a7f6",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Lesson 1 - Getting Started",
|
|
||||||
"Video Presentation Styles",
|
|
||||||
],
|
|
||||||
"score": 1.1870136,
|
|
||||||
"title": "Connecting a Circuit and a Circuit Diagram",
|
|
||||||
"type": "video",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@video+block@636541acbae448d98ab484b028c9a7f6",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@python_grader",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 2: Get Interactive",
|
|
||||||
"Homework - Labs and Demos",
|
|
||||||
"Code Grader",
|
|
||||||
],
|
|
||||||
"score": 1.0107487,
|
|
||||||
"title": "CAPA",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@python_grader",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@problem+block@9cee77a606ea4c1aa5440e0ea5d0f618",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Lesson 1 - Getting Started",
|
|
||||||
"Interactive Questions",
|
|
||||||
],
|
|
||||||
"score": 0.96387196,
|
|
||||||
"title": "Interactive Questions",
|
|
||||||
"type": "capa",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@problem+block@9cee77a606ea4c1aa5440e0ea5d0f618",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@030e35c4756a4ddc8d40b95fbbfff4d4",
|
|
||||||
"location": Array [
|
|
||||||
"Introduction",
|
|
||||||
"Demo Course Overview",
|
|
||||||
"Introduction: Video and Sequences",
|
|
||||||
],
|
|
||||||
"score": 0.8844358,
|
|
||||||
"title": "Blank HTML Page",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@030e35c4756a4ddc8d40b95fbbfff4d4",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@html_49b4494da2f7",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 3: Be Social",
|
|
||||||
"Lesson 3 - Be Social",
|
|
||||||
"Discussion Forums",
|
|
||||||
],
|
|
||||||
"score": 0.8803684,
|
|
||||||
"title": "Discussion Forums",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@html_49b4494da2f7",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@f4a39219742149f781a1dda6f43a623c",
|
|
||||||
"location": Array [
|
|
||||||
"About Exams and Certificates",
|
|
||||||
"edX Exams",
|
|
||||||
"Overall Grade Performance",
|
|
||||||
],
|
|
||||||
"score": 0.87981963,
|
|
||||||
"title": "Overall Grade",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@f4a39219742149f781a1dda6f43a623c",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@87fa6792d79f4862be098e5169e93339",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 3: Be Social",
|
|
||||||
"Lesson 3 - Be Social",
|
|
||||||
"Homework - Find Your Study Buddy",
|
|
||||||
],
|
|
||||||
"score": 0.84284115,
|
|
||||||
"title": "Blank HTML Page",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@87fa6792d79f4862be098e5169e93339",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@6018785795994726950614ce7d0f38c5",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 3: Be Social",
|
|
||||||
"Homework - Find Your Study Buddy",
|
|
||||||
"Homework - Find Your Study Buddy",
|
|
||||||
],
|
|
||||||
"score": 0.84284115,
|
|
||||||
"title": "Find Your Study Buddy",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@6018785795994726950614ce7d0f38c5",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@f9f3a25e7bab46e583fd1fbbd7a2f6a0",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 3: Be Social",
|
|
||||||
"Lesson 3 - Be Social",
|
|
||||||
"Be Social",
|
|
||||||
],
|
|
||||||
"score": 0.84210813,
|
|
||||||
"title": "Be Social",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@f9f3a25e7bab46e583fd1fbbd7a2f6a0",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@8293139743f34377817d537b69911530",
|
|
||||||
"location": Array [
|
|
||||||
"About Exams and Certificates",
|
|
||||||
"edX Exams",
|
|
||||||
"EdX Exams",
|
|
||||||
],
|
|
||||||
"score": 0.8306555,
|
|
||||||
"title": "EdX Exams",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@8293139743f34377817d537b69911530",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@html+block@9d5104b502f24ee89c3d2f4ce9d347cf",
|
|
||||||
"location": Array [
|
|
||||||
"Example Week 1: Getting Started",
|
|
||||||
"Lesson 1 - Getting Started",
|
|
||||||
"When Are Your Exams? ",
|
|
||||||
],
|
|
||||||
"score": 0.82610154,
|
|
||||||
"title": "When Are Your Exams? ",
|
|
||||||
"type": "text",
|
|
||||||
"url": "/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+DemoX+Demo_Course+type@html+block@9d5104b502f24ee89c3d2f4ce9d347cf",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"contentHits": 0,
|
|
||||||
"id": "random-element-id",
|
|
||||||
"location": null,
|
|
||||||
"score": 0.82610154,
|
|
||||||
"title": "External Course Link Test",
|
|
||||||
"type": "unknown",
|
|
||||||
"url": "https://www.edx.org",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"total": 29,
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
@@ -5,13 +5,25 @@
|
|||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
border-top: 1px solid $light-300;
|
width: 100%;
|
||||||
z-index: $zindex-modal; // Bootstrap's z-index layer for Modals.
|
height: 100%;
|
||||||
|
max-width: none;
|
||||||
|
margin: 0;
|
||||||
|
border-top: 1px solid var(--pgn-color-light-300);
|
||||||
|
z-index: var(--pgn-elevation-modal-zindex); // Bootstrap's z-index layer for Modals.
|
||||||
|
|
||||||
|
&__form {
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
.h2 {
|
||||||
|
margin-right: 2.5rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
&__close {
|
&__close {
|
||||||
position: absolute !important; // For some reason it gets overridden
|
position: absolute !important; // For some reason it gets overridden
|
||||||
top: 0.5rem;
|
top: 0;
|
||||||
right: 1rem;
|
right: 0;
|
||||||
font-size: 1.5rem;
|
font-size: 1.5rem;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
}
|
}
|
||||||
@@ -35,7 +47,7 @@
|
|||||||
|
|
||||||
&__results-summary {
|
&__results-summary {
|
||||||
font-size: .9rem;
|
font-size: .9rem;
|
||||||
color: $gray-500;
|
color: var(--pgn-color-gray-500);
|
||||||
padding: 1rem 0 .5rem;
|
padding: 1rem 0 .5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -50,7 +62,7 @@
|
|||||||
margin-top: 1.5rem;
|
margin-top: 1.5rem;
|
||||||
|
|
||||||
&__empty {
|
&__empty {
|
||||||
color: $gray-500;
|
color: var(--pgn-color-gray-500);
|
||||||
padding: 6rem 0;
|
padding: 6rem 0;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
@@ -64,17 +76,17 @@
|
|||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
background: $light-300;
|
background: var(--pgn-color-light-300);
|
||||||
}
|
}
|
||||||
|
|
||||||
&:not(:first-child) {
|
&:not(:first-child) {
|
||||||
border-top: 1px solid $light-300;
|
border-top: 1px solid var(--pgn-color-light-300);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&__icon {
|
&__icon {
|
||||||
padding: 0.375rem 0 0 0.375rem;
|
padding: 0.375rem 0 0 0.375rem;
|
||||||
color: $gray-300;
|
color: var(--pgn-color-gray-300);
|
||||||
}
|
}
|
||||||
|
|
||||||
&__info {
|
&__info {
|
||||||
@@ -87,7 +99,7 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
line-height: 2.5;
|
line-height: 2.5;
|
||||||
font-size: 0.875rem;
|
font-size: 0.875rem;
|
||||||
color: $black;
|
color: var(--pgn-color-black);
|
||||||
|
|
||||||
> span {
|
> span {
|
||||||
display: block;
|
display: block;
|
||||||
@@ -101,7 +113,7 @@
|
|||||||
font-variant-numeric: lining-nums tabular-nums;
|
font-variant-numeric: lining-nums tabular-nums;
|
||||||
min-width: 1.25rem;
|
min-width: 1.25rem;
|
||||||
line-height: 1rem;
|
line-height: 1rem;
|
||||||
background: $light-300;
|
background: var(--pgn-color-light-300);
|
||||||
border-radius: 99rem;
|
border-radius: 99rem;
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
margin-left: 0.375rem;
|
margin-left: 0.375rem;
|
||||||
@@ -113,7 +125,7 @@
|
|||||||
&__breadcrumbs {
|
&__breadcrumbs {
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 1.25rem;
|
gap: 1.25rem;
|
||||||
color: $gray-500;
|
color: var(--pgn-color-gray-500);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
list-style: none;
|
list-style: none;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
@@ -144,17 +156,24 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.courseware-search-results-tabs {
|
.courseware-search-results-tabs {
|
||||||
border-bottom-color: $gray-400 !important;
|
border-bottom-color: var(--pgn-color-gray-400) !important;
|
||||||
|
|
||||||
&.nav-tabs .nav-link.active {
|
&.nav-tabs .nav-link.active {
|
||||||
border-bottom-width: 4px !important;
|
border-bottom-width: 4px !important;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: map-get($grid-breakpoints, 'md')) {
|
@media (--pgn-size-breakpoint-min-width-md) {
|
||||||
.courseware-search__content {
|
.courseware-search {
|
||||||
padding-top: 8rem;
|
&__close {
|
||||||
|
right: -2.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
&__content {
|
||||||
|
padding-top: 8rem;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
body._search-no-scroll {
|
body._search-no-scroll {
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { renderHook, act } from '@testing-library/react-hooks';
|
import { renderHook, act, waitFor } from '@testing-library/react';
|
||||||
import { useParams, useSearchParams } from 'react-router-dom';
|
import { useParams, useSearchParams } from 'react-router-dom';
|
||||||
import { useSelector } from 'react-redux';
|
import { useSelector } from 'react-redux';
|
||||||
import { fetchCoursewareSearchSettings } from '../data/thunks';
|
import { fetchCoursewareSearchSettings } from '../data/thunks';
|
||||||
@@ -38,13 +38,13 @@ describe('CoursewareSearch Hooks', () => {
|
|||||||
|
|
||||||
it('should return true if feature is enabled', async () => {
|
it('should return true if feature is enabled', async () => {
|
||||||
const hook = await renderTestHook();
|
const hook = await renderTestHook();
|
||||||
await hook.waitFor(() => expect(fetchCoursewareSearchSettings).toBeCalledTimes(1));
|
await waitFor(() => expect(fetchCoursewareSearchSettings).toBeCalledTimes(1));
|
||||||
expect(hook.result.current).toBe(true);
|
expect(hook.result.current).toBe(true);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return false if feature is disabled', async () => {
|
it('should return false if feature is disabled', async () => {
|
||||||
const hook = await renderTestHook(false);
|
const hook = await renderTestHook(false);
|
||||||
await hook.waitFor(() => expect(fetchCoursewareSearchSettings).toBeCalledTimes(1));
|
await waitFor(() => expect(fetchCoursewareSearchSettings).toBeCalledTimes(1));
|
||||||
expect(hook.result.current).toBe(false);
|
expect(hook.result.current).toBe(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -125,7 +125,7 @@ describe('CoursewareSearch Hooks', () => {
|
|||||||
it('should return the element bounding box', async () => {
|
it('should return the element bounding box', async () => {
|
||||||
const hook = await renderTestHook({ elementId: 'test', mockedInfo });
|
const hook = await renderTestHook({ elementId: 'test', mockedInfo });
|
||||||
|
|
||||||
hook.waitFor(() => expect(getBoundingClientRectSpy).toHaveBeenCalled());
|
await waitFor(() => expect(getBoundingClientRectSpy).toHaveBeenCalled());
|
||||||
|
|
||||||
expect(hook.result.current).toEqual(mockedInfo);
|
expect(hook.result.current).toEqual(mockedInfo);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ describe('mapSearchResponse', () => {
|
|||||||
response = mapSearchResponse(camelCaseObject(mockedResponse));
|
response = mapSearchResponse(camelCaseObject(mockedResponse));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should match snapshot', () => {
|
it('should match number of results', () => {
|
||||||
expect(response).toMatchSnapshot();
|
expect(response.results.length).toBe(mockedResponse.results.length);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should match expected filters', () => {
|
it('should match expected filters', () => {
|
||||||
@@ -24,6 +24,25 @@ describe('mapSearchResponse', () => {
|
|||||||
];
|
];
|
||||||
expect(response.filters).toEqual(expectedFilters);
|
expect(response.filters).toEqual(expectedFilters);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should match expected results', () => {
|
||||||
|
const mockFirstResult = mockedResponse.results[0];
|
||||||
|
const expectedFirstResult = {
|
||||||
|
id: mockFirstResult.data.id,
|
||||||
|
title: mockFirstResult.data.content.display_name,
|
||||||
|
type: mockFirstResult.data.content_type.toLowerCase(),
|
||||||
|
location: mockFirstResult.data.location,
|
||||||
|
url: mockFirstResult.data.url,
|
||||||
|
contentHits: 0,
|
||||||
|
score: mockFirstResult.score,
|
||||||
|
};
|
||||||
|
expect(response.results[0]).toEqual(expectedFirstResult);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should match expected ms and max score', () => {
|
||||||
|
expect(response.maxScore).toBe(mockedResponse.max_score);
|
||||||
|
expect(response.ms).toBe(mockedResponse.took);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('when the a keyword is provided', () => {
|
describe('when the a keyword is provided', () => {
|
||||||
|
|||||||
@@ -2,79 +2,84 @@ import { defineMessages } from '@edx/frontend-platform/i18n';
|
|||||||
|
|
||||||
const messages = defineMessages({
|
const messages = defineMessages({
|
||||||
searchOpenAction: {
|
searchOpenAction: {
|
||||||
id: 'learn.coursewareSerch.openAction',
|
id: 'learn.coursewareSearch.openAction',
|
||||||
defaultMessage: 'Search within this course',
|
defaultMessage: 'Search within this course',
|
||||||
description: 'Aria-label for a button that will pop up Courseware Search.',
|
description: 'Aria-label for a button that will pop up Courseware Search.',
|
||||||
},
|
},
|
||||||
|
contentSearchButton: {
|
||||||
|
id: 'learn.coursewareSearch.contentSearchButton',
|
||||||
|
defaultMessage: 'Content search',
|
||||||
|
description: 'Text for a button that will pop up Courseware Search.',
|
||||||
|
},
|
||||||
searchSubmitLabel: {
|
searchSubmitLabel: {
|
||||||
id: 'learn.coursewareSerch.submitLabel',
|
id: 'learn.coursewareSearch.submitLabel',
|
||||||
defaultMessage: 'Search',
|
defaultMessage: 'Search',
|
||||||
description: 'Button label that will submit Courseware Search.',
|
description: 'Button label that will submit Courseware Search.',
|
||||||
},
|
},
|
||||||
searchClearAction: {
|
searchClearAction: {
|
||||||
id: 'learn.coursewareSerch.clearAction',
|
id: 'learn.coursewareSearch.clearAction',
|
||||||
defaultMessage: 'Clear search',
|
defaultMessage: 'Clear search',
|
||||||
description: 'Button label that will the current Courseware Search input.',
|
description: 'Button label that will the current Courseware Search input.',
|
||||||
},
|
},
|
||||||
searchCloseAction: {
|
searchCloseAction: {
|
||||||
id: 'learn.coursewareSerch.closeAction',
|
id: 'learn.coursewareSearch.closeAction',
|
||||||
defaultMessage: 'Close the search form',
|
defaultMessage: 'Close the search form',
|
||||||
description: 'Aria-label for a button that will close Courseware Search.',
|
description: 'Aria-label for a button that will close Courseware Search.',
|
||||||
},
|
},
|
||||||
searchModuleTitle: {
|
searchModuleTitle: {
|
||||||
id: 'learn.coursewareSerch.searchModuleTitle',
|
id: 'learn.coursewareSearch.searchModuleTitle',
|
||||||
defaultMessage: 'Search this course',
|
defaultMessage: 'Search this course',
|
||||||
description: 'Title for the Courseware Search module.',
|
description: 'Title for the Courseware Search module.',
|
||||||
},
|
},
|
||||||
searchBarPlaceholderText: {
|
searchBarPlaceholderText: {
|
||||||
id: 'learn.coursewareSerch.searchBarPlaceholderText',
|
id: 'learn.coursewareSearch.searchBarPlaceholderText',
|
||||||
defaultMessage: 'Search',
|
defaultMessage: 'Search',
|
||||||
description: 'Placeholder text for the Courseware Search input control',
|
description: 'Placeholder text for the Courseware Search input control',
|
||||||
},
|
},
|
||||||
loading: {
|
loading: {
|
||||||
id: 'learn.coursewareSerch.loading',
|
id: 'learn.coursewareSearch.loading',
|
||||||
defaultMessage: 'Searching...',
|
defaultMessage: 'Searching...',
|
||||||
description: 'Screen reader text to use on the spinner while the search is performing.',
|
description: 'Screen reader text to use on the spinner while the search is performing.',
|
||||||
},
|
},
|
||||||
searchResultsNone: {
|
searchResultsNone: {
|
||||||
id: 'learn.coursewareSerch.searchResultsNone',
|
id: 'learn.coursewareSearch.searchResultsNone',
|
||||||
defaultMessage: 'No results found.',
|
defaultMessage: 'No results found.',
|
||||||
description: 'Text to show when the Courseware Search found no results matching the criteria.',
|
description: 'Text to show when the Courseware Search found no results matching the criteria.',
|
||||||
},
|
},
|
||||||
searchResultsLabel: {
|
searchResultsLabel: {
|
||||||
id: 'learn.coursewareSerch.searchResultsLabel',
|
id: 'learn.coursewareSearch.searchResultsLabel',
|
||||||
defaultMessage: 'Results for "{keyword}":',
|
defaultMessage: 'Results for "{keyword}":',
|
||||||
description: 'Text to show above the search results response list.',
|
description: 'Text to show above the search results response list.',
|
||||||
},
|
},
|
||||||
searchResultsError: {
|
searchResultsError: {
|
||||||
id: 'learn.coursewareSerch.searchResultsError',
|
id: 'learn.coursewareSearch.searchResultsError',
|
||||||
defaultMessage: 'There was an error on the search process. Please try again in a few minutes. If the problem persists, please contact the support team.',
|
defaultMessage: 'There was an error on the search process. Please try again in a few minutes. If the problem persists, please contact the support team.',
|
||||||
description: 'Error message to show to the users when there\'s an error with the endpoint or the returned payload format.',
|
description: 'Error message to show to the users when there\'s an error with the endpoint or the returned payload format.',
|
||||||
},
|
},
|
||||||
|
|
||||||
// These are translations for labeling the filters
|
// These are translations for labeling the filters
|
||||||
'filter:all': {
|
'filter:all': {
|
||||||
id: 'learn.coursewareSerch.filter:all',
|
id: 'learn.coursewareSearch.filter:all',
|
||||||
defaultMessage: 'All content',
|
defaultMessage: 'All content',
|
||||||
description: 'Label for the search results filter that shows all content (no filter).',
|
description: 'Label for the search results filter that shows all content (no filter).',
|
||||||
},
|
},
|
||||||
'filter:text': {
|
'filter:text': {
|
||||||
id: 'learn.coursewareSerch.filter:text',
|
id: 'learn.coursewareSearch.filter:text',
|
||||||
defaultMessage: 'Text',
|
defaultMessage: 'Text',
|
||||||
description: 'Label for the search results filter that shows results with text content.',
|
description: 'Label for the search results filter that shows results with text content.',
|
||||||
},
|
},
|
||||||
'filter:video': {
|
'filter:video': {
|
||||||
id: 'learn.coursewareSerch.filter:video',
|
id: 'learn.coursewareSearch.filter:video',
|
||||||
defaultMessage: 'Video',
|
defaultMessage: 'Video',
|
||||||
description: 'Label for the search results filter that shows results with video content.',
|
description: 'Label for the search results filter that shows results with video content.',
|
||||||
},
|
},
|
||||||
'filter:sequence': {
|
'filter:sequence': {
|
||||||
id: 'learn.coursewareSerch.filter:sequence',
|
id: 'learn.coursewareSearch.filter:sequence',
|
||||||
defaultMessage: 'Section',
|
defaultMessage: 'Section',
|
||||||
description: 'Label for the search results filter that shows results with section content.',
|
description: 'Label for the search results filter that shows results with section content.',
|
||||||
},
|
},
|
||||||
'filter:other': {
|
'filter:other': {
|
||||||
id: 'learn.coursewareSerch.filter:other',
|
id: 'learn.coursewareSearch.filter:other',
|
||||||
defaultMessage: 'Other',
|
defaultMessage: 'Other',
|
||||||
description: 'Label for the search results filter that shows results with other content.',
|
description: 'Label for the search results filter that shows results with other content.',
|
||||||
},
|
},
|
||||||
@@ -6,6 +6,7 @@ Factory.define('courseHomeMetadata')
|
|||||||
.option('host', 'http://localhost:18000')
|
.option('host', 'http://localhost:18000')
|
||||||
.attrs({
|
.attrs({
|
||||||
title: 'Demonstration Course',
|
title: 'Demonstration Course',
|
||||||
|
is_new_discussion_sidebar_view_enabled: false,
|
||||||
is_self_paced: false,
|
is_self_paced: false,
|
||||||
is_enrolled: false,
|
is_enrolled: false,
|
||||||
is_staff: false,
|
is_staff: false,
|
||||||
|
|||||||
@@ -31,7 +31,6 @@ Factory.define('outlineTabData')
|
|||||||
course_access_redirect: false,
|
course_access_redirect: false,
|
||||||
has_scheduled_content: null,
|
has_scheduled_content: null,
|
||||||
access_expiration: null,
|
access_expiration: null,
|
||||||
can_show_upgrade_sock: false,
|
|
||||||
cert_data: {
|
cert_data: {
|
||||||
cert_status: null,
|
cert_status: null,
|
||||||
cert_web_view_url: null,
|
cert_web_view_url: null,
|
||||||
|
|||||||
@@ -1,927 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`Data layer integration tests Test fetchDatesTab Should fetch, normalize, and save metadata 1`] = `
|
|
||||||
Object {
|
|
||||||
"courseHome": Object {
|
|
||||||
"courseId": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"courseStatus": "loaded",
|
|
||||||
"proctoringPanelStatus": "loading",
|
|
||||||
"showSearch": false,
|
|
||||||
"targetUserId": undefined,
|
|
||||||
"toastBodyLink": null,
|
|
||||||
"toastBodyText": null,
|
|
||||||
"toastHeader": "",
|
|
||||||
},
|
|
||||||
"courseware": Object {
|
|
||||||
"courseId": null,
|
|
||||||
"courseStatus": "loading",
|
|
||||||
"sequenceId": null,
|
|
||||||
"sequenceMightBeUnit": false,
|
|
||||||
"sequenceStatus": "loading",
|
|
||||||
},
|
|
||||||
"learningAssistant": ObjectContaining {
|
|
||||||
"conversationId": Any<String>,
|
|
||||||
},
|
|
||||||
"models": Object {
|
|
||||||
"courseHomeMeta": Object {
|
|
||||||
"course-v1:edX+DemoX+Demo_Course": Object {
|
|
||||||
"canViewCertificate": true,
|
|
||||||
"celebrations": null,
|
|
||||||
"courseAccess": Object {
|
|
||||||
"additionalContextUserMessage": null,
|
|
||||||
"developerMessage": null,
|
|
||||||
"errorCode": null,
|
|
||||||
"hasAccess": true,
|
|
||||||
"userFragment": null,
|
|
||||||
"userMessage": null,
|
|
||||||
},
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"isEnrolled": false,
|
|
||||||
"isMasquerading": false,
|
|
||||||
"isSelfPaced": false,
|
|
||||||
"isStaff": false,
|
|
||||||
"number": "DemoX",
|
|
||||||
"org": "edX",
|
|
||||||
"originalUserIsStaff": false,
|
|
||||||
"start": "2013-02-05T05:00:00Z",
|
|
||||||
"tabs": Array [
|
|
||||||
Object {
|
|
||||||
"slug": "outline",
|
|
||||||
"title": "Course",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/course/",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "discussion",
|
|
||||||
"title": "Discussion",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/discussion/forum/",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "wiki",
|
|
||||||
"title": "Wiki",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/course_wiki",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "progress",
|
|
||||||
"title": "Progress",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/progress",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "instructor",
|
|
||||||
"title": "Instructor",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/instructor",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "dates",
|
|
||||||
"title": "Dates",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/dates",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"title": "Demonstration Course",
|
|
||||||
"userTimezone": "UTC",
|
|
||||||
"username": "MockUser",
|
|
||||||
"verifiedMode": Object {
|
|
||||||
"accessExpirationDate": null,
|
|
||||||
"currency": "USD",
|
|
||||||
"currencySymbol": "$",
|
|
||||||
"price": 149,
|
|
||||||
"sku": "8CF08E5",
|
|
||||||
"upgradeUrl": "http://localhost:18130/basket/add/?sku=8CF08E5",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"dates": Object {
|
|
||||||
"course-v1:edX+DemoX+Demo_Course": Object {
|
|
||||||
"courseDateBlocks": Array [
|
|
||||||
Object {
|
|
||||||
"date": "2020-05-01T17:59:41Z",
|
|
||||||
"dateType": "course-start-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "",
|
|
||||||
"title": "Course Starts",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"complete": true,
|
|
||||||
"date": "2020-05-04T02:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"title": "Multi Badges Completed",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2020-05-05T02:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"title": "Multi Badges Past Due",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2020-05-27T02:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Both Past Due 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2020-05-27T02:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Both Past Due 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"complete": true,
|
|
||||||
"date": "2020-05-28T08:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "One Completed/Due 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2020-05-28T08:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "One Completed/Due 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"complete": true,
|
|
||||||
"date": "2020-05-29T08:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Both Completed 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"complete": true,
|
|
||||||
"date": "2020-05-29T08:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Both Completed 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"date": "2020-06-16T17:59:40.942669Z",
|
|
||||||
"dateType": "verified-upgrade-deadline",
|
|
||||||
"description": "Don't miss the opportunity to highlight your new knowledge and skills by earning a verified certificate.",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Upgrade to Verified Certificate",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-17T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": false,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "One Verified 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-17T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "One Verified 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-17T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": "ORA Dates are set by the instructor, and can't be changed",
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "ORA Verified 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-18T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": false,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Both Verified 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-18T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": false,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Both Verified 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-19T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"title": "One Unreleased 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-19T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "One Unreleased 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-20T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"title": "Both Unreleased 1",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"date": "2030-08-20T05:59:40.942669Z",
|
|
||||||
"dateType": "assignment-due-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"title": "Both Unreleased 2",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"date": "2030-08-23T00:00:00Z",
|
|
||||||
"dateType": "course-end-date",
|
|
||||||
"description": "",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"link": "",
|
|
||||||
"title": "Course Ends",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"date": "2030-09-01T00:00:00Z",
|
|
||||||
"dateType": "verification-deadline-date",
|
|
||||||
"description": "You must successfully complete verification before this date to qualify for a Verified Certificate.",
|
|
||||||
"extraInfo": null,
|
|
||||||
"learnerHasAccess": false,
|
|
||||||
"link": "https://example.com/",
|
|
||||||
"title": "Verification Deadline",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"datesBannerInfo": Object {
|
|
||||||
"contentTypeGatingEnabled": false,
|
|
||||||
"missedDeadlines": false,
|
|
||||||
"missedGatedContent": false,
|
|
||||||
"verifiedUpgradeLink": "http://localhost:18130/basket/add/?sku=8CF08E5",
|
|
||||||
},
|
|
||||||
"hasEnded": false,
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"learnerIsFullAccess": true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"plugins": Object {},
|
|
||||||
"recommendations": Object {
|
|
||||||
"recommendationsStatus": "loading",
|
|
||||||
},
|
|
||||||
"specialExams": Object {
|
|
||||||
"activeAttempt": null,
|
|
||||||
"allowProctoringOptOut": false,
|
|
||||||
"apiErrorMsg": "",
|
|
||||||
"exam": Object {
|
|
||||||
"attempt": Object {
|
|
||||||
"attempt_code": "",
|
|
||||||
"attempt_id": null,
|
|
||||||
"attempt_status": "",
|
|
||||||
"course_id": "",
|
|
||||||
"desktop_application_js_url": "",
|
|
||||||
"exam_display_name": "",
|
|
||||||
"exam_started_poll_url": "",
|
|
||||||
"exam_type": "",
|
|
||||||
"exam_url_path": "",
|
|
||||||
"external_id": "",
|
|
||||||
"in_timed_exam": true,
|
|
||||||
"ping_interval": null,
|
|
||||||
"taking_as_proctored": true,
|
|
||||||
"time_remaining_seconds": null,
|
|
||||||
"use_legacy_attempt_api": true,
|
|
||||||
},
|
|
||||||
"backend": "",
|
|
||||||
"content_id": "",
|
|
||||||
"course_id": "",
|
|
||||||
"due_date": null,
|
|
||||||
"exam_name": "",
|
|
||||||
"external_id": "",
|
|
||||||
"hide_after_due": false,
|
|
||||||
"id": null,
|
|
||||||
"is_active": true,
|
|
||||||
"is_practice_exam": false,
|
|
||||||
"is_proctored": false,
|
|
||||||
"prerequisite_status": Object {
|
|
||||||
"are_prerequisites_satisifed": true,
|
|
||||||
"declined_prerequisites": Array [],
|
|
||||||
"failed_prerequisites": Array [],
|
|
||||||
"pending_prerequisites": Array [],
|
|
||||||
"satisfied_prerequisites": Array [],
|
|
||||||
},
|
|
||||||
"time_limit_mins": null,
|
|
||||||
"type": "",
|
|
||||||
},
|
|
||||||
"examAccessToken": Object {
|
|
||||||
"exam_access_token": "",
|
|
||||||
"exam_access_token_expiration": "",
|
|
||||||
},
|
|
||||||
"isLoading": true,
|
|
||||||
"proctoringSettings": Object {
|
|
||||||
"exam_proctoring_backend": Object {
|
|
||||||
"download_url": "",
|
|
||||||
"instructions": Array [],
|
|
||||||
"name": "",
|
|
||||||
"rules": Object {},
|
|
||||||
},
|
|
||||||
"integration_specific_email": "",
|
|
||||||
"learner_notification_from_email": "",
|
|
||||||
"provider_name": "",
|
|
||||||
"provider_tech_support_email": "",
|
|
||||||
"provider_tech_support_phone": "",
|
|
||||||
"provider_tech_support_url": "",
|
|
||||||
},
|
|
||||||
"timeIsOver": false,
|
|
||||||
},
|
|
||||||
"tours": Object {
|
|
||||||
"showCoursewareTour": false,
|
|
||||||
"showExistingUserCourseHomeTour": false,
|
|
||||||
"showNewUserCourseHomeModal": false,
|
|
||||||
"showNewUserCourseHomeTour": false,
|
|
||||||
"toursEnabled": false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`Data layer integration tests Test fetchOutlineTab Should fetch, normalize, and save metadata 1`] = `
|
|
||||||
Object {
|
|
||||||
"courseHome": Object {
|
|
||||||
"courseId": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"courseStatus": "loaded",
|
|
||||||
"proctoringPanelStatus": "loading",
|
|
||||||
"showSearch": false,
|
|
||||||
"targetUserId": undefined,
|
|
||||||
"toastBodyLink": null,
|
|
||||||
"toastBodyText": null,
|
|
||||||
"toastHeader": "",
|
|
||||||
},
|
|
||||||
"courseware": Object {
|
|
||||||
"courseId": null,
|
|
||||||
"courseStatus": "loading",
|
|
||||||
"sequenceId": null,
|
|
||||||
"sequenceMightBeUnit": false,
|
|
||||||
"sequenceStatus": "loading",
|
|
||||||
},
|
|
||||||
"learningAssistant": ObjectContaining {
|
|
||||||
"conversationId": Any<String>,
|
|
||||||
},
|
|
||||||
"models": Object {
|
|
||||||
"courseHomeMeta": Object {
|
|
||||||
"course-v1:edX+DemoX+Demo_Course": Object {
|
|
||||||
"canViewCertificate": true,
|
|
||||||
"celebrations": null,
|
|
||||||
"courseAccess": Object {
|
|
||||||
"additionalContextUserMessage": null,
|
|
||||||
"developerMessage": null,
|
|
||||||
"errorCode": null,
|
|
||||||
"hasAccess": true,
|
|
||||||
"userFragment": null,
|
|
||||||
"userMessage": null,
|
|
||||||
},
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"isEnrolled": false,
|
|
||||||
"isMasquerading": false,
|
|
||||||
"isSelfPaced": false,
|
|
||||||
"isStaff": false,
|
|
||||||
"number": "DemoX",
|
|
||||||
"org": "edX",
|
|
||||||
"originalUserIsStaff": false,
|
|
||||||
"start": "2013-02-05T05:00:00Z",
|
|
||||||
"tabs": Array [
|
|
||||||
Object {
|
|
||||||
"slug": "outline",
|
|
||||||
"title": "Course",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/course/",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "discussion",
|
|
||||||
"title": "Discussion",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/discussion/forum/",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "wiki",
|
|
||||||
"title": "Wiki",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/course_wiki",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "progress",
|
|
||||||
"title": "Progress",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/progress",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "instructor",
|
|
||||||
"title": "Instructor",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/instructor",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "dates",
|
|
||||||
"title": "Dates",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/dates",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"title": "Demonstration Course",
|
|
||||||
"userTimezone": "UTC",
|
|
||||||
"username": "MockUser",
|
|
||||||
"verifiedMode": Object {
|
|
||||||
"accessExpirationDate": null,
|
|
||||||
"currency": "USD",
|
|
||||||
"currencySymbol": "$",
|
|
||||||
"price": 149,
|
|
||||||
"sku": "8CF08E5",
|
|
||||||
"upgradeUrl": "http://localhost:18130/basket/add/?sku=8CF08E5",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"outline": Object {
|
|
||||||
"course-v1:edX+DemoX+Demo_Course": Object {
|
|
||||||
"accessExpiration": null,
|
|
||||||
"canShowUpgradeSock": false,
|
|
||||||
"certData": Object {
|
|
||||||
"certStatus": null,
|
|
||||||
"certWebViewUrl": null,
|
|
||||||
"certificateAvailableDate": null,
|
|
||||||
},
|
|
||||||
"courseBlocks": Object {
|
|
||||||
"courses": Object {
|
|
||||||
"block-v1:edX+DemoX+Demo_Course+type@course+block@bcdabcdabcdabcdabcdabcdabcdabcd3": Object {
|
|
||||||
"hasScheduledContent": false,
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"sectionIds": Array [
|
|
||||||
"block-v1:edX+DemoX+Demo_Course+type@chapter+block@bcdabcdabcdabcdabcdabcdabcdabcd2",
|
|
||||||
],
|
|
||||||
"title": "bcdabcdabcdabcdabcdabcdabcdabcd3",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sections": Object {
|
|
||||||
"block-v1:edX+DemoX+Demo_Course+type@chapter+block@bcdabcdabcdabcdabcdabcdabcdabcd2": Object {
|
|
||||||
"complete": false,
|
|
||||||
"courseId": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"hideFromTOC": undefined,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@chapter+block@bcdabcdabcdabcdabcdabcdabcdabcd2",
|
|
||||||
"resumeBlock": false,
|
|
||||||
"sequenceIds": Array [
|
|
||||||
"block-v1:edX+DemoX+Demo_Course+type@sequential+block@bcdabcdabcdabcdabcdabcdabcdabcd1",
|
|
||||||
],
|
|
||||||
"title": "Title of Section",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sequences": Object {
|
|
||||||
"block-v1:edX+DemoX+Demo_Course+type@sequential+block@bcdabcdabcdabcdabcdabcdabcdabcd1": Object {
|
|
||||||
"complete": false,
|
|
||||||
"description": null,
|
|
||||||
"due": null,
|
|
||||||
"effortActivities": 2,
|
|
||||||
"effortTime": 15,
|
|
||||||
"hideFromTOC": undefined,
|
|
||||||
"icon": null,
|
|
||||||
"id": "block-v1:edX+DemoX+Demo_Course+type@sequential+block@bcdabcdabcdabcdabcdabcdabcdabcd1",
|
|
||||||
"navigationDisabled": undefined,
|
|
||||||
"sectionId": "block-v1:edX+DemoX+Demo_Course+type@chapter+block@bcdabcdabcdabcdabcdabcdabcdabcd2",
|
|
||||||
"showLink": true,
|
|
||||||
"title": "Title of Sequence",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"courseGoals": Object {
|
|
||||||
"daysPerWeek": null,
|
|
||||||
"goalOptions": Array [],
|
|
||||||
"selectedGoal": null,
|
|
||||||
"subscribedToReminders": null,
|
|
||||||
"weeklyLearningGoalEnabled": false,
|
|
||||||
},
|
|
||||||
"courseTools": Array [
|
|
||||||
Object {
|
|
||||||
"analyticsId": "edx.bookmarks",
|
|
||||||
"title": "Bookmarks",
|
|
||||||
"url": "https://example.com/bookmarks",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"datesBannerInfo": Object {
|
|
||||||
"contentTypeGatingEnabled": false,
|
|
||||||
"missedDeadlines": false,
|
|
||||||
"missedGatedContent": false,
|
|
||||||
},
|
|
||||||
"datesWidget": Object {
|
|
||||||
"courseDateBlocks": Array [],
|
|
||||||
},
|
|
||||||
"enableProctoredExams": undefined,
|
|
||||||
"enrollAlert": Object {
|
|
||||||
"canEnroll": true,
|
|
||||||
"extraText": "Contact the administrator.",
|
|
||||||
},
|
|
||||||
"enrollmentMode": undefined,
|
|
||||||
"handoutsHtml": "<ul><li>Handout 1</li></ul>",
|
|
||||||
"hasEnded": undefined,
|
|
||||||
"hasScheduledContent": null,
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"offer": null,
|
|
||||||
"resumeCourse": Object {
|
|
||||||
"hasVisitedCourse": false,
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/jump_to/block-v1:edX+Test+Block@12345abcde",
|
|
||||||
},
|
|
||||||
"timeOffsetMillis": 0,
|
|
||||||
"userHasPassingGrade": undefined,
|
|
||||||
"verifiedMode": Object {
|
|
||||||
"accessExpirationDate": "2050-01-01T12:00:00",
|
|
||||||
"currency": "USD",
|
|
||||||
"currencySymbol": "$",
|
|
||||||
"price": 149,
|
|
||||||
"sku": "ABCD1234",
|
|
||||||
"upgradeUrl": "http://localhost:18000/dashboard",
|
|
||||||
},
|
|
||||||
"welcomeMessageHtml": "<p>Welcome to this course!</p>",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"plugins": Object {},
|
|
||||||
"recommendations": Object {
|
|
||||||
"recommendationsStatus": "loading",
|
|
||||||
},
|
|
||||||
"specialExams": Object {
|
|
||||||
"activeAttempt": null,
|
|
||||||
"allowProctoringOptOut": false,
|
|
||||||
"apiErrorMsg": "",
|
|
||||||
"exam": Object {
|
|
||||||
"attempt": Object {
|
|
||||||
"attempt_code": "",
|
|
||||||
"attempt_id": null,
|
|
||||||
"attempt_status": "",
|
|
||||||
"course_id": "",
|
|
||||||
"desktop_application_js_url": "",
|
|
||||||
"exam_display_name": "",
|
|
||||||
"exam_started_poll_url": "",
|
|
||||||
"exam_type": "",
|
|
||||||
"exam_url_path": "",
|
|
||||||
"external_id": "",
|
|
||||||
"in_timed_exam": true,
|
|
||||||
"ping_interval": null,
|
|
||||||
"taking_as_proctored": true,
|
|
||||||
"time_remaining_seconds": null,
|
|
||||||
"use_legacy_attempt_api": true,
|
|
||||||
},
|
|
||||||
"backend": "",
|
|
||||||
"content_id": "",
|
|
||||||
"course_id": "",
|
|
||||||
"due_date": null,
|
|
||||||
"exam_name": "",
|
|
||||||
"external_id": "",
|
|
||||||
"hide_after_due": false,
|
|
||||||
"id": null,
|
|
||||||
"is_active": true,
|
|
||||||
"is_practice_exam": false,
|
|
||||||
"is_proctored": false,
|
|
||||||
"prerequisite_status": Object {
|
|
||||||
"are_prerequisites_satisifed": true,
|
|
||||||
"declined_prerequisites": Array [],
|
|
||||||
"failed_prerequisites": Array [],
|
|
||||||
"pending_prerequisites": Array [],
|
|
||||||
"satisfied_prerequisites": Array [],
|
|
||||||
},
|
|
||||||
"time_limit_mins": null,
|
|
||||||
"type": "",
|
|
||||||
},
|
|
||||||
"examAccessToken": Object {
|
|
||||||
"exam_access_token": "",
|
|
||||||
"exam_access_token_expiration": "",
|
|
||||||
},
|
|
||||||
"isLoading": true,
|
|
||||||
"proctoringSettings": Object {
|
|
||||||
"exam_proctoring_backend": Object {
|
|
||||||
"download_url": "",
|
|
||||||
"instructions": Array [],
|
|
||||||
"name": "",
|
|
||||||
"rules": Object {},
|
|
||||||
},
|
|
||||||
"integration_specific_email": "",
|
|
||||||
"learner_notification_from_email": "",
|
|
||||||
"provider_name": "",
|
|
||||||
"provider_tech_support_email": "",
|
|
||||||
"provider_tech_support_phone": "",
|
|
||||||
"provider_tech_support_url": "",
|
|
||||||
},
|
|
||||||
"timeIsOver": false,
|
|
||||||
},
|
|
||||||
"tours": Object {
|
|
||||||
"showCoursewareTour": false,
|
|
||||||
"showExistingUserCourseHomeTour": false,
|
|
||||||
"showNewUserCourseHomeModal": false,
|
|
||||||
"showNewUserCourseHomeTour": false,
|
|
||||||
"toursEnabled": false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`Data layer integration tests Test fetchProgressTab Should fetch, normalize, and save metadata 1`] = `
|
|
||||||
Object {
|
|
||||||
"courseHome": Object {
|
|
||||||
"courseId": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"courseStatus": "loaded",
|
|
||||||
"proctoringPanelStatus": "loading",
|
|
||||||
"showSearch": false,
|
|
||||||
"targetUserId": undefined,
|
|
||||||
"toastBodyLink": null,
|
|
||||||
"toastBodyText": null,
|
|
||||||
"toastHeader": "",
|
|
||||||
},
|
|
||||||
"courseware": Object {
|
|
||||||
"courseId": null,
|
|
||||||
"courseStatus": "loading",
|
|
||||||
"sequenceId": null,
|
|
||||||
"sequenceMightBeUnit": false,
|
|
||||||
"sequenceStatus": "loading",
|
|
||||||
},
|
|
||||||
"learningAssistant": ObjectContaining {
|
|
||||||
"conversationId": Any<String>,
|
|
||||||
},
|
|
||||||
"models": Object {
|
|
||||||
"courseHomeMeta": Object {
|
|
||||||
"course-v1:edX+DemoX+Demo_Course": Object {
|
|
||||||
"canViewCertificate": true,
|
|
||||||
"celebrations": null,
|
|
||||||
"courseAccess": Object {
|
|
||||||
"additionalContextUserMessage": null,
|
|
||||||
"developerMessage": null,
|
|
||||||
"errorCode": null,
|
|
||||||
"hasAccess": true,
|
|
||||||
"userFragment": null,
|
|
||||||
"userMessage": null,
|
|
||||||
},
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"isEnrolled": false,
|
|
||||||
"isMasquerading": false,
|
|
||||||
"isSelfPaced": false,
|
|
||||||
"isStaff": false,
|
|
||||||
"number": "DemoX",
|
|
||||||
"org": "edX",
|
|
||||||
"originalUserIsStaff": false,
|
|
||||||
"start": "2013-02-05T05:00:00Z",
|
|
||||||
"tabs": Array [
|
|
||||||
Object {
|
|
||||||
"slug": "outline",
|
|
||||||
"title": "Course",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/course/",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "discussion",
|
|
||||||
"title": "Discussion",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/discussion/forum/",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "wiki",
|
|
||||||
"title": "Wiki",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/course_wiki",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "progress",
|
|
||||||
"title": "Progress",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/progress",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "instructor",
|
|
||||||
"title": "Instructor",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/instructor",
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"slug": "dates",
|
|
||||||
"title": "Dates",
|
|
||||||
"url": "http://localhost:18000/courses/course-v1:edX+DemoX+Demo_Course/dates",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"title": "Demonstration Course",
|
|
||||||
"userTimezone": "UTC",
|
|
||||||
"username": "MockUser",
|
|
||||||
"verifiedMode": Object {
|
|
||||||
"accessExpirationDate": null,
|
|
||||||
"currency": "USD",
|
|
||||||
"currencySymbol": "$",
|
|
||||||
"price": 149,
|
|
||||||
"sku": "8CF08E5",
|
|
||||||
"upgradeUrl": "http://localhost:18130/basket/add/?sku=8CF08E5",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"progress": Object {
|
|
||||||
"course-v1:edX+DemoX+Demo_Course": Object {
|
|
||||||
"accessExpiration": null,
|
|
||||||
"certificateData": Object {},
|
|
||||||
"completionSummary": Object {
|
|
||||||
"completeCount": 1,
|
|
||||||
"incompleteCount": 1,
|
|
||||||
"lockedCount": 0,
|
|
||||||
},
|
|
||||||
"courseGrade": Object {
|
|
||||||
"isPassing": true,
|
|
||||||
"letterGrade": "pass",
|
|
||||||
"percent": 1,
|
|
||||||
},
|
|
||||||
"courseId": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"creditCourseRequirements": null,
|
|
||||||
"end": "3027-03-31T00:00:00Z",
|
|
||||||
"enrollmentMode": "audit",
|
|
||||||
"gradesFeatureIsFullyLocked": false,
|
|
||||||
"gradesFeatureIsPartiallyLocked": false,
|
|
||||||
"gradingPolicy": Object {
|
|
||||||
"assignmentPolicies": Array [
|
|
||||||
Object {
|
|
||||||
"averageGrade": "1.00",
|
|
||||||
"numDroppable": 1,
|
|
||||||
"shortLabel": "HW",
|
|
||||||
"type": "Homework",
|
|
||||||
"weight": 1,
|
|
||||||
"weightedGrade": 1,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"gradeRange": Object {
|
|
||||||
"pass": 0.75,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"hasScheduledContent": false,
|
|
||||||
"id": "course-v1:edX+DemoX+Demo_Course",
|
|
||||||
"sectionScores": Array [
|
|
||||||
Object {
|
|
||||||
"displayName": "First section",
|
|
||||||
"subsections": Array [
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"blockKey": "block-v1:edX+DemoX+Demo_Course+type@sequential+block@12345",
|
|
||||||
"displayName": "First subsection",
|
|
||||||
"hasGradedAssignment": true,
|
|
||||||
"learnerHasAccess": true,
|
|
||||||
"numPointsEarned": 0,
|
|
||||||
"numPointsPossible": 3,
|
|
||||||
"percentGraded": 0,
|
|
||||||
"problemScores": Array [
|
|
||||||
Object {
|
|
||||||
"earned": 0,
|
|
||||||
"possible": 1,
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"earned": 0,
|
|
||||||
"possible": 1,
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"earned": 0,
|
|
||||||
"possible": 1,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"showCorrectness": "always",
|
|
||||||
"showGrades": true,
|
|
||||||
"url": "http://learning.edx.org/course/course-v1:edX+Test+run/first_subsection",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
Object {
|
|
||||||
"displayName": "Second section",
|
|
||||||
"subsections": Array [
|
|
||||||
Object {
|
|
||||||
"assignmentType": "Homework",
|
|
||||||
"displayName": "Second subsection",
|
|
||||||
"hasGradedAssignment": true,
|
|
||||||
"numPointsEarned": 1,
|
|
||||||
"numPointsPossible": 1,
|
|
||||||
"percentGraded": 1,
|
|
||||||
"problemScores": Array [
|
|
||||||
Object {
|
|
||||||
"earned": 1,
|
|
||||||
"possible": 1,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"showCorrectness": "always",
|
|
||||||
"showGrades": true,
|
|
||||||
"url": "http://learning.edx.org/course/course-v1:edX+Test+run/second_subsection",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
"studioUrl": "http://studio.edx.org/settings/grading/course-v1:edX+Test+run",
|
|
||||||
"userHasPassingGrade": false,
|
|
||||||
"verificationData": Object {
|
|
||||||
"link": null,
|
|
||||||
"status": "none",
|
|
||||||
"statusDate": null,
|
|
||||||
},
|
|
||||||
"verifiedMode": null,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"plugins": Object {},
|
|
||||||
"recommendations": Object {
|
|
||||||
"recommendationsStatus": "loading",
|
|
||||||
},
|
|
||||||
"specialExams": Object {
|
|
||||||
"activeAttempt": null,
|
|
||||||
"allowProctoringOptOut": false,
|
|
||||||
"apiErrorMsg": "",
|
|
||||||
"exam": Object {
|
|
||||||
"attempt": Object {
|
|
||||||
"attempt_code": "",
|
|
||||||
"attempt_id": null,
|
|
||||||
"attempt_status": "",
|
|
||||||
"course_id": "",
|
|
||||||
"desktop_application_js_url": "",
|
|
||||||
"exam_display_name": "",
|
|
||||||
"exam_started_poll_url": "",
|
|
||||||
"exam_type": "",
|
|
||||||
"exam_url_path": "",
|
|
||||||
"external_id": "",
|
|
||||||
"in_timed_exam": true,
|
|
||||||
"ping_interval": null,
|
|
||||||
"taking_as_proctored": true,
|
|
||||||
"time_remaining_seconds": null,
|
|
||||||
"use_legacy_attempt_api": true,
|
|
||||||
},
|
|
||||||
"backend": "",
|
|
||||||
"content_id": "",
|
|
||||||
"course_id": "",
|
|
||||||
"due_date": null,
|
|
||||||
"exam_name": "",
|
|
||||||
"external_id": "",
|
|
||||||
"hide_after_due": false,
|
|
||||||
"id": null,
|
|
||||||
"is_active": true,
|
|
||||||
"is_practice_exam": false,
|
|
||||||
"is_proctored": false,
|
|
||||||
"prerequisite_status": Object {
|
|
||||||
"are_prerequisites_satisifed": true,
|
|
||||||
"declined_prerequisites": Array [],
|
|
||||||
"failed_prerequisites": Array [],
|
|
||||||
"pending_prerequisites": Array [],
|
|
||||||
"satisfied_prerequisites": Array [],
|
|
||||||
},
|
|
||||||
"time_limit_mins": null,
|
|
||||||
"type": "",
|
|
||||||
},
|
|
||||||
"examAccessToken": Object {
|
|
||||||
"exam_access_token": "",
|
|
||||||
"exam_access_token_expiration": "",
|
|
||||||
},
|
|
||||||
"isLoading": true,
|
|
||||||
"proctoringSettings": Object {
|
|
||||||
"exam_proctoring_backend": Object {
|
|
||||||
"download_url": "",
|
|
||||||
"instructions": Array [],
|
|
||||||
"name": "",
|
|
||||||
"rules": Object {},
|
|
||||||
},
|
|
||||||
"integration_specific_email": "",
|
|
||||||
"learner_notification_from_email": "",
|
|
||||||
"provider_name": "",
|
|
||||||
"provider_tech_support_email": "",
|
|
||||||
"provider_tech_support_phone": "",
|
|
||||||
"provider_tech_support_url": "",
|
|
||||||
},
|
|
||||||
"timeIsOver": false,
|
|
||||||
},
|
|
||||||
"tours": Object {
|
|
||||||
"showCoursewareTour": false,
|
|
||||||
"showExistingUserCourseHomeTour": false,
|
|
||||||
"showNewUserCourseHomeModal": false,
|
|
||||||
"showNewUserCourseHomeTour": false,
|
|
||||||
"toursEnabled": false,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
@@ -18,7 +18,7 @@ const calculateAssignmentTypeGrades = (points, assignmentWeight, numDroppable) =
|
|||||||
// Calculate the average grade for the assignment and round it. This rounding is not ideal and does not accurately
|
// Calculate the average grade for the assignment and round it. This rounding is not ideal and does not accurately
|
||||||
// reflect what a learner's grade would be, however, we must have parity with the current grading behavior that
|
// reflect what a learner's grade would be, however, we must have parity with the current grading behavior that
|
||||||
// exists in edx-platform.
|
// exists in edx-platform.
|
||||||
averageGrade = (points.reduce((a, b) => a + b, 0) / points.length).toFixed(2);
|
averageGrade = (points.reduce((a, b) => a + b, 0) / points.length).toFixed(4);
|
||||||
weightedGrade = averageGrade * assignmentWeight;
|
weightedGrade = averageGrade * assignmentWeight;
|
||||||
}
|
}
|
||||||
return { averageGrade, weightedGrade };
|
return { averageGrade, weightedGrade };
|
||||||
@@ -289,9 +289,17 @@ export async function getProgressTabData(courseId, targetUserId) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export async function getProctoringInfoData(courseId, username) {
|
export async function getProctoringInfoData(courseId, username) {
|
||||||
let url = `${getConfig().LMS_BASE_URL}/api/edx_proctoring/v1/user_onboarding/status?is_learning_mfe=true&course_id=${encodeURIComponent(courseId)}`;
|
let url;
|
||||||
if (username) {
|
if (!getConfig().EXAMS_BASE_URL) {
|
||||||
url += `&username=${encodeURIComponent(username)}`;
|
url = `${getConfig().LMS_BASE_URL}/api/edx_proctoring/v1/user_onboarding/status?is_learning_mfe=true&course_id=${encodeURIComponent(courseId)}`;
|
||||||
|
if (username) {
|
||||||
|
url += `&username=${encodeURIComponent(username)}`;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
url = `${getConfig().EXAMS_BASE_URL}/api/v1/student/course_id/${encodeURIComponent(courseId)}/onboarding`;
|
||||||
|
if (username) {
|
||||||
|
url += `?username=${encodeURIComponent(username)}`;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const { data } = await getAuthenticatedHttpClient().get(url);
|
const { data } = await getAuthenticatedHttpClient().get(url);
|
||||||
@@ -359,7 +367,6 @@ export async function getOutlineTabData(courseId) {
|
|||||||
} = tabData;
|
} = tabData;
|
||||||
|
|
||||||
const accessExpiration = camelCaseObject(data.access_expiration);
|
const accessExpiration = camelCaseObject(data.access_expiration);
|
||||||
const canShowUpgradeSock = data.can_show_upgrade_sock;
|
|
||||||
const certData = camelCaseObject(data.cert_data);
|
const certData = camelCaseObject(data.cert_data);
|
||||||
const courseBlocks = data.course_blocks ? normalizeOutlineBlocks(courseId, data.course_blocks.blocks) : {};
|
const courseBlocks = data.course_blocks ? normalizeOutlineBlocks(courseId, data.course_blocks.blocks) : {};
|
||||||
const courseGoals = camelCaseObject(data.course_goals);
|
const courseGoals = camelCaseObject(data.course_goals);
|
||||||
@@ -381,7 +388,6 @@ export async function getOutlineTabData(courseId) {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
accessExpiration,
|
accessExpiration,
|
||||||
canShowUpgradeSock,
|
|
||||||
certData,
|
certData,
|
||||||
courseBlocks,
|
courseBlocks,
|
||||||
courseGoals,
|
courseGoals,
|
||||||
@@ -449,7 +455,7 @@ export async function unsubscribeFromCourseGoal(token) {
|
|||||||
.then(res => camelCaseObject(res));
|
.then(res => camelCaseObject(res));
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function getCoursewareSearchEnabledFlag(courseId) {
|
export async function getCoursewareSearchEnabled(courseId) {
|
||||||
const url = new URL(`${getConfig().LMS_BASE_URL}/courses/${courseId}/courseware-search/enabled/`);
|
const url = new URL(`${getConfig().LMS_BASE_URL}/courses/${courseId}/courseware-search/enabled/`);
|
||||||
const { data } = await getAuthenticatedHttpClient().get(url.href);
|
const { data } = await getAuthenticatedHttpClient().get(url.href);
|
||||||
return { enabled: data.enabled || false };
|
return { enabled: data.enabled || false };
|
||||||
|
|||||||
@@ -46,7 +46,6 @@ describe('Course Home Service', () => {
|
|||||||
willRespondWith: {
|
willRespondWith: {
|
||||||
status: 200,
|
status: 200,
|
||||||
body: {
|
body: {
|
||||||
can_show_upgrade_sock: boolean(false),
|
|
||||||
verified_mode: like({
|
verified_mode: like({
|
||||||
access_expiration_date: null,
|
access_expiration_date: null,
|
||||||
currency: 'USD',
|
currency: 'USD',
|
||||||
@@ -89,11 +88,11 @@ describe('Course Home Service', () => {
|
|||||||
}),
|
}),
|
||||||
title: string('Demonstration Course'),
|
title: string('Demonstration Course'),
|
||||||
username: string('edx'),
|
username: string('edx'),
|
||||||
|
has_course_author_access: boolean(true),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
const normalizedTabData = {
|
const normalizedTabData = {
|
||||||
canShowUpgradeSock: false,
|
|
||||||
verifiedMode: {
|
verifiedMode: {
|
||||||
accessExpirationDate: null,
|
accessExpirationDate: null,
|
||||||
currency: 'USD',
|
currency: 'USD',
|
||||||
@@ -133,6 +132,7 @@ describe('Course Home Service', () => {
|
|||||||
],
|
],
|
||||||
title: 'Demonstration Course',
|
title: 'Demonstration Course',
|
||||||
username: 'edx',
|
username: 'edx',
|
||||||
|
hasCourseAuthorAccess: true,
|
||||||
};
|
};
|
||||||
const response = getCourseHomeCourseMetadata(courseId, 'outline');
|
const response = getCourseHomeCourseMetadata(courseId, 'outline');
|
||||||
expect(response).toBeTruthy();
|
expect(response).toBeTruthy();
|
||||||
|
|||||||
@@ -90,14 +90,14 @@ describe('Data layer integration tests', () => {
|
|||||||
|
|
||||||
const state = store.getState();
|
const state = store.getState();
|
||||||
expect(state.courseHome.courseStatus).toEqual('loaded');
|
expect(state.courseHome.courseStatus).toEqual('loaded');
|
||||||
expect(state).toMatchSnapshot({
|
expect(state).toEqual(expect.objectContaining({
|
||||||
// The Xpert chatbot (frontend-lib-learning-assistant) generates a unique UUID
|
// The Xpert chatbot (frontend-lib-learning-assistant) generates a unique UUID
|
||||||
// to keep track of conversations. This causes snapshots to fail, because this UUID
|
// to keep track of conversations. This UUID is generated on each run.
|
||||||
// is generated on each run of the snapshot. Instead, we use an asymmetric matcher here.
|
// Instead, we use an asymmetric matcher here.
|
||||||
learningAssistant: expect.objectContaining({
|
learningAssistant: expect.objectContaining({
|
||||||
conversationId: expect.any(String),
|
conversationId: expect.any(String),
|
||||||
}),
|
}),
|
||||||
});
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
it.each([401, 403, 404])(
|
it.each([401, 403, 404])(
|
||||||
@@ -137,14 +137,14 @@ describe('Data layer integration tests', () => {
|
|||||||
|
|
||||||
const state = store.getState();
|
const state = store.getState();
|
||||||
expect(state.courseHome.courseStatus).toEqual('loaded');
|
expect(state.courseHome.courseStatus).toEqual('loaded');
|
||||||
expect(state).toMatchSnapshot({
|
expect(state).toEqual(expect.objectContaining({
|
||||||
// The Xpert chatbot (frontend-lib-learning-assistant) generates a unique UUID
|
// The Xpert chatbot (frontend-lib-learning-assistant) generates a unique UUID
|
||||||
// to keep track of conversations. This causes snapshots to fail, because this UUID
|
// to keep track of conversations. This UUID is generated on each run.
|
||||||
// is generated on each run of the snapshot. Instead, we use an asymmetric matcher here.
|
// Instead, we use an asymmetric matcher here.
|
||||||
learningAssistant: expect.objectContaining({
|
learningAssistant: expect.objectContaining({
|
||||||
conversationId: expect.any(String),
|
conversationId: expect.any(String),
|
||||||
}),
|
}),
|
||||||
});
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
it.each([401, 403, 404])(
|
it.each([401, 403, 404])(
|
||||||
@@ -185,14 +185,14 @@ describe('Data layer integration tests', () => {
|
|||||||
|
|
||||||
const state = store.getState();
|
const state = store.getState();
|
||||||
expect(state.courseHome.courseStatus).toEqual('loaded');
|
expect(state.courseHome.courseStatus).toEqual('loaded');
|
||||||
expect(state).toMatchSnapshot({
|
expect(state).toEqual(expect.objectContaining({
|
||||||
// The Xpert chatbot (frontend-lib-learning-assistant) generates a unique UUID
|
// The Xpert chatbot (frontend-lib-learning-assistant) generates a unique UUID
|
||||||
// to keep track of conversations. This causes snapshots to fail, because this UUID
|
// to keep track of conversations. This UUID is generated on each run.
|
||||||
// is generated on each run of the snapshot. Instead, we use an asymmetric matcher here.
|
// Instead, we use an asymmetric matcher here.
|
||||||
learningAssistant: expect.objectContaining({
|
learningAssistant: expect.objectContaining({
|
||||||
conversationId: expect.any(String),
|
conversationId: expect.any(String),
|
||||||
}),
|
}),
|
||||||
});
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Should handle the url including a targetUserId', async () => {
|
it('Should handle the url including a targetUserId', async () => {
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
/* eslint-disable no-param-reassign */
|
/* eslint-disable no-param-reassign */
|
||||||
import { createSlice } from '@reduxjs/toolkit';
|
import { createSlice } from '@reduxjs/toolkit';
|
||||||
|
|
||||||
export const LOADING = 'loading';
|
import {
|
||||||
export const LOADED = 'loaded';
|
LOADING,
|
||||||
export const FAILED = 'failed';
|
LOADED,
|
||||||
export const DENIED = 'denied';
|
FAILED,
|
||||||
|
DENIED,
|
||||||
|
} from '@src/constants';
|
||||||
|
|
||||||
const slice = createSlice({
|
const slice = createSlice({
|
||||||
name: 'course-home',
|
name: 'course-home',
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ import {
|
|||||||
postDismissWelcomeMessage,
|
postDismissWelcomeMessage,
|
||||||
postRequestCert,
|
postRequestCert,
|
||||||
getLiveTabIframe,
|
getLiveTabIframe,
|
||||||
getCoursewareSearchEnabledFlag,
|
getCoursewareSearchEnabled,
|
||||||
searchCourseContentFromAPI,
|
searchCourseContentFromAPI,
|
||||||
} from './api';
|
} from './api';
|
||||||
|
|
||||||
@@ -159,7 +159,7 @@ export function processEvent(eventData, getTabData) {
|
|||||||
|
|
||||||
export async function fetchCoursewareSearchSettings(courseId) {
|
export async function fetchCoursewareSearchSettings(courseId) {
|
||||||
try {
|
try {
|
||||||
const { enabled } = await getCoursewareSearchEnabledFlag(courseId);
|
const { enabled } = await getCoursewareSearchEnabled(courseId);
|
||||||
return { enabled };
|
return { enabled };
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return { enabled: false };
|
return { enabled: false };
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useSelector } from 'react-redux';
|
import { useSelector } from 'react-redux';
|
||||||
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
|
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import Timeline from './timeline/Timeline';
|
import Timeline from './timeline/Timeline';
|
||||||
@@ -14,7 +14,8 @@ import ShiftDatesAlert from '../suggested-schedule-messaging/ShiftDatesAlert';
|
|||||||
import UpgradeToCompleteAlert from '../suggested-schedule-messaging/UpgradeToCompleteAlert';
|
import UpgradeToCompleteAlert from '../suggested-schedule-messaging/UpgradeToCompleteAlert';
|
||||||
import UpgradeToShiftDatesAlert from '../suggested-schedule-messaging/UpgradeToShiftDatesAlert';
|
import UpgradeToShiftDatesAlert from '../suggested-schedule-messaging/UpgradeToShiftDatesAlert';
|
||||||
|
|
||||||
const DatesTab = ({ intl }) => {
|
const DatesTab = () => {
|
||||||
|
const intl = useIntl();
|
||||||
const {
|
const {
|
||||||
courseId,
|
courseId,
|
||||||
} = useSelector(state => state.courseHome);
|
} = useSelector(state => state.courseHome);
|
||||||
@@ -59,8 +60,4 @@ const DatesTab = ({ intl }) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
DatesTab.propTypes = {
|
export default DatesTab;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(DatesTab);
|
|
||||||
|
|||||||
@@ -135,6 +135,7 @@ describe('DatesTab', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('shows extra info', async () => {
|
it('shows extra info', async () => {
|
||||||
|
const user = userEvent.setup();
|
||||||
const { items } = await getDay('Sat, Aug 17, 2030');
|
const { items } = await getDay('Sat, Aug 17, 2030');
|
||||||
expect(items).toHaveLength(3);
|
expect(items).toHaveLength(3);
|
||||||
|
|
||||||
@@ -142,10 +143,12 @@ describe('DatesTab', () => {
|
|||||||
const tipText = "ORA Dates are set by the instructor, and can't be changed";
|
const tipText = "ORA Dates are set by the instructor, and can't be changed";
|
||||||
|
|
||||||
expect(screen.queryByText(tipText)).toBeNull(); // tooltip does not start in DOM
|
expect(screen.queryByText(tipText)).toBeNull(); // tooltip does not start in DOM
|
||||||
userEvent.hover(tipIcon);
|
await user.hover(tipIcon);
|
||||||
const tooltip = screen.getByText(tipText); // now it's there
|
screen.getByText(tipText); // now it's there
|
||||||
userEvent.unhover(tipIcon);
|
await user.unhover(tipIcon);
|
||||||
await waitForElementToBeRemoved(tooltip); // and it's gone again
|
await waitFor(() => {
|
||||||
|
expect(screen.queryByText(tipText)).toBeNull(); // and it's gone again
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -5,8 +5,7 @@ import { useSelector } from 'react-redux';
|
|||||||
import {
|
import {
|
||||||
FormattedDate,
|
FormattedDate,
|
||||||
FormattedTime,
|
FormattedTime,
|
||||||
injectIntl,
|
useIntl,
|
||||||
intlShape,
|
|
||||||
} from '@edx/frontend-platform/i18n';
|
} from '@edx/frontend-platform/i18n';
|
||||||
import { Tooltip, OverlayTrigger } from '@openedx/paragon';
|
import { Tooltip, OverlayTrigger } from '@openedx/paragon';
|
||||||
import { faInfoCircle } from '@fortawesome/free-solid-svg-icons';
|
import { faInfoCircle } from '@fortawesome/free-solid-svg-icons';
|
||||||
@@ -20,10 +19,10 @@ import { isLearnerAssignment } from '../utils';
|
|||||||
const Day = ({
|
const Day = ({
|
||||||
date,
|
date,
|
||||||
first,
|
first,
|
||||||
intl,
|
|
||||||
items,
|
items,
|
||||||
last,
|
last,
|
||||||
}) => {
|
}) => {
|
||||||
|
const intl = useIntl();
|
||||||
const {
|
const {
|
||||||
courseId,
|
courseId,
|
||||||
} = useSelector(state => state.courseHome);
|
} = useSelector(state => state.courseHome);
|
||||||
@@ -108,7 +107,6 @@ const Day = ({
|
|||||||
Day.propTypes = {
|
Day.propTypes = {
|
||||||
date: PropTypes.objectOf(Date).isRequired,
|
date: PropTypes.objectOf(Date).isRequired,
|
||||||
first: PropTypes.bool,
|
first: PropTypes.bool,
|
||||||
intl: intlShape.isRequired,
|
|
||||||
items: PropTypes.arrayOf(PropTypes.shape({
|
items: PropTypes.arrayOf(PropTypes.shape({
|
||||||
date: PropTypes.string,
|
date: PropTypes.string,
|
||||||
dateType: PropTypes.string,
|
dateType: PropTypes.string,
|
||||||
@@ -126,4 +124,4 @@ Day.defaultProps = {
|
|||||||
last: false,
|
last: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(Day);
|
export default Day;
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import { getConfig } from '@edx/frontend-platform';
|
import { getConfig } from '@edx/frontend-platform';
|
||||||
import { injectIntl } from '@edx/frontend-platform/i18n';
|
|
||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { useSelector } from 'react-redux';
|
import { useSelector } from 'react-redux';
|
||||||
import { useParams, generatePath, useNavigate } from 'react-router-dom';
|
import { useParams, generatePath, useNavigate } from 'react-router-dom';
|
||||||
@@ -30,6 +29,4 @@ const DiscussionTab = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
DiscussionTab.propTypes = {};
|
export default DiscussionTab;
|
||||||
|
|
||||||
export default injectIntl(DiscussionTab);
|
|
||||||
|
|||||||
@@ -1,16 +1,17 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import { useParams } from 'react-router-dom';
|
import { useParams } from 'react-router-dom';
|
||||||
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
|
|
||||||
import { LearningHeader as Header } from '@edx/frontend-component-header';
|
import HeaderSlot from '../../plugin-slots/HeaderSlot';
|
||||||
import PageLoading from '../../generic/PageLoading';
|
import PageLoading from '../../generic/PageLoading';
|
||||||
import { unsubscribeFromCourseGoal } from '../data/api';
|
import { unsubscribeFromCourseGoal } from '../data/api';
|
||||||
|
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import ResultPage from './ResultPage';
|
import ResultPage from './ResultPage';
|
||||||
|
|
||||||
const GoalUnsubscribe = ({ intl }) => {
|
const GoalUnsubscribe = () => {
|
||||||
|
const intl = useIntl();
|
||||||
const { token } = useParams();
|
const { token } = useParams();
|
||||||
const [error, setError] = useState(false);
|
const [error, setError] = useState(false);
|
||||||
const [isLoading, setIsLoading] = useState(true);
|
const [isLoading, setIsLoading] = useState(true);
|
||||||
@@ -38,7 +39,7 @@ const GoalUnsubscribe = ({ intl }) => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Header showUserDropdown={false} />
|
<HeaderSlot showUserDropdown={false} />
|
||||||
<main id="main-content" className="container my-5 text-center">
|
<main id="main-content" className="container my-5 text-center">
|
||||||
{isLoading && (
|
{isLoading && (
|
||||||
<PageLoading srMessage={`${intl.formatMessage(messages.loading)}`} />
|
<PageLoading srMessage={`${intl.formatMessage(messages.loading)}`} />
|
||||||
@@ -51,8 +52,4 @@ const GoalUnsubscribe = ({ intl }) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
GoalUnsubscribe.propTypes = {
|
export default GoalUnsubscribe;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(GoalUnsubscribe);
|
|
||||||
|
|||||||
@@ -1,28 +1,26 @@
|
|||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { getConfig } from '@edx/frontend-platform';
|
import { getConfig } from '@edx/frontend-platform';
|
||||||
import { FormattedMessage, injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import { Button, Hyperlink } from '@openedx/paragon';
|
import { Button, Hyperlink } from '@openedx/paragon';
|
||||||
|
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import { ReactComponent as UnsubscribeIcon } from './unsubscribe.svg';
|
import { ReactComponent as UnsubscribeIcon } from './unsubscribe.svg';
|
||||||
|
|
||||||
const ResultPage = ({ courseTitle, error, intl }) => {
|
const ResultPage = ({ courseTitle, error }) => {
|
||||||
const errorDescription = (
|
const intl = useIntl();
|
||||||
<FormattedMessage
|
const errorDescription = intl.formatMessage(
|
||||||
id="learning.goals.unsubscribe.errorDescription"
|
messages.errorDescription,
|
||||||
defaultMessage="We were unable to unsubscribe you from goal reminder emails. Please try again later or {contactSupport} for help."
|
{
|
||||||
values={{
|
contactSupport: (
|
||||||
contactSupport: (
|
<Hyperlink
|
||||||
<Hyperlink
|
className="text-reset"
|
||||||
className="text-reset"
|
style={{ textDecoration: 'underline' }}
|
||||||
style={{ textDecoration: 'underline' }}
|
destination={`${getConfig().CONTACT_URL}`}
|
||||||
destination={`${getConfig().CONTACT_URL}`}
|
>
|
||||||
>
|
{intl.formatMessage(messages.contactSupport)}
|
||||||
{intl.formatMessage(messages.contactSupport)}
|
</Hyperlink>
|
||||||
</Hyperlink>
|
),
|
||||||
),
|
},
|
||||||
}}
|
|
||||||
/>
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const header = error
|
const header = error
|
||||||
@@ -54,7 +52,6 @@ ResultPage.defaultProps = {
|
|||||||
ResultPage.propTypes = {
|
ResultPage.propTypes = {
|
||||||
courseTitle: PropTypes.string,
|
courseTitle: PropTypes.string,
|
||||||
error: PropTypes.bool,
|
error: PropTypes.bool,
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(ResultPage);
|
export default ResultPage;
|
||||||
|
|||||||
@@ -16,6 +16,11 @@ const messages = defineMessages({
|
|||||||
defaultMessage: 'Something went wrong',
|
defaultMessage: 'Something went wrong',
|
||||||
description: 'It indicate that the unsubscribing request has failed',
|
description: 'It indicate that the unsubscribing request has failed',
|
||||||
},
|
},
|
||||||
|
errorDescription: {
|
||||||
|
id: 'learning.goals.unsubscribe.errorDescription',
|
||||||
|
defaultMessage: 'We were unable to unsubscribe you from goal reminder emails. Please try again later or {contactSupport} for help.',
|
||||||
|
description: 'Message that notifies user that unsubscribing failed and to try again',
|
||||||
|
},
|
||||||
goToDashboard: {
|
goToDashboard: {
|
||||||
id: 'learning.goals.unsubscribe.goToDashboard',
|
id: 'learning.goals.unsubscribe.goToDashboard',
|
||||||
defaultMessage: 'Go to dashboard',
|
defaultMessage: 'Go to dashboard',
|
||||||
@@ -65,6 +65,7 @@ const DateSummary = ({
|
|||||||
)}
|
)}
|
||||||
{!linkedTitle && dateBlock.link && (
|
{!linkedTitle && dateBlock.link && (
|
||||||
<a
|
<a
|
||||||
|
id={dateBlock.dateType === 'verified-upgrade-deadline' ? 'date-verified-upgrade-deadline' : ''}
|
||||||
href={dateBlock.link}
|
href={dateBlock.link}
|
||||||
onClick={dateBlock.dateType === 'verified-upgrade-deadline' ? logVerifiedUpgradeClick : () => {}}
|
onClick={dateBlock.dateType === 'verified-upgrade-deadline' ? logVerifiedUpgradeClick : () => {}}
|
||||||
className="description-link"
|
className="description-link"
|
||||||
|
|||||||
@@ -1,10 +1,11 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import { useEffect, useRef, useState } from 'react';
|
||||||
import { useLocation, useNavigate } from 'react-router-dom';
|
import { useLocation, useNavigate } from 'react-router-dom';
|
||||||
import { useSelector } from 'react-redux';
|
import { useSelector } from 'react-redux';
|
||||||
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
||||||
import { getAuthenticatedUser } from '@edx/frontend-platform/auth';
|
import { getAuthenticatedUser } from '@edx/frontend-platform/auth';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { useIntl } from '@edx/frontend-platform/i18n';
|
||||||
import { Button } from '@openedx/paragon';
|
import { Button } from '@openedx/paragon';
|
||||||
|
import { CourseOutlineTabNotificationsSlot } from '../../plugin-slots/CourseOutlineTabNotificationsSlot';
|
||||||
import { AlertList } from '../../generic/user-messages';
|
import { AlertList } from '../../generic/user-messages';
|
||||||
|
|
||||||
import CourseDates from './widgets/CourseDates';
|
import CourseDates from './widgets/CourseDates';
|
||||||
@@ -14,9 +15,7 @@ import WeeklyLearningGoalCard from './widgets/WeeklyLearningGoalCard';
|
|||||||
import CourseTools from './widgets/CourseTools';
|
import CourseTools from './widgets/CourseTools';
|
||||||
import { fetchOutlineTab } from '../data';
|
import { fetchOutlineTab } from '../data';
|
||||||
import messages from './messages';
|
import messages from './messages';
|
||||||
import Section from './Section';
|
|
||||||
import ShiftDatesAlert from '../suggested-schedule-messaging/ShiftDatesAlert';
|
import ShiftDatesAlert from '../suggested-schedule-messaging/ShiftDatesAlert';
|
||||||
import UpgradeNotification from '../../generic/upgrade-notification/UpgradeNotification';
|
|
||||||
import UpgradeToShiftDatesAlert from '../suggested-schedule-messaging/UpgradeToShiftDatesAlert';
|
import UpgradeToShiftDatesAlert from '../suggested-schedule-messaging/UpgradeToShiftDatesAlert';
|
||||||
import useCertificateAvailableAlert from './alerts/certificate-status-alert';
|
import useCertificateAvailableAlert from './alerts/certificate-status-alert';
|
||||||
import useCourseEndAlert from './alerts/course-end-alert';
|
import useCourseEndAlert from './alerts/course-end-alert';
|
||||||
@@ -27,8 +26,10 @@ import { useModel } from '../../generic/model-store';
|
|||||||
import WelcomeMessage from './widgets/WelcomeMessage';
|
import WelcomeMessage from './widgets/WelcomeMessage';
|
||||||
import ProctoringInfoPanel from './widgets/ProctoringInfoPanel';
|
import ProctoringInfoPanel from './widgets/ProctoringInfoPanel';
|
||||||
import AccountActivationAlert from '../../alerts/logistration-alert/AccountActivationAlert';
|
import AccountActivationAlert from '../../alerts/logistration-alert/AccountActivationAlert';
|
||||||
|
import CourseHomeSectionOutlineSlot from '../../plugin-slots/CourseHomeSectionOutlineSlot';
|
||||||
|
|
||||||
const OutlineTab = ({ intl }) => {
|
const OutlineTab = () => {
|
||||||
|
const intl = useIntl();
|
||||||
const {
|
const {
|
||||||
courseId,
|
courseId,
|
||||||
proctoringPanelStatus,
|
proctoringPanelStatus,
|
||||||
@@ -38,11 +39,11 @@ const OutlineTab = ({ intl }) => {
|
|||||||
isSelfPaced,
|
isSelfPaced,
|
||||||
org,
|
org,
|
||||||
title,
|
title,
|
||||||
userTimezone,
|
|
||||||
} = useModel('courseHomeMeta', courseId);
|
} = useModel('courseHomeMeta', courseId);
|
||||||
|
|
||||||
|
const expandButtonRef = useRef();
|
||||||
|
|
||||||
const {
|
const {
|
||||||
accessExpiration,
|
|
||||||
courseBlocks: {
|
courseBlocks: {
|
||||||
courses,
|
courses,
|
||||||
sections,
|
sections,
|
||||||
@@ -51,20 +52,12 @@ const OutlineTab = ({ intl }) => {
|
|||||||
selectedGoal,
|
selectedGoal,
|
||||||
weeklyLearningGoalEnabled,
|
weeklyLearningGoalEnabled,
|
||||||
} = {},
|
} = {},
|
||||||
datesBannerInfo,
|
|
||||||
datesWidget: {
|
datesWidget: {
|
||||||
courseDateBlocks,
|
courseDateBlocks,
|
||||||
},
|
},
|
||||||
enableProctoredExams,
|
enableProctoredExams,
|
||||||
offer,
|
|
||||||
timeOffsetMillis,
|
|
||||||
verifiedMode,
|
|
||||||
} = useModel('outline', courseId);
|
} = useModel('outline', courseId);
|
||||||
|
|
||||||
const {
|
|
||||||
marketingUrl,
|
|
||||||
} = useModel('coursewareMeta', courseId);
|
|
||||||
|
|
||||||
const [expandAll, setExpandAll] = useState(false);
|
const [expandAll, setExpandAll] = useState(false);
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@@ -158,27 +151,21 @@ const OutlineTab = ({ intl }) => {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
<StartOrResumeCourseCard />
|
<StartOrResumeCourseCard />
|
||||||
<WelcomeMessage courseId={courseId} />
|
<WelcomeMessage courseId={courseId} nextElementRef={expandButtonRef} />
|
||||||
{rootCourseId && (
|
{rootCourseId && (
|
||||||
<>
|
<>
|
||||||
<div className="row w-100 m-0 mb-3 justify-content-end">
|
<div id="expand-button-row" className="row w-100 m-0 mb-3 justify-content-end">
|
||||||
<div className="col-12 col-md-auto p-0">
|
<div className="col-12 col-md-auto p-0">
|
||||||
<Button variant="outline-primary" block onClick={() => { setExpandAll(!expandAll); }}>
|
<Button ref={expandButtonRef} variant="outline-primary" block onClick={() => { setExpandAll(!expandAll); }}>
|
||||||
{expandAll ? intl.formatMessage(messages.collapseAll) : intl.formatMessage(messages.expandAll)}
|
{expandAll ? intl.formatMessage(messages.collapseAll) : intl.formatMessage(messages.expandAll)}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ol id="courseHome-outline" className="list-unstyled">
|
<CourseHomeSectionOutlineSlot
|
||||||
{courses[rootCourseId].sectionIds.map((sectionId) => (
|
expandAll={expandAll}
|
||||||
<Section
|
sectionIds={courses[rootCourseId].sectionIds}
|
||||||
key={sectionId}
|
sections={sections}
|
||||||
courseId={courseId}
|
/>
|
||||||
defaultOpen={sections[sectionId].resumeBlock}
|
|
||||||
expand={expandAll}
|
|
||||||
section={sections[sectionId]}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</ol>
|
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
@@ -194,19 +181,7 @@ const OutlineTab = ({ intl }) => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<CourseTools />
|
<CourseTools />
|
||||||
<UpgradeNotification
|
<CourseOutlineTabNotificationsSlot courseId={courseId} />
|
||||||
offer={offer}
|
|
||||||
verifiedMode={verifiedMode}
|
|
||||||
accessExpiration={accessExpiration}
|
|
||||||
contentTypeGatingEnabled={datesBannerInfo.contentTypeGatingEnabled}
|
|
||||||
marketingUrl={marketingUrl}
|
|
||||||
upsellPageName="course_home"
|
|
||||||
userTimezone={userTimezone}
|
|
||||||
shouldDisplayBorder
|
|
||||||
timeOffsetMillis={timeOffsetMillis}
|
|
||||||
courseId={courseId}
|
|
||||||
org={org}
|
|
||||||
/>
|
|
||||||
<CourseDates />
|
<CourseDates />
|
||||||
<CourseHandouts />
|
<CourseHandouts />
|
||||||
</div>
|
</div>
|
||||||
@@ -216,8 +191,4 @@ const OutlineTab = ({ intl }) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
OutlineTab.propTypes = {
|
export default OutlineTab;
|
||||||
intl: intlShape.isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(OutlineTab);
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import React from 'react';
|
|||||||
import { MemoryRouter } from 'react-router-dom';
|
import { MemoryRouter } from 'react-router-dom';
|
||||||
import { Factory } from 'rosie';
|
import { Factory } from 'rosie';
|
||||||
import { getConfig } from '@edx/frontend-platform';
|
import { getConfig } from '@edx/frontend-platform';
|
||||||
import { sendTrackEvent, sendTrackingLogEvent } from '@edx/frontend-platform/analytics';
|
import { sendTrackEvent } from '@edx/frontend-platform/analytics';
|
||||||
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
|
import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
|
||||||
import MockAdapter from 'axios-mock-adapter';
|
import MockAdapter from 'axios-mock-adapter';
|
||||||
import Cookies from 'js-cookie';
|
import Cookies from 'js-cookie';
|
||||||
@@ -54,7 +54,7 @@ describe('Outline Tab', () => {
|
|||||||
const goalUrl = `${getConfig().LMS_BASE_URL}/api/course_home/save_course_goal`;
|
const goalUrl = `${getConfig().LMS_BASE_URL}/api/course_home/save_course_goal`;
|
||||||
const masqueradeUrl = `${getConfig().LMS_BASE_URL}/courses/${courseId}/masquerade`;
|
const masqueradeUrl = `${getConfig().LMS_BASE_URL}/courses/${courseId}/masquerade`;
|
||||||
const outlineUrl = `${getConfig().LMS_BASE_URL}/api/course_home/outline/${courseId}`;
|
const outlineUrl = `${getConfig().LMS_BASE_URL}/api/course_home/outline/${courseId}`;
|
||||||
const proctoringInfoUrl = `${getConfig().LMS_BASE_URL}/api/edx_proctoring/v1/user_onboarding/status?is_learning_mfe=true&course_id=${encodeURIComponent(courseId)}&username=MockUser`;
|
const proctoringInfoUrl = `${getConfig().EXAMS_BASE_URL}/api/v1/student/course_id/${encodeURIComponent(courseId)}/onboarding?username=MockUser`;
|
||||||
|
|
||||||
const store = initializeStore();
|
const store = initializeStore();
|
||||||
const defaultMetadata = Factory.build('courseHomeMetadata');
|
const defaultMetadata = Factory.build('courseHomeMetadata');
|
||||||
@@ -132,7 +132,18 @@ describe('Outline Tab', () => {
|
|||||||
expect(expandedSectionNode).toHaveAttribute('aria-expanded', 'true');
|
expect(expandedSectionNode).toHaveAttribute('aria-expanded', 'true');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('includes outline_tab_notifications_slot', async () => {
|
||||||
|
const { courseBlocks } = await buildMinimalCourseBlocks(courseId, 'Title', { resumeBlock: true });
|
||||||
|
setTabData({
|
||||||
|
course_blocks: { blocks: courseBlocks.blocks },
|
||||||
|
});
|
||||||
|
await fetchAndRender();
|
||||||
|
|
||||||
|
expect(screen.getByTestId('org.openedx.frontend.learning.course_outline_tab_notifications.v1')).toBeInTheDocument();
|
||||||
|
});
|
||||||
|
|
||||||
it('handles expand/collapse all button click', async () => {
|
it('handles expand/collapse all button click', async () => {
|
||||||
|
const user = userEvent.setup();
|
||||||
await fetchAndRender();
|
await fetchAndRender();
|
||||||
// Button renders as "Expand All"
|
// Button renders as "Expand All"
|
||||||
const expandButton = screen.getByRole('button', { name: 'Expand all' });
|
const expandButton = screen.getByRole('button', { name: 'Expand all' });
|
||||||
@@ -143,11 +154,11 @@ describe('Outline Tab', () => {
|
|||||||
expect(collapsedSectionNode).toHaveAttribute('aria-expanded', 'false');
|
expect(collapsedSectionNode).toHaveAttribute('aria-expanded', 'false');
|
||||||
|
|
||||||
// Click to expand section
|
// Click to expand section
|
||||||
userEvent.click(expandButton);
|
await user.click(expandButton);
|
||||||
await waitFor(() => expect(collapsedSectionNode).toHaveAttribute('aria-expanded', 'true'));
|
await waitFor(() => expect(collapsedSectionNode).toHaveAttribute('aria-expanded', 'true'));
|
||||||
|
|
||||||
// Click to collapse section
|
// Click to collapse section
|
||||||
userEvent.click(expandButton);
|
await user.click(expandButton);
|
||||||
await waitFor(() => expect(collapsedSectionNode).toHaveAttribute('aria-expanded', 'false'));
|
await waitFor(() => expect(collapsedSectionNode).toHaveAttribute('aria-expanded', 'false'));
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -157,7 +168,7 @@ describe('Outline Tab', () => {
|
|||||||
course_blocks: { blocks: courseBlocks.blocks },
|
course_blocks: { blocks: courseBlocks.blocks },
|
||||||
});
|
});
|
||||||
await fetchAndRender();
|
await fetchAndRender();
|
||||||
expect(screen.getByTitle('Completed section')).toBeInTheDocument();
|
expect(screen.getByLabelText('Completed section')).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('displays correct icon for incomplete assignment', async () => {
|
it('displays correct icon for incomplete assignment', async () => {
|
||||||
@@ -166,7 +177,7 @@ describe('Outline Tab', () => {
|
|||||||
course_blocks: { blocks: courseBlocks.blocks },
|
course_blocks: { blocks: courseBlocks.blocks },
|
||||||
});
|
});
|
||||||
await fetchAndRender();
|
await fetchAndRender();
|
||||||
expect(screen.getByTitle('Incomplete section')).toBeInTheDocument();
|
expect(screen.getByLabelText('Incomplete section')).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('SequenceLink displays link', async () => {
|
it('SequenceLink displays link', async () => {
|
||||||
@@ -265,21 +276,34 @@ describe('Outline Tab', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it('renders show more/less button and handles click', async () => {
|
it('renders show more/less button and handles click', async () => {
|
||||||
|
const user = userEvent.setup();
|
||||||
expect(screen.getByTestId('alert-container-welcome')).toBeInTheDocument();
|
expect(screen.getByTestId('alert-container-welcome')).toBeInTheDocument();
|
||||||
let showMoreButton = screen.getByRole('button', { name: 'Show More' });
|
let showMoreButton = screen.getByRole('button', { name: 'Show More' });
|
||||||
expect(showMoreButton).toBeInTheDocument();
|
expect(showMoreButton).toBeInTheDocument();
|
||||||
|
|
||||||
userEvent.click(showMoreButton);
|
await user.click(showMoreButton);
|
||||||
let showLessButton = screen.getByRole('button', { name: 'Show Less' });
|
let showLessButton = screen.getByRole('button', { name: 'Show Less' });
|
||||||
expect(showLessButton).toBeInTheDocument();
|
expect(showLessButton).toBeInTheDocument();
|
||||||
expect(screen.getByTestId('long-welcome-message-iframe')).toBeInTheDocument();
|
expect(screen.getByTestId('long-welcome-message-iframe')).toBeInTheDocument();
|
||||||
|
|
||||||
userEvent.click(showLessButton);
|
await user.click(showLessButton);
|
||||||
showLessButton = screen.queryByRole('button', { name: 'Show Less' });
|
showLessButton = screen.queryByRole('button', { name: 'Show Less' });
|
||||||
expect(showLessButton).not.toBeInTheDocument();
|
expect(showLessButton).not.toBeInTheDocument();
|
||||||
showMoreButton = screen.getByRole('button', { name: 'Show More' });
|
showMoreButton = screen.getByRole('button', { name: 'Show More' });
|
||||||
expect(showMoreButton).toBeInTheDocument();
|
expect(showMoreButton).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('dismisses message', async () => {
|
||||||
|
expect(screen.getByTestId('alert-container-welcome')).toBeInTheDocument();
|
||||||
|
const dismissButton = screen.queryByRole('button', { name: 'Dismiss' });
|
||||||
|
const expandButton = screen.queryByRole('button', { name: 'Expand all' });
|
||||||
|
|
||||||
|
fireEvent.click(dismissButton);
|
||||||
|
|
||||||
|
expect(expandButton).toHaveFocus();
|
||||||
|
|
||||||
|
expect(screen.queryByText('Welcome Message')).toBeNull();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('ignores comments and misformatted HTML', async () => {
|
it('ignores comments and misformatted HTML', async () => {
|
||||||
@@ -1166,80 +1190,6 @@ describe('Outline Tab', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('Upgrade Card', () => {
|
|
||||||
it('renders title when upgrade is available', async () => {
|
|
||||||
await fetchAndRender();
|
|
||||||
expect(screen.queryByRole('heading', { name: 'Pursue a verified certificate' })).toBeInTheDocument();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('displays link to upgrade', async () => {
|
|
||||||
await fetchAndRender();
|
|
||||||
expect(screen.getByRole('link', { name: 'Upgrade for $149' })).toBeInTheDocument();
|
|
||||||
});
|
|
||||||
|
|
||||||
it('viewing upgrade card sends analytics', async () => {
|
|
||||||
sendTrackEvent.mockClear();
|
|
||||||
sendTrackingLogEvent.mockClear();
|
|
||||||
await fetchAndRender();
|
|
||||||
|
|
||||||
expect(sendTrackEvent).toHaveBeenCalledTimes(1);
|
|
||||||
expect(sendTrackEvent).toHaveBeenCalledWith('Promotion Viewed', {
|
|
||||||
org_key: 'edX',
|
|
||||||
courserun_key: courseId,
|
|
||||||
creative: 'sidebarupsell',
|
|
||||||
name: 'In-Course Verification Prompt',
|
|
||||||
position: 'sidebar-message',
|
|
||||||
promotion_id: 'courseware_verified_certificate_upsell',
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(sendTrackingLogEvent).toHaveBeenCalledTimes(1);
|
|
||||||
expect(sendTrackingLogEvent).toHaveBeenCalledWith('edx.bi.course.upgrade.sidebarupsell.displayed', {
|
|
||||||
org_key: 'edX',
|
|
||||||
courserun_key: courseId,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
it('clicking upgrade link sends analytics', async () => {
|
|
||||||
await fetchAndRender();
|
|
||||||
|
|
||||||
// Clearing after render to remove any events sent on view (ex. 'Promotion Viewed')
|
|
||||||
sendTrackEvent.mockClear();
|
|
||||||
sendTrackingLogEvent.mockClear();
|
|
||||||
const upgradeButton = screen.getByRole('link', { name: 'Upgrade for $149' });
|
|
||||||
|
|
||||||
fireEvent.click(upgradeButton);
|
|
||||||
|
|
||||||
expect(sendTrackEvent).toHaveBeenCalledTimes(2);
|
|
||||||
expect(sendTrackEvent).toHaveBeenNthCalledWith(1, 'Promotion Clicked', {
|
|
||||||
org_key: 'edX',
|
|
||||||
courserun_key: courseId,
|
|
||||||
creative: 'sidebarupsell',
|
|
||||||
name: 'In-Course Verification Prompt',
|
|
||||||
position: 'sidebar-message',
|
|
||||||
promotion_id: 'courseware_verified_certificate_upsell',
|
|
||||||
});
|
|
||||||
expect(sendTrackEvent).toHaveBeenNthCalledWith(2, 'edx.bi.ecommerce.upsell_links_clicked', {
|
|
||||||
org_key: 'edX',
|
|
||||||
courserun_key: courseId,
|
|
||||||
linkCategory: 'green_upgrade',
|
|
||||||
linkName: 'course_home_green',
|
|
||||||
linkType: 'button',
|
|
||||||
pageName: 'course_home',
|
|
||||||
});
|
|
||||||
|
|
||||||
expect(sendTrackingLogEvent).toHaveBeenCalledTimes(2);
|
|
||||||
expect(sendTrackingLogEvent).toHaveBeenNthCalledWith(1, 'edx.bi.course.upgrade.sidebarupsell.clicked', {
|
|
||||||
org_key: 'edX',
|
|
||||||
courserun_key: courseId,
|
|
||||||
});
|
|
||||||
expect(sendTrackingLogEvent).toHaveBeenNthCalledWith(2, 'edx.course.enrollment.upgrade.clicked', {
|
|
||||||
org_key: 'edX',
|
|
||||||
courserun_key: courseId,
|
|
||||||
location: 'sidebar-message',
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('Account Activation Alert', () => {
|
describe('Account Activation Alert', () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
const intersectionObserverMock = () => ({
|
const intersectionObserverMock = () => ({
|
||||||
|
|||||||
@@ -1,137 +0,0 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
|
||||||
import { Collapsible, IconButton, Icon } from '@openedx/paragon';
|
|
||||||
import { faCheckCircle as fasCheckCircle, faMinus, faPlus } from '@fortawesome/free-solid-svg-icons';
|
|
||||||
import { faCheckCircle as farCheckCircle } from '@fortawesome/free-regular-svg-icons';
|
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
|
||||||
|
|
||||||
import { DisabledVisible } from '@openedx/paragon/icons';
|
|
||||||
import SequenceLink from './SequenceLink';
|
|
||||||
import { useModel } from '../../generic/model-store';
|
|
||||||
|
|
||||||
import genericMessages from '../../generic/messages';
|
|
||||||
import messages from './messages';
|
|
||||||
|
|
||||||
const Section = ({
|
|
||||||
courseId,
|
|
||||||
defaultOpen,
|
|
||||||
expand,
|
|
||||||
intl,
|
|
||||||
section,
|
|
||||||
}) => {
|
|
||||||
const {
|
|
||||||
complete,
|
|
||||||
sequenceIds,
|
|
||||||
title,
|
|
||||||
hideFromTOC,
|
|
||||||
} = section;
|
|
||||||
const {
|
|
||||||
courseBlocks: {
|
|
||||||
sequences,
|
|
||||||
},
|
|
||||||
} = useModel('outline', courseId);
|
|
||||||
|
|
||||||
const [open, setOpen] = useState(defaultOpen);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
setOpen(expand);
|
|
||||||
}, [expand]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
setOpen(defaultOpen);
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const sectionTitle = (
|
|
||||||
<div className="d-flex row w-100 m-0">
|
|
||||||
<div className="col-auto p-0">
|
|
||||||
{complete ? (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={fasCheckCircle}
|
|
||||||
fixedWidth
|
|
||||||
className="float-left mt-1 text-success"
|
|
||||||
aria-hidden="true"
|
|
||||||
title={intl.formatMessage(messages.completedSection)}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={farCheckCircle}
|
|
||||||
fixedWidth
|
|
||||||
className="float-left mt-1 text-gray-400"
|
|
||||||
aria-hidden="true"
|
|
||||||
title={intl.formatMessage(messages.incompleteSection)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div className="col-7 ml-3 p-0 font-weight-bold text-dark-500">
|
|
||||||
<span className="align-middle col-6">{title}</span>
|
|
||||||
<span className="sr-only">
|
|
||||||
, {intl.formatMessage(complete ? messages.completedSection : messages.incompleteSection)}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
{hideFromTOC && (
|
|
||||||
<div className="row">
|
|
||||||
{hideFromTOC && (
|
|
||||||
<span className="small d-flex align-content-end">
|
|
||||||
<Icon className="mr-2" src={DisabledVisible} data-testid="hide-from-toc-section-icon" />
|
|
||||||
<span data-testid="hide-from-toc-section-text">
|
|
||||||
{intl.formatMessage(messages.hiddenSection)}
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<li>
|
|
||||||
<Collapsible
|
|
||||||
className="mb-2"
|
|
||||||
styling="card-lg"
|
|
||||||
title={sectionTitle}
|
|
||||||
open={open}
|
|
||||||
onToggle={() => { setOpen(!open); }}
|
|
||||||
iconWhenClosed={(
|
|
||||||
<IconButton
|
|
||||||
alt={intl.formatMessage(messages.openSection)}
|
|
||||||
icon={faPlus}
|
|
||||||
onClick={() => { setOpen(true); }}
|
|
||||||
size="sm"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
iconWhenOpen={(
|
|
||||||
<IconButton
|
|
||||||
alt={intl.formatMessage(genericMessages.close)}
|
|
||||||
icon={faMinus}
|
|
||||||
onClick={() => { setOpen(false); }}
|
|
||||||
size="sm"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<ol className="list-unstyled">
|
|
||||||
{sequenceIds.map((sequenceId, index) => (
|
|
||||||
<SequenceLink
|
|
||||||
key={sequenceId}
|
|
||||||
id={sequenceId}
|
|
||||||
courseId={courseId}
|
|
||||||
sequence={sequences[sequenceId]}
|
|
||||||
first={index === 0}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</ol>
|
|
||||||
</Collapsible>
|
|
||||||
</li>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
Section.propTypes = {
|
|
||||||
courseId: PropTypes.string.isRequired,
|
|
||||||
defaultOpen: PropTypes.bool.isRequired,
|
|
||||||
expand: PropTypes.bool.isRequired,
|
|
||||||
intl: intlShape.isRequired,
|
|
||||||
section: PropTypes.shape().isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(Section);
|
|
||||||
@@ -1,148 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
import PropTypes from 'prop-types';
|
|
||||||
import classNames from 'classnames';
|
|
||||||
import { Link } from 'react-router-dom';
|
|
||||||
import {
|
|
||||||
FormattedMessage,
|
|
||||||
FormattedTime,
|
|
||||||
injectIntl,
|
|
||||||
intlShape,
|
|
||||||
} from '@edx/frontend-platform/i18n';
|
|
||||||
import { faCheckCircle as fasCheckCircle } from '@fortawesome/free-solid-svg-icons';
|
|
||||||
import { faCheckCircle as farCheckCircle } from '@fortawesome/free-regular-svg-icons';
|
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
|
||||||
|
|
||||||
import { Icon } from '@openedx/paragon';
|
|
||||||
import { Block } from '@openedx/paragon/icons';
|
|
||||||
import EffortEstimate from '../../shared/effort-estimate';
|
|
||||||
import { useModel } from '../../generic/model-store';
|
|
||||||
import messages from './messages';
|
|
||||||
|
|
||||||
const SequenceLink = ({
|
|
||||||
id,
|
|
||||||
intl,
|
|
||||||
courseId,
|
|
||||||
first,
|
|
||||||
sequence,
|
|
||||||
}) => {
|
|
||||||
const {
|
|
||||||
complete,
|
|
||||||
description,
|
|
||||||
due,
|
|
||||||
showLink,
|
|
||||||
title,
|
|
||||||
hideFromTOC,
|
|
||||||
} = sequence;
|
|
||||||
const {
|
|
||||||
userTimezone,
|
|
||||||
} = useModel('outline', courseId);
|
|
||||||
|
|
||||||
const timezoneFormatArgs = userTimezone ? { timeZone: userTimezone } : {};
|
|
||||||
|
|
||||||
const coursewareUrl = <Link to={`/course/${courseId}/${id}`}>{title}</Link>;
|
|
||||||
const displayTitle = showLink ? coursewareUrl : title;
|
|
||||||
|
|
||||||
const dueDateMessage = (
|
|
||||||
<FormattedMessage
|
|
||||||
id="learning.outline.sequence-due-date-set"
|
|
||||||
defaultMessage="{description} due {assignmentDue}"
|
|
||||||
description="Used below an assignment title"
|
|
||||||
values={{
|
|
||||||
assignmentDue: (
|
|
||||||
<FormattedTime
|
|
||||||
key={`${id}-due`}
|
|
||||||
day="numeric"
|
|
||||||
month="short"
|
|
||||||
year="numeric"
|
|
||||||
timeZoneName="short"
|
|
||||||
value={due}
|
|
||||||
{...timezoneFormatArgs}
|
|
||||||
/>
|
|
||||||
),
|
|
||||||
description: description || '',
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
const noDueDateMessage = (
|
|
||||||
<FormattedMessage
|
|
||||||
id="learning.outline.sequence-due-date-not-set"
|
|
||||||
defaultMessage="{description}"
|
|
||||||
description="Used below an assignment title"
|
|
||||||
values={{
|
|
||||||
assignmentDue: (
|
|
||||||
<FormattedTime
|
|
||||||
key={`${id}-due`}
|
|
||||||
day="numeric"
|
|
||||||
month="short"
|
|
||||||
year="numeric"
|
|
||||||
timeZoneName="short"
|
|
||||||
value={due}
|
|
||||||
{...timezoneFormatArgs}
|
|
||||||
/>
|
|
||||||
),
|
|
||||||
description: description || '',
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<li>
|
|
||||||
<div className={classNames('', { 'mt-2 pt-2 border-top border-light': !first })}>
|
|
||||||
<div className="row w-100 m-0">
|
|
||||||
<div className="col-auto p-0">
|
|
||||||
{complete ? (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={fasCheckCircle}
|
|
||||||
fixedWidth
|
|
||||||
className="float-left text-success mt-1"
|
|
||||||
aria-hidden="true"
|
|
||||||
title={intl.formatMessage(messages.completedAssignment)}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<FontAwesomeIcon
|
|
||||||
icon={farCheckCircle}
|
|
||||||
fixedWidth
|
|
||||||
className="float-left text-gray-400 mt-1"
|
|
||||||
aria-hidden="true"
|
|
||||||
title={intl.formatMessage(messages.incompleteAssignment)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div className="col-10 p-0 ml-3 text-break">
|
|
||||||
<span className="align-middle">{displayTitle}</span>
|
|
||||||
<span className="sr-only">
|
|
||||||
, {intl.formatMessage(complete ? messages.completedAssignment : messages.incompleteAssignment)}
|
|
||||||
</span>
|
|
||||||
<EffortEstimate className="ml-3 align-middle" block={sequence} />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{hideFromTOC && (
|
|
||||||
<div className="row w-100 my-2 mx-4 pl-3">
|
|
||||||
<span className="small d-flex">
|
|
||||||
<Icon className="mr-2" src={Block} data-testid="hide-from-toc-sequence-link-icon" />
|
|
||||||
<span data-testid="hide-from-toc-sequence-link-text">
|
|
||||||
{intl.formatMessage(messages.hiddenSequenceLink)}
|
|
||||||
</span>
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className="row w-100 m-0 ml-3 pl-3">
|
|
||||||
<small className="text-body pl-2">
|
|
||||||
{due ? dueDateMessage : noDueDateMessage}
|
|
||||||
</small>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
SequenceLink.propTypes = {
|
|
||||||
id: PropTypes.string.isRequired,
|
|
||||||
intl: intlShape.isRequired,
|
|
||||||
courseId: PropTypes.string.isRequired,
|
|
||||||
first: PropTypes.bool.isRequired,
|
|
||||||
sequence: PropTypes.shape().isRequired,
|
|
||||||
};
|
|
||||||
|
|
||||||
export default injectIntl(SequenceLink);
|
|
||||||
@@ -3,8 +3,7 @@ import PropTypes from 'prop-types';
|
|||||||
import {
|
import {
|
||||||
FormattedDate,
|
FormattedDate,
|
||||||
FormattedMessage,
|
FormattedMessage,
|
||||||
injectIntl,
|
useIntl,
|
||||||
intlShape,
|
|
||||||
} from '@edx/frontend-platform/i18n';
|
} from '@edx/frontend-platform/i18n';
|
||||||
import { Alert, Button } from '@openedx/paragon';
|
import { Alert, Button } from '@openedx/paragon';
|
||||||
import { useDispatch } from 'react-redux';
|
import { useDispatch } from 'react-redux';
|
||||||
@@ -25,7 +24,8 @@ export const CERT_STATUS_TYPE = {
|
|||||||
UNVERIFIED: 'unverified',
|
UNVERIFIED: 'unverified',
|
||||||
};
|
};
|
||||||
|
|
||||||
const CertificateStatusAlert = ({ intl, payload }) => {
|
const CertificateStatusAlert = ({ payload }) => {
|
||||||
|
const intl = useIntl();
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const {
|
const {
|
||||||
certificateAvailableDate,
|
certificateAvailableDate,
|
||||||
@@ -192,7 +192,6 @@ const CertificateStatusAlert = ({ intl, payload }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
CertificateStatusAlert.propTypes = {
|
CertificateStatusAlert.propTypes = {
|
||||||
intl: intlShape.isRequired,
|
|
||||||
payload: PropTypes.shape({
|
payload: PropTypes.shape({
|
||||||
certificateAvailableDate: PropTypes.string,
|
certificateAvailableDate: PropTypes.string,
|
||||||
certStatus: PropTypes.string,
|
certStatus: PropTypes.string,
|
||||||
@@ -210,4 +209,4 @@ CertificateStatusAlert.propTypes = {
|
|||||||
}).isRequired,
|
}).isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(CertificateStatusAlert);
|
export default CertificateStatusAlert;
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import { getConfig } from '@edx/frontend-platform';
|
import { getConfig } from '@edx/frontend-platform';
|
||||||
import { injectIntl, intlShape, FormattedMessage } from '@edx/frontend-platform/i18n';
|
import { useIntl, FormattedMessage } from '@edx/frontend-platform/i18n';
|
||||||
import { getLoginRedirectUrl } from '@edx/frontend-platform/auth';
|
import { getLoginRedirectUrl } from '@edx/frontend-platform/auth';
|
||||||
import { Alert, Button, Hyperlink } from '@openedx/paragon';
|
import { Alert, Button, Hyperlink } from '@openedx/paragon';
|
||||||
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
||||||
@@ -14,7 +14,8 @@ import outlineMessages from '../../messages';
|
|||||||
import useEnrollClickHandler from '../../../../alerts/enrollment-alert/clickHook';
|
import useEnrollClickHandler from '../../../../alerts/enrollment-alert/clickHook';
|
||||||
import { useModel } from '../../../../generic/model-store';
|
import { useModel } from '../../../../generic/model-store';
|
||||||
|
|
||||||
const PrivateCourseAlert = ({ intl, payload }) => {
|
const PrivateCourseAlert = ({ payload }) => {
|
||||||
|
const intl = useIntl();
|
||||||
const {
|
const {
|
||||||
anonymousUser,
|
anonymousUser,
|
||||||
canEnroll,
|
canEnroll,
|
||||||
@@ -103,7 +104,6 @@ const PrivateCourseAlert = ({ intl, payload }) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
PrivateCourseAlert.propTypes = {
|
PrivateCourseAlert.propTypes = {
|
||||||
intl: intlShape.isRequired,
|
|
||||||
payload: PropTypes.shape({
|
payload: PropTypes.shape({
|
||||||
anonymousUser: PropTypes.bool,
|
anonymousUser: PropTypes.bool,
|
||||||
canEnroll: PropTypes.bool,
|
canEnroll: PropTypes.bool,
|
||||||
@@ -111,4 +111,4 @@ PrivateCourseAlert.propTypes = {
|
|||||||
}).isRequired,
|
}).isRequired,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default injectIntl(PrivateCourseAlert);
|
export default PrivateCourseAlert;
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user