Добавил спиннер (#5)
This commit is contained in:
		
							parent
							
								
									f09536f4aa
								
							
						
					
					
						commit
						b858d70675
					
				| @ -4,6 +4,7 @@ import { Global, css  } from '@emotion/react' | |||||||
| import { BrowserRouter } from 'react-router-dom'; | import { BrowserRouter } from 'react-router-dom'; | ||||||
| import ruLocale from 'dayjs/locale/ru'; | import ruLocale from 'dayjs/locale/ru'; | ||||||
| import dayjs from 'dayjs'; | import dayjs from 'dayjs'; | ||||||
|  | import { ChakraProvider } from '@chakra-ui/react' | ||||||
| 
 | 
 | ||||||
| import { Dashboard } from './dashboard'; | import { Dashboard } from './dashboard'; | ||||||
| 
 | 
 | ||||||
| @ -11,6 +12,7 @@ dayjs.locale('ru', ruLocale); | |||||||
| 
 | 
 | ||||||
| const App = ({ store }) => { | const App = ({ store }) => { | ||||||
|     return( |     return( | ||||||
|  |       <ChakraProvider> | ||||||
|           <BrowserRouter> |           <BrowserRouter> | ||||||
|               <Helmet> |               <Helmet> | ||||||
|                   <meta name="viewport" content="width=device-width, user-scalable=no" /> |                   <meta name="viewport" content="width=device-width, user-scalable=no" /> | ||||||
| @ -18,7 +20,6 @@ const App = ({ store }) => { | |||||||
|               </Helmet> |               </Helmet> | ||||||
|               <Global |               <Global | ||||||
|                   styles={css` |                   styles={css` | ||||||
|                     * { box-sizing: border-box; } |  | ||||||
|                       html { |                       html { | ||||||
|                           height: 100%; |                           height: 100%; | ||||||
|                           max-width: 100%; |                           max-width: 100%; | ||||||
| @ -40,7 +41,8 @@ const App = ({ store }) => { | |||||||
|                           radial-gradient( |                           radial-gradient( | ||||||
|                               farthest-side at bottom right, |                               farthest-side at bottom right, | ||||||
|                               rgba(0, 195, 255, 0.648),  |                               rgba(0, 195, 255, 0.648),  | ||||||
|                             rgba(255, 255, 255, 0) 65% |                               rgb(239 244 246) 65% | ||||||
|  |                               /* rgba(255, 255, 255, 0) 65% */ | ||||||
|                           ); |                           ); | ||||||
|                           height: 100%; |                           height: 100%; | ||||||
|                           font-family: KiyosunaSans, Montserrat, RFKrabuler, sans-serif; |                           font-family: KiyosunaSans, Montserrat, RFKrabuler, sans-serif; | ||||||
| @ -77,6 +79,7 @@ const App = ({ store }) => { | |||||||
|               /> |               /> | ||||||
|               <Dashboard store={store} /> |               <Dashboard store={store} /> | ||||||
|           </BrowserRouter> |           </BrowserRouter> | ||||||
|  |         </ChakraProvider> | ||||||
|     ) |     ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,34 +1,82 @@ | |||||||
| import React, { useEffect, Suspense } from "react"; | import React, { useEffect, Suspense } from 'react' | ||||||
| import { Routes, Route, useNavigate } from "react-router-dom"; | import { Routes, Route, useNavigate } from 'react-router-dom' | ||||||
| import { Provider } from "react-redux"; | import { Provider } from 'react-redux' | ||||||
| 
 | 
 | ||||||
| import { | import { | ||||||
|   CourseListPage, |   CourseListPage, | ||||||
|   LessonDetailsPage, |   LessonDetailsPage, | ||||||
|   LessonListPage, |   LessonListPage, | ||||||
|   UserPage |   UserPage, | ||||||
| } from './pages' | } from './pages' | ||||||
| import { getNavigationsValue } from "@ijl/cli"; | import { getNavigationsValue } from '@ijl/cli' | ||||||
|  | import { Box, Container, Spinner, VStack } from '@chakra-ui/react' | ||||||
| 
 | 
 | ||||||
| const Redirect = ({ path }) => { | const Redirect = ({ path }) => { | ||||||
|   const navigate = useNavigate(); |   const navigate = useNavigate() | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     navigate(path); |     navigate(path) | ||||||
|   }, []); |   }, []) | ||||||
| 
 | 
 | ||||||
|   return null; |   return null | ||||||
| }; | } | ||||||
| 
 | 
 | ||||||
| const Wrapper = ({ children }) => <Suspense fallback="...">{children}</Suspense> | const Wrapper = ({ children }) => ( | ||||||
|  |   <Suspense | ||||||
|  |     fallback={ | ||||||
|  |       <Container> | ||||||
|  |         <VStack> | ||||||
|  |           <Box mt="150"> | ||||||
|  |           <Spinner | ||||||
|  |             thickness="4px" | ||||||
|  |             speed="0.65s" | ||||||
|  |             emptyColor="gray.200" | ||||||
|  |             color="blue.500" | ||||||
|  |             size="xl" | ||||||
|  |           /></Box> | ||||||
|  |         </VStack> | ||||||
|  |       </Container> | ||||||
|  |     } | ||||||
|  |   > | ||||||
|  |     {children} | ||||||
|  |   </Suspense> | ||||||
|  | ) | ||||||
| 
 | 
 | ||||||
| export const Dashboard = ({ store }) => ( | export const Dashboard = ({ store }) => ( | ||||||
|   <Provider store={store}> |   <Provider store={store}> | ||||||
|     <Routes> |     <Routes> | ||||||
|       <Route path={getNavigationsValue('journal.main')} element={<Wrapper><CourseListPage /></Wrapper>} /> |       <Route | ||||||
|       <Route path={`${getNavigationsValue('journal.main')}/lessons-list/:courseId`} element={<Wrapper><LessonListPage /></Wrapper>} /> |         path={getNavigationsValue('journal.main')} | ||||||
|       <Route path={`${getNavigationsValue('journal.main')}/u/:lessonId/:accessId`} element={<Wrapper><UserPage /></Wrapper>} /> |         element={ | ||||||
|       <Route path={`${getNavigationsValue('journal.main')}/lesson/:courseId/:lessonId`} element={<Wrapper><LessonDetailsPage /></Wrapper>} /> |           <Wrapper> | ||||||
|  |             <CourseListPage /> | ||||||
|  |           </Wrapper> | ||||||
|  |         } | ||||||
|  |       /> | ||||||
|  |       <Route | ||||||
|  |         path={`${getNavigationsValue('journal.main')}/lessons-list/:courseId`} | ||||||
|  |         element={ | ||||||
|  |           <Wrapper> | ||||||
|  |             <LessonListPage /> | ||||||
|  |           </Wrapper> | ||||||
|  |         } | ||||||
|  |       /> | ||||||
|  |       <Route | ||||||
|  |         path={`${getNavigationsValue('journal.main')}/u/:lessonId/:accessId`} | ||||||
|  |         element={ | ||||||
|  |           <Wrapper> | ||||||
|  |             <UserPage /> | ||||||
|  |           </Wrapper> | ||||||
|  |         } | ||||||
|  |       /> | ||||||
|  |       <Route | ||||||
|  |         path={`${getNavigationsValue('journal.main')}/lesson/:courseId/:lessonId`} | ||||||
|  |         element={ | ||||||
|  |           <Wrapper> | ||||||
|  |             <LessonDetailsPage /> | ||||||
|  |           </Wrapper> | ||||||
|  |         } | ||||||
|  |       /> | ||||||
|     </Routes> |     </Routes> | ||||||
|   </Provider> |   </Provider> | ||||||
| ); | ) | ||||||
|  | |||||||
| @ -1,8 +1,7 @@ | |||||||
| import React, { useCallback, useEffect, useRef, useState } from 'react' | import React, { useCallback, useEffect, useRef, useState } from 'react' | ||||||
| import dayjs from 'dayjs' | import dayjs from 'dayjs' | ||||||
| import { Link } from 'react-router-dom' | import { Link as ConnectedLink } from 'react-router-dom' | ||||||
| import { getConfigValue, getNavigationsValue } from '@ijl/cli' | import { getNavigationsValue } from '@ijl/cli' | ||||||
| 
 |  | ||||||
| import { | import { | ||||||
|   Box, |   Box, | ||||||
|   CardHeader, |   CardHeader, | ||||||
| @ -12,107 +11,191 @@ import { | |||||||
|   Stack, |   Stack, | ||||||
|   StackDivider, |   StackDivider, | ||||||
|   Button, |   Button, | ||||||
|   UnorderedList, |   Card, | ||||||
|   Heading, |   Heading, | ||||||
|   Tooltip, |   Tooltip, | ||||||
|  |   Spinner, | ||||||
|  |   Container, | ||||||
|  |   VStack, | ||||||
|  |   Link, | ||||||
|  |   Input, | ||||||
|  |   CloseButton, | ||||||
|  |   FormControl, | ||||||
|  |   FormLabel, | ||||||
|  |   FormHelperText, | ||||||
|  |   Center, | ||||||
|  |   FormErrorMessage, | ||||||
| } from '@chakra-ui/react' | } from '@chakra-ui/react' | ||||||
|  | import { useForm, Controller } from 'react-hook-form' | ||||||
| 
 | 
 | ||||||
| import { | import { ErrorSpan, MainWrapper } from './style' | ||||||
|   ArrowImg, |  | ||||||
|   IconButton, |  | ||||||
|   InputElement, |  | ||||||
|   InputLabel, |  | ||||||
|   InputWrapper, |  | ||||||
|   StartWrapper, |  | ||||||
|   Papper, |  | ||||||
|   ErrorSpan, |  | ||||||
|   Cross, |  | ||||||
|   AddButton, |  | ||||||
|   MainWrapper, |  | ||||||
|   StyledCard, |  | ||||||
| } from './style' |  | ||||||
| 
 | 
 | ||||||
| import { linkOpen, moreDetails } from '../assets' |  | ||||||
| 
 |  | ||||||
| import arrow from '../assets/36-arrow-right.svg' |  | ||||||
| import { keycloak } from '../__data__/kc' |  | ||||||
| import { useAppSelector } from '../__data__/store' | import { useAppSelector } from '../__data__/store' | ||||||
| import { api } from '../__data__/api/api' | import { api } from '../__data__/api/api' | ||||||
| import { isTeacher } from '../utils/user' | import { isTeacher } from '../utils/user' | ||||||
|  | import { AddIcon, ArrowDownIcon, ArrowUpIcon, LinkIcon } from '@chakra-ui/icons' | ||||||
|  | 
 | ||||||
|  | interface NewCourseForm { | ||||||
|  |   startDt: string | ||||||
|  |   name: string | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| const CoursesList = () => { | const CoursesList = () => { | ||||||
|   const user = useAppSelector((s) => s.user) |   const user = useAppSelector((s) => s.user) | ||||||
|   const { data, isLoading, error } = api.useCoursesListQuery() |   const { data, isLoading, error } = api.useCoursesListQuery() | ||||||
|   const [createCourse, crcQuery] = api.useCreateUpdateCourseMutation() |   const [createUpdateCourse, crucQuery] = api.useCreateUpdateCourseMutation() | ||||||
|   const [value, setValue] = useState('') |   const [value, setValue] = useState('') | ||||||
|   const [showForm, setShowForm] = useState(false) |   const [showForm, setShowForm] = useState(false) | ||||||
|   const [showDetails, setShowDetails] = useState(false) |   const [courseDetailsOpenedId, setCourseDetailsOpenedId] = useState< | ||||||
|  |     string | null | ||||||
|  |   >(null) | ||||||
| 
 | 
 | ||||||
|   const handleChange = useCallback( |   const { control, handleSubmit, reset } = useForm<NewCourseForm>({ | ||||||
|     (event) => { |     mode: 'all', | ||||||
|       setValue(event.target.value.toUpperCase()) |   }) | ||||||
|     }, | 
 | ||||||
|     [setValue], |   const onSubmit = ({ startDt, name }) => { | ||||||
|   ) |     createUpdateCourse({ name, startDt }) | ||||||
|   const handleSubmit = useCallback( |   } | ||||||
|     (event) => { |  | ||||||
|       event.preventDefault() |  | ||||||
|       createCourse({ name: value }) |  | ||||||
|     }, |  | ||||||
|     [value], |  | ||||||
|   ) |  | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     if (crcQuery.isSuccess) { |     if (crucQuery.isSuccess) { | ||||||
|       setValue('') |       reset() | ||||||
|  |     } | ||||||
|  |   }, [crucQuery.isSuccess]) | ||||||
|  | 
 | ||||||
|  |   if (isLoading) { | ||||||
|  |     return ( | ||||||
|  |       <Container maxW="container.xl"> | ||||||
|  |         <Center h="300px"> | ||||||
|  |           <Spinner | ||||||
|  |             thickness="4px" | ||||||
|  |             speed="0.65s" | ||||||
|  |             emptyColor="gray.200" | ||||||
|  |             color="blue.500" | ||||||
|  |             size="xl" | ||||||
|  |           /> | ||||||
|  |         </Center> | ||||||
|  |       </Container> | ||||||
|  |     ) | ||||||
|   } |   } | ||||||
|   }, [crcQuery.isSuccess]) |  | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <MainWrapper> |     <Container maxW="container.xl"> | ||||||
|       <StartWrapper> |  | ||||||
|       {isTeacher(user) && ( |       {isTeacher(user) && ( | ||||||
|           <> |         <Box mt="15" mb="15"> | ||||||
|             {showForm ? ( |           {!showForm ? ( | ||||||
|               <Papper> |             <Card align="left"> | ||||||
|                 <Cross role="button" onClick={() => setShowForm(false)}> |               <CardHeader display="flex"> | ||||||
|                   X |                 <Heading as="h2" mt="0"> | ||||||
|                 </Cross> |                   Создание курса | ||||||
|                 <form onSubmit={handleSubmit}> |                 </Heading> | ||||||
|                   <InputWrapper> |                 <CloseButton ml="auto" onClick={() => setShowForm(false)} /> | ||||||
|                     <InputLabel htmlFor="input"> |               </CardHeader> | ||||||
|                       Название новой лекции: |               <CardBody> | ||||||
|                     </InputLabel> |                 <form onSubmit={handleSubmit(onSubmit)}> | ||||||
|                     <InputElement |                   <VStack spacing={10} align="left"> | ||||||
|  |                     <Controller | ||||||
|  |                       control={control} | ||||||
|  |                       name="startDt" | ||||||
|  |                       rules={{ required: 'Обязательное поле' }} | ||||||
|  |                       render={({ field, fieldState }) => ( | ||||||
|  |                         <FormControl isRequired> | ||||||
|  |                           <FormLabel>Дата начала</FormLabel> | ||||||
|  |                           <Input | ||||||
|  |                             {...field} | ||||||
|  |                             placeholder="Select Date and Time" | ||||||
|  |                             size="md" | ||||||
|  |                             type="datetime-local" | ||||||
|  |                           /> | ||||||
|  |                           {fieldState.error ? ( | ||||||
|  |                             <FormErrorMessage> | ||||||
|  |                               {fieldState.error.message} | ||||||
|  |                             </FormErrorMessage> | ||||||
|  |                           ) : ( | ||||||
|  |                             <FormHelperText> | ||||||
|  |                               Укажите дату начала курса | ||||||
|  |                             </FormHelperText> | ||||||
|  |                           )} | ||||||
|  |                         </FormControl> | ||||||
|  |                       )} | ||||||
|  |                     /> | ||||||
|  |                     <FormControl isRequired> | ||||||
|  |                       <FormLabel>Название новой лекции:</FormLabel> | ||||||
|  |                       <Input | ||||||
|                         value={value} |                         value={value} | ||||||
|                         onChange={handleChange} |                         onChange={handleChange} | ||||||
|                       id="input" |                         placeholder="КФУ-24-2" | ||||||
|                       type="text" |                         size="md" | ||||||
|                       autoComplete="off" |  | ||||||
|                       /> |                       /> | ||||||
|                     <IconButton type="submit"> |                       <FormHelperText></FormHelperText> | ||||||
|                       <ArrowImg src={arrow} /> |                     </FormControl> | ||||||
|                     </IconButton> | 
 | ||||||
|                   </InputWrapper> |                     <Box mt={10}> | ||||||
|  |                       <Button | ||||||
|  |                         size="lg" | ||||||
|  |                         leftIcon={<AddIcon />} | ||||||
|  |                         colorScheme="blue" | ||||||
|  |                       > | ||||||
|  |                         Создать | ||||||
|  |                       </Button> | ||||||
|  |                     </Box> | ||||||
|  |                   </VStack> | ||||||
|  | 
 | ||||||
|                   {crcQuery?.error && ( |                   {crcQuery?.error && ( | ||||||
|                     <ErrorSpan>{(crcQuery?.error as any).error}</ErrorSpan> |                     <ErrorSpan>{(crcQuery?.error as any).error}</ErrorSpan> | ||||||
|                   )} |                   )} | ||||||
|                 </form> |                 </form> | ||||||
|               </Papper> |               </CardBody> | ||||||
|  |             </Card> | ||||||
|           ) : ( |           ) : ( | ||||||
|               <AddButton onClick={() => setShowForm(true)}>Добавить</AddButton> |             <Box p="2" m="2"> | ||||||
|  |               <Button | ||||||
|  |                 leftIcon={<AddIcon />} | ||||||
|  |                 colorScheme="green" | ||||||
|  |                 onClick={() => setShowForm(true)} | ||||||
|  |               > | ||||||
|  |                 Добавить | ||||||
|  |               </Button> | ||||||
|  |             </Box> | ||||||
|           )} |           )} | ||||||
|           </> |         </Box> | ||||||
|       )} |       )} | ||||||
|         <UnorderedList spacing={3}> |       <VStack as="ul" align="stretch"> | ||||||
|           {data?.body?.map((course) => ( |         {data?.body?.map((c) => ( | ||||||
|             <StyledCard key={course._id} align="left"> |           <CourseCard | ||||||
|  |             isOpened={courseDetailsOpenedId === c._id} | ||||||
|  |             key={c._id} | ||||||
|  |             course={c} | ||||||
|  |             openDetails={() => | ||||||
|  |               courseDetailsOpenedId === c._id | ||||||
|  |                 ? setCourseDetailsOpenedId(null) | ||||||
|  |                 : setCourseDetailsOpenedId(c._id) | ||||||
|  |             } | ||||||
|  |           /> | ||||||
|  |         ))} | ||||||
|  |       </VStack> | ||||||
|  |     </Container> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const CourseCard = ({ course, isOpened, openDetails }) => { | ||||||
|  |   const [getLessonList, lessonList] = api.useLazyLessonListQuery() | ||||||
|  |   useEffect(() => { | ||||||
|  |     if (isOpened) { | ||||||
|  |       getLessonList(course._id, true) | ||||||
|  |     } | ||||||
|  |   }, [isOpened]) | ||||||
|  |   const user = useAppSelector((s) => s.user) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Card key={course._id} align="left"> | ||||||
|       <CardHeader> |       <CardHeader> | ||||||
|         <Heading as="h2" mt="0"> |         <Heading as="h2" mt="0"> | ||||||
|           {course.name} |           {course.name} | ||||||
|         </Heading> |         </Heading> | ||||||
|       </CardHeader> |       </CardHeader> | ||||||
|               {showDetails && ( |       {isOpened && ( | ||||||
|         <CardBody mt="16px"> |         <CardBody mt="16px"> | ||||||
|           <Stack divider={<StackDivider />} spacing="8px"> |           <Stack divider={<StackDivider />} spacing="8px"> | ||||||
|             <Box as="span" textAlign="left"> |             <Box as="span" textAlign="left"> | ||||||
| @ -121,46 +204,59 @@ const CoursesList = () => { | |||||||
|             <Box as="span" textAlign="left"> |             <Box as="span" textAlign="left"> | ||||||
|               Количество занятий - {course.lessons.length} |               Количество занятий - {course.lessons.length} | ||||||
|             </Box> |             </Box> | ||||||
|  |             {lessonList.isFetching ? ( | ||||||
|  |               <Spinner /> | ||||||
|  |             ) : ( | ||||||
|  |               <> | ||||||
|  |                 <Heading as="h3" mt={4} mb={3} size="lg"> | ||||||
|  |                   Список занятий: | ||||||
|  |                 </Heading> | ||||||
|  |                 <Stack> | ||||||
|  |                   {lessonList.data?.body?.map((lesson) => ( | ||||||
|  |                     <Link | ||||||
|  |                       as={ConnectedLink} | ||||||
|  |                       to={ | ||||||
|  |                         isTeacher(user) | ||||||
|  |                           ? `${getNavigationsValue('journal.main')}/lesson/${course._id}/${lesson._id}` | ||||||
|  |                           : '' | ||||||
|  |                       } | ||||||
|  |                     > | ||||||
|  |                       {lesson.name} | ||||||
|  |                     </Link> | ||||||
|  |                   ))} | ||||||
|  |                 </Stack> | ||||||
|  |               </> | ||||||
|  |             )} | ||||||
|           </Stack> |           </Stack> | ||||||
|         </CardBody> |         </CardBody> | ||||||
|       )} |       )} | ||||||
|       <CardFooter> |       <CardFooter> | ||||||
|                 <ButtonGroup spacing="12" mt="16px"> |         <ButtonGroup spacing="4" mt="16px"> | ||||||
|                   <Tooltip |           <Tooltip label="На страницу с лекциями" fontSize="12px" top="16px"> | ||||||
|                     label="На страницу с лекциями" |             <Button | ||||||
|                     fontSize="12px" |               leftIcon={<LinkIcon />} | ||||||
|                     top="16px" |               as={ConnectedLink} | ||||||
|                   > |               colorScheme="blue" | ||||||
|                     <Button variant="ghost" border="none" bg="#ffffff"> |  | ||||||
|                       <Link |  | ||||||
|               to={`${getNavigationsValue('journal.main')}/lessons-list/${course._id}`} |               to={`${getNavigationsValue('journal.main')}/lessons-list/${course._id}`} | ||||||
|             > |             > | ||||||
|                         <img src={linkOpen} alt="Перейти к лекциям" /> |               Открыть | ||||||
|                       </Link> |  | ||||||
|             </Button> |             </Button> | ||||||
|           </Tooltip> |           </Tooltip> | ||||||
|           <Tooltip label="Детали" fontSize="12px" top="16px"> |           <Tooltip label="Детали" fontSize="12px" top="16px"> | ||||||
|             <Button |             <Button | ||||||
|                       variant="ghost" |               colorScheme="blue" | ||||||
|                       border="none" |               variant="outline" | ||||||
|                       bg="#ffffff" |               leftIcon={isOpened ? <ArrowUpIcon /> : <ArrowDownIcon />} | ||||||
|                       onClick={() => { |               loadingText="Загрузка" | ||||||
|                         showDetails |               isLoading={lessonList.isFetching} | ||||||
|                           ? setShowDetails(null) |               onClick={openDetails} | ||||||
|                           : setShowDetails(true) |  | ||||||
|                       }} |  | ||||||
|                       cursor="pointer" |  | ||||||
|             > |             > | ||||||
|                       <img src={moreDetails} alt="Просмотреть детали" /> |               Просмотреть детали | ||||||
|             </Button> |             </Button> | ||||||
|           </Tooltip> |           </Tooltip> | ||||||
|         </ButtonGroup> |         </ButtonGroup> | ||||||
|       </CardFooter> |       </CardFooter> | ||||||
|             </StyledCard> |     </Card> | ||||||
|           ))} |  | ||||||
|         </UnorderedList> |  | ||||||
|       </StartWrapper> |  | ||||||
|     </MainWrapper> |  | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user