diff --git a/src/frontend/js/components/DashboardCourseList/_styles.scss b/src/frontend/js/components/DashboardCourseList/_styles.scss
index b14eb45853..d5a203a41f 100644
--- a/src/frontend/js/components/DashboardCourseList/_styles.scss
+++ b/src/frontend/js/components/DashboardCourseList/_styles.scss
@@ -13,9 +13,34 @@ $r-course-glimpse-dashboard-gutter: rem-calc(30px) !default;
margin-bottom: 10px;
}
+ //
+ // Loading transition
+ //
+ .fade-in-enter {
+ opacity: 0;
+ }
+ .fade-in-enter-active {
+ opacity: 1;
+ transition: opacity 300ms;
+ }
+ .fade-in-exit {
+ display: none;
+ opacity: 0;
+ }
+ .fade-in-exit-active {
+ display: none;
+ opacity: 0;
+ }
+
//
// Course Glimpse in dashboards
//
+ &__placeholder {
+ display: flex;
+ justify-content: center;
+ align-items: center;
+ height: rem-calc(360px);
+ }
.dashboard {
&__course-glimpse-list {
diff --git a/src/frontend/js/components/DashboardCourseList/index.tsx b/src/frontend/js/components/DashboardCourseList/index.tsx
index 96da6ef4bc..473c28f595 100644
--- a/src/frontend/js/components/DashboardCourseList/index.tsx
+++ b/src/frontend/js/components/DashboardCourseList/index.tsx
@@ -1,3 +1,5 @@
+import { useRef } from 'react';
+import { CSSTransition } from 'react-transition-group';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
import { Link } from 'react-router-dom';
import queryString from 'query-string';
@@ -7,6 +9,7 @@ import { useCourses, TeacherCourseSearchFilters } from 'hooks/useCourses';
import { getDashboardRoutePath } from 'widgets/Dashboard/utils/dashboardRoutes';
import { TeacherDashboardPaths } from 'widgets/Dashboard/utils/teacherRouteMessages';
import context from 'utils/context';
+import useIsLoading from 'hooks/useIsLoading';
const messages = defineMessages({
loading: {
@@ -30,6 +33,10 @@ const DashboardCourseList = ({ titleTranslated, filters }: DashboardCourseListPr
states: { fetching },
} = coursesResults;
+ const isLoading = useIsLoading([fetching], 300);
+
+ const fadeInNodeRef = useRef(null);
+
return (
{titleTranslated && (
@@ -41,20 +48,31 @@ const DashboardCourseList = ({ titleTranslated, filters }: DashboardCourseListPr
{titleTranslated}
)}
- {fetching && (
-
-
-
-
-
- )}
- {!fetching && courses.length && (
-
+ {isLoading && (
+
+
+
+
+
+
+
)}
+
+
+
+
+
+
);
};
diff --git a/src/frontend/js/hooks/useIsLoading.ts b/src/frontend/js/hooks/useIsLoading.ts
new file mode 100644
index 0000000000..0a68096a1b
--- /dev/null
+++ b/src/frontend/js/hooks/useIsLoading.ts
@@ -0,0 +1,43 @@
+import { useEffect, useState } from 'react';
+
+// await new Promise(resolve => setTimeout(resolve, 1000))
+
+const useIsLoading = (loadingStates: boolean[], delayInMillisecond = 500) => {
+ const [isLoading, setIsLoading] = useState(!loadingStates.every((v) => v === false));
+ const [timerId, setTimerId] = useState>();
+ const [canStopLoading, setCanStopLoading] = useState(true);
+
+ const startLoading = () => {
+ setCanStopLoading(false);
+ setIsLoading(true);
+ setTimerId(
+ setTimeout(() => {
+ setCanStopLoading(true);
+ }, delayInMillisecond),
+ );
+ };
+
+ const stopLoading = () => {
+ setIsLoading(false);
+ clearTimeout(timerId);
+ setTimerId(undefined);
+ };
+
+ useEffect(() => {
+ const newIsLoading = !loadingStates.every((v) => v === false);
+ if (newIsLoading && !timerId) {
+ startLoading();
+ }
+ }, [loadingStates, timerId]);
+
+ useEffect(() => {
+ const newIsLoading = !loadingStates.every((v) => v === false);
+ if (isLoading && !newIsLoading && canStopLoading) {
+ stopLoading();
+ }
+ }, [loadingStates, isLoading, canStopLoading]);
+
+ return isLoading;
+};
+
+export default useIsLoading;
diff --git a/src/frontend/package.json b/src/frontend/package.json
index f614008750..e66dfeb14a 100644
--- a/src/frontend/package.json
+++ b/src/frontend/package.json
@@ -75,6 +75,7 @@
"@types/react-autosuggest": "10.1.6",
"@types/react-dom": "18.0.11",
"@types/react-modal": "3.13.1",
+ "@types/react-transition-group": "4.4.5",
"@types/uuid": "9.0.1",
"@typescript-eslint/eslint-plugin": "5.58.0",
"@typescript-eslint/parser": "5.58.0",
@@ -122,6 +123,7 @@
"react-intl": "6.3.2",
"react-modal": "3.16.1",
"react-router-dom": "6.10.0",
+ "react-transition-group": "4.4.5",
"sass": "1.62.0",
"source-map-loader": "4.0.1",
"storybook": "7.0.3",
diff --git a/src/frontend/yarn.lock b/src/frontend/yarn.lock
index d20ae85c3d..b366695530 100644
--- a/src/frontend/yarn.lock
+++ b/src/frontend/yarn.lock
@@ -1076,7 +1076,7 @@
resolved "https://registry.yarnpkg.com/@babel/regjsgen/-/regjsgen-0.8.0.tgz#f0ba69b075e1f05fb2825b7fad991e7adbb18310"
integrity sha512-x/rqGMdzj+fWZvCOYForTghzbtqPDZ5gPwaoNGHdgDfF2QA/XZbCBp4Moo5scrkAMPhB7z26XM/AaHuIJdgauA==
-"@babel/runtime@^7.0.0", "@babel/runtime@^7.12.5", "@babel/runtime@^7.14.8", "@babel/runtime@^7.17.8", "@babel/runtime@^7.20.7", "@babel/runtime@^7.7.6", "@babel/runtime@^7.8.4", "@babel/runtime@^7.9.2":
+"@babel/runtime@^7.0.0", "@babel/runtime@^7.12.5", "@babel/runtime@^7.14.8", "@babel/runtime@^7.17.8", "@babel/runtime@^7.20.7", "@babel/runtime@^7.5.5", "@babel/runtime@^7.7.6", "@babel/runtime@^7.8.4", "@babel/runtime@^7.8.7", "@babel/runtime@^7.9.2":
version "7.21.0"
resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.21.0.tgz#5b55c9d394e5fcf304909a8b00c07dc217b56673"
integrity sha512-xwII0//EObnq89Ji5AKYQaRYiW/nZ3llSv29d49IuxPhKbtJoLP+9QUUZ4nVragQVtaVGeZrpB+ZtG/Pdy/POw==
@@ -3268,6 +3268,13 @@
dependencies:
"@types/react" "*"
+"@types/react-transition-group@4.4.5":
+ version "4.4.5"
+ resolved "https://registry.yarnpkg.com/@types/react-transition-group/-/react-transition-group-4.4.5.tgz#aae20dcf773c5aa275d5b9f7cdbca638abc5e416"
+ integrity sha512-juKD/eiSM3/xZYzjuzH6ZwpP+/lejltmiS3QEzV/vmb/Q8+HfDmxu+Baga8UEMGBqV88Nbg4l2hY/K2DkyaLLA==
+ dependencies:
+ "@types/react" "*"
+
"@types/react@*", "@types/react@16 || 17 || 18", "@types/react@18.0.34", "@types/react@>=16":
version "18.0.34"
resolved "https://registry.yarnpkg.com/@types/react/-/react-18.0.34.tgz#e553444a578f023e6e1ac499514688fb80b0a984"
@@ -5008,6 +5015,14 @@ dom-converter@^0.2.0:
dependencies:
utila "~0.4"
+dom-helpers@^5.0.1:
+ version "5.2.1"
+ resolved "https://registry.yarnpkg.com/dom-helpers/-/dom-helpers-5.2.1.tgz#d9400536b2bf8225ad98fe052e029451ac40e902"
+ integrity sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==
+ dependencies:
+ "@babel/runtime" "^7.8.7"
+ csstype "^3.0.2"
+
dom-serializer@^1.0.1:
version "1.4.1"
resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-1.4.1.tgz#de5d41b1aea290215dc45a6dae8adcf1d32e2d30"
@@ -8676,7 +8691,7 @@ prompts@^2.0.1, prompts@^2.4.0:
kleur "^3.0.3"
sisteransi "^1.0.5"
-prop-types@^15.7.2, prop-types@^15.8.1:
+prop-types@^15.6.2, prop-types@^15.7.2, prop-types@^15.8.1:
version "15.8.1"
resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.8.1.tgz#67d87bf1a694f48435cf332c24af10214a3140b5"
integrity sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==
@@ -8981,6 +8996,16 @@ react-themeable@^1.1.0:
dependencies:
object-assign "^3.0.0"
+react-transition-group@4.4.5:
+ version "4.4.5"
+ resolved "https://registry.yarnpkg.com/react-transition-group/-/react-transition-group-4.4.5.tgz#e53d4e3f3344da8521489fbef8f2581d42becdd1"
+ integrity sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==
+ dependencies:
+ "@babel/runtime" "^7.5.5"
+ dom-helpers "^5.0.1"
+ loose-envify "^1.4.0"
+ prop-types "^15.6.2"
+
react@18.2.0:
version "18.2.0"
resolved "https://registry.yarnpkg.com/react/-/react-18.2.0.tgz#555bd98592883255fa00de14f1151a917b5d77d5"