feat: complete profile page with profile edit, project likes and UI updates
This commit is contained in:
parent
105eab9bad
commit
06a5d46e1f
@ -5,7 +5,13 @@ import { z } from "zod"
|
|||||||
|
|
||||||
import { and, eq, sql } from "drizzle-orm"
|
import { and, eq, sql } from "drizzle-orm"
|
||||||
import * as schema from "./schema"
|
import * as schema from "./schema"
|
||||||
import { sandbox, user, usersToSandboxes } from "./schema"
|
import {
|
||||||
|
Sandbox,
|
||||||
|
sandbox,
|
||||||
|
sandboxLikes,
|
||||||
|
user,
|
||||||
|
usersToSandboxes,
|
||||||
|
} from "./schema"
|
||||||
|
|
||||||
export interface Env {
|
export interface Env {
|
||||||
DB: D1Database
|
DB: D1Database
|
||||||
@ -18,6 +24,13 @@ export interface Env {
|
|||||||
|
|
||||||
// npm run generate
|
// npm run generate
|
||||||
// npx wrangler d1 execute d1-sandbox --local --file=./drizzle/<FILE>
|
// npx wrangler d1 execute d1-sandbox --local --file=./drizzle/<FILE>
|
||||||
|
interface SandboxWithLiked extends Sandbox {
|
||||||
|
liked: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserResponse extends Omit<schema.User, "sandbox"> {
|
||||||
|
sandbox: SandboxWithLiked[]
|
||||||
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
async fetch(
|
async fetch(
|
||||||
@ -258,33 +271,147 @@ export default {
|
|||||||
.get()
|
.get()
|
||||||
|
|
||||||
return success
|
return success
|
||||||
|
} else if (path === "/api/sandbox/like") {
|
||||||
|
if (method === "POST") {
|
||||||
|
const likeSchema = z.object({
|
||||||
|
sandboxId: z.string(),
|
||||||
|
userId: z.string(),
|
||||||
|
})
|
||||||
|
|
||||||
|
try {
|
||||||
|
const body = await request.json()
|
||||||
|
const { sandboxId, userId } = likeSchema.parse(body)
|
||||||
|
|
||||||
|
// Check if user has already liked
|
||||||
|
const existingLike = await db.query.sandboxLikes.findFirst({
|
||||||
|
where: (likes, { and, eq }) =>
|
||||||
|
and(eq(likes.sandboxId, sandboxId), eq(likes.userId, userId)),
|
||||||
|
})
|
||||||
|
|
||||||
|
if (existingLike) {
|
||||||
|
// Unlike
|
||||||
|
await db
|
||||||
|
.delete(sandboxLikes)
|
||||||
|
.where(
|
||||||
|
and(
|
||||||
|
eq(sandboxLikes.sandboxId, sandboxId),
|
||||||
|
eq(sandboxLikes.userId, userId)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
await db
|
||||||
|
.update(sandbox)
|
||||||
|
.set({
|
||||||
|
likeCount: sql`${sandbox.likeCount} - 1`,
|
||||||
|
})
|
||||||
|
.where(eq(sandbox.id, sandboxId))
|
||||||
|
|
||||||
|
return json({
|
||||||
|
message: "Unlike successful",
|
||||||
|
liked: false,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// Like
|
||||||
|
await db.insert(sandboxLikes).values({
|
||||||
|
sandboxId,
|
||||||
|
userId,
|
||||||
|
createdAt: new Date(),
|
||||||
|
})
|
||||||
|
|
||||||
|
await db
|
||||||
|
.update(sandbox)
|
||||||
|
.set({
|
||||||
|
likeCount: sql`${sandbox.likeCount} + 1`,
|
||||||
|
})
|
||||||
|
.where(eq(sandbox.id, sandboxId))
|
||||||
|
|
||||||
|
return json({
|
||||||
|
message: "Like successful",
|
||||||
|
liked: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
return new Response("Invalid request format", { status: 400 })
|
||||||
|
}
|
||||||
|
} else if (method === "GET") {
|
||||||
|
const params = url.searchParams
|
||||||
|
const sandboxId = params.get("sandboxId")
|
||||||
|
const userId = params.get("userId")
|
||||||
|
|
||||||
|
if (!sandboxId || !userId) {
|
||||||
|
return invalidRequest
|
||||||
|
}
|
||||||
|
|
||||||
|
const like = await db.query.sandboxLikes.findFirst({
|
||||||
|
where: (likes, { and, eq }) =>
|
||||||
|
and(eq(likes.sandboxId, sandboxId), eq(likes.userId, userId)),
|
||||||
|
})
|
||||||
|
|
||||||
|
return json({
|
||||||
|
liked: !!like,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
return methodNotAllowed
|
||||||
|
}
|
||||||
} else if (path === "/api/user") {
|
} else if (path === "/api/user") {
|
||||||
if (method === "GET") {
|
if (method === "GET") {
|
||||||
const params = url.searchParams
|
const params = url.searchParams
|
||||||
|
|
||||||
if (params.has("id")) {
|
if (params.has("id")) {
|
||||||
const id = params.get("id") as string
|
const id = params.get("id") as string
|
||||||
|
|
||||||
const res = await db.query.user.findFirst({
|
const res = await db.query.user.findFirst({
|
||||||
where: (user, { eq }) => eq(user.id, id),
|
where: (user, { eq }) => eq(user.id, id),
|
||||||
with: {
|
with: {
|
||||||
sandbox: {
|
sandbox: {
|
||||||
orderBy: (sandbox, { desc }) => [desc(sandbox.createdAt)],
|
orderBy: (sandbox, { desc }) => [desc(sandbox.createdAt)],
|
||||||
|
with: {
|
||||||
|
likes: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
usersToSandboxes: true,
|
usersToSandboxes: true,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
if (res) {
|
||||||
|
const transformedUser: UserResponse = {
|
||||||
|
...res,
|
||||||
|
sandbox: res.sandbox.map(
|
||||||
|
(sb): SandboxWithLiked => ({
|
||||||
|
...sb,
|
||||||
|
liked: sb.likes.some((like) => like.userId === id),
|
||||||
|
})
|
||||||
|
),
|
||||||
|
}
|
||||||
|
return json(transformedUser)
|
||||||
|
}
|
||||||
return json(res ?? {})
|
return json(res ?? {})
|
||||||
} else if (params.has("username")) {
|
} else if (params.has("username")) {
|
||||||
const username = params.get("username") as string
|
const username = params.get("username") as string
|
||||||
|
const userId = params.get("currentUserId")
|
||||||
const res = await db.query.user.findFirst({
|
const res = await db.query.user.findFirst({
|
||||||
where: (user, { eq }) => eq(user.username, username),
|
where: (user, { eq }) => eq(user.username, username),
|
||||||
with: {
|
with: {
|
||||||
sandbox: {
|
sandbox: {
|
||||||
orderBy: (sandbox, { desc }) => [desc(sandbox.createdAt)],
|
orderBy: (sandbox, { desc }) => [desc(sandbox.createdAt)],
|
||||||
|
with: {
|
||||||
|
likes: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
usersToSandboxes: true,
|
usersToSandboxes: true,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
if (res) {
|
||||||
|
const transformedUser: UserResponse = {
|
||||||
|
...res,
|
||||||
|
sandbox: res.sandbox.map(
|
||||||
|
(sb): SandboxWithLiked => ({
|
||||||
|
...sb,
|
||||||
|
liked: sb.likes.some((like) => like.userId === userId),
|
||||||
|
})
|
||||||
|
),
|
||||||
|
}
|
||||||
|
return json(transformedUser)
|
||||||
|
}
|
||||||
return json(res ?? {})
|
return json(res ?? {})
|
||||||
} else {
|
} else {
|
||||||
const res = await db.select().from(user).all()
|
const res = await db.select().from(user).all()
|
||||||
@ -326,6 +453,57 @@ export default {
|
|||||||
await db.delete(user).where(eq(user.id, id))
|
await db.delete(user).where(eq(user.id, id))
|
||||||
return success
|
return success
|
||||||
} else return invalidRequest
|
} else return invalidRequest
|
||||||
|
} else if (method === "PUT") {
|
||||||
|
const updateUserSchema = z.object({
|
||||||
|
id: z.string(),
|
||||||
|
name: z.string().optional(),
|
||||||
|
email: z.string().email().optional(),
|
||||||
|
username: z.string().optional(),
|
||||||
|
avatarUrl: z.string().optional(),
|
||||||
|
generations: z.number().optional(),
|
||||||
|
})
|
||||||
|
|
||||||
|
try {
|
||||||
|
const body = await request.json()
|
||||||
|
const validatedData = updateUserSchema.parse(body)
|
||||||
|
|
||||||
|
const { id, username, ...updateData } = validatedData
|
||||||
|
|
||||||
|
// If username is being updated, check for existing username
|
||||||
|
if (username) {
|
||||||
|
const existingUser = await db
|
||||||
|
.select()
|
||||||
|
.from(user)
|
||||||
|
.where(eq(user.username, username))
|
||||||
|
.get()
|
||||||
|
if (existingUser && existingUser.id !== id) {
|
||||||
|
return json({ error: "Username already exists" }, { status: 409 })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const cleanUpdateData = {
|
||||||
|
...updateData,
|
||||||
|
...(username ? { username } : {}),
|
||||||
|
}
|
||||||
|
|
||||||
|
const res = await db
|
||||||
|
.update(user)
|
||||||
|
.set(cleanUpdateData)
|
||||||
|
.where(eq(user.id, id))
|
||||||
|
.returning()
|
||||||
|
.get()
|
||||||
|
|
||||||
|
if (!res) {
|
||||||
|
return json({ error: "User not found" }, { status: 404 })
|
||||||
|
}
|
||||||
|
|
||||||
|
return json({ res })
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
return json({ error: error.errors }, { status: 400 })
|
||||||
|
}
|
||||||
|
return json({ error: "Internal server error" }, { status: 500 })
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
return methodNotAllowed
|
return methodNotAllowed
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
import { createId } from "@paralleldrive/cuid2"
|
import { createId } from "@paralleldrive/cuid2"
|
||||||
import { relations } from "drizzle-orm"
|
import { relations, sql } from "drizzle-orm"
|
||||||
import { integer, sqliteTable, text } from "drizzle-orm/sqlite-core"
|
import { integer, primaryKey, sqliteTable, text } from "drizzle-orm/sqlite-core"
|
||||||
import { sql } from "drizzle-orm"
|
|
||||||
|
|
||||||
|
// #region Tables
|
||||||
export const user = sqliteTable("user", {
|
export const user = sqliteTable("user", {
|
||||||
id: text("id")
|
id: text("id")
|
||||||
.$defaultFn(() => createId())
|
.$defaultFn(() => createId())
|
||||||
@ -12,18 +12,14 @@ export const user = sqliteTable("user", {
|
|||||||
email: text("email").notNull(),
|
email: text("email").notNull(),
|
||||||
username: text("username").notNull().unique(),
|
username: text("username").notNull().unique(),
|
||||||
avatarUrl: text("avatarUrl"),
|
avatarUrl: text("avatarUrl"),
|
||||||
createdAt: integer("createdAt", { mode: "timestamp_ms" })
|
createdAt: integer("createdAt", { mode: "timestamp_ms" }).default(
|
||||||
.default(sql`CURRENT_TIMESTAMP`),
|
sql`CURRENT_TIMESTAMP`
|
||||||
|
),
|
||||||
generations: integer("generations").default(0),
|
generations: integer("generations").default(0),
|
||||||
})
|
})
|
||||||
|
|
||||||
export type User = typeof user.$inferSelect
|
export type User = typeof user.$inferSelect
|
||||||
|
|
||||||
export const userRelations = relations(user, ({ many }) => ({
|
|
||||||
sandbox: many(sandbox),
|
|
||||||
usersToSandboxes: many(usersToSandboxes),
|
|
||||||
}))
|
|
||||||
|
|
||||||
export const sandbox = sqliteTable("sandbox", {
|
export const sandbox = sqliteTable("sandbox", {
|
||||||
id: text("id")
|
id: text("id")
|
||||||
.$defaultFn(() => createId())
|
.$defaultFn(() => createId())
|
||||||
@ -32,8 +28,9 @@ export const sandbox = sqliteTable("sandbox", {
|
|||||||
name: text("name").notNull(),
|
name: text("name").notNull(),
|
||||||
type: text("type").notNull(),
|
type: text("type").notNull(),
|
||||||
visibility: text("visibility", { enum: ["public", "private"] }),
|
visibility: text("visibility", { enum: ["public", "private"] }),
|
||||||
createdAt: integer("createdAt", { mode: "timestamp_ms" })
|
createdAt: integer("createdAt", { mode: "timestamp_ms" }).default(
|
||||||
.default(sql`CURRENT_TIMESTAMP`),
|
sql`CURRENT_TIMESTAMP`
|
||||||
|
),
|
||||||
userId: text("user_id")
|
userId: text("user_id")
|
||||||
.notNull()
|
.notNull()
|
||||||
.references(() => user.id),
|
.references(() => user.id),
|
||||||
@ -43,13 +40,23 @@ export const sandbox = sqliteTable("sandbox", {
|
|||||||
|
|
||||||
export type Sandbox = typeof sandbox.$inferSelect
|
export type Sandbox = typeof sandbox.$inferSelect
|
||||||
|
|
||||||
export const sandboxRelations = relations(sandbox, ({ one, many }) => ({
|
export const sandboxLikes = sqliteTable(
|
||||||
author: one(user, {
|
"sandbox_likes",
|
||||||
fields: [sandbox.userId],
|
{
|
||||||
references: [user.id],
|
userId: text("user_id")
|
||||||
}),
|
.notNull()
|
||||||
usersToSandboxes: many(usersToSandboxes),
|
.references(() => user.id),
|
||||||
}))
|
sandboxId: text("sandbox_id")
|
||||||
|
.notNull()
|
||||||
|
.references(() => sandbox.id),
|
||||||
|
createdAt: integer("createdAt", { mode: "timestamp_ms" }).default(
|
||||||
|
sql`CURRENT_TIMESTAMP`
|
||||||
|
),
|
||||||
|
},
|
||||||
|
(table) => ({
|
||||||
|
pk: primaryKey({ columns: [table.sandboxId, table.userId] }),
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
export const usersToSandboxes = sqliteTable("users_to_sandboxes", {
|
export const usersToSandboxes = sqliteTable("users_to_sandboxes", {
|
||||||
userId: text("userId")
|
userId: text("userId")
|
||||||
@ -61,6 +68,33 @@ export const usersToSandboxes = sqliteTable("users_to_sandboxes", {
|
|||||||
sharedOn: integer("sharedOn", { mode: "timestamp_ms" }),
|
sharedOn: integer("sharedOn", { mode: "timestamp_ms" }),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
// #region Relations
|
||||||
|
export const userRelations = relations(user, ({ many }) => ({
|
||||||
|
sandbox: many(sandbox),
|
||||||
|
usersToSandboxes: many(usersToSandboxes),
|
||||||
|
likes: many(sandboxLikes),
|
||||||
|
}))
|
||||||
|
|
||||||
|
export const sandboxRelations = relations(sandbox, ({ one, many }) => ({
|
||||||
|
author: one(user, {
|
||||||
|
fields: [sandbox.userId],
|
||||||
|
references: [user.id],
|
||||||
|
}),
|
||||||
|
usersToSandboxes: many(usersToSandboxes),
|
||||||
|
likes: many(sandboxLikes),
|
||||||
|
}))
|
||||||
|
|
||||||
|
export const sandboxLikesRelations = relations(sandboxLikes, ({ one }) => ({
|
||||||
|
user: one(user, {
|
||||||
|
fields: [sandboxLikes.userId],
|
||||||
|
references: [user.id],
|
||||||
|
}),
|
||||||
|
sandbox: one(sandbox, {
|
||||||
|
fields: [sandboxLikes.sandboxId],
|
||||||
|
references: [sandbox.id],
|
||||||
|
}),
|
||||||
|
}))
|
||||||
|
|
||||||
export const usersToSandboxesRelations = relations(
|
export const usersToSandboxesRelations = relations(
|
||||||
usersToSandboxes,
|
usersToSandboxes,
|
||||||
({ one }) => ({
|
({ one }) => ({
|
||||||
@ -74,3 +108,5 @@ export const usersToSandboxesRelations = relations(
|
|||||||
}),
|
}),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// #endregion
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
import ProfilePage from "@/components/profile"
|
import ProfilePage from "@/components/profile"
|
||||||
import ProfileNavbar from "@/components/profile/navbar"
|
import ProfileNavbar from "@/components/profile/navbar"
|
||||||
import { Sandbox, User } from "@/lib/types"
|
import { SandboxWithLiked, User } from "@/lib/types"
|
||||||
import { currentUser } from "@clerk/nextjs"
|
import { currentUser } from "@clerk/nextjs"
|
||||||
|
import { notFound } from "next/navigation"
|
||||||
|
|
||||||
export default async function Page({
|
export default async function Page({
|
||||||
params: { username: rawUsername },
|
params: { username: rawUsername },
|
||||||
@ -9,11 +10,11 @@ export default async function Page({
|
|||||||
params: { username: string }
|
params: { username: string }
|
||||||
}) {
|
}) {
|
||||||
const username = decodeURIComponent(rawUsername).replace("@", "")
|
const username = decodeURIComponent(rawUsername).replace("@", "")
|
||||||
const currentLoggedInUser = await currentUser()
|
const loggedInClerkUser = await currentUser()
|
||||||
console.log(username)
|
|
||||||
const [profileRespnse, dbUserResponse] = await Promise.all([
|
const [profileOwnerResponse, loggedInUserResponse] = await Promise.all([
|
||||||
fetch(
|
fetch(
|
||||||
`${process.env.NEXT_PUBLIC_DATABASE_WORKER_URL}/api/user?username=${username}`,
|
`${process.env.NEXT_PUBLIC_DATABASE_WORKER_URL}/api/user?username=${username}¤tUserId=${loggedInClerkUser?.id}`,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `${process.env.NEXT_PUBLIC_WORKERS_KEY}`,
|
Authorization: `${process.env.NEXT_PUBLIC_WORKERS_KEY}`,
|
||||||
@ -21,7 +22,7 @@ export default async function Page({
|
|||||||
}
|
}
|
||||||
),
|
),
|
||||||
fetch(
|
fetch(
|
||||||
`${process.env.NEXT_PUBLIC_DATABASE_WORKER_URL}/api/user?id=${currentLoggedInUser?.id}`,
|
`${process.env.NEXT_PUBLIC_DATABASE_WORKER_URL}/api/user?id=${loggedInClerkUser?.id}`,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `${process.env.NEXT_PUBLIC_WORKERS_KEY}`,
|
Authorization: `${process.env.NEXT_PUBLIC_WORKERS_KEY}`,
|
||||||
@ -30,30 +31,35 @@ export default async function Page({
|
|||||||
),
|
),
|
||||||
])
|
])
|
||||||
|
|
||||||
const userProfile = (await profileRespnse.json()) as User
|
const profileOwner = (await profileOwnerResponse.json()) as User
|
||||||
const dbUserData = (await dbUserResponse.json()) as User
|
const loggedInUser = (await loggedInUserResponse.json()) as User
|
||||||
const publicSandboxes: Sandbox[] = []
|
|
||||||
const privateSandboxes: Sandbox[] = []
|
|
||||||
|
|
||||||
userProfile?.sandbox?.forEach((sandbox) => {
|
if (!Boolean(profileOwner?.id)) {
|
||||||
|
notFound()
|
||||||
|
}
|
||||||
|
const publicSandboxes: SandboxWithLiked[] = []
|
||||||
|
const privateSandboxes: SandboxWithLiked[] = []
|
||||||
|
|
||||||
|
profileOwner?.sandbox?.forEach((sandbox) => {
|
||||||
if (sandbox.visibility === "public") {
|
if (sandbox.visibility === "public") {
|
||||||
publicSandboxes.push(sandbox)
|
publicSandboxes.push(sandbox as SandboxWithLiked)
|
||||||
} else if (sandbox.visibility === "private") {
|
} else if (sandbox.visibility === "private") {
|
||||||
privateSandboxes.push(sandbox)
|
privateSandboxes.push(sandbox as SandboxWithLiked)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
const hasCurrentUser = Boolean(dbUserData?.id)
|
|
||||||
|
const isUserLoggedIn = Boolean(loggedInUser?.id)
|
||||||
return (
|
return (
|
||||||
<div className="">
|
<section>
|
||||||
<ProfileNavbar userData={dbUserData} />
|
<ProfileNavbar userData={loggedInUser} />
|
||||||
<ProfilePage
|
<ProfilePage
|
||||||
publicSandboxes={publicSandboxes}
|
publicSandboxes={publicSandboxes}
|
||||||
privateSandboxes={
|
privateSandboxes={
|
||||||
userProfile?.id === dbUserData.id ? privateSandboxes : []
|
profileOwner?.id === loggedInUser.id ? privateSandboxes : []
|
||||||
}
|
}
|
||||||
user={userProfile}
|
profileOwner={profileOwner}
|
||||||
currentUser={hasCurrentUser ? dbUserData : null}
|
loggedInUser={isUserLoggedIn ? loggedInUser : null}
|
||||||
/>
|
/>
|
||||||
</div>
|
</section>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -26,7 +26,7 @@ export default function ProjectCardDropdown({
|
|||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
e.stopPropagation()
|
e.stopPropagation()
|
||||||
}}
|
}}
|
||||||
className="h-6 w-6 flex items-center justify-center transition-colors bg-transparent hover:bg-muted-foreground/25 rounded-sm outline-foreground"
|
className="h-6 w-6 z-10 flex items-center justify-center transition-colors bg-transparent hover:bg-muted-foreground/25 rounded-sm outline-foreground"
|
||||||
>
|
>
|
||||||
<Ellipsis className="w-4 h-4" />
|
<Ellipsis className="w-4 h-4" />
|
||||||
</DropdownMenuTrigger>
|
</DropdownMenuTrigger>
|
||||||
|
@ -1,13 +1,26 @@
|
|||||||
"use client"
|
"use client"
|
||||||
|
|
||||||
|
import { Button } from "@/components/ui/button"
|
||||||
import { Card } from "@/components/ui/card"
|
import { Card } from "@/components/ui/card"
|
||||||
|
import { toggleLike } from "@/lib/actions"
|
||||||
import { projectTemplates } from "@/lib/data"
|
import { projectTemplates } from "@/lib/data"
|
||||||
import { Sandbox } from "@/lib/types"
|
import { Sandbox } from "@/lib/types"
|
||||||
|
import { cn } from "@/lib/utils"
|
||||||
|
import { useUser } from "@clerk/nextjs"
|
||||||
import { AnimatePresence, motion } from "framer-motion"
|
import { AnimatePresence, motion } from "framer-motion"
|
||||||
import { Clock, Eye, Globe, Heart, Lock } from "lucide-react"
|
import { Clock, Eye, Globe, Heart, Lock } from "lucide-react"
|
||||||
import Image from "next/image"
|
import Image from "next/image"
|
||||||
|
import Link from "next/link"
|
||||||
import { useRouter } from "next/navigation"
|
import { useRouter } from "next/navigation"
|
||||||
import { memo, useEffect, useMemo, useState } from "react"
|
import {
|
||||||
|
memo,
|
||||||
|
MouseEventHandler,
|
||||||
|
useEffect,
|
||||||
|
useMemo,
|
||||||
|
useOptimistic,
|
||||||
|
useState,
|
||||||
|
useTransition,
|
||||||
|
} from "react"
|
||||||
import ProjectCardDropdown from "./dropdown"
|
import ProjectCardDropdown from "./dropdown"
|
||||||
import { CanvasRevealEffect } from "./revealEffect"
|
import { CanvasRevealEffect } from "./revealEffect"
|
||||||
|
|
||||||
@ -18,6 +31,7 @@ type BaseProjectCardProps = {
|
|||||||
visibility: "public" | "private"
|
visibility: "public" | "private"
|
||||||
createdAt: Date
|
createdAt: Date
|
||||||
likeCount: number
|
likeCount: number
|
||||||
|
liked?: boolean
|
||||||
viewCount: number
|
viewCount: number
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,16 +73,19 @@ const formatDate = (date: Date): string => {
|
|||||||
|
|
||||||
const ProjectMetadata = memo(
|
const ProjectMetadata = memo(
|
||||||
({
|
({
|
||||||
|
id,
|
||||||
visibility,
|
visibility,
|
||||||
createdAt,
|
createdAt,
|
||||||
likeCount,
|
likeCount,
|
||||||
|
liked,
|
||||||
viewCount,
|
viewCount,
|
||||||
}: Pick<
|
}: Pick<
|
||||||
BaseProjectCardProps,
|
BaseProjectCardProps,
|
||||||
"visibility" | "createdAt" | "likeCount" | "viewCount"
|
"visibility" | "createdAt" | "likeCount" | "liked" | "viewCount" | "id"
|
||||||
>) => {
|
>) => {
|
||||||
|
const { user } = useUser()
|
||||||
const [date, setDate] = useState<string>()
|
const [date, setDate] = useState<string>()
|
||||||
|
const Icon = visibility === "private" ? Lock : Globe
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setDate(formatDate(new Date(createdAt)))
|
setDate(formatDate(new Date(createdAt)))
|
||||||
}, [createdAt])
|
}, [createdAt])
|
||||||
@ -76,23 +93,23 @@ const ProjectMetadata = memo(
|
|||||||
return (
|
return (
|
||||||
<div className="flex flex-col text-muted-foreground space-y-2 text-sm z-10">
|
<div className="flex flex-col text-muted-foreground space-y-2 text-sm z-10">
|
||||||
<div className="flex items-center justify-between">
|
<div className="flex items-center justify-between">
|
||||||
<div className="flex items-center">
|
<div className="flex items-center gap-2">
|
||||||
{visibility === "private" ? (
|
<Icon className="size-4" />
|
||||||
<>
|
<span className="text-xs">
|
||||||
<Lock className="size-4 mr-2" /> Private
|
{visibility === "private" ? "Private" : "Public"}
|
||||||
</>
|
</span>
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<Globe className="size-4 mr-2" /> Public
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex gap-4">
|
<div className="flex gap-3">
|
||||||
<div className="flex items-center">
|
<div className="flex items-center gap-2">
|
||||||
<Clock className="size-4 mr-2" /> {date}
|
<Clock className="size-4" /> <span className="text-xs">{date}</span>
|
||||||
</div>
|
</div>
|
||||||
<StatItem icon={Heart} value={likeCount} />
|
<LikeButton
|
||||||
|
sandboxId={id}
|
||||||
|
initialIsLiked={!!liked}
|
||||||
|
initialLikeCount={likeCount}
|
||||||
|
userId={user?.id ?? null}
|
||||||
|
/>
|
||||||
<StatItem icon={Eye} value={viewCount} />
|
<StatItem icon={Eye} value={viewCount} />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -102,6 +119,63 @@ const ProjectMetadata = memo(
|
|||||||
|
|
||||||
ProjectMetadata.displayName = "ProjectMetadata"
|
ProjectMetadata.displayName = "ProjectMetadata"
|
||||||
|
|
||||||
|
interface LikeButtonProps {
|
||||||
|
sandboxId: string
|
||||||
|
userId: string | null
|
||||||
|
initialLikeCount: number
|
||||||
|
initialIsLiked: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
export function LikeButton({
|
||||||
|
sandboxId,
|
||||||
|
userId,
|
||||||
|
initialLikeCount,
|
||||||
|
initialIsLiked,
|
||||||
|
}: LikeButtonProps) {
|
||||||
|
// Optimistic state for like status and count
|
||||||
|
const [{ isLiked, likeCount }, optimisticUpdateLike] = useOptimistic(
|
||||||
|
{ isLiked: initialIsLiked, likeCount: initialLikeCount },
|
||||||
|
(state, optimisticValue: boolean) => {
|
||||||
|
return {
|
||||||
|
isLiked: optimisticValue,
|
||||||
|
likeCount: state.likeCount + (optimisticValue ? 1 : -1),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const [isPending, startTransition] = useTransition()
|
||||||
|
|
||||||
|
const handleLike: MouseEventHandler<HTMLButtonElement> = async (e) => {
|
||||||
|
e.stopPropagation() // Prevent click event from bubbling up which leads to navigation to /code/:id
|
||||||
|
if (!userId) return
|
||||||
|
|
||||||
|
startTransition(async () => {
|
||||||
|
const newLikeState = !isLiked
|
||||||
|
try {
|
||||||
|
optimisticUpdateLike(newLikeState)
|
||||||
|
await toggleLike(sandboxId, userId)
|
||||||
|
} catch (error) {
|
||||||
|
console.log("error", error)
|
||||||
|
optimisticUpdateLike(!newLikeState)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
variant="ghost"
|
||||||
|
size="sm"
|
||||||
|
disabled={!userId || isPending}
|
||||||
|
onClick={handleLike}
|
||||||
|
className="gap-1 px-1 rounded-full"
|
||||||
|
>
|
||||||
|
<Heart
|
||||||
|
className={cn("size-4", isLiked ? "stroke-red-500 fill-red-500" : "")}
|
||||||
|
/>
|
||||||
|
<span className="text-xs">{likeCount}</span>
|
||||||
|
</Button>
|
||||||
|
)
|
||||||
|
}
|
||||||
function ProjectCardComponent({
|
function ProjectCardComponent({
|
||||||
id,
|
id,
|
||||||
name,
|
name,
|
||||||
@ -150,7 +224,11 @@ function ProjectCardComponent({
|
|||||||
className={`
|
className={`
|
||||||
group/canvas-card p-4 h-48 flex flex-col justify-between items-start
|
group/canvas-card p-4 h-48 flex flex-col justify-between items-start
|
||||||
hover:border-muted-foreground/50 relative overflow-hidden transition-all
|
hover:border-muted-foreground/50 relative overflow-hidden transition-all
|
||||||
${props.isAuthenticated && props.deletingId === id ? "opacity-50" : ""}
|
${
|
||||||
|
props.isAuthenticated && props.deletingId === id
|
||||||
|
? "opacity-50 pointer-events-none cursor-events-none"
|
||||||
|
: "cursor-pointer"
|
||||||
|
}
|
||||||
`}
|
`}
|
||||||
>
|
>
|
||||||
<AnimatePresence>
|
<AnimatePresence>
|
||||||
@ -178,9 +256,12 @@ function ProjectCardComponent({
|
|||||||
width={20}
|
width={20}
|
||||||
height={20}
|
height={20}
|
||||||
/>
|
/>
|
||||||
<div className="font-medium static whitespace-nowrap w-full text-ellipsis overflow-hidden">
|
<Link
|
||||||
|
href={`/code/${id}`}
|
||||||
|
className="font-medium static whitespace-nowrap w-full text-ellipsis overflow-hidden before:content-[''] before:absolute before:z-0 before:top-0 before:left-0 before:w-full before:h-full before:rounded-xl"
|
||||||
|
>
|
||||||
{name}
|
{name}
|
||||||
</div>
|
</Link>
|
||||||
{props.isAuthenticated && (
|
{props.isAuthenticated && (
|
||||||
<ProjectCardDropdown
|
<ProjectCardDropdown
|
||||||
onVisibilityChange={handleVisibilityChange}
|
onVisibilityChange={handleVisibilityChange}
|
||||||
@ -195,6 +276,8 @@ function ProjectCardComponent({
|
|||||||
createdAt={createdAt}
|
createdAt={createdAt}
|
||||||
likeCount={likeCount}
|
likeCount={likeCount}
|
||||||
viewCount={viewCount}
|
viewCount={viewCount}
|
||||||
|
id={id}
|
||||||
|
liked={props.liked}
|
||||||
/>
|
/>
|
||||||
</Card>
|
</Card>
|
||||||
)
|
)
|
||||||
|
@ -2,8 +2,6 @@
|
|||||||
|
|
||||||
import { deleteSandbox, updateSandbox } from "@/lib/actions"
|
import { deleteSandbox, updateSandbox } from "@/lib/actions"
|
||||||
import { Sandbox } from "@/lib/types"
|
import { Sandbox } from "@/lib/types"
|
||||||
import { cn } from "@/lib/utils"
|
|
||||||
import Link from "next/link"
|
|
||||||
import { useEffect, useMemo, useState } from "react"
|
import { useEffect, useMemo, useState } from "react"
|
||||||
import { toast } from "sonner"
|
import { toast } from "sonner"
|
||||||
import ProjectCard from "./projectCard"
|
import ProjectCard from "./projectCard"
|
||||||
@ -71,24 +69,14 @@ export default function DashboardProjects({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<Link
|
<ProjectCard
|
||||||
key={sandbox.id}
|
key={sandbox.id}
|
||||||
href={`/code/${sandbox.id}`}
|
onVisibilityChange={onVisibilityChange}
|
||||||
className={cn(
|
onDelete={onDelete}
|
||||||
"transition-all focus-visible:outline-none focus-visible:ring-offset-2 focus-visible:ring-offset-background focus-visible:ring-2 focus-visible:ring-ring rounded-lg",
|
deletingId={deletingId}
|
||||||
deletingId === sandbox.id
|
isAuthenticated
|
||||||
? "pointer-events-none opacity-50 cursor-events-none"
|
{...sandbox}
|
||||||
: "cursor-pointer"
|
/>
|
||||||
)}
|
|
||||||
>
|
|
||||||
<ProjectCard
|
|
||||||
onVisibilityChange={onVisibilityChange}
|
|
||||||
onDelete={onDelete}
|
|
||||||
deletingId={deletingId}
|
|
||||||
isAuthenticated
|
|
||||||
{...sandbox}
|
|
||||||
/>
|
|
||||||
</Link>
|
|
||||||
)
|
)
|
||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
"use client"
|
"use client"
|
||||||
|
|
||||||
|
import NewProjectModal from "@/components/dashboard/newProject"
|
||||||
import ProjectCard from "@/components/dashboard/projectCard/"
|
import ProjectCard from "@/components/dashboard/projectCard/"
|
||||||
import { Button } from "@/components/ui/button"
|
import { Button } from "@/components/ui/button"
|
||||||
import {
|
import {
|
||||||
@ -13,64 +14,118 @@ import {
|
|||||||
HoverCardContent,
|
HoverCardContent,
|
||||||
HoverCardTrigger,
|
HoverCardTrigger,
|
||||||
} from "@/components/ui/hover-card"
|
} from "@/components/ui/hover-card"
|
||||||
|
import { Label } from "@/components/ui/label"
|
||||||
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
|
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
|
||||||
import { deleteSandbox, updateSandbox } from "@/lib/actions"
|
import { deleteSandbox, updateSandbox, updateUser } from "@/lib/actions"
|
||||||
import { MAX_FREE_GENERATION } from "@/lib/constant"
|
import { MAX_FREE_GENERATION } from "@/lib/constant"
|
||||||
import { Sandbox, User } from "@/lib/types"
|
import { SandboxWithLiked, User } from "@/lib/types"
|
||||||
import { cn } from "@/lib/utils"
|
import { useUser } from "@clerk/nextjs"
|
||||||
import { Heart, LucideIcon, Package2, PlusCircle, Sparkles } from "lucide-react"
|
import {
|
||||||
import Link from "next/link"
|
Edit,
|
||||||
import { useMemo, useState } from "react"
|
Heart,
|
||||||
|
Info,
|
||||||
|
Loader2,
|
||||||
|
LucideIcon,
|
||||||
|
Package2,
|
||||||
|
PlusCircle,
|
||||||
|
Sparkles,
|
||||||
|
X,
|
||||||
|
} from "lucide-react"
|
||||||
|
import { useRouter } from "next/navigation"
|
||||||
|
import { Fragment, useCallback, useEffect, useMemo, useState } from "react"
|
||||||
|
import { useFormState, useFormStatus } from "react-dom"
|
||||||
import { toast } from "sonner"
|
import { toast } from "sonner"
|
||||||
import Avatar from "../ui/avatar"
|
import Avatar from "../ui/avatar"
|
||||||
import { Badge } from "../ui/badge"
|
import { Badge } from "../ui/badge"
|
||||||
|
import { Input } from "../ui/input"
|
||||||
import { Progress } from "../ui/progress"
|
import { Progress } from "../ui/progress"
|
||||||
|
|
||||||
|
// #region Profile Page
|
||||||
export default function ProfilePage({
|
export default function ProfilePage({
|
||||||
publicSandboxes,
|
publicSandboxes,
|
||||||
privateSandboxes,
|
privateSandboxes,
|
||||||
user,
|
profileOwner,
|
||||||
currentUser,
|
loggedInUser,
|
||||||
}: {
|
}: {
|
||||||
publicSandboxes: Sandbox[]
|
publicSandboxes: SandboxWithLiked[]
|
||||||
privateSandboxes: Sandbox[]
|
privateSandboxes: SandboxWithLiked[]
|
||||||
user: User
|
profileOwner: User
|
||||||
currentUser: User | null
|
loggedInUser: User | null
|
||||||
}) {
|
}) {
|
||||||
const [deletingId, setDeletingId] = useState<string>("")
|
const isOwnProfile = profileOwner.id === loggedInUser?.id
|
||||||
const isLoggedIn = Boolean(currentUser)
|
|
||||||
const hasPublicSandboxes = publicSandboxes.length > 0
|
|
||||||
const hasPrivateSandboxes = privateSandboxes.length > 0
|
|
||||||
|
|
||||||
const onVisibilityChange = useMemo(
|
const sandboxes = useMemo(() => {
|
||||||
() => async (sandbox: Pick<Sandbox, "id" | "name" | "visibility">) => {
|
const allSandboxes = isOwnProfile
|
||||||
const newVisibility =
|
|
||||||
sandbox.visibility === "public" ? "private" : "public"
|
|
||||||
toast(`Project ${sandbox.name} is now ${newVisibility}.`)
|
|
||||||
await updateSandbox({
|
|
||||||
id: sandbox.id,
|
|
||||||
visibility: newVisibility,
|
|
||||||
})
|
|
||||||
},
|
|
||||||
[]
|
|
||||||
)
|
|
||||||
|
|
||||||
const onDelete = useMemo(
|
|
||||||
() => async (sandbox: Pick<Sandbox, "id" | "name">) => {
|
|
||||||
setDeletingId(sandbox.id)
|
|
||||||
toast(`Project ${sandbox.name} deleted.`)
|
|
||||||
await deleteSandbox(sandbox.id)
|
|
||||||
setDeletingId("")
|
|
||||||
},
|
|
||||||
[]
|
|
||||||
)
|
|
||||||
const stats = useMemo(() => {
|
|
||||||
const allSandboxes = isLoggedIn
|
|
||||||
? [...publicSandboxes, ...privateSandboxes]
|
? [...publicSandboxes, ...privateSandboxes]
|
||||||
: publicSandboxes
|
: publicSandboxes
|
||||||
|
|
||||||
const totalSandboxes = allSandboxes.length
|
return allSandboxes
|
||||||
const totalLikes = allSandboxes.reduce(
|
}, [isOwnProfile, publicSandboxes, privateSandboxes])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className="container mx-auto p-6 grid grid-cols-1 md:grid-cols-3 gap-6">
|
||||||
|
<div className="md:col-span-1">
|
||||||
|
<ProfileCard
|
||||||
|
name={profileOwner.name}
|
||||||
|
username={profileOwner.username}
|
||||||
|
avatarUrl={profileOwner.avatarUrl}
|
||||||
|
sandboxes={sandboxes}
|
||||||
|
joinedDate={profileOwner.createdAt}
|
||||||
|
generations={isOwnProfile ? loggedInUser.generations : undefined}
|
||||||
|
isOwnProfile={isOwnProfile}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="md:col-span-2">
|
||||||
|
<SandboxesPanel
|
||||||
|
{...{
|
||||||
|
publicSandboxes,
|
||||||
|
privateSandboxes,
|
||||||
|
isOwnProfile,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// #endregion
|
||||||
|
|
||||||
|
// #region Profile Card
|
||||||
|
function ProfileCard({
|
||||||
|
name,
|
||||||
|
username,
|
||||||
|
avatarUrl,
|
||||||
|
sandboxes,
|
||||||
|
joinedDate,
|
||||||
|
generations,
|
||||||
|
isOwnProfile,
|
||||||
|
}: {
|
||||||
|
name: string
|
||||||
|
username: string
|
||||||
|
avatarUrl: string | null
|
||||||
|
sandboxes: SandboxWithLiked[]
|
||||||
|
joinedDate: Date
|
||||||
|
generations?: number
|
||||||
|
isOwnProfile: boolean
|
||||||
|
}) {
|
||||||
|
const { user } = useUser()
|
||||||
|
const router = useRouter()
|
||||||
|
const [isEditing, setIsEditing] = useState(false)
|
||||||
|
const [formState, formAction] = useFormState(updateUser, {})
|
||||||
|
const joinedAt = useMemo(() => {
|
||||||
|
const date = new Date(joinedDate).toLocaleDateString("en-US", {
|
||||||
|
month: "long",
|
||||||
|
year: "numeric",
|
||||||
|
})
|
||||||
|
return `Joined ${date}`
|
||||||
|
}, [joinedDate])
|
||||||
|
const toggleEdit = useCallback(() => {
|
||||||
|
setIsEditing((s) => !s)
|
||||||
|
}, [])
|
||||||
|
const stats = useMemo(() => {
|
||||||
|
const totalSandboxes = sandboxes.length
|
||||||
|
const totalLikes = sandboxes.reduce(
|
||||||
(sum, sandbox) => sum + sandbox.likeCount,
|
(sum, sandbox) => sum + sandbox.likeCount,
|
||||||
0
|
0
|
||||||
)
|
)
|
||||||
@ -80,160 +135,299 @@ export default function ProfilePage({
|
|||||||
totalSandboxes === 1 ? "1 sandbox" : `${totalSandboxes} sandboxes`,
|
totalSandboxes === 1 ? "1 sandbox" : `${totalSandboxes} sandboxes`,
|
||||||
likes: totalLikes === 1 ? "1 like" : `${totalLikes} likes`,
|
likes: totalLikes === 1 ? "1 like" : `${totalLikes} likes`,
|
||||||
}
|
}
|
||||||
}, [isLoggedIn, publicSandboxes, privateSandboxes])
|
}, [sandboxes])
|
||||||
const joinDate = useMemo(
|
|
||||||
() =>
|
|
||||||
new Date(user.createdAt).toLocaleDateString("en-US", {
|
|
||||||
month: "long",
|
|
||||||
year: "numeric",
|
|
||||||
}),
|
|
||||||
[user.createdAt]
|
|
||||||
)
|
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if ("message" in formState) {
|
||||||
|
toast.success(formState.message as String)
|
||||||
|
toggleEdit()
|
||||||
|
if ("newRoute" in formState && typeof formState.newRoute === "string") {
|
||||||
|
router.replace(formState.newRoute)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ("error" in formState) {
|
||||||
|
const error = formState.error
|
||||||
|
if (typeof error === "string") {
|
||||||
|
toast.error(error)
|
||||||
|
} else {
|
||||||
|
toast.error("An Error Occured")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [formState])
|
||||||
return (
|
return (
|
||||||
<>
|
<Card className="mb-6 md:mb-0 sticky top-6">
|
||||||
<div className="container mx-auto p-6 grid grid-cols-1 md:grid-cols-3 gap-6">
|
{isOwnProfile && (
|
||||||
<div className="md:col-span-1">
|
<Button
|
||||||
<Card className="mb-6 md:mb-0 sticky top-6">
|
onClick={toggleEdit}
|
||||||
<CardContent className="flex flex-col gap-3 items-center pt-6">
|
aria-label={isEditing ? "close edit form" : "open edit form"}
|
||||||
<Avatar
|
size="smIcon"
|
||||||
name={user.name}
|
variant="secondary"
|
||||||
avatarUrl={user.avatarUrl}
|
className="rounded-full absolute top-2 right-2"
|
||||||
className="size-36"
|
>
|
||||||
/>
|
{isEditing ? <X className="size-4" /> : <Edit className="size-4" />}
|
||||||
|
|
||||||
<CardTitle className="text-2xl">{user.name}</CardTitle>
|
|
||||||
<CardDescription>{`@${user.username}`}</CardDescription>
|
|
||||||
<div className="flex gap-6">
|
|
||||||
<StatsItem icon={Package2} label={stats.sandboxes} />
|
|
||||||
<StatsItem icon={Heart} label={stats.likes} />
|
|
||||||
</div>
|
|
||||||
<div className="flex flex-col items-center gap-2">
|
|
||||||
<p className="text-xs text-muted-foreground">
|
|
||||||
{`Joined ${joinDate}`}
|
|
||||||
</p>
|
|
||||||
{isLoggedIn && <SubscriptionBadge user={currentUser!} />}
|
|
||||||
</div>
|
|
||||||
</CardContent>
|
|
||||||
</Card>
|
|
||||||
</div>
|
|
||||||
<div className="md:col-span-2">
|
|
||||||
<Tabs defaultValue="public">
|
|
||||||
<TabsList className="mb-4">
|
|
||||||
<TabsTrigger value="public">Public</TabsTrigger>
|
|
||||||
{isLoggedIn && <TabsTrigger value="private">Private</TabsTrigger>}
|
|
||||||
</TabsList>
|
|
||||||
<TabsContent value="public">
|
|
||||||
{hasPublicSandboxes ? (
|
|
||||||
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
|
||||||
{publicSandboxes.map((sandbox) => {
|
|
||||||
return (
|
|
||||||
<Link
|
|
||||||
key={sandbox.id}
|
|
||||||
href={`/code/${sandbox.id}`}
|
|
||||||
className={cn(
|
|
||||||
"transition-all focus-visible:outline-none focus-visible:ring-offset-2 focus-visible:ring-offset-background focus-visible:ring-2 focus-visible:ring-ring rounded-lg",
|
|
||||||
deletingId === sandbox.id
|
|
||||||
? "pointer-events-none opacity-50 cursor-events-none"
|
|
||||||
: "cursor-pointer"
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
{isLoggedIn ? (
|
|
||||||
<ProjectCard
|
|
||||||
onVisibilityChange={onVisibilityChange}
|
|
||||||
onDelete={onDelete}
|
|
||||||
deletingId={deletingId}
|
|
||||||
isAuthenticated
|
|
||||||
{...sandbox}
|
|
||||||
/>
|
|
||||||
) : (
|
|
||||||
<ProjectCard isAuthenticated={false} {...sandbox} />
|
|
||||||
)}
|
|
||||||
</Link>
|
|
||||||
)
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<EmptyState
|
|
||||||
title="No public sandboxes yet"
|
|
||||||
description={
|
|
||||||
isLoggedIn
|
|
||||||
? "Create your first public sandbox to share your work with the world!"
|
|
||||||
: "Login to create public sandboxes"
|
|
||||||
}
|
|
||||||
isLoggedIn={isLoggedIn}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</TabsContent>
|
|
||||||
{isLoggedIn && (
|
|
||||||
<TabsContent value="private">
|
|
||||||
{hasPrivateSandboxes ? (
|
|
||||||
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
|
||||||
{privateSandboxes.map((sandbox) => (
|
|
||||||
<Link
|
|
||||||
key={sandbox.id}
|
|
||||||
href={`/code/${sandbox.id}`}
|
|
||||||
className={cn(
|
|
||||||
"transition-all focus-visible:outline-none focus-visible:ring-offset-2 focus-visible:ring-offset-background focus-visible:ring-2 focus-visible:ring-ring rounded-lg",
|
|
||||||
deletingId === sandbox.id
|
|
||||||
? "pointer-events-none opacity-50 cursor-events-none"
|
|
||||||
: "cursor-pointer"
|
|
||||||
)}
|
|
||||||
>
|
|
||||||
<ProjectCard
|
|
||||||
onVisibilityChange={onVisibilityChange}
|
|
||||||
onDelete={onDelete}
|
|
||||||
deletingId={deletingId}
|
|
||||||
isAuthenticated
|
|
||||||
{...sandbox}
|
|
||||||
/>
|
|
||||||
</Link>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<EmptyState
|
|
||||||
title="No private sandboxes yet"
|
|
||||||
description={
|
|
||||||
isLoggedIn
|
|
||||||
? "Create your first private sandbox to start working on your personal projects!"
|
|
||||||
: "Login to create private sandboxes"
|
|
||||||
}
|
|
||||||
isLoggedIn={isLoggedIn}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</TabsContent>
|
|
||||||
)}
|
|
||||||
</Tabs>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
function EmptyState({
|
|
||||||
title,
|
|
||||||
description,
|
|
||||||
isLoggedIn,
|
|
||||||
}: {
|
|
||||||
title: string
|
|
||||||
description: string
|
|
||||||
isLoggedIn: boolean
|
|
||||||
}) {
|
|
||||||
return (
|
|
||||||
<Card className="flex flex-col items-center justify-center p-6 text-center h-[300px]">
|
|
||||||
<PlusCircle className="h-12 w-12 text-muted-foreground mb-4" />
|
|
||||||
<CardTitle className="text-xl mb-2">{title}</CardTitle>
|
|
||||||
<CardDescription className="mb-4">{description}</CardDescription>
|
|
||||||
{isLoggedIn && (
|
|
||||||
<Button>
|
|
||||||
<PlusCircle className="h-4 w-4 mr-2" />
|
|
||||||
Create Sandbox
|
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
|
<CardContent className="flex flex-col gap-4 items-center pt-6">
|
||||||
|
<Avatar name={name} avatarUrl={avatarUrl} className="size-36" />
|
||||||
|
|
||||||
|
{!isEditing ? (
|
||||||
|
<div className="space-y-2">
|
||||||
|
<CardTitle className="text-2xl text-center">{name}</CardTitle>
|
||||||
|
<CardDescription className="text-center">{`@${username}`}</CardDescription>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<form action={formAction} className="flex flex-col gap-2">
|
||||||
|
<Input
|
||||||
|
name="id"
|
||||||
|
placeholder="ID"
|
||||||
|
className="hidden "
|
||||||
|
value={user?.id}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
name="oldUsername"
|
||||||
|
placeholder="ID"
|
||||||
|
className="hidden "
|
||||||
|
value={user?.username ?? undefined}
|
||||||
|
/>
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Label htmlFor="input-name">Name</Label>
|
||||||
|
<Input
|
||||||
|
id="input-name"
|
||||||
|
name="name"
|
||||||
|
placeholder="Name"
|
||||||
|
defaultValue={name}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="space-y-1">
|
||||||
|
<Label htmlFor="input-username">User name</Label>
|
||||||
|
<div className="relative">
|
||||||
|
<Input
|
||||||
|
id="input-username"
|
||||||
|
className="peer ps-6"
|
||||||
|
type="text"
|
||||||
|
name="username"
|
||||||
|
placeholder="Username"
|
||||||
|
defaultValue={username}
|
||||||
|
/>
|
||||||
|
<span className="pointer-events-none absolute inset-y-0 start-0 flex items-center justify-center ps-2 text-sm text-muted-foreground peer-disabled:opacity-50">
|
||||||
|
@
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<SubmitButton />
|
||||||
|
</form>
|
||||||
|
)}
|
||||||
|
{!isEditing && (
|
||||||
|
<>
|
||||||
|
<div className="flex gap-6">
|
||||||
|
<StatsItem icon={Package2} label={stats.sandboxes} />
|
||||||
|
<StatsItem icon={Heart} label={stats.likes} />
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col items-center gap-2">
|
||||||
|
<p className="text-xs text-muted-foreground">{joinedAt}</p>
|
||||||
|
{typeof generations === "number" && (
|
||||||
|
<SubscriptionBadge generations={generations} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</CardContent>
|
||||||
</Card>
|
</Card>
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function SubmitButton() {
|
||||||
|
const { pending } = useFormStatus()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Button size="sm" type="submit" className="w-full mt-2" disabled={pending}>
|
||||||
|
{pending && <Loader2 className="animate-spin mr-2 h-4 w-4" />}
|
||||||
|
Save
|
||||||
|
</Button>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// #endregion
|
||||||
|
|
||||||
|
// #region Sandboxes Panel
|
||||||
|
function SandboxesPanel({
|
||||||
|
publicSandboxes,
|
||||||
|
privateSandboxes,
|
||||||
|
isOwnProfile,
|
||||||
|
}: {
|
||||||
|
publicSandboxes: SandboxWithLiked[]
|
||||||
|
privateSandboxes: SandboxWithLiked[]
|
||||||
|
isOwnProfile: boolean
|
||||||
|
}) {
|
||||||
|
const [deletingId, setDeletingId] = useState<string>("")
|
||||||
|
const hasPublicSandboxes = publicSandboxes.length > 0
|
||||||
|
const hasPrivateSandboxes = privateSandboxes.length > 0
|
||||||
|
|
||||||
|
const onVisibilityChange = useMemo(
|
||||||
|
() =>
|
||||||
|
async (sandbox: Pick<SandboxWithLiked, "id" | "name" | "visibility">) => {
|
||||||
|
const newVisibility =
|
||||||
|
sandbox.visibility === "public" ? "private" : "public"
|
||||||
|
toast(`Project ${sandbox.name} is now ${newVisibility}.`)
|
||||||
|
await updateSandbox({
|
||||||
|
id: sandbox.id,
|
||||||
|
visibility: newVisibility,
|
||||||
|
})
|
||||||
|
},
|
||||||
|
[]
|
||||||
|
)
|
||||||
|
|
||||||
|
const onDelete = useMemo(
|
||||||
|
() => async (sandbox: Pick<SandboxWithLiked, "id" | "name">) => {
|
||||||
|
setDeletingId(sandbox.id)
|
||||||
|
toast(`Project ${sandbox.name} deleted.`)
|
||||||
|
await deleteSandbox(sandbox.id)
|
||||||
|
setDeletingId("")
|
||||||
|
},
|
||||||
|
[]
|
||||||
|
)
|
||||||
|
if (!isOwnProfile) {
|
||||||
|
return (
|
||||||
|
<div className="">
|
||||||
|
{hasPublicSandboxes ? (
|
||||||
|
<>
|
||||||
|
<h2 className="font-semibold text-xl mb-4">Sandboxes</h2>
|
||||||
|
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
||||||
|
{publicSandboxes.map((sandbox) => {
|
||||||
|
return (
|
||||||
|
<Fragment key={sandbox.id}>
|
||||||
|
{isOwnProfile ? (
|
||||||
|
<ProjectCard
|
||||||
|
onVisibilityChange={onVisibilityChange}
|
||||||
|
onDelete={onDelete}
|
||||||
|
deletingId={deletingId}
|
||||||
|
isAuthenticated
|
||||||
|
{...sandbox}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<ProjectCard isAuthenticated={false} {...sandbox} />
|
||||||
|
)}
|
||||||
|
</Fragment>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<EmptyState type="private" isOwnProfile={isOwnProfile} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Tabs defaultValue="public">
|
||||||
|
<TabsList className="mb-4">
|
||||||
|
<TabsTrigger value="public">Public</TabsTrigger>
|
||||||
|
<TabsTrigger value="private">Private</TabsTrigger>
|
||||||
|
</TabsList>
|
||||||
|
<TabsContent value="public">
|
||||||
|
{hasPublicSandboxes ? (
|
||||||
|
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
||||||
|
{publicSandboxes.map((sandbox) => {
|
||||||
|
return (
|
||||||
|
<Fragment key={sandbox.id}>
|
||||||
|
{isOwnProfile ? (
|
||||||
|
<ProjectCard
|
||||||
|
onVisibilityChange={onVisibilityChange}
|
||||||
|
onDelete={onDelete}
|
||||||
|
deletingId={deletingId}
|
||||||
|
isAuthenticated
|
||||||
|
{...sandbox}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<ProjectCard isAuthenticated={false} {...sandbox} />
|
||||||
|
)}
|
||||||
|
</Fragment>
|
||||||
|
)
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<EmptyState type="public" isOwnProfile={isOwnProfile} />
|
||||||
|
)}
|
||||||
|
</TabsContent>
|
||||||
|
<TabsContent value="private">
|
||||||
|
{hasPrivateSandboxes ? (
|
||||||
|
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
||||||
|
{privateSandboxes.map((sandbox) => (
|
||||||
|
<ProjectCard
|
||||||
|
key={sandbox.id}
|
||||||
|
onVisibilityChange={onVisibilityChange}
|
||||||
|
onDelete={onDelete}
|
||||||
|
deletingId={deletingId}
|
||||||
|
isAuthenticated
|
||||||
|
{...sandbox}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<EmptyState type="private" isOwnProfile={isOwnProfile} />
|
||||||
|
)}
|
||||||
|
</TabsContent>
|
||||||
|
</Tabs>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// #endregion
|
||||||
|
|
||||||
|
// #region Empty State
|
||||||
|
function EmptyState({
|
||||||
|
type,
|
||||||
|
isOwnProfile,
|
||||||
|
}: {
|
||||||
|
type: "public" | "private"
|
||||||
|
isOwnProfile: boolean
|
||||||
|
}) {
|
||||||
|
const [newProjectModalOpen, setNewProjectModalOpen] = useState(false)
|
||||||
|
|
||||||
|
const text = useMemo(() => {
|
||||||
|
let title: string
|
||||||
|
let description: string
|
||||||
|
switch (type) {
|
||||||
|
case "public":
|
||||||
|
title = "No public sandboxes yet"
|
||||||
|
description = isOwnProfile
|
||||||
|
? "Create your first public sandbox to share your work with the world!"
|
||||||
|
: "user has no public sandboxes"
|
||||||
|
|
||||||
|
case "private":
|
||||||
|
title = "No private sandboxes yet"
|
||||||
|
description = isOwnProfile
|
||||||
|
? "Create your first private sandbox to start working on your personal projects!"
|
||||||
|
: "user has no private sandboxes"
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
title,
|
||||||
|
description,
|
||||||
|
}
|
||||||
|
}, [type, isOwnProfile])
|
||||||
|
const openModal = useCallback(() => setNewProjectModalOpen(true), [])
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Card className="flex flex-col items-center justify-center p-6 text-center h-[300px]">
|
||||||
|
<PlusCircle className="h-12 w-12 text-muted-foreground mb-4" />
|
||||||
|
<CardTitle className="text-xl mb-2">{text.title}</CardTitle>
|
||||||
|
<CardDescription className="mb-4">{text.description}</CardDescription>
|
||||||
|
{isOwnProfile && (
|
||||||
|
<Button onClick={openModal}>
|
||||||
|
<PlusCircle className="h-4 w-4 mr-2" />
|
||||||
|
Create Sandbox
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</Card>
|
||||||
|
<NewProjectModal
|
||||||
|
open={newProjectModalOpen}
|
||||||
|
setOpen={setNewProjectModalOpen}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
// #endregion
|
||||||
|
|
||||||
|
// #region StatsItem
|
||||||
interface StatsItemProps {
|
interface StatsItemProps {
|
||||||
icon: LucideIcon
|
icon: LucideIcon
|
||||||
label: string
|
label: string
|
||||||
@ -245,31 +439,39 @@ const StatsItem = ({ icon: Icon, label }: StatsItemProps) => (
|
|||||||
<span className="text-sm text-muted-foreground">{label}</span>
|
<span className="text-sm text-muted-foreground">{label}</span>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
|
// #endregion
|
||||||
|
|
||||||
const SubscriptionBadge = ({ user }: { user: User }) => {
|
// #region Sub Badge
|
||||||
|
const SubscriptionBadge = ({ generations }: { generations: number }) => {
|
||||||
return (
|
return (
|
||||||
<HoverCard>
|
<div className="flex gap-2 items-center">
|
||||||
<HoverCardTrigger>
|
<Badge variant="secondary" className="text-sm cursor-pointer">
|
||||||
<Badge variant="secondary" className="text-xs cursor-pointer">
|
Free
|
||||||
Free
|
</Badge>
|
||||||
</Badge>
|
<HoverCard>
|
||||||
</HoverCardTrigger>
|
<HoverCardTrigger>
|
||||||
<HoverCardContent>
|
<Button variant="ghost" size="smIcon">
|
||||||
<div className="w-full space-y-2">
|
<Info size={20} />
|
||||||
<div className="flex justify-between text-sm">
|
</Button>
|
||||||
<span className="font-medium">AI Generations</span>
|
</HoverCardTrigger>
|
||||||
<span>{`${user.generations} / ${MAX_FREE_GENERATION}`}</span>
|
<HoverCardContent>
|
||||||
|
<div className="w-full space-y-2">
|
||||||
|
<div className="flex justify-between text-sm">
|
||||||
|
<span className="font-medium">AI Generations</span>
|
||||||
|
<span>{`${generations} / ${MAX_FREE_GENERATION}`}</span>
|
||||||
|
</div>
|
||||||
|
<Progress
|
||||||
|
value={generations}
|
||||||
|
max={MAX_FREE_GENERATION}
|
||||||
|
className="w-full"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
<Progress
|
<Button size="sm" className="w-full mt-4">
|
||||||
value={user?.generations!}
|
<Sparkles className="mr-2 h-4 w-4" /> Upgrade to Pro
|
||||||
max={MAX_FREE_GENERATION}
|
</Button>
|
||||||
className="w-full"
|
</HoverCardContent>
|
||||||
/>
|
</HoverCard>
|
||||||
</div>
|
</div>
|
||||||
<Button size="sm" className="w-full mt-4">
|
|
||||||
<Sparkles className="mr-2 h-4 w-4" /> Upgrade to Pro
|
|
||||||
</Button>
|
|
||||||
</HoverCardContent>
|
|
||||||
</HoverCard>
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
// #endregion
|
||||||
|
@ -4,6 +4,7 @@ import UserButton from "@/components/ui/userButton"
|
|||||||
import { User } from "@/lib/types"
|
import { User } from "@/lib/types"
|
||||||
import Image from "next/image"
|
import Image from "next/image"
|
||||||
import Link from "next/link"
|
import Link from "next/link"
|
||||||
|
import { Button } from "../ui/button"
|
||||||
|
|
||||||
export default function ProfileNavbar({ userData }: { userData: User }) {
|
export default function ProfileNavbar({ userData }: { userData: User }) {
|
||||||
return (
|
return (
|
||||||
@ -19,7 +20,13 @@ export default function ProfileNavbar({ userData }: { userData: User }) {
|
|||||||
</div>
|
</div>
|
||||||
<div className="flex items-center space-x-4">
|
<div className="flex items-center space-x-4">
|
||||||
<ThemeSwitcher />
|
<ThemeSwitcher />
|
||||||
{Boolean(userData?.id) ? <UserButton userData={userData} /> : null}
|
{Boolean(userData?.id) ? (
|
||||||
|
<UserButton userData={userData} />
|
||||||
|
) : (
|
||||||
|
<Link href="/sign-in">
|
||||||
|
<Button>Login</Button>
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</nav>
|
</nav>
|
||||||
)
|
)
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
"use server"
|
"use server"
|
||||||
|
|
||||||
import { revalidatePath } from "next/cache"
|
import { revalidatePath } from "next/cache"
|
||||||
|
import { z } from "zod"
|
||||||
|
|
||||||
export async function createSandbox(body: {
|
export async function createSandbox(body: {
|
||||||
type: string
|
type: string
|
||||||
@ -91,3 +92,95 @@ export async function unshareSandbox(sandboxId: string, userId: string) {
|
|||||||
|
|
||||||
revalidatePath(`/code/${sandboxId}`)
|
revalidatePath(`/code/${sandboxId}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function toggleLike(sandboxId: string, userId: string) {
|
||||||
|
await fetch(
|
||||||
|
`${process.env.NEXT_PUBLIC_DATABASE_WORKER_URL}/api/sandbox/like`,
|
||||||
|
{
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Authorization: `${process.env.NEXT_PUBLIC_WORKERS_KEY}`,
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ sandboxId, userId }),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
revalidatePath(`/[username]`, "page")
|
||||||
|
revalidatePath(`/dashboard`, "page")
|
||||||
|
}
|
||||||
|
|
||||||
|
const UpdateErrorSchema = z.object({
|
||||||
|
error: z
|
||||||
|
.union([
|
||||||
|
z.string(),
|
||||||
|
z.array(
|
||||||
|
z.object({
|
||||||
|
path: z.array(z.string()),
|
||||||
|
message: z.string(),
|
||||||
|
})
|
||||||
|
),
|
||||||
|
])
|
||||||
|
.optional(),
|
||||||
|
})
|
||||||
|
|
||||||
|
export async function updateUser(prevState: any, formData: FormData) {
|
||||||
|
const data = Object.fromEntries(formData)
|
||||||
|
|
||||||
|
const schema = z.object({
|
||||||
|
id: z.string(),
|
||||||
|
username: z.string(),
|
||||||
|
oldUsername: z.string(),
|
||||||
|
name: z.string(),
|
||||||
|
})
|
||||||
|
console.log(data)
|
||||||
|
|
||||||
|
try {
|
||||||
|
const validatedData = schema.parse(data)
|
||||||
|
|
||||||
|
const changedUsername = validatedData.username !== validatedData.oldUsername
|
||||||
|
const res = await fetch(
|
||||||
|
`${process.env.NEXT_PUBLIC_DATABASE_WORKER_URL}/api/user`,
|
||||||
|
{
|
||||||
|
method: "PUT",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Authorization: `${process.env.NEXT_PUBLIC_WORKERS_KEY}`,
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
id: validatedData.id,
|
||||||
|
username: data.username ?? undefined,
|
||||||
|
name: data.name ?? undefined,
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const responseData = await res.json()
|
||||||
|
|
||||||
|
// Validate the response using our error schema
|
||||||
|
const parseResult = UpdateErrorSchema.safeParse(responseData)
|
||||||
|
|
||||||
|
if (!parseResult.success) {
|
||||||
|
return { error: "Unexpected error occurred" }
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parseResult.data.error) {
|
||||||
|
return parseResult.data
|
||||||
|
}
|
||||||
|
|
||||||
|
if (changedUsername) {
|
||||||
|
const newRoute = `/@${validatedData.username}`
|
||||||
|
return { message: "Successfully updated", newRoute }
|
||||||
|
}
|
||||||
|
revalidatePath(`/[username]`, "page")
|
||||||
|
return { message: "Successfully updated" }
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof z.ZodError) {
|
||||||
|
console.log(error)
|
||||||
|
return {
|
||||||
|
error: error.errors?.[0].message,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { error: "An unexpected error occurred" }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -23,7 +23,9 @@ export type Sandbox = {
|
|||||||
viewCount: number
|
viewCount: number
|
||||||
usersToSandboxes: UsersToSandboxes[]
|
usersToSandboxes: UsersToSandboxes[]
|
||||||
}
|
}
|
||||||
|
export type SandboxWithLiked = Sandbox & {
|
||||||
|
liked: boolean
|
||||||
|
}
|
||||||
export type UsersToSandboxes = {
|
export type UsersToSandboxes = {
|
||||||
userId: string
|
userId: string
|
||||||
sandboxId: string
|
sandboxId: string
|
||||||
|
Loading…
x
Reference in New Issue
Block a user