Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
105 commits
Select commit Hold shift + click to select a range
f4dcdc0
fix: modify api.ts
Benjtalkshow Oct 14, 2025
b90d4ed
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Oct 16, 2025
9ea81a5
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Oct 16, 2025
5be269f
fix: remove google auth buttom
Benjtalkshow Oct 16, 2025
490dcb2
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Oct 20, 2025
842fd48
fix: fixes responsive fixes on organization
Benjtalkshow Oct 20, 2025
596a7f8
fix: minor fixes
Benjtalkshow Nov 6, 2025
96fee24
fix: minor fixes
Benjtalkshow Nov 6, 2025
9dfb149
fix: modify create organization
Benjtalkshow Nov 7, 2025
a194d90
fix: modify create organization
Benjtalkshow Nov 7, 2025
b2ceee0
fix: fix organization permission
Benjtalkshow Nov 8, 2025
9ea97d1
fix: merge into main
Benjtalkshow Nov 8, 2025
adb4629
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 8, 2025
0c11420
fix: merge into main
Benjtalkshow Nov 8, 2025
1f5ec24
feat: hackathon overview page
Benjtalkshow Nov 11, 2025
bca1ef6
feat: hackathon overview page
Benjtalkshow Nov 11, 2025
9202b4f
feat: implement participant overview
Benjtalkshow Nov 12, 2025
096f265
feat: implement participant overview
Benjtalkshow Nov 12, 2025
b3478d0
feat: implement resources tab
Benjtalkshow Nov 12, 2025
e83a0be
feat: implement the submission tab
Benjtalkshow Nov 12, 2025
398be9b
feat: implement comment tab
Benjtalkshow Nov 12, 2025
de546b1
fix: implement provider for hackathon
Benjtalkshow Nov 14, 2025
0fd2690
fix: implement provider for hackathon
Benjtalkshow Nov 14, 2025
0d7417f
fix: minor fixes
Benjtalkshow Nov 15, 2025
4d3efee
fix: merge branch 'main' of https://github.com/Benjtalkshow/boundless…
Benjtalkshow Nov 15, 2025
83893e4
fix: hackathon banner
Benjtalkshow Nov 15, 2025
8013d62
fix: hackathon banner
Benjtalkshow Nov 15, 2025
3fb1323
fix: fix hackthon conflict
Benjtalkshow Nov 15, 2025
b7fc94f
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 15, 2025
0594ac8
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 15, 2025
7e08bc1
fix: fix organization page
Benjtalkshow Nov 15, 2025
22c12c1
fix: fix organization page
Benjtalkshow Nov 16, 2025
08e5be5
fix: fix organization page
Benjtalkshow Nov 16, 2025
8e6bded
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 16, 2025
5e6c9e6
fix: use transform
Benjtalkshow Nov 17, 2025
9135f2e
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 17, 2025
c2409f4
fix: add tagline
Benjtalkshow Nov 18, 2025
b08579a
fix: add tagline
Benjtalkshow Nov 18, 2025
4b8ef58
fix: fix conflict
Benjtalkshow Nov 18, 2025
0ee756e
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 19, 2025
ab0384d
fix: minor fixes
Benjtalkshow Nov 20, 2025
91b5c18
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 20, 2025
67c9fee
fix: minor fixes
Benjtalkshow Nov 21, 2025
9b2029b
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 21, 2025
1fca425
fix: fix timeline and prizes
Benjtalkshow Nov 23, 2025
3c057a0
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 23, 2025
a75f5ee
fix: correct timeline events
Benjtalkshow Nov 26, 2025
da0d27f
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 26, 2025
3a0c19b
fix: implement registration deadline policy
Benjtalkshow Nov 27, 2025
c616182
fix: implement registration deadline policy
Benjtalkshow Nov 27, 2025
9a98077
fix: implement registration deadline policy
Benjtalkshow Nov 27, 2025
3ac17f8
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 28, 2025
386a338
feat: implement leave hackathon
Benjtalkshow Nov 28, 2025
f4ed466
feat: implement leave hackathon
Benjtalkshow Nov 28, 2025
12a1705
feat: implement leave hackathon
Benjtalkshow Nov 28, 2025
dd60321
fix: delete hackathon
Benjtalkshow Nov 28, 2025
a4e22b0
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 28, 2025
19dd6ee
fix: implement invite participants
Benjtalkshow Nov 30, 2025
8211c60
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Nov 30, 2025
fdebd0b
fix: implement participant profile viewing
Benjtalkshow Dec 1, 2025
f7fad5c
feat: fetch participants team
Benjtalkshow Dec 1, 2025
c110717
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Dec 1, 2025
68b1a2d
fix: redesign hackathon banner
Benjtalkshow Dec 1, 2025
9cf2a9e
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Dec 1, 2025
d662148
fix: fix hackthon card
Benjtalkshow Dec 4, 2025
6037e99
fix: Resolve conflict: delete middleware.ts
Benjtalkshow Dec 5, 2025
bc80ddb
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Dec 5, 2025
5569590
fix: Resolve conflict in ProfileHeader.tsx
Benjtalkshow Dec 9, 2025
dc1be4f
fix: fix search bar in blog page
Benjtalkshow Dec 24, 2025
67abffb
fix: fix search bar in blog page
Benjtalkshow Dec 24, 2025
a40739f
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Dec 24, 2025
1c4965f
fix: fix search bar in blog page
Benjtalkshow Dec 24, 2025
43a4b9b
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Dec 24, 2025
54bd42e
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Jan 4, 2026
b79d391
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Jan 7, 2026
5c906a5
fix: fix error in fetching team posts
Benjtalkshow Jan 10, 2026
b000eb7
feat: implement create team, get my team
Benjtalkshow Jan 10, 2026
8e95934
feat: implement create team, get my team
Benjtalkshow Jan 11, 2026
59586e4
feat: implement hackathon project submission flow
Benjtalkshow Jan 12, 2026
d812fec
feat: implement voting for submission
Benjtalkshow Jan 14, 2026
d501680
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Jan 20, 2026
e62d570
fix: team formation updates
Benjtalkshow Feb 3, 2026
15a581b
fix: fix conflict
Benjtalkshow Feb 3, 2026
4d89af9
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Feb 3, 2026
2589770
fix: implement team invitation
Benjtalkshow Feb 5, 2026
ed26db6
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Feb 5, 2026
1d20c57
feat: hackathon submissions bulk actions, ranking and ui refinements
Benjtalkshow Feb 5, 2026
aa259cd
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Feb 5, 2026
18747e4
fix: implement empty state for hackathons page
Benjtalkshow Feb 6, 2026
76a9124
feat: Implement submission visibility and explore submissions
Benjtalkshow Feb 6, 2026
e0fb888
feat: Implement winners tab for hackathon winners display
Benjtalkshow Feb 7, 2026
2775ba8
feat: implement hackathon analytics
Benjtalkshow Feb 11, 2026
199c698
fix: fix coderabbit corrections
Benjtalkshow Feb 11, 2026
e7e78dd
fix: fix coderabbit corrections
Benjtalkshow Feb 11, 2026
736b900
fix: fix coderabbit corrections
Benjtalkshow Feb 11, 2026
e92b634
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Feb 11, 2026
e289ce6
fix: fix conflict
Benjtalkshow Feb 11, 2026
4e45528
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Feb 11, 2026
29c281d
fix: fix organization settings data persistence, hackathondrafts and …
Benjtalkshow Feb 11, 2026
20a31e2
feat: fix user profile
Benjtalkshow Feb 12, 2026
415434e
Merge branch 'main' of https://github.com/Benjtalkshow/boundless into…
Benjtalkshow Feb 12, 2026
8497f6d
feat: Implemented the participant-facing and hackathon organizers ann…
Benjtalkshow Feb 13, 2026
5f53c32
feat: Implement judging dashboard and detailed breakdowns
Benjtalkshow Feb 15, 2026
5ab6866
feat: Implement judging dashboard and detailed breakdowns
Benjtalkshow Feb 15, 2026
a135b9a
fix: fix coderabbit corrections
Benjtalkshow Feb 15, 2026
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,199 @@
'use client';

import React, { useState, useEffect } from 'react';
import { useParams, useRouter } from 'next/navigation';
import { Megaphone, ArrowLeft, Calendar, User, Pin } from 'lucide-react';
import {
getAnnouncementDetails,
GetHackathonBySlug,
type HackathonAnnouncement,
} from '@/lib/api/hackathons/index';
import { useMarkdown } from '@/hooks/use-markdown';
import { BoundlessButton } from '@/components/buttons';
import Loading from '@/components/Loading';
import { Badge } from '@/components/ui/badge';

export default function AnnouncementDetailPage() {
const params = useParams();
const router = useRouter();
const announcementId = params.announcementId as string;
const slug = params.slug as string;

const [announcement, setAnnouncement] =
useState<HackathonAnnouncement | null>(null);
const [hackathonName, setHackathonName] = useState<string>('');
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);

useEffect(() => {
async function fetchDetails() {
if (!announcementId) return;
try {
setLoading(true);
const [announcementData, hackathonData] = await Promise.all([
getAnnouncementDetails(announcementId),
GetHackathonBySlug(slug),
]);
setAnnouncement(announcementData);
setHackathonName(hackathonData.data.name);
} catch (err) {
console.error('Failed to fetch details:', err);
setError(
'Failed to load announcement. It may have been deleted or moved.'
);
} finally {
setLoading(false);
}
}
fetchDetails();
}, [announcementId, slug]);

const { styledContent, loading: markdownLoading } = useMarkdown(
announcement?.content || '',
{
breaks: true,
gfm: true,
}
);

if (loading) {
return (
<div className='flex min-h-screen items-center justify-center bg-black'>
<Loading />
</div>
);
}

if (error || !announcement) {
return (
<div className='flex min-h-screen flex-col items-center justify-center bg-black p-6 text-center'>
<div className='mb-6 flex h-20 w-20 items-center justify-center rounded-full bg-red-500/10'>
<Megaphone className='h-10 w-10 text-red-500' />
</div>
<h1 className='mb-2 text-2xl font-bold text-white'>
Announcement Not Found
</h1>
<p className='mb-8 text-zinc-400'>
{error || "We couldn't find the announcement you're looking for."}
</p>
<BoundlessButton onClick={() => router.back()} variant='outline'>
<ArrowLeft className='mr-2 h-4 w-4' />
Go Back
</BoundlessButton>
</div>
);
}

return (
<div className='min-h-screen bg-black pb-24'>
{/* Top Header */}
<div className='sticky top-0 z-10 border-b border-zinc-900 bg-black/80 backdrop-blur-md'>
<div className='mx-auto max-w-4xl items-center justify-between px-6 py-4'>
<button
onClick={() => window.close()}
className='flex items-center gap-2 text-sm text-zinc-400 transition-colors hover:text-white'
>
<ArrowLeft className='h-4 w-4' />
Close Tab
</button>
<div className='flex items-center gap-2 text-zinc-500'>
<Megaphone className='text-primary h-4 w-4' />
<span className='text-xs font-medium tracking-widest uppercase'>
{hackathonName ? `${hackathonName} • ` : ''}Announcement
</span>
</div>
</div>
</div>

<div className='mx-auto max-w-4xl px-6 pt-12'>
{/* Title & Metadata */}
<div className='mb-12 space-y-6 border-b border-zinc-900 pb-12'>
<div className='flex flex-wrap items-center gap-3'>
{hackathonName && (
<Badge
variant='outline'
className='border-primary/30 text-primary bg-primary/5 font-bold tracking-tighter uppercase'
>
{hackathonName}
</Badge>
)}
{announcement.isPinned && (
<Badge
variant='secondary'
className='bg-primary/10 text-primary border-primary/20 flex items-center gap-1 tracking-tighter uppercase'
>
<Pin className='h-3 w-3 fill-current' />
Pinned
</Badge>
)}
<Badge
variant='outline'
className='tracking-tighter text-zinc-500 uppercase'
>
Published
</Badge>
</div>

<h1 className='text-2xl font-bold tracking-tight text-white'>
{announcement.title}
</h1>

<div className='flex flex-wrap items-center gap-6 text-sm text-zinc-400'>
<div className='flex items-center gap-2'>
<div className='flex h-8 w-8 items-center justify-center rounded-full bg-zinc-800 text-zinc-500 uppercase'>
{announcement.author?.name?.charAt(0) || (
<User className='h-4 w-4' />
)}
</div>
<span className='font-medium text-zinc-200'>
{announcement.author?.name || 'Organizer'}
</span>
</div>

<div className='flex items-center gap-2'>
<Calendar className='h-4 w-4 text-zinc-600' />
<span>
{new Date(
announcement.publishedAt || announcement.createdAt
).toLocaleDateString(undefined, {
weekday: 'long',
month: 'long',
day: 'numeric',
year: 'numeric',
})}
</span>
</div>
</div>
</div>

{/* Content */}
<div className='prose prose-invert prose-primary max-w-none'>
{markdownLoading ? (
<div className='flex h-32 items-center justify-center'>
<div className='border-primary h-6 w-6 animate-spin rounded-full border-2 border-t-transparent' />
</div>
) : (
<div className='text-lg leading-relaxed text-zinc-200'>
{styledContent}
</div>
)}
</div>

{/* Footer */}
<div className='mt-24 rounded-2xl border border-zinc-900 bg-zinc-950/50 p-8 text-center'>
<h3 className='mb-2 font-bold text-white'>End of Announcement</h3>
<p className='mb-6 text-sm text-zinc-500'>
This announcement was published by the hackathon organizers.
</p>
<BoundlessButton
onClick={() => window.close()}
variant='outline'
size='sm'
>
Return to Hackathon
</BoundlessButton>
</div>
</div>
</div>
);
}
57 changes: 51 additions & 6 deletions app/(landing)/hackathons/[slug]/page.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,13 @@ import { HackathonParticipants } from '@/components/hackathons/participants/hack
import { useCommentSystem } from '@/hooks/use-comment-system';
import { CommentEntityType } from '@/types/comment';
import { useTeamPosts } from '@/hooks/hackathon/use-team-posts';
import {
listAnnouncements,
type HackathonAnnouncement,
} from '@/lib/api/hackathons/index';
import { HackathonWinner } from '@/lib/api/hackathons';
import { Megaphone } from 'lucide-react';
import { AnnouncementsTab } from '@/components/hackathons/announcements/AnnouncementsTab';

export default function HackathonPage() {
const router = useRouter();
Expand Down Expand Up @@ -63,6 +69,29 @@ export default function HackathonPage() {
autoFetch: !!hackathonId,
});

// Fetch announcements for public view
const [announcements, setAnnouncements] = useState<HackathonAnnouncement[]>(
[]
);
const [announcementsLoading, setAnnouncementsLoading] = useState(false);

useEffect(() => {
async function fetchAnnouncements() {
if (!hackathonId) return;
try {
setAnnouncementsLoading(true);
const data = await listAnnouncements(hackathonId);
// Only show published announcements for public view
setAnnouncements(data.filter(a => !a.isDraft));
} catch (error) {
console.error('Failed to fetch announcements:', error);
} finally {
setAnnouncementsLoading(false);
}
}
fetchAnnouncements();
}, [hackathonId]);

const hackathonTabs = useMemo(() => {
const hasParticipants =
Array.isArray(currentHackathon?.participants) &&
Expand All @@ -71,9 +100,7 @@ export default function HackathonPage() {
const hasResources = currentHackathon?.resources?.[0];
const participantType = currentHackathon?.participantType;
const isTeamHackathon =
participantType === 'TEAM' ||
participantType === 'TEAM_OR_INDIVIDUAL' ||
participantType === 'INDIVIDUAL';
participantType === 'TEAM' || participantType === 'TEAM_OR_INDIVIDUAL';
const isTabEnabled =
currentHackathon?.enabledTabs?.includes('joinATeamTab') !== false;

Expand Down Expand Up @@ -110,6 +137,16 @@ export default function HackathonPage() {
},
]
: []),
...(announcements.length > 0
? [
{
id: 'announcements',
label: 'Announcements',
badge: announcements.length,
icon: Megaphone,
},
]
: []),
{
id: 'submission',
label: 'Submissions',
Expand Down Expand Up @@ -149,6 +186,7 @@ export default function HackathonPage() {
teamPosts.length,
hackathonId,
winners,
announcements,
]);

// Refresh hackathon data
Expand Down Expand Up @@ -202,8 +240,7 @@ export default function HackathonPage() {
// Team formation availability
const isTeamHackathon =
currentHackathon?.participantType === 'TEAM' ||
currentHackathon?.participantType === 'TEAM_OR_INDIVIDUAL' ||
currentHackathon?.participantType === 'INDIVIDUAL';
currentHackathon?.participantType === 'TEAM_OR_INDIVIDUAL';
const isTeamFormationEnabled =
isTeamHackathon &&
currentHackathon?.enabledTabs?.includes('joinATeamTab') !== false;
Expand Down Expand Up @@ -300,6 +337,7 @@ export default function HackathonPage() {
isTeamFormationEnabled,
registrationDeadlinePolicy: currentHackathon.registrationDeadlinePolicy, // Now matches API casing
registrationDeadline: currentHackathon.registrationDeadline,
participantType: currentHackathon.participantType,
onJoinClick: handleJoinClick,
onLeaveClick: handleLeaveClick,
isLeaving,
Expand Down Expand Up @@ -342,7 +380,7 @@ export default function HackathonPage() {
currency: tier.currency,
passMark: tier.passMark,
description: tier.description,
prizeAmount: tier.prizeAmount, // Keep as string to match PrizeTier interface
prizeAmount: tier.prizeAmount,
}))}
totalPrizePool={currentHackathon.prizeTiers
.reduce(
Expand Down Expand Up @@ -371,6 +409,13 @@ export default function HackathonPage() {
<HackathonParticipants />
)}

{activeTab === 'announcements' && announcements.length > 0 && (
<AnnouncementsTab
announcements={announcements}
hackathonSlug={hackathonId}
/>
)}

{activeTab === 'submission' && (
<SubmissionTab
organizationId={currentHackathon.organizationId}
Expand Down
Loading
Loading