mirror of
https://github.com/spliit-app/spliit.git
synced 2026-03-07 20:59:05 +01:00
Improve loading states
This commit is contained in:
@@ -7,11 +7,13 @@ import {
|
|||||||
CardHeader,
|
CardHeader,
|
||||||
CardTitle,
|
CardTitle,
|
||||||
} from '@/components/ui/card'
|
} from '@/components/ui/card'
|
||||||
|
import { Skeleton } from '@/components/ui/skeleton'
|
||||||
import { getGroup, getGroupExpenses } from '@/lib/api'
|
import { getGroup, getGroupExpenses } from '@/lib/api'
|
||||||
import { Plus } from 'lucide-react'
|
import { Plus } from 'lucide-react'
|
||||||
import { Metadata } from 'next'
|
import { Metadata } from 'next'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
import { notFound } from 'next/navigation'
|
import { notFound } from 'next/navigation'
|
||||||
|
import { Suspense } from 'react'
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
title: 'Expenses',
|
title: 'Expenses',
|
||||||
@@ -22,11 +24,6 @@ export default async function GroupExpensesPage({
|
|||||||
}: {
|
}: {
|
||||||
params: { groupId: string }
|
params: { groupId: string }
|
||||||
}) {
|
}) {
|
||||||
const group = await getGroup(groupId)
|
|
||||||
if (!group) notFound()
|
|
||||||
|
|
||||||
const expenses = await getGroupExpenses(groupId)
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card className="mb-4">
|
<Card className="mb-4">
|
||||||
<div className="flex flex-1">
|
<div className="flex flex-1">
|
||||||
@@ -46,13 +43,40 @@ export default async function GroupExpensesPage({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<CardContent className="p-0">
|
<CardContent className="p-0">
|
||||||
<ExpenseList
|
<Suspense
|
||||||
expenses={expenses}
|
fallback={[0, 1, 2].map((i) => (
|
||||||
groupId={groupId}
|
<div
|
||||||
currency={group.currency}
|
key={i}
|
||||||
participants={group.participants}
|
className="border-t flex justify-between items-center px-6 py-4 text-sm"
|
||||||
/>
|
>
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
<Skeleton className="h-4 w-16 rounded-full" />
|
||||||
|
<Skeleton className="h-4 w-32 rounded-full" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<Skeleton className="h-4 w-16 rounded-full" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
>
|
||||||
|
<Expenses groupId={groupId} />
|
||||||
|
</Suspense>
|
||||||
</CardContent>
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function Expenses({ groupId }: { groupId: string }) {
|
||||||
|
const group = await getGroup(groupId)
|
||||||
|
if (!group) notFound()
|
||||||
|
const expenses = await getGroupExpenses(group.id)
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ExpenseList
|
||||||
|
expenses={expenses}
|
||||||
|
groupId={group.id}
|
||||||
|
currency={group.currency}
|
||||||
|
participants={group.participants}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
import { getGroupsAction } from '@/app/groups/actions'
|
import { getGroupsAction } from '@/app/groups/actions'
|
||||||
import { getRecentGroups } from '@/app/groups/recent-groups-helpers'
|
import { getRecentGroups } from '@/app/groups/recent-groups-helpers'
|
||||||
import { Button } from '@/components/ui/button'
|
import { Button } from '@/components/ui/button'
|
||||||
|
import { Skeleton } from '@/components/ui/skeleton'
|
||||||
import { getGroups } from '@/lib/api'
|
import { getGroups } from '@/lib/api'
|
||||||
import { Calendar, Loader2, Users } from 'lucide-react'
|
import { Calendar, Loader2, Users } from 'lucide-react'
|
||||||
import Link from 'next/link'
|
import Link from 'next/link'
|
||||||
@@ -96,9 +97,10 @@ export function RecentGroupList() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<div className="flex justify-between">
|
||||||
<Loader2 className="w-3 h-3 mr-1 inline animate-spin" />
|
<Skeleton className="h-4 w-6 rounded-full" />
|
||||||
</>
|
<Skeleton className="h-4 w-24 rounded-full" />
|
||||||
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
15
src/components/ui/skeleton.tsx
Normal file
15
src/components/ui/skeleton.tsx
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
|
||||||
|
function Skeleton({
|
||||||
|
className,
|
||||||
|
...props
|
||||||
|
}: React.HTMLAttributes<HTMLDivElement>) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={cn("animate-pulse rounded-md bg-muted", className)}
|
||||||
|
{...props}
|
||||||
|
/>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export { Skeleton }
|
||||||
@@ -3,6 +3,7 @@ import { PrismaClient } from '@prisma/client'
|
|||||||
let prisma: PrismaClient
|
let prisma: PrismaClient
|
||||||
|
|
||||||
export async function getPrisma() {
|
export async function getPrisma() {
|
||||||
|
// await delay(1000)
|
||||||
if (!prisma) {
|
if (!prisma) {
|
||||||
if (process.env.NODE_ENV === 'production') {
|
if (process.env.NODE_ENV === 'production') {
|
||||||
prisma = new PrismaClient()
|
prisma = new PrismaClient()
|
||||||
|
|||||||
Reference in New Issue
Block a user