-
Notifications
You must be signed in to change notification settings - Fork 14
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #64 from deepraj21/main
feat: added star to project
- Loading branch information
Showing
14 changed files
with
454 additions
and
488 deletions.
There are no files selected for viewing
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
import { useState, useEffect } from "react"; | ||
import { CircleIcon, StarIcon } from "@radix-ui/react-icons"; | ||
import { Button } from "@/components/ui/button"; | ||
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card"; | ||
import { FaStar as FilledStarIcon } from "react-icons/fa"; | ||
|
||
interface ProjectProps { | ||
project: { | ||
projectId: string; | ||
title: string; | ||
description: string; | ||
repoLink: string; | ||
starCount: number; | ||
tags: string[]; | ||
}; | ||
} | ||
|
||
const backendUrl = import.meta.env.VITE_BACKEND_URL || 'http://localhost:5000'; | ||
|
||
export function ProjectCard({ project }: ProjectProps) { | ||
const [isStarred, setIsStarred] = useState(false); | ||
const [starCount, setStarCount] = useState(project.starCount); | ||
const username = localStorage.getItem('devhub_username'); | ||
|
||
// Fetch initial star state from server/localStorage or logic to check if user has starred | ||
useEffect(() => { | ||
const fetchStarState = async () => { | ||
const starredStatus = localStorage.getItem(`starred_${project.projectId}`); | ||
setIsStarred(!!starredStatus); | ||
}; | ||
fetchStarState(); | ||
}, [project.projectId]); | ||
|
||
const handleStarClick = async () => { | ||
if (isStarred) return; // Prevent multiple stars | ||
|
||
try { | ||
const response = await fetch(`${backendUrl}/profile/${username}/projects/${project.projectId}/star`, { | ||
method: "POST", | ||
headers: { "Content-Type": "application/json" }, | ||
}); | ||
|
||
if (response.ok) { | ||
setStarCount((prevCount) => prevCount + 1); | ||
setIsStarred(true); | ||
localStorage.setItem(`starred_${project.projectId}`, "true"); | ||
} else { | ||
console.error("Failed to star the project"); | ||
} | ||
} catch (error) { | ||
console.error("Error starring the project:", error); | ||
} | ||
}; | ||
|
||
return ( | ||
<Card> | ||
<CardHeader className="grid grid-cols-[1fr_110px] items-start gap-4 space-y-0"> | ||
<div className="space-y-1"> | ||
<CardTitle>{project.title}</CardTitle> | ||
<CardDescription>{project.description}</CardDescription> | ||
</div> | ||
<div className="flex items-center rounded-md bg-secondary text-secondary-foreground"> | ||
<Button | ||
variant="secondary" | ||
className=" shadow-none" | ||
onClick={handleStarClick} | ||
disabled={isStarred} | ||
> | ||
{isStarred ? ( | ||
<FilledStarIcon className="mr-2 h-4 w-4 text-yellow-400" /> | ||
) : ( | ||
<StarIcon className="mr-2 h-4 w-4" /> | ||
)} | ||
Star | ||
</Button> | ||
</div> | ||
</CardHeader> | ||
<CardContent> | ||
<div className="flex space-x-4 text-sm text-muted-foreground"> | ||
<div className="flex items-center"> | ||
<CircleIcon className="mr-1 h-3 w-3 fill-sky-400 text-sky-400" /> | ||
{project.tags.join(", ") || "No Tags"} | ||
</div> | ||
<div className="flex items-center"> | ||
<FilledStarIcon className="mr-1 h-3 w-3 text-yellow-400" /> | ||
{starCount} Stars | ||
</div> | ||
</div> | ||
</CardContent> | ||
</Card> | ||
); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
import * as React from "react" | ||
import * as AlertDialogPrimitive from "@radix-ui/react-alert-dialog" | ||
|
||
import { cn } from "@/lib/utils" | ||
import { buttonVariants } from "@/components/ui/button" | ||
|
||
const AlertDialog = AlertDialogPrimitive.Root | ||
|
||
const AlertDialogTrigger = AlertDialogPrimitive.Trigger | ||
|
||
const AlertDialogPortal = AlertDialogPrimitive.Portal | ||
|
||
const AlertDialogOverlay = React.forwardRef< | ||
React.ElementRef<typeof AlertDialogPrimitive.Overlay>, | ||
React.ComponentPropsWithoutRef<typeof AlertDialogPrimitive.Overlay> | ||
>(({ className, ...props }, ref) => ( | ||
<AlertDialogPrimitive.Overlay | ||
className={cn( | ||
"fixed inset-0 z-50 bg-black/80 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0", | ||
className | ||
)} | ||
{...props} | ||
ref={ref} | ||
/> | ||
)) | ||
AlertDialogOverlay.displayName = AlertDialogPrimitive.Overlay.displayName | ||
|
||
const AlertDialogContent = React.forwardRef< | ||
React.ElementRef<typeof AlertDialogPrimitive.Content>, | ||
React.ComponentPropsWithoutRef<typeof AlertDialogPrimitive.Content> | ||
>(({ className, ...props }, ref) => ( | ||
<AlertDialogPortal> | ||
<AlertDialogOverlay /> | ||
<AlertDialogPrimitive.Content | ||
ref={ref} | ||
className={cn( | ||
"fixed left-[50%] top-[50%] z-50 grid w-[90%] sm:w-auto translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg", | ||
className | ||
)} | ||
{...props} | ||
/> | ||
</AlertDialogPortal> | ||
)) | ||
AlertDialogContent.displayName = AlertDialogPrimitive.Content.displayName | ||
|
||
const AlertDialogHeader = ({ | ||
className, | ||
...props | ||
}: React.HTMLAttributes<HTMLDivElement>) => ( | ||
<div | ||
className={cn( | ||
"flex flex-col space-y-2 text-center sm:text-left", | ||
className | ||
)} | ||
{...props} | ||
/> | ||
) | ||
AlertDialogHeader.displayName = "AlertDialogHeader" | ||
|
||
const AlertDialogFooter = ({ | ||
className, | ||
...props | ||
}: React.HTMLAttributes<HTMLDivElement>) => ( | ||
<div | ||
className={cn( | ||
"flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2", | ||
className | ||
)} | ||
{...props} | ||
/> | ||
) | ||
AlertDialogFooter.displayName = "AlertDialogFooter" | ||
|
||
const AlertDialogTitle = React.forwardRef< | ||
React.ElementRef<typeof AlertDialogPrimitive.Title>, | ||
React.ComponentPropsWithoutRef<typeof AlertDialogPrimitive.Title> | ||
>(({ className, ...props }, ref) => ( | ||
<AlertDialogPrimitive.Title | ||
ref={ref} | ||
className={cn("text-lg font-semibold", className)} | ||
{...props} | ||
/> | ||
)) | ||
AlertDialogTitle.displayName = AlertDialogPrimitive.Title.displayName | ||
|
||
const AlertDialogDescription = React.forwardRef< | ||
React.ElementRef<typeof AlertDialogPrimitive.Description>, | ||
React.ComponentPropsWithoutRef<typeof AlertDialogPrimitive.Description> | ||
>(({ className, ...props }, ref) => ( | ||
<AlertDialogPrimitive.Description | ||
ref={ref} | ||
className={cn("text-sm text-muted-foreground", className)} | ||
{...props} | ||
/> | ||
)) | ||
AlertDialogDescription.displayName = | ||
AlertDialogPrimitive.Description.displayName | ||
|
||
const AlertDialogAction = React.forwardRef< | ||
React.ElementRef<typeof AlertDialogPrimitive.Action>, | ||
React.ComponentPropsWithoutRef<typeof AlertDialogPrimitive.Action> | ||
>(({ className, ...props }, ref) => ( | ||
<AlertDialogPrimitive.Action | ||
ref={ref} | ||
className={cn(buttonVariants(), className)} | ||
{...props} | ||
/> | ||
)) | ||
AlertDialogAction.displayName = AlertDialogPrimitive.Action.displayName | ||
|
||
const AlertDialogCancel = React.forwardRef< | ||
React.ElementRef<typeof AlertDialogPrimitive.Cancel>, | ||
React.ComponentPropsWithoutRef<typeof AlertDialogPrimitive.Cancel> | ||
>(({ className, ...props }, ref) => ( | ||
<AlertDialogPrimitive.Cancel | ||
ref={ref} | ||
className={cn( | ||
buttonVariants({ variant: "outline" }), | ||
"mt-2 sm:mt-0", | ||
className | ||
)} | ||
{...props} | ||
/> | ||
)) | ||
AlertDialogCancel.displayName = AlertDialogPrimitive.Cancel.displayName | ||
|
||
export { | ||
AlertDialog, | ||
AlertDialogPortal, | ||
AlertDialogOverlay, | ||
AlertDialogTrigger, | ||
AlertDialogContent, | ||
AlertDialogHeader, | ||
AlertDialogFooter, | ||
AlertDialogTitle, | ||
AlertDialogDescription, | ||
AlertDialogAction, | ||
AlertDialogCancel, | ||
} |
Oops, something went wrong.