From 22e68939136978d2719a13309fe48ecb4320f073 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=EB=A5=98=EC=A0=95=EC=9A=B0?= <88191233+jw-r@users.noreply.github.com> Date: Sat, 21 Dec 2024 19:48:36 +0900 Subject: [PATCH] feat: random quiz (#314) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat: random quiz swiper 구현 * feat: random quiz * feat: infinite random quiz --- src/app/(routes)/quiz/random/page.tsx | 10 +- src/features/category/config/index.ts | 22 +- .../collection/components/collection.tsx | 3 +- .../components/create-collection-form.tsx | 8 +- .../quiz-view/components/quiz-option.tsx | 4 +- .../quiz/screen/random-quiz-view/index.tsx | 231 ++++++++++-------- src/features/quiz/utils/index.ts | 8 +- src/requests/collection/client.ts | 11 + src/requests/collection/hooks.ts | 9 + src/requests/collection/server.ts | 11 + src/requests/directory/client.tsx | 2 +- src/requests/directory/hooks.ts | 4 +- src/requests/directory/server.ts | 19 ++ src/requests/quiz/hooks.ts | 2 +- src/shared/configs/endpoint.ts | 2 + src/types/collection.d.ts | 7 + src/types/quiz.d.ts | 18 ++ 17 files changed, 233 insertions(+), 138 deletions(-) create mode 100644 src/requests/directory/server.ts diff --git a/src/app/(routes)/quiz/random/page.tsx b/src/app/(routes)/quiz/random/page.tsx index e93a8b98..db166b93 100644 --- a/src/app/(routes)/quiz/random/page.tsx +++ b/src/app/(routes)/quiz/random/page.tsx @@ -1,10 +1,14 @@ import RandomQuizView from '@/features/quiz/screen/random-quiz-view' -import { getBookmarkedCollections } from '@/requests/collection/server' +import { getDirectories } from '@/requests/directory/server' const RandomQuiz = async () => { - const bookmarkedCollections = await getBookmarkedCollections() + const directories = await getDirectories() - return + const directoriesHasDocuments = directories.directories.filter( + (directory) => directory.documentCount > 0 + ) + + return } export default RandomQuiz diff --git a/src/features/category/config/index.ts b/src/features/category/config/index.ts index fa362ae4..8a69c6e5 100644 --- a/src/features/category/config/index.ts +++ b/src/features/category/config/index.ts @@ -1,57 +1,57 @@ type Category = { - code: Collection.Field + id: Collection.Field name: string emoji: string } export const CATEGORIES: Category[] = [ { - code: 'IT', + id: 'IT', name: 'IT·프로그래밍', emoji: '🤖', }, { - code: 'LAW', + id: 'LAW', name: '법률·정치', emoji: '📖', }, { - code: 'BUSINESS_ECONOMY', + id: 'BUSINESS_ECONOMY', name: '경제·경영', emoji: '💰', }, { - code: 'SOCIETY_POLITICS', + id: 'SOCIETY_POLITICS', name: '사회·정치', emoji: '⚖️', }, { - code: 'LANGUAGE', + id: 'LANGUAGE', name: '언어', emoji: '💬', }, { - code: 'MEDICINE_PHARMACY', + id: 'MEDICINE_PHARMACY', name: '의학·약학', emoji: '🩺', }, { - code: 'ART', + id: 'ART', name: '예술', emoji: '🎨', }, { - code: 'SCIENCE_ENGINEERING', + id: 'SCIENCE_ENGINEERING', name: '과학·공학', emoji: '🔬', }, { - code: 'HISTORY_PHILOSOPHY', + id: 'HISTORY_PHILOSOPHY', name: '역사·철학', emoji: '📜', }, { - code: 'OTHER', + id: 'OTHER', name: '기타', emoji: '♾️', }, diff --git a/src/features/collection/components/collection.tsx b/src/features/collection/components/collection.tsx index 73c45a50..667c55ef 100644 --- a/src/features/collection/components/collection.tsx +++ b/src/features/collection/components/collection.tsx @@ -34,8 +34,7 @@ const Collection = ({ }: Props) => { const { mutate: bookmarkMutate } = useBookmarkMutation() - const categoryLabel = - CATEGORIES.find((categoryItem) => categoryItem.code === category)?.name ?? '' + const categoryLabel = CATEGORIES.find((categoryItem) => categoryItem.id === category)?.name ?? '' return (
diff --git a/src/features/collection/components/create-collection-form.tsx b/src/features/collection/components/create-collection-form.tsx index 6f2eff69..68e5aa68 100644 --- a/src/features/collection/components/create-collection-form.tsx +++ b/src/features/collection/components/create-collection-form.tsx @@ -51,7 +51,7 @@ const CreateCollectionForm = () => { const [emoji, setEmoji] = useState('🥹') const [title, setTitle] = useState('') const [description, setDescription] = useState('') - const [categoryCode, setCategoryCode] = useState(CATEGORIES[0]?.code ?? 'IT') + const [categoryCode, setCategoryCode] = useState(CATEGORIES[0]?.id ?? 'IT') const { data: directoryQuizzesData, isLoading: directoryQuizzesLoading } = useDirectoryQuizzes(selectedDirectoryId) @@ -225,7 +225,7 @@ const CreateCollectionForm = () => {
category.code === categoryCode)?.name ?? ''} + title={CATEGORIES.find((category) => category.id === categoryCode)?.name ?? ''} />
@@ -235,10 +235,10 @@ const CreateCollectionForm = () => {
{CATEGORIES.map((category) => ( - + setCategoryCode(category.code)} + onClick={() => setCategoryCode(category.id)} /> ))} diff --git a/src/features/quiz/screen/quiz-view/components/quiz-option.tsx b/src/features/quiz/screen/quiz-view/components/quiz-option.tsx index 40b1bc2a..b50e155f 100644 --- a/src/features/quiz/screen/quiz-view/components/quiz-option.tsx +++ b/src/features/quiz/screen/quiz-view/components/quiz-option.tsx @@ -6,7 +6,7 @@ import { QUIZ_ANIMATION_DURATION } from '@/features/quiz/config' import { cn } from '@/shared/lib/utils' interface QuizOptionsProps { - quiz: Quiz.Item + quiz: Quiz.Item | Quiz.RandomItem currentResult: Quiz.Result | null onAnswer: (params: { id: number; isRight: boolean; choseAnswer: string }) => void className?: HTMLElement['className'] @@ -31,7 +31,7 @@ const QuizOptions = ({ quiz, currentResult, onAnswer, className }: QuizOptionsPr if (quiz.quizType === 'MULTIPLE_CHOICE') { return ( [] } -type CategoryWithQuizzesAndCollectionName = { - category: (typeof CATEGORIES)[number] - quizzes: (Quiz.Item & { tag: string })[] -} - -const RandomQuizView = ({ collections, directories }: Props) => { - const [categoriesWithQuizzes, setCategoriesWithQuizzes] = useState([]) - const [] = useState() - - // 디렉토리에 생성된 모든 랜덤 퀴즈 가져옴 - - const randomQuizList = [...quizzes] // 임시 +const RandomQuizView = ({ directories }: Props) => { + const router = useRouter() + const [randomQuizList, setRandomQuizList] = useState([]) const [repository, setRepository] = useState<'directory' | 'collection'>('directory') const [activeDirectoryIndex, setActiveDirectoryIndex] = useState(0) const [activeCategoryIndex, setActiveCategoryIndex] = useState(0) const activeDirectoryId = useMemo( - () => mockDirectories[activeDirectoryIndex]?.id, - [activeDirectoryIndex] + () => directories[activeDirectoryIndex]?.id, + [activeDirectoryIndex, directories] + ) + const activeCategoryId = useMemo(() => CATEGORIES[activeCategoryIndex]?.id, [activeCategoryIndex]) + + const { data: randomCollectionQuizzesData } = useRandomCollectionQuizzes(activeCategoryId) + const randomCollectionQuizzes = useMemo( + () => randomCollectionQuizzesData?.quizzes ?? [], + [randomCollectionQuizzesData?.quizzes] + ) + + const { data: randomDirectoryQuizzesData } = useDirectoryQuizzes(activeDirectoryId) + const randomDirectoryQuizzes = useMemo( + () => randomDirectoryQuizzesData?.quizzes ?? [], + [randomDirectoryQuizzesData?.quizzes] ) - const activeCategoryCode = 1 const [openExplanation, setOpenExplanation] = useState(false) const { currentIndex, navigateToNext } = useQuizNavigation() - const { handleNext, quizResults, setQuizResults } = useQuizState({ + const { quizResults, setQuizResults } = useQuizState({ quizCount: randomQuizList.length, currentIndex, }) + const currentQuiz = randomQuizList[currentIndex] + const currentResult = quizResults[currentIndex] as Exclude< + (typeof quizResults)[number], + undefined + > + const handleSlideChange = (index: number) => { if (repository === 'directory') { setActiveDirectoryIndex(index) @@ -69,12 +80,13 @@ const RandomQuizView = ({ collections, directories }: Props) => { setOpenExplanation(false) } - const hasNextQuiz = handleNext(currentIndex, randomQuizList.length) - if (hasNextQuiz) { - navigateToNext(currentIndex) - } else { - // TODO: 종료 로직 추가 + if (repository === 'directory') { + // API 요청 } + + // 무한히 반복되기 위함 + setRandomQuizList((prev) => [...prev, currentQuiz!]) + navigateToNext(currentIndex) } const onAnswer = ({ @@ -104,15 +116,39 @@ const RandomQuizView = ({ collections, directories }: Props) => { } } - const currentQuiz = randomQuizList[currentIndex] - const currentResult = quizResults[currentIndex] as Exclude< - (typeof quizResults)[number], - undefined - > + const [SwiperContainerWidth, setSwiperContainerWidth] = useState(0) + const swiperContainerRef = useRef(null) + useLayoutEffect(() => { + if (swiperContainerRef.current) { + setSwiperContainerWidth(swiperContainerRef.current.clientWidth) + } + const handleResize = () => { + if (swiperContainerRef.current) { + setSwiperContainerWidth(swiperContainerRef.current.clientWidth) + } + } + window.addEventListener('resize', handleResize) + return () => window.removeEventListener('resize', handleResize) + }, []) - if (!currentQuiz) { - notFound() - } + const slideItems = repository === 'directory' ? directories : CATEGORIES + + useEffect(() => { + if (repository === 'directory') { + setRandomQuizList(randomDirectoryQuizzes) + } else { + setRandomQuizList(randomCollectionQuizzes) + } + router.replace('/quiz/random') + setQuizResults([]) + }, [ + router, + repository, + randomCollectionQuizzes, + randomDirectoryQuizzes, + setRandomQuizList, + setQuizResults, + ]) return (
@@ -132,26 +168,32 @@ const RandomQuizView = ({ collections, directories }: Props) => {
{/* 문제 영역 */} -
- - {currentQuiz.document.name} - - - - {currentQuiz.question} - - - -
+ {currentQuiz ? ( +
+ + + {currentQuiz.document?.name || currentQuiz.collection?.name} + + + + + {currentQuiz.question} + + + +
+ ) : ( + + )} {/* 탭 영역 */} {
{/* Swiper 영역 */} -
+
{ initialSlide={repository === 'directory' ? activeDirectoryIndex : activeCategoryIndex} onSlideChange={(data) => handleSlideChange(data.activeIndex)} > - {mockDirectories.map((item, index) => { + {slideItems.map((item, index) => { const isActive = repository === 'directory' ? index === activeDirectoryIndex @@ -191,10 +234,15 @@ const RandomQuizView = ({ collections, directories }: Props) => { return ( - ) @@ -207,10 +255,10 @@ const RandomQuizView = ({ collections, directories }: Props) => {
@@ -219,17 +267,18 @@ const RandomQuizView = ({ collections, directories }: Props) => { export default RandomQuizView -interface CategoryItemProps { +interface SlideItemProps { isActive: boolean data: { - id: number + id: number | string name: string emoji: string } variant: 'directory' | 'collection' + quizCount: number } -const CategoryItem = ({ isActive, data, variant }: CategoryItemProps) => { +const SlideItem = ({ isActive, data, variant, quizCount }: SlideItemProps) => { const styles = { directory: { background: { @@ -255,9 +304,13 @@ const CategoryItem = ({ isActive, data, variant }: CategoryItemProps) => {
@@ -267,9 +320,9 @@ const CategoryItem = ({ isActive, data, variant }: CategoryItemProps) => { > {data.name} - {isActive && ( + {isActive && quizCount > 0 && ( - 232문제 + {quizCount}문제 )}
@@ -277,41 +330,3 @@ const CategoryItem = ({ isActive, data, variant }: CategoryItemProps) => {
) } - -const mockDirectories = [ - { - id: 1, - name: '파이썬기본문법과응용', - emoji: '❄️', - }, - { - id: 2, - name: '통계학이론', - emoji: '🌱', - }, - { - id: 3, - name: '제테크상식', - emoji: '💵', - }, - { - id: 4, - name: '전공 공부', - emoji: '🔥', - }, - { - id: 5, - name: '세계사 1급', - emoji: '🌍', - }, - { - id: 6, - name: '강의 복기', - emoji: '✏️', - }, - { - id: 7, - name: '통계학이론', - emoji: '🌂', - }, -] diff --git a/src/features/quiz/utils/index.ts b/src/features/quiz/utils/index.ts index 823f07b2..e0846ea8 100644 --- a/src/features/quiz/utils/index.ts +++ b/src/features/quiz/utils/index.ts @@ -4,16 +4,16 @@ export const getOptionCondition = ( rightAnswer: string ) => { if (!isQuizSolved(result)) return 'IDLE' - if (result.answer === true && result.choseAnswer === option) return 'RIGHT' - if (result.answer === false && result.choseAnswer === option) return 'WRONG' + if (result?.answer === true && result?.choseAnswer === option) return 'RIGHT' + if (result?.answer === false && result?.choseAnswer === option) return 'WRONG' if (option === rightAnswer) return 'RIGHT' return 'DISABLED' } export const getOXCondition = (result: Quiz.Result | null) => { if (!isQuizSolved(result)) return 'IDLE' - if (result.answer === true && result.choseAnswer === 'correct') return 'RIGHT' - if (result.answer === false && result.choseAnswer === 'correct') return 'WRONG' + if (result?.answer === true && result?.choseAnswer === 'correct') return 'RIGHT' + if (result?.answer === false && result?.choseAnswer === 'correct') return 'WRONG' return 'WRONG' } diff --git a/src/requests/collection/client.ts b/src/requests/collection/client.ts index cde8c45e..640dae4b 100644 --- a/src/requests/collection/client.ts +++ b/src/requests/collection/client.ts @@ -75,3 +75,14 @@ export const getCollectionInfo = async ({ collectionId }: { collectionId: number throw error } } + +export const getRandomCollectionQuizzes = async ({ categoryId }: { categoryId: string }) => { + try { + const { data } = await http.get( + API_ENDPOINTS.COLLECTION.GET.RANDOM_QUIZZES(categoryId) + ) + return data + } catch (error) { + throw error + } +} diff --git a/src/requests/collection/hooks.ts b/src/requests/collection/hooks.ts index 77f22e47..efb6c994 100644 --- a/src/requests/collection/hooks.ts +++ b/src/requests/collection/hooks.ts @@ -8,6 +8,7 @@ import { createBookmark, getMyCollections, getCollectionInfo, + getRandomCollectionQuizzes, } from './client' export const useCollections = () => { @@ -38,6 +39,14 @@ export const useBookmarkedCollections = () => { }) } +export const useRandomCollectionQuizzes = (categoryId?: string) => { + return useQuery({ + queryKey: ['randomCollectionQuizzes', categoryId], + queryFn: async () => getRandomCollectionQuizzes({ categoryId: categoryId! }), + enabled: categoryId != null, + }) +} + export const useCreateCollection = () => { const queryClient = getQueryClient() diff --git a/src/requests/collection/server.ts b/src/requests/collection/server.ts index 951a8c2c..094fc8ce 100644 --- a/src/requests/collection/server.ts +++ b/src/requests/collection/server.ts @@ -25,3 +25,14 @@ export const getBookmarkedCollections = async () => { throw error } } + +export const getMyCollections = async () => { + try { + const { data } = await httpServer.get( + API_ENDPOINTS.COLLECTION.GET.MY_COLLECTIONS + ) + return data + } catch (error) { + throw error + } +} diff --git a/src/requests/directory/client.tsx b/src/requests/directory/client.tsx index 9034393e..62604f9f 100644 --- a/src/requests/directory/client.tsx +++ b/src/requests/directory/client.tsx @@ -6,7 +6,7 @@ import { http } from '@/shared/lib/axios/http' /** * 모든 디렉토리 가져오기 */ -export const fetchDirectories = async () => { +export const getDirectories = async () => { try { const { data } = await http.get( API_ENDPOINTS.DIRECTORY.GET.ALL diff --git a/src/requests/directory/hooks.ts b/src/requests/directory/hooks.ts index 6218ec47..91ae15ba 100644 --- a/src/requests/directory/hooks.ts +++ b/src/requests/directory/hooks.ts @@ -4,7 +4,7 @@ import { useMutation, useQuery } from '@tanstack/react-query' import { createDirectory, deleteDirectory, - fetchDirectories, + getDirectories, fetchDirectory, updateDirectoryInfo, } from './client' @@ -17,7 +17,7 @@ import { queries } from '@/shared/lib/tanstack-query/query-keys' export const useDirectories = () => { return useQuery({ queryKey: ['directories'], - queryFn: async () => fetchDirectories(), + queryFn: async () => getDirectories(), }) } diff --git a/src/requests/directory/server.ts b/src/requests/directory/server.ts new file mode 100644 index 00000000..aaf3f854 --- /dev/null +++ b/src/requests/directory/server.ts @@ -0,0 +1,19 @@ +'use server' + +import { API_ENDPOINTS } from '@/shared/configs/endpoint' +import { httpServer } from '@/shared/lib/axios/http-server' + +/** + * 모든 디렉토리 가져오기 + */ +export const getDirectories = async () => { + try { + const { data } = await httpServer.get( + API_ENDPOINTS.DIRECTORY.GET.ALL + ) + return data + } catch (error: unknown) { + console.error(error) + throw error + } +} diff --git a/src/requests/quiz/hooks.ts b/src/requests/quiz/hooks.ts index 948870c9..27978db2 100644 --- a/src/requests/quiz/hooks.ts +++ b/src/requests/quiz/hooks.ts @@ -18,7 +18,7 @@ import { // }) // } -export const useDirectoryQuizzes = (directoryId: number | null) => { +export const useDirectoryQuizzes = (directoryId?: number) => { return useQuery({ queryKey: ['directoryQuizzes', directoryId], queryFn: async () => getDirectoryQuizzes({ directoryId: directoryId! }), diff --git a/src/shared/configs/endpoint.ts b/src/shared/configs/endpoint.ts index 184f5473..8d558f19 100644 --- a/src/shared/configs/endpoint.ts +++ b/src/shared/configs/endpoint.ts @@ -34,6 +34,8 @@ export const API_ENDPOINTS = { BOOKMARKED: '/collections/bookmarked-collections', /** GET /collections-analysis - 컬렉션 분석 */ ANALYSIS: '/collections-analysis', + /** GET /collections/{collection_category}/quizzes - 북마크하거나 소유한 컬렉션 분야별로 모든 퀴즈 랜덤하게 가져오기 */ + RANDOM_QUIZZES: (categoryId: string) => `/collections/${categoryId}/quizzes`, }, POST: { /** POST /collections - 컬렉션 생성 */ diff --git a/src/types/collection.d.ts b/src/types/collection.d.ts index b109a1ff..ce0c8934 100644 --- a/src/types/collection.d.ts +++ b/src/types/collection.d.ts @@ -102,6 +102,13 @@ declare global { paths['/api/v2/collections-analysis']['get']['responses']['200']['content']['application/json;charset=UTF-8'] > + /** GET /api/v2/collections/{collection_category}/quizzes + * 북마크하거나 소유한 컬렉션 분야별로 모든 퀴즈 랜덤하게 가져오기 + */ + type GetRandomCollectionQuizzes = DeepRequired< + paths['/api/v2/collections/{collection_category}/quizzes']['get']['responses']['200']['content']['application/json;charset=UTF-8'] + > + /** POST /api/v2/collections * 컬렉션 생성 */ diff --git a/src/types/quiz.d.ts b/src/types/quiz.d.ts index 36afec0f..80fb6f12 100644 --- a/src/types/quiz.d.ts +++ b/src/types/quiz.d.ts @@ -17,6 +17,23 @@ type Metadata = { directory: DeepRequired } +type RandomQuiz = { + id: number + question: string + answer: string + explanation: string + options: string[] + quizType: 'MIX_UP' | 'MULTIPLE_CHOICE' + document?: { + id: number + name: string + } + collection?: { + id: number + name: string + } +} + declare global { /** <참고> * OX : correct/incorrect @@ -25,6 +42,7 @@ declare global { declare namespace Quiz { type Item = QuizItem type ReplayType = ReplayQuizType + type RandomItem = RandomQuiz type Type = QuizType type ReplayType = QuizType | 'RANDOM'