This is a submission for the GitHub Finish-Up-A-Thon Challenge
What I Built
CampusBeat 2.0 β a React Native campus super-app for students across 17 colleges in Odisha, India.
It started in 2023 as a simple notice board aggregator: scrape college websites so students didn't have to visit them. It worked. Students used it. Then life happened, and it sat untouched on GitHub for three years.
This challenge gave me the push to finally open that repo again. What I found was equal parts embarrassing and educational.
What it is now:
- π° Real-time notices for 17 colleges β ITER, KIIT, NIT Rourkela, IIT, and more
- π¨ Complete UI overhaul β warm cream Γ charcoal Γ coral editorial design
- π 3D tiltable campus identity card you can share with friends
- π Bookmarks β save any notice, grouped by college
- π¬ Real-time campus chat rooms powered by Socket.io
- π Campus marketplace β buy/sell within your college
- π Push notifications via Firebase
Demo
The original app from July 2023: LinkedIn post
CampusBeat 2.0 β running on device:
Onboarding screen - with beautiful animation
Login screen β editorial serif heading, Lottie animation, warm ink hero

Register screen β custom college picker bottom sheet with live search

Home screen β quote card, college notice feed, floating tab bar

Profile screen β 3D tiltable campus card with holographic shimmer

Share modal β drag to rotate the card, share natively

News Explorer β college chips, notice type tabs, live banner

Marketplace β buy and sell within your college

Bookmark - your persistent news
Chat screen - live interaction within colleges
The Comeback Story
What I found after 3 years
Opening an old repo is humbling. Here is what I walked into.
The dead API. The home screen showed a daily quote β except quotable.io had shut down. Every user was silently seeing the hardcoded fallback for three years:
"Villains are not bad, they are just real and true."
Nobody told me.
The login race condition.
// 2023 β broken
dispatch(login(email, password)).then(() => {
if (user) navigation.navigate("Main");
// user is ALWAYS null here β Redux hasn't updated yet
});
Redux updates asynchronously. By the time .then() fires, user in that closure is still the stale pre-dispatch value. Nobody ever got navigated after logging in on a fresh install.
The AsyncStorage timestamp bug.
// currentTime is a number
// storedTimestamp is a STRING (AsyncStorage always returns strings)
currentTime - storedTimestamp < 24 * 60 * 60 * 1000
// number - string = NaN, NaN < anything = false
// Cache NEVER worked. Re-fetched the dead API every single app open.
The copy-paste bug. This exact if/else chain existed copy-pasted in two separate files:
if (selectedSubNews === "General Notice") setPrefix("gn");
if (selectedSubNews === "Exam Notice") setPrefix("en");
// ... 8 more branches in Feed.js AND GeneralNews.js
CORS locked to localhost. The deployed backend had origin: "http://localhost:3000". The mobile app in production couldn't talk to it at all.
Open scrape routes. Anyone could hit /odisha/update to trigger all 17 college scrapers simultaneously. No auth, no rate limiting.
What I changed
Design was the most visible transformation. Out: harsh #051E2D dark blue with #38A2E0 neon accents. In: #F2EDE4 warm cream, #1C1917 ink, #C8432A warm coral β a calm editorial palette that Gen Z actually responds to.
Bug fixes landed in this order:
// Fixed login β useEffect pattern, no closure stale value
useEffect(() => {
if (isAuthenticated) navigation.replace("Main");
}, [isAuthenticated]);
// Fixed AsyncStorage cache β parseInt the stored string
const fresh = storedData && storedTs &&
(now - parseInt(storedTs, 10)) < CACHE_DURATION_MS;
// Fixed prefix map β single shared utility, no duplication
// utils/prefixMap.js
export const getPrefix = (s) => PREFIX_MAP[s]?.prefix ?? "";
Architecture cleanup:
-
constants/api.jsβ all URLs in one place, no more scattered hardcoded strings -
utils/prefixMap.jsβ single source of truth for notice type β API prefix -
constants/style.jsβ full design token system: palette, typography, spacing, radius, shadows, college accent colors for all 17 colleges
New features:
- Socket.io campus chat rooms with typing indicators and message history
- Marketplace with category filtering, pagination, owner-only delete
- Bookmarks persisted to AsyncStorage, grouped by college, swipe-to-delete
- 3D campus card using
PanResponderβrotateX/rotateYinterpolation with holographic shimmer - Custom college picker bottom sheet (replaced the native
Pickerwith search + styled list)
The dependency upgrade tax
The project was 3 years old. Updating to current versions silently broke five things:
| Package | What broke |
|---|---|
mongoose 9.x |
mongoose.set("strictQuery") removed β server wouldn't start |
reanimated 4.x |
Layout renamed to LinearTransition
|
| React 19 | Hooks inside .map() callbacks crash in strict mode |
express 5.x |
Path-to-regexp v8 breaking wildcard route changes |
None of these gave helpful error messages. This is the real cost of abandoning a project β every year of neglect adds one more silent explosion to debug.
The tab flicker fix
The most satisfying bug to solve. The tab bar was flickering on every press. Root cause: paddingHorizontal: withSpring(...) inside useAnimatedStyle.
Animating layout props on the UI thread forces React Native to re-measure and re-layout every frame β that's the flicker. The fix was to stop animating layout props entirely:
// BEFORE β flicker
const pillStyle = useAnimatedStyle(() => ({
backgroundColor: interpolateColor(progress.value, [0, 1], [...]),
paddingHorizontal: withSpring(isActive ? 14 : 10), // β layout prop, causes flicker
}));
// AFTER β no flicker
// Static conditional styles for padding, only animate color
<Animated.View style={[
styles.tabPill,
isActive ? styles.tabPillActive : styles.tabPillInactive, // static
pillStyle, // only backgroundColor animated
]}>
Also: all 5 tab screens stay permanently mounted using opacity: 0 / pointerEvents: "none" toggle. The old condition && <Screen /> pattern unmounted and remounted on every switch β that's what caused the full-screen flash between tabs.
My Experience with GitHub Copilot
I want to be honest: Copilot didn't write the app. But it made me faster in specific, measurable ways.
Bug pattern recognition was the biggest win. When I showed it the login .then() code and described the symptom, it immediately identified the Redux closure staleness issue and suggested the useEffect on isAuthenticated pattern. That would have taken me 30 minutes of debugging.
Boilerplate acceleration. The Socket.io room management β join, message history fetch, typing emit, disconnect cleanup β Copilot scaffolded this from a single comment describing what I wanted. I wrote the business logic on top.
Code review. Asking Copilot to review specific functions caught the parseInt missing from the timestamp comparison, and spotted the duplicate prefix mapping across two files. Fresh eyes, instantly.
Animation scaffolding. I described the home screen bubble animation in plain English: "two morphing orbs that bounce off screen edges with organic shape, glow halos, and a blend effect when they meet." Copilot gave me the requestAnimationFrame loop structure and radial gradient layer setup. Saved an hour of Skia documentation reading.
The most honest framing: Copilot is a very good pair programmer who reads docs faster than I do. The design decisions, UX flows, architecture choices β mine. Copilot helped me build what I envisioned, faster.
Tech Stack
Frontend β React Native + Expo SDK 56, Reanimated 4.x, @shopify/react-native-skia, react-native-gesture-handler, socket.io-client, Redux Toolkit 2.x
Backend β Node.js + Express 5, MongoDB + Mongoose 9, Socket.io, Firebase Admin, Cheerio + Axios (web scraping)
Colleges supported β ITER Β· KIIT Β· CGU Β· OUTR Β· GITA Β· RAJDHANI Β· UTKAL Β· GEC Β· TRIDENT Β· NIT Rourkela Β· SILICON Β· IIT Bhubaneswar Β· IIT Delhi Β· IIT Bombay Β· IIT Kharagpur Β· IIT Madras Β· IIT Kanpur
Built in Bhubaneswar, Odisha. For every student who's tired of their college website.
Top comments (0)