Skip to content

Commit

Permalink
Add loaders to show loading state for questions (#156)
Browse files Browse the repository at this point in the history
Part of #86 

Added a library of simple React loaders https://uiball.com/loaders/.
Can reuse it wherever you need if you encounter a loading state.
  • Loading branch information
ckcherry23 authored Oct 24, 2023
1 parent c9ab65b commit 7eb8c27
Show file tree
Hide file tree
Showing 5 changed files with 547 additions and 610 deletions.
4 changes: 4 additions & 0 deletions frontend/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,10 @@
"@radix-ui/react-tabs": "^1.0.4",
"@tanstack/react-query": "^5.0.0",
"@tanstack/react-table": "^8.10.4",
"@types/node": "20.6.0",
"@types/react": "18.2.21",
"@types/react-dom": "18.2.7",
"@uiball/loaders": "^1.3.0",
"autoprefixer": "10.4.15",
"class-variance-authority": "^0.7.0",
"clsx": "^2.0.0",
Expand Down
15 changes: 10 additions & 5 deletions frontend/src/components/questions/data-table.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,8 @@ import { AuthContext } from "../../contexts/AuthContext";
import { User } from "firebase/auth";


import { DotWave } from '@uiball/loaders'

interface DataTableProps<TData, TValue> {
columns: ColumnDef<TData, TValue>[];
isEditable?: boolean;
Expand Down Expand Up @@ -153,9 +155,12 @@ export function DataTable<TData, TValue>({
</div>
<div className="rounded-md border">
{loading ? (
// todo make loading look good
<div className="h-64 flex items-center justify-center">
Loading...
<DotWave
size={47}
speed={1}
color="white"
/>
</div>
) : (
<Table>
Expand All @@ -168,9 +173,9 @@ export function DataTable<TData, TValue>({
{header.isPlaceholder
? null
: flexRender(
header.column.columnDef.header,
header.getContext()
)}
header.column.columnDef.header,
header.getContext()
)}
</TableHead>
);
})}
Expand Down
8 changes: 4 additions & 4 deletions frontend/src/components/room/description.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@ export default function Description({
return (
<Card className={`m-2 ml-0 px-6 h-full ${className} overflow-y-auto overflow-x-wrap pb-4`}>
<div className="flex flex-row items-center justify-between py-2">
<div className="flex gap-4 items-center justify-center">
<TypographyH2>{question.title}</TypographyH2>
<div className="flex items-center justify-center">
<TypographyH2 className="w-fit">{question.title}</TypographyH2>
<Badge variant="secondary" className="h-min">
<TypographySmall className="text-[#27CA40]">
{question.difficulty}
Expand All @@ -30,14 +30,14 @@ export default function Description({
</div>
<div className="flex gap-2">
{question.topics.map((tag) => (
<Badge variant="outline" className="" key={tag}>
<Badge variant="outline" key={tag}>
<TypographySmall>{tag}</TypographySmall>
</Badge>
))}
</div>
<div className="py-6">
<TypographySmall>
<div dangerouslySetInnerHTML={{ __html: question.description }} className="max-w-2xl overflow-x-auto"></div>
<div dangerouslySetInnerHTML={{ __html: question.description }} className="w-[40vw] overflow-x-auto"></div>
</TypographySmall>
</div>
</Card>
Expand Down
68 changes: 39 additions & 29 deletions frontend/src/pages/questions/[id]/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,8 +7,9 @@ import { useContext, useEffect, useState } from "react";
import { Question } from "../../../types/QuestionTypes";
import { AuthContext } from "@/contexts/AuthContext";
import { fetchQuestion } from "../../api/questionHandler";
import { MrMiyagi } from "@uiball/loaders";

export default function Room() {
export default function Questions() {
const router = useRouter();
const questionId = router.query.id as string;
const [question, setQuestion] = useState<Question | null>(null);
Expand Down Expand Up @@ -36,40 +37,49 @@ export default function Room() {
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [questionId, authIsReady, currentUser]);

if (loading) return (<p>Loading...</p>);

if (question === null) return (<p>Question not found</p>);
if (question === null && !loading) return (<p>Question not found</p>);

// implement some on change solo save logic here - user side most likely

return (
<div className="h-[calc(100vh-80px)] px-12 py-6">
<div className="flex h-full">
<Tabs defaultValue="description" className="flex-1">
<TabsList>
<TabsTrigger value="description">
<TypographyBody>Description</TypographyBody>
</TabsTrigger>
<TabsTrigger value="solution">
<TypographyBody>Solution</TypographyBody>
</TabsTrigger>
</TabsList>
<TabsContent value="description" className="h-[79vh]">
<Description
question={question}
className="h-full"
/>
</TabsContent>
<TabsContent value="solution">{question.solution}</TabsContent>
</Tabs>
<div className="flex-1">
<CodeEditor
defaultValue={question.defaultCode.python}
onChange={setAnswer}
text={answer}
{!router.isReady || question === null || loading ?
<div className="flex w-full h-full justify-center items-center">
<MrMiyagi
size={35}
lineWeight={3.5}
speed={1}
color="white"
/>
</div>
</div>
</div> :
(
<div className="flex h-full">
<Tabs defaultValue="description" className="flex-1">
<TabsList>
<TabsTrigger value="description">
<TypographyBody>Description</TypographyBody>
</TabsTrigger>
<TabsTrigger value="solution">
<TypographyBody>Solution</TypographyBody>
</TabsTrigger>
</TabsList>
<TabsContent value="description" className="h-[79vh]">
<Description
question={question}
className="h-full"
/>
</TabsContent>
<TabsContent value="solution">{question.solution}</TabsContent>
</Tabs>
<div className="flex-1">
<CodeEditor
defaultValue={question.defaultCode.python}
onChange={setAnswer}
text={answer}
/>
</div>
</div>
)}
</div>
);
}
Loading

0 comments on commit 7eb8c27

Please sign in to comment.