From ab2b9e2bffa5a903bc16662de386f49cb3f433ea Mon Sep 17 00:00:00 2001 From: Mai Nguyen <123816878+in-mai-space@users.noreply.github.com> Date: Tue, 18 Jun 2024 00:09:12 +0700 Subject: [PATCH] feat: event & club preview components (#1034) Co-authored-by: Alder Whiteford --- backend/entities/clubs/events/transactions.go | 2 +- frontend/lib/package.json | 2 +- frontend/lib/src/types/index.ts | 1 + frontend/mobile/package.json | 2 +- .../mobile/src/app/(app)/(tabs)/_layout.tsx | 94 ++++---- .../mobile/src/app/(app)/(tabs)/index.tsx | 42 +--- frontend/mobile/src/app/(app)/_layout.tsx | 11 + frontend/mobile/src/app/(app)/club/[id].tsx | 220 ++++++++++++++++++ .../mobile/src/app/(app)/club/_layout.tsx | 13 ++ .../app/(app)/club/components/skeleton.tsx | 46 ++++ frontend/mobile/src/app/(app)/event/[id].tsx | 162 +++++-------- .../src/app/(app)/event/components/about.tsx | 36 +-- .../(app)/event/components/description.tsx | 5 +- .../app/(app)/event/components/location.tsx | 27 +-- .../app/(app)/event/components/overview.tsx | 40 ++-- .../app/(app)/event/components/register.tsx | 2 - .../event/components/upcoming-events.tsx | 5 +- .../components/AboutSection/AboutSection.tsx | 34 +++ .../AnimatedImageHeader.tsx | 61 +++++ .../components/Calendar/Day.tsx | 4 +- .../components/ClubIcon/ClubIcon.tsx | 3 +- .../components/ClubPage/ClubPage.tsx | 172 -------------- .../components/ClubPage/ClubPageHeader.tsx | 23 -- .../RecruitmentInfo/ClubRecruitmentInfo.tsx | 10 +- .../components/EventCard/EventCardList.tsx | 12 +- .../EventCard/Variants/EventCardBig.tsx | 4 +- .../EventCard/Variants/EventCardCalendar.tsx | 17 +- .../EventCard/Variants/EventCardSmall.tsx | 2 +- .../components/PageError/PageError.tsx} | 4 +- .../components/Preview/Club/ClubPreview.tsx | 125 ++++++++++ .../Preview/Club/ClubPreviewSkeleton.tsx | 38 +++ .../components/Preview/Event/EventPreview.tsx | 125 ++++++++++ .../Preview/Event/EventPreviewSkeleton.tsx | 38 +++ .../components/Preview/PreviewError.tsx | 14 ++ .../(design-system)/components/Tag/Tags.tsx | 27 +++ .../mobile/src/app/(design-system)/index.ts | 3 + frontend/mobile/src/app/_layout.tsx | 25 +- frontend/mobile/src/hooks/useClub.ts | 75 ++++++ frontend/mobile/src/hooks/useEvent.ts | 70 ++++++ frontend/mobile/src/hooks/usePreview.ts | 56 +++++ frontend/mobile/src/store/slices/clubSlice.ts | 68 ++++++ .../mobile/src/store/slices/eventSlice.ts | 69 ++++++ frontend/mobile/src/store/store.ts | 6 +- frontend/mobile/yarn.lock | 8 +- 44 files changed, 1307 insertions(+), 496 deletions(-) create mode 100644 frontend/mobile/src/app/(app)/club/[id].tsx create mode 100644 frontend/mobile/src/app/(app)/club/_layout.tsx create mode 100644 frontend/mobile/src/app/(app)/club/components/skeleton.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/AboutSection/AboutSection.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/AnimatedImageHeader/AnimatedImageHeader.tsx delete mode 100644 frontend/mobile/src/app/(design-system)/components/ClubPage/ClubPage.tsx delete mode 100644 frontend/mobile/src/app/(design-system)/components/ClubPage/ClubPageHeader.tsx rename frontend/mobile/src/app/{(app)/event/components/error.tsx => (design-system)/components/PageError/PageError.tsx} (90%) create mode 100644 frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreview.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreviewSkeleton.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreview.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreviewSkeleton.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/Preview/PreviewError.tsx create mode 100644 frontend/mobile/src/app/(design-system)/components/Tag/Tags.tsx create mode 100644 frontend/mobile/src/hooks/useClub.ts create mode 100644 frontend/mobile/src/hooks/useEvent.ts create mode 100644 frontend/mobile/src/hooks/usePreview.ts create mode 100644 frontend/mobile/src/store/slices/clubSlice.ts create mode 100644 frontend/mobile/src/store/slices/eventSlice.ts diff --git a/backend/entities/clubs/events/transactions.go b/backend/entities/clubs/events/transactions.go index 9d4ddf98a..dcdd159f3 100644 --- a/backend/entities/clubs/events/transactions.go +++ b/backend/entities/clubs/events/transactions.go @@ -14,7 +14,7 @@ func GetClubEvents(db *gorm.DB, clubID uuid.UUID, pageInfo fiberpaginate.PageInf db = cache.SetUseCache(db, true) var events []models.Event - if err := db.Where("club_id = ?", clubID).Scopes(utilities.IntoScope(pageInfo, db)).Find(&events).Error; err != nil { + if err := db.Where("host = ?", clubID).Scopes(utilities.IntoScope(pageInfo, db)).Find(&events).Error; err != nil { return nil, err } diff --git a/frontend/lib/package.json b/frontend/lib/package.json index c08e89d8b..fd4707354 100644 --- a/frontend/lib/package.json +++ b/frontend/lib/package.json @@ -1,6 +1,6 @@ { "name": "@generatesac/lib", - "version": "0.0.170", + "version": "0.0.171", "type": "module", "main": "dist/index.js", "types": "dist/index.d.ts", diff --git a/frontend/lib/src/types/index.ts b/frontend/lib/src/types/index.ts index 693dbacf2..6352c6a66 100644 --- a/frontend/lib/src/types/index.ts +++ b/frontend/lib/src/types/index.ts @@ -10,3 +10,4 @@ export * from "./event"; export * from "./file"; export * from "./pointOfContact"; export * from "./verification"; +export * from "./recruitment"; diff --git a/frontend/mobile/package.json b/frontend/mobile/package.json index 743b29a97..d2ee424e5 100644 --- a/frontend/mobile/package.json +++ b/frontend/mobile/package.json @@ -25,7 +25,7 @@ "@fortawesome/free-solid-svg-icons": "^6.5.2", "@fortawesome/react-fontawesome": "^0.2.2", "@fortawesome/react-native-fontawesome": "^0.3.2", - "@generatesac/lib": "0.0.170", + "@generatesac/lib": "0.0.171", "@gorhom/bottom-sheet": "^4.6.3", "@hookform/resolvers": "^3.4.2", "@react-native-async-storage/async-storage": "^1.23.1", diff --git a/frontend/mobile/src/app/(app)/(tabs)/_layout.tsx b/frontend/mobile/src/app/(app)/(tabs)/_layout.tsx index e7996b90b..7733fc27d 100644 --- a/frontend/mobile/src/app/(app)/(tabs)/_layout.tsx +++ b/frontend/mobile/src/app/(app)/(tabs)/_layout.tsx @@ -35,54 +35,56 @@ const TabBarIcon: React.FC = ({ focused, icon }) => ( const Layout = () => { return ( - - - TabBarLabel({ focused, title: 'Home' }), - tabBarIcon: ({ focused }) => - TabBarIcon({ focused, icon: faHouse }) + <> + - - TabBarLabel({ focused, title: 'Calendar' }), - tabBarIcon: ({ focused }) => - TabBarIcon({ focused, icon: faCalendarDays }) + sceneContainerStyle={{ + backgroundColor: 'white' }} - /> - - TabBarLabel({ focused, title: 'Profile' }), - tabBarIcon: ({ focused }) => - TabBarIcon({ focused, icon: faUser }) - }} - /> - + > + + TabBarLabel({ focused, title: 'Home' }), + tabBarIcon: ({ focused }) => + TabBarIcon({ focused, icon: faHouse }) + }} + /> + + TabBarLabel({ focused, title: 'Calendar' }), + tabBarIcon: ({ focused }) => + TabBarIcon({ focused, icon: faCalendarDays }) + }} + /> + + TabBarLabel({ focused, title: 'Profile' }), + tabBarIcon: ({ focused }) => + TabBarIcon({ focused, icon: faUser }) + }} + /> + + ); }; diff --git a/frontend/mobile/src/app/(app)/(tabs)/index.tsx b/frontend/mobile/src/app/(app)/(tabs)/index.tsx index dfc535ff1..7c612d70e 100644 --- a/frontend/mobile/src/app/(app)/(tabs)/index.tsx +++ b/frontend/mobile/src/app/(app)/(tabs)/index.tsx @@ -1,47 +1,7 @@ import React from 'react'; -import { Pressable, StyleSheet } from 'react-native'; - -import { router } from 'expo-router'; - -import { Box, Text } from '@/src/app/(design-system)'; -import { EventCard } from '@/src/app/(design-system)/components/EventCard'; const HomePage = () => { - const item = { - name: 'Your Event Name', - host: 'Your Club Name', - start_time: new Date(), - end_time: new Date() - }; - - return ( - - Home - router.push(`/event/1`)}> - - - - ); + return <>; }; -const styles = StyleSheet.create({ - container: { - flex: 1, - alignItems: 'center', - justifyContent: 'center' - }, - contentContainer: { - flex: 1, - alignItems: 'center' - } -}); - export default HomePage; diff --git a/frontend/mobile/src/app/(app)/_layout.tsx b/frontend/mobile/src/app/(app)/_layout.tsx index c981213db..c136c2c31 100644 --- a/frontend/mobile/src/app/(app)/_layout.tsx +++ b/frontend/mobile/src/app/(app)/_layout.tsx @@ -17,6 +17,17 @@ const Layout = () => { } }} /> + { + const { id } = useLocalSearchParams<{ id: string }>(); + const { width } = Dimensions.get('window'); + const IMG_HEIGHT = width; + + const scrollRef = useAnimatedRef(); + const scrollOffset = useScrollViewOffset(scrollRef); + + const bottomSheet = useRef(null); + + const club = useAppSelector((state) => state.club); + const { setRetriggerFetch, apiLoading, apiError } = useClub(id as string); + + const headerAnimatedStyle = useAnimatedStyle(() => { + return { + transform: [ + { + translateY: interpolate( + scrollOffset.value, + [-IMG_HEIGHT, 0, IMG_HEIGHT], + [0, 0, -IMG_HEIGHT * 0.75] + ) + } + ] + }; + }); + + return ( + + ( + + + + ) + }} + /> + + {apiLoading ? ( + + ) : apiError ? ( + + ) : ( + <> + + + + + + + + + + {club.name} + + + bottomSheet.current?.snapToIndex(0) + } + type="club" + /> + {club.recruitment?.is_recruiting && ( + + )} + + + Recruiting + + + + + + Upcoming Events + + {club.events.length > 0 ? ( + <> + + + + ) : ( + <> + )} + + + + + )} + + + + ); +}; + +export default ClubPage; diff --git a/frontend/mobile/src/app/(app)/club/_layout.tsx b/frontend/mobile/src/app/(app)/club/_layout.tsx new file mode 100644 index 000000000..026d149b4 --- /dev/null +++ b/frontend/mobile/src/app/(app)/club/_layout.tsx @@ -0,0 +1,13 @@ +import React from 'react'; + +import { Stack } from 'expo-router'; + +const Layout = () => { + return ( + + + + ); +}; + +export default Layout; diff --git a/frontend/mobile/src/app/(app)/club/components/skeleton.tsx b/frontend/mobile/src/app/(app)/club/components/skeleton.tsx new file mode 100644 index 000000000..0e1cf3c9f --- /dev/null +++ b/frontend/mobile/src/app/(app)/club/components/skeleton.tsx @@ -0,0 +1,46 @@ +import { Skeleton } from '@rneui/base'; + +import { Box, Colors } from '@/src/app/(design-system)'; + +const ClubPageSkeleton = () => { + return ( + + + + + + + + + + + + + ); +}; + +export default ClubPageSkeleton; diff --git a/frontend/mobile/src/app/(app)/event/[id].tsx b/frontend/mobile/src/app/(app)/event/[id].tsx index 64cc02bf9..e867f04f6 100644 --- a/frontend/mobile/src/app/(app)/event/[id].tsx +++ b/frontend/mobile/src/app/(app)/event/[id].tsx @@ -1,4 +1,4 @@ -import { useEffect, useRef, useState } from 'react'; +import { useRef } from 'react'; import { Dimensions } from 'react-native'; import Animated, { interpolate, @@ -9,18 +9,20 @@ import Animated, { import { Stack, useLocalSearchParams } from 'expo-router'; -import { EventType, clubApi } from '@generatesac/lib'; -import { eventApi } from '@generatesac/lib'; +import { EventType } from '@generatesac/lib'; import BottomSheet from '@gorhom/bottom-sheet'; import { Arrow, Box, KebabMenu } from '@/src/app/(design-system)'; import { SACColors } from '@/src/app/(design-system)'; import { Button } from '@/src/app/(design-system)/components/Button/Button'; import { description, events, tags } from '@/src/consts/event-page'; +import useEvent from '@/src/hooks/useEvent'; +import { useAppSelector } from '@/src/store/store'; +import AnimatedImageHeader from '../../(design-system)/components/AnimatedImageHeader/AnimatedImageHeader'; +import PageError from '../../(design-system)/components/PageError/PageError'; import { AboutEvent } from './components/about'; import { Description } from './components/description'; -import EventPageError from './components/error'; import { Location } from './components/location'; import { Overview } from './components/overview'; import { RegisterBottomSheet } from './components/register'; @@ -33,7 +35,7 @@ const MockEvent = { clubId: 'generate', // uuid logo: '', eventName: 'Generate Spring Showcase 2024', - color: 'aqua' as SACColors, + color: 'darkRed' as SACColors, club: 'Generate', startTime: new Date('2024-06-01T10:00:00'), endTime: new Date('2024-06-01T12:00:00'), @@ -56,39 +58,12 @@ const EventPage = () => { const scrollRef = useAnimatedRef(); const scrollOffset = useScrollViewOffset(scrollRef); - const [retriggerFetch, setRetriggerFetch] = useState(false); - - const [ - getEvent, - { isLoading: eventLoading, error: eventError, data: event } - ] = eventApi.useLazyEventQuery(); - const [getClub, { isLoading: clubLoading, error: clubError, data: club }] = - clubApi.useLazyClubQuery(); - const [ - getEventTags, - { isLoading: tagsLoading, error: tagsError, data: tags } - ] = eventApi.useLazyEventTagsQuery(); + const event = useAppSelector((state) => state.event); + const { name: clubName, logo: clubLogo } = useAppSelector( + (state) => state.club + ); - const imageAnimatedStyle = useAnimatedStyle(() => { - return { - transform: [ - { - translateY: interpolate( - scrollOffset.value, - [-IMG_HEIGHT, 0, IMG_HEIGHT], - [-IMG_HEIGHT / 2, 0, IMG_HEIGHT * 0.75] - ) - }, - { - scale: interpolate( - scrollOffset.value, - [-IMG_HEIGHT, 0, IMG_HEIGHT], - [2, 1, 1] - ) - } - ] - }; - }); + const { setRetriggerFetch, apiLoading, apiError } = useEvent(id as string); const headerAnimatedStyle = useAnimatedStyle(() => { return { @@ -104,22 +79,6 @@ const EventPage = () => { }; }); - useEffect(() => { - // Fetch events - getEvent(id as string).then(({ data: eventData }) => { - if (eventData) { - // Fetch club - getClub(eventData.host as string); - // Fetch tags: - getEventTags(eventData.id); - } - }); - }, [retriggerFetch, id, getClub, getEvent, getEventTags]); - - const apiLoading = eventLoading || clubLoading || tagsLoading; - const apiError = eventError || clubError || tagsError; - const allData = event && tags && club; - return ( { {apiLoading ? ( ) : apiError ? ( - + ) : ( - allData && ( - <> - - - - - - - - - - + + + + + - bottomSheet.current?.snapToIndex(0) - } + club={clubName} + startTime={event.start_time} + endTime={event.end_time} + location={event.location || 'ISEC'} + type={event.event_type} /> - + register.current?.snapToIndex(0) } - /> + > + Register + - {/* */} + + bottomSheet.current?.snapToIndex(0) + } + /> + - - ) + {/* */} + + )} @@ -233,6 +178,7 @@ const EventPage = () => { ); diff --git a/frontend/mobile/src/app/(app)/event/components/about.tsx b/frontend/mobile/src/app/(app)/event/components/about.tsx index 438eb6aad..442237dd6 100644 --- a/frontend/mobile/src/app/(app)/event/components/about.tsx +++ b/frontend/mobile/src/app/(app)/event/components/about.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import { Linking, TouchableOpacity } from 'react-native'; +import { Linking } from 'react-native'; import { faVideo } from '@fortawesome/free-solid-svg-icons'; import { FontAwesomeIcon } from '@fortawesome/react-native-fontawesome'; @@ -12,9 +12,11 @@ import { Text, textColorVariants } from '@/src/app/(design-system)'; -import { Tag as TagComponent } from '@/src/app/(design-system)'; +import { AboutSection } from '@/src/app/(design-system)/components/AboutSection/AboutSection'; import { Button } from '@/src/app/(design-system)/components/Button/Button'; +import { PageTags } from '../../../(design-system)/components/Tag/Tags'; + interface AboutEventProps { tags: Tag[]; description: string; @@ -36,31 +38,15 @@ export const AboutEvent = ({ } }; - const renderTag = (item: Tag) => ( - - {item.name} - - ); - return ( - - About Event - - - - - {description} - - - Read More... - - - - - {tags.length >= 5 - ? tags.slice(0, 5).map((item) => renderTag(item)) - : tags.map((item) => renderTag(item))} + + + {zoomLink && ( - - - {club.name} - - {tags.map((tag) => ( - - {tag.name} - - ))} - - - About Us - {club.description} - - - - - - Recruiting - - - - Upcoming Events - - - - - Leadership - - - - - ); -}; diff --git a/frontend/mobile/src/app/(design-system)/components/ClubPage/ClubPageHeader.tsx b/frontend/mobile/src/app/(design-system)/components/ClubPage/ClubPageHeader.tsx deleted file mode 100644 index ba81006b2..000000000 --- a/frontend/mobile/src/app/(design-system)/components/ClubPage/ClubPageHeader.tsx +++ /dev/null @@ -1,23 +0,0 @@ -import React from 'react'; -import Animated from 'react-native-reanimated'; - -import { Box } from '../Box/Box'; - -interface AnimatedImageBoxProps { - uri: string; - animatedStyle: any; -} - -export const AnimatedImageBox: React.FC = ({ - uri, - animatedStyle -}) => { - return ( - - - - ); -}; diff --git a/frontend/mobile/src/app/(design-system)/components/ClubRecruitment/RecruitmentInfo/ClubRecruitmentInfo.tsx b/frontend/mobile/src/app/(design-system)/components/ClubRecruitment/RecruitmentInfo/ClubRecruitmentInfo.tsx index 32d191b15..bfe2313f7 100644 --- a/frontend/mobile/src/app/(design-system)/components/ClubRecruitment/RecruitmentInfo/ClubRecruitmentInfo.tsx +++ b/frontend/mobile/src/app/(design-system)/components/ClubRecruitment/RecruitmentInfo/ClubRecruitmentInfo.tsx @@ -15,11 +15,9 @@ interface RecruitmentInfoProps { recruitmentCycle: RecruitmentCycle; recruitingType: RecruitmentType; isRecruiting?: boolean; - recruitmentText: string; } export const RecruitmentInfo = ({ - recruitmentText, color, recruitmentCycle, recruitingType, @@ -35,12 +33,8 @@ export const RecruitmentInfo = ({ /> = ({ events }) => { +export const EventCardList: React.FC = ({ + events, + club +}) => { const renderEventCard = ({ item }: { item: Event }) => { return ( router.push(`/event/${item.id}`)}> diff --git a/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardBig.tsx b/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardBig.tsx index c5bedf986..1edd303d7 100644 --- a/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardBig.tsx +++ b/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardBig.tsx @@ -7,7 +7,7 @@ import { router } from 'expo-router'; import { Image } from '@rneui/base'; import { Box, Text } from '@/src/app/(design-system)'; -import { calculateDuration, createOptions, eventTime } from '@/src/utils/time'; +import { createOptions, eventTime } from '@/src/utils/time'; interface EventCardBigProps { event: string; @@ -46,8 +46,6 @@ export const EventCardBig: React.FC = ({ width="100%" > {event} - - {calculateDuration(startTime, endTime)} {club} diff --git a/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardCalendar.tsx b/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardCalendar.tsx index 0a6d769be..32047c78b 100644 --- a/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardCalendar.tsx +++ b/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardCalendar.tsx @@ -1,12 +1,15 @@ import React from 'react'; import { TouchableOpacity } from 'react-native'; -import { router } from 'expo-router'; - import { Tag } from '@generatesac/lib'; import { Avatar, Image } from '@rneui/base'; import { Box, Text, createStyles } from '@/src/app/(design-system)'; +import { + setEventId, + setEventShouldPreview +} from '@/src/store/slices/eventSlice'; +import { useAppDispatch } from '@/src/store/store'; import { createOptions, eventTime, happeningNow } from '@/src/utils/time'; import { EventTags } from '../EventCardTags/EventCardTags'; @@ -35,11 +38,15 @@ export const EventCardCalendar: React.FC = ({ }) => { const isHappening = happeningNow(startTime, endTime); const isPast = new Date(endTime).getTime() < new Date().getTime(); + const dispatch = useAppDispatch(); return ( router.navigate(`/event/${eventId}`)} + onPress={() => { + dispatch(setEventId(eventId)); + dispatch(setEventShouldPreview(true)); + }} > {isHappening && ( @@ -63,8 +70,8 @@ export const EventCardCalendar: React.FC = ({ /> - {event.length >= 28 - ? `${event.slice(0, 28).trim()}...` + {event.length >= 26 + ? `${event.slice(0, 26).trim()}...` : event} diff --git a/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardSmall.tsx b/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardSmall.tsx index 392cef2fb..740e71105 100644 --- a/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardSmall.tsx +++ b/frontend/mobile/src/app/(design-system)/components/EventCard/Variants/EventCardSmall.tsx @@ -27,7 +27,7 @@ export const EventCardSmall: React.FC = ({ image }) => { const screenWidth = Dimensions.get('window').width; - const boxWidth = screenWidth * 0.4; + const boxWidth = screenWidth * 0.41; return ( diff --git a/frontend/mobile/src/app/(app)/event/components/error.tsx b/frontend/mobile/src/app/(design-system)/components/PageError/PageError.tsx similarity index 90% rename from frontend/mobile/src/app/(app)/event/components/error.tsx rename to frontend/mobile/src/app/(design-system)/components/PageError/PageError.tsx index 9fb570dc1..5fcb1ef28 100644 --- a/frontend/mobile/src/app/(app)/event/components/error.tsx +++ b/frontend/mobile/src/app/(design-system)/components/PageError/PageError.tsx @@ -8,7 +8,7 @@ type EventPageErrorProps = { refetch: React.Dispatch>; }; -const EventPageError = ({ refetch }: EventPageErrorProps) => { +const PageError = ({ refetch }: EventPageErrorProps) => { return ( @@ -30,4 +30,4 @@ const EventPageError = ({ refetch }: EventPageErrorProps) => { ); }; -export default EventPageError; +export default PageError; diff --git a/frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreview.tsx b/frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreview.tsx new file mode 100644 index 000000000..b9cfc6a5e --- /dev/null +++ b/frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreview.tsx @@ -0,0 +1,125 @@ +import React, { forwardRef, useCallback, useState } from 'react'; + +import { router } from 'expo-router'; + +import BottomSheet, { BottomSheetBackdrop } from '@gorhom/bottom-sheet'; +import { SerializedError } from '@reduxjs/toolkit'; +import { FetchBaseQueryError } from '@reduxjs/toolkit/query'; +import { Avatar } from '@rneui/base'; + +import { setClubShouldPreview } from '@/src/store/slices/clubSlice'; +import { useAppDispatch, useAppSelector } from '@/src/store/store'; + +import { Box } from '../../Box/Box'; +import { Button } from '../../Button/Button'; +import { Text } from '../../Text/Text'; +import PreviewError from '../PreviewError'; +import ClubPreviewSkeleton from './ClubPreviewSkeleton'; + +interface ClubPreviewProps { + clubId: string; + isLoading: boolean; + error: FetchBaseQueryError | SerializedError | undefined; +} + +type Ref = BottomSheet; +const LOGO_HEIGHT = 77; + +export const ClubPreview = forwardRef( + ({ clubId, isLoading, error }, ref) => { + const club = useAppSelector((state) => state.club); + const dispatch = useAppDispatch(); + + const renderBackdrop = useCallback( + (props: any) => ( + + ), + [] + ); + + const ClubLogo = () => { + return () => ( + + + + ); + }; + + const NoLogo = () => { + return () => <>; + }; + + const [logo, setLogo] = useState(NoLogo); + + return ( + { + toIndex === 0 && !error + ? setLogo(ClubLogo) + : setLogo(NoLogo); + }} + backgroundStyle={{ backgroundColor: 'white' }} + backdropComponent={renderBackdrop} + onClose={() => dispatch(setClubShouldPreview(false))} + > + + + {isLoading || club.id === '' ? ( + + ) : error ? ( + + ) : ( + <> + + {club.name} + + + Who are we? + {`${club.preview}...`} + + + + )} + + + ); + } +); diff --git a/frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreviewSkeleton.tsx b/frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreviewSkeleton.tsx new file mode 100644 index 000000000..d406be081 --- /dev/null +++ b/frontend/mobile/src/app/(design-system)/components/Preview/Club/ClubPreviewSkeleton.tsx @@ -0,0 +1,38 @@ +import { Skeleton } from '@rneui/base'; + +import { Colors } from '../../../shared/colors'; +import { Box } from '../../Box/Box'; + +const ClubPreviewSkeleton = () => { + return ( + + + + + + + + + + + + + + + ); +}; + +export default ClubPreviewSkeleton; diff --git a/frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreview.tsx b/frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreview.tsx new file mode 100644 index 000000000..648efba7a --- /dev/null +++ b/frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreview.tsx @@ -0,0 +1,125 @@ +import React, { forwardRef, useCallback } from 'react'; +import { TouchableOpacity } from 'react-native-gesture-handler'; + +import { router } from 'expo-router'; + +import BottomSheet, { BottomSheetBackdrop } from '@gorhom/bottom-sheet'; +import { SerializedError } from '@reduxjs/toolkit'; +import { FetchBaseQueryError } from '@reduxjs/toolkit/query'; +import { Avatar } from '@rneui/base'; + +import { setEventShouldPreview } from '@/src/store/slices/eventSlice'; +import { useAppDispatch, useAppSelector } from '@/src/store/store'; + +import { Box } from '../../Box/Box'; +import { Button } from '../../Button/Button'; +import { Text } from '../../Text/Text'; +import PreviewError from '../PreviewError'; +import EventPreviewSkeleton from './EventPreviewSkeleton'; + +interface EventPreviewProps { + eventId: string; + isLoading: boolean; + error: FetchBaseQueryError | SerializedError | undefined; +} + +type Ref = BottomSheet; + +export const EventPreview = forwardRef( + ({ eventId, isLoading, error }, ref) => { + const event = useAppSelector((state) => state.event); + const { name: clubName, logo: clubLogo } = useAppSelector( + (state) => state.club + ); + + const dispatch = useAppDispatch(); + + const renderBackdrop = useCallback( + (props: any) => ( + + ), + [] + ); + + return ( + dispatch(setEventShouldPreview(false))} + > + + {isLoading || event.id === '' ? ( + + ) : error ? ( + + ) : ( + <> + + {`${event.name.slice(0, 32)}${event.name.length > 32 ? '...' : ''}`} + + + { + dispatch(setEventShouldPreview(false)); + router.push(`/club/${event.host}`); + }} + > + + + Hosted by {clubName} + + + + + + + About Event + {`${event.description.slice(0, 220).trim()}${event.description.length > 220 ? '...' : ''}`} + + + + )} + + + ); + } +); diff --git a/frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreviewSkeleton.tsx b/frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreviewSkeleton.tsx new file mode 100644 index 000000000..e9a365191 --- /dev/null +++ b/frontend/mobile/src/app/(design-system)/components/Preview/Event/EventPreviewSkeleton.tsx @@ -0,0 +1,38 @@ +import { Skeleton } from '@rneui/base'; + +import { Colors } from '../../../shared/colors'; +import { Box } from '../../Box/Box'; + +const EventPreviewSkeleton = () => { + return ( + + + + + + + + + + + + + + + ); +}; + +export default EventPreviewSkeleton; diff --git a/frontend/mobile/src/app/(design-system)/components/Preview/PreviewError.tsx b/frontend/mobile/src/app/(design-system)/components/Preview/PreviewError.tsx new file mode 100644 index 000000000..635442ec0 --- /dev/null +++ b/frontend/mobile/src/app/(design-system)/components/Preview/PreviewError.tsx @@ -0,0 +1,14 @@ +import { Box } from '../Box/Box'; +import { Text } from '../Text/Text'; + +const PreviewError = () => { + return ( + + + Something went wrong, please try again later. + + + ); +}; + +export default PreviewError; diff --git a/frontend/mobile/src/app/(design-system)/components/Tag/Tags.tsx b/frontend/mobile/src/app/(design-system)/components/Tag/Tags.tsx new file mode 100644 index 000000000..67e21f0ab --- /dev/null +++ b/frontend/mobile/src/app/(design-system)/components/Tag/Tags.tsx @@ -0,0 +1,27 @@ +import { Tag } from '@generatesac/lib'; + +import { SACColors } from '../../shared/colors'; +import { Box } from '../Box/Box'; +import { Text } from '../Text/Text'; +import { Tag as TagComponent } from './Tag'; + +interface PageTags { + tags: Tag[]; + color: SACColors; +} + +export const PageTags: React.FC = ({ tags, color }) => { + const renderTag = (item: Tag) => ( + + {item.name} + + ); + + return ( + + {tags.length >= 5 + ? tags.slice(0, 5).map((item) => renderTag(item)) + : tags.map((item) => renderTag(item))} + + ); +}; diff --git a/frontend/mobile/src/app/(design-system)/index.ts b/frontend/mobile/src/app/(design-system)/index.ts index c45d74daa..ed1baac5d 100644 --- a/frontend/mobile/src/app/(design-system)/index.ts +++ b/frontend/mobile/src/app/(design-system)/index.ts @@ -12,3 +12,6 @@ export * from './components/Dropdown/SelectOne'; export * from './components/Dropdown/Multiselect'; export * from './components/PointOfContactCard/PointOfContactCard'; export * from './components/Tag/Tag'; +export * from './components/Preview/Event/EventPreview'; +export * from './components/Preview/Club/ClubPreview'; +export * from './components/Tag/Tags'; diff --git a/frontend/mobile/src/app/_layout.tsx b/frontend/mobile/src/app/_layout.tsx index abcba3931..f35b22cc3 100644 --- a/frontend/mobile/src/app/_layout.tsx +++ b/frontend/mobile/src/app/_layout.tsx @@ -10,9 +10,10 @@ import { StatusBar } from 'expo-status-bar'; import FontAwesome from '@expo/vector-icons/FontAwesome'; import { ThemeProvider } from '@shopify/restyle'; +import usePreview from '../hooks/usePreview'; import StoreProvider from '../store/StoreProvider'; import { useAppSelector } from '../store/store'; -import { theme } from './(design-system)'; +import { ClubPreview, EventPreview, theme } from './(design-system)'; export { ErrorBoundary } from 'expo-router'; @@ -20,6 +21,16 @@ SplashScreen.preventAutoHideAsync(); const InitalLayout = () => { const { accessToken } = useAppSelector((state) => state.user); + const { + eventPreviewRef, + eventId, + eventApiError, + eventApiLoading, + clubPreviewRef, + clubId, + clubApiError, + clubApiLoading + } = usePreview(); useEffect(() => { if (!accessToken) { @@ -36,6 +47,18 @@ const InitalLayout = () => { + + ); }; diff --git a/frontend/mobile/src/hooks/useClub.ts b/frontend/mobile/src/hooks/useClub.ts new file mode 100644 index 000000000..3f1726069 --- /dev/null +++ b/frontend/mobile/src/hooks/useClub.ts @@ -0,0 +1,75 @@ +import { useEffect, useState } from 'react'; + +import { clubApi } from '@generatesac/lib'; + +import { setClub, setClubEvents, setClubTags } from '../store/slices/clubSlice'; +import { useAppDispatch, useAppSelector } from '../store/store'; + +const useClub = (id: string) => { + const [retriggerFetch, setRetriggerFetch] = useState(false); + + const [getClub, { isLoading: clubLoading, error: clubError }] = + clubApi.useLazyClubQuery(); + const [getClubTags, { isLoading: tagsLoading, error: tagsError }] = + clubApi.useLazyClubTagsQuery(); + const [getEvents, { isLoading: eventsLoading, error: eventsError }] = + clubApi.useLazyClubEventsQuery(); + + const dispatch = useAppDispatch(); + const { shouldPreview, id: clubId } = useAppSelector((state) => state.club); + + const apiLoading = clubLoading || tagsLoading || eventsLoading; + const apiError = clubError || tagsError || eventsError; + + useEffect(() => { + if ((id !== '' && (id !== clubId || shouldPreview)) || apiError) { + getClub(id).then(({ data: clubData }) => { + if (clubData) { + dispatch(setClub(clubData)); + } + }); + + getClubTags(id).then(({ data: tagData }) => { + if (tagData) { + dispatch(setClubTags(tagData)); + } + }); + + getEvents({ id }).then(({ data: eventData }) => { + if (eventData) { + const sortedEvents = [...eventData]; + sortedEvents.sort((a, b) => { + return ( + new Date(a.start_time).getTime() - + new Date(b.start_time).getTime() + ); + }); + const events = sortedEvents.filter( + (event) => + new Date(event.end_time).getTime() > + new Date().getTime() + ); + dispatch(setClubEvents(events)); + } + }); + } + }, [ + retriggerFetch, + id, + getClub, + getClubTags, + getEvents, + dispatch, + clubId, + apiError, + shouldPreview + ]); + + return { + setRetriggerFetch, + apiLoading, + apiError + }; +}; + +export default useClub; diff --git a/frontend/mobile/src/hooks/useEvent.ts b/frontend/mobile/src/hooks/useEvent.ts new file mode 100644 index 000000000..6cb2778a2 --- /dev/null +++ b/frontend/mobile/src/hooks/useEvent.ts @@ -0,0 +1,70 @@ +import { useEffect, useState } from 'react'; + +import { clubApi, eventApi } from '@generatesac/lib'; + +import { setClub } from '../store/slices/clubSlice'; +import { setEvent, setEventTags } from '../store/slices/eventSlice'; +import { useAppDispatch, useAppSelector } from '../store/store'; + +const useEvent = (id: string) => { + const [retriggerFetch, setRetriggerFetch] = useState(false); + + const [getEvent, { isLoading: eventLoading, error: eventError }] = + eventApi.useLazyEventQuery(); + const [getClub, { isLoading: clubLoading, error: clubError }] = + clubApi.useLazyClubQuery(); + const [getEventTags, { isLoading: tagsLoading, error: tagsError }] = + eventApi.useLazyEventTagsQuery(); + + const dispatch = useAppDispatch(); + const { shouldPreview, id: eventId } = useAppSelector( + (state) => state.event + ); + + const apiLoading = eventLoading || clubLoading || tagsLoading; + const apiError = eventError || clubError || tagsError; + + useEffect(() => { + if ((id !== '' && (id !== eventId || shouldPreview)) || apiError) { + console.log('fetching data!'); + // Fetch events + getEvent(id).then(({ data: eventData }) => { + dispatch(setEvent(eventData)); + if (eventData) { + // Fetch club + getClub(eventData.host as string).then( + ({ data: clubData }) => { + if (clubData) { + dispatch(setClub(clubData)); + } + } + ); + } + }); + + getEventTags(id).then(({ data: tagData }) => { + if (tagData) { + dispatch(setEventTags(tagData)); + } + }); + } + }, [ + retriggerFetch, + id, + getClub, + getEvent, + getEventTags, + dispatch, + eventId, + shouldPreview, + apiError + ]); + + return { + setRetriggerFetch, + apiLoading, + apiError + }; +}; + +export default useEvent; diff --git a/frontend/mobile/src/hooks/usePreview.ts b/frontend/mobile/src/hooks/usePreview.ts new file mode 100644 index 000000000..8cba55d54 --- /dev/null +++ b/frontend/mobile/src/hooks/usePreview.ts @@ -0,0 +1,56 @@ +import { useEffect, useRef } from 'react'; + +import BottomSheet from '@gorhom/bottom-sheet'; + +import { setClubShouldPreview } from '../store/slices/clubSlice'; +import { setEventShouldPreview } from '../store/slices/eventSlice'; +import { useAppDispatch, useAppSelector } from '../store/store'; +import useClub from './useClub'; +import useEvent from './useEvent'; + +const usePreview = () => { + const dispatch = useAppDispatch(); + const eventPreviewRef = useRef(null); + const clubPreviewRef = useRef(null); + const { shouldPreview: eventShouldPreview, id: eventId } = useAppSelector( + (state) => state.event + ); + const { shouldPreview: clubShouldPreview, id: clubId } = useAppSelector( + (state) => state.club + ); + const { apiLoading: eventApiLoading, apiError: eventApiError } = + useEvent(eventId); + const { apiLoading: clubApiLoading, apiError: clubApiError } = + useClub(clubId); + + useEffect(() => { + dispatch(setEventShouldPreview(false)); + dispatch(setClubShouldPreview(false)); + }, [dispatch]); + + useEffect(() => { + if (eventShouldPreview) { + eventPreviewRef.current?.snapToIndex(0); + } else { + eventPreviewRef.current?.close(); + } + if (clubShouldPreview) { + clubPreviewRef.current?.snapToIndex(0); + } else { + clubPreviewRef.current?.close(); + } + }, [eventShouldPreview, clubShouldPreview]); + + return { + eventPreviewRef, + clubPreviewRef, + eventId, + clubId, + eventApiLoading, + eventApiError, + clubApiLoading, + clubApiError + }; +}; + +export default usePreview; diff --git a/frontend/mobile/src/store/slices/clubSlice.ts b/frontend/mobile/src/store/slices/clubSlice.ts new file mode 100644 index 000000000..3f44c9d12 --- /dev/null +++ b/frontend/mobile/src/store/slices/clubSlice.ts @@ -0,0 +1,68 @@ +import { Club, Event, Tag } from '@generatesac/lib'; +import { createSlice } from '@reduxjs/toolkit'; + +type ClubState = { + shouldPreview?: boolean; + tags: Tag[]; + events: Event[]; +}; + +const initialState: Club & ClubState = { + id: '', + created_at: '', + updated_at: '', + name: '', + description: '', + preview: '', + num_members: 0, + logo: '', + weekly_time_committment: 0, + one_word_to_describe_us: '', + tags: [], + events: [] +}; + +export const clubSlice = createSlice({ + name: 'club', + initialState, + reducers: { + setClubId: (state, action) => { + state.id = action.payload; + }, + setClub: (state, action) => { + state.id = action.payload.id; + state.created_at = action.payload.created_at; + state.updated_at = action.payload.updated_at; + state.name = action.payload.name; + state.description = action.payload.description; + state.preview = action.payload.preview; + state.num_members = action.payload.num_members; + state.logo = action.payload.logo; + state.weekly_time_committment = + action.payload.weekly_time_committment; + state.one_word_to_describe_us = + action.payload.one_word_to_describe_us; + state.recruitment = action.payload.recruitment; + }, + setClubTags: (state, action) => { + state.tags = action.payload; + }, + setClubEvents: (state, action) => { + state.events = action.payload; + }, + resetClub: () => initialState, + setClubShouldPreview: (state, action) => { + state.shouldPreview = action.payload; + } + } +}); + +export const { + setClubId, + setClub, + setClubTags, + setClubEvents, + resetClub, + setClubShouldPreview +} = clubSlice.actions; +export default clubSlice.reducer; diff --git a/frontend/mobile/src/store/slices/eventSlice.ts b/frontend/mobile/src/store/slices/eventSlice.ts new file mode 100644 index 000000000..cf43da36a --- /dev/null +++ b/frontend/mobile/src/store/slices/eventSlice.ts @@ -0,0 +1,69 @@ +import { Event, Tag } from '@generatesac/lib'; +import { createSlice } from '@reduxjs/toolkit'; + +type EventClubState = { + shouldPreview?: boolean; + tags: Tag[]; +}; + +const initialState: Event & EventClubState = { + id: '', + created_at: '', + updated_at: '', + name: '', + description: '', + start_time: '', + end_time: '', + location: '', + event_type: 'hybrid', + is_archived: false, + is_draft: false, + is_recurring: false, + is_public: false, + preview: '', + host: '', + tags: [] +}; + +export const eventSlice = createSlice({ + name: 'event', + initialState, + reducers: { + setEventId: (state, action) => { + state.id = action.payload; + }, + setEvent: (state, action) => { + state.id = action.payload.id; + state.created_at = action.payload.created_at; + state.updated_at = action.payload.updated_at; + state.name = action.payload.name; + state.description = action.payload.description; + state.start_time = action.payload.start_time; + state.end_time = action.payload.end_time; + state.location = action.payload.location; + state.event_type = action.payload.event_type; + state.is_archived = action.payload.is_archived; + state.is_draft = action.payload.is_draft; + state.is_recurring = action.payload.is_recurring; + state.is_public = action.payload.is_public; + state.preview = action.payload.preview; + state.host = action.payload.host; + }, + setEventTags: (state, action) => { + state.tags = action.payload; + }, + resetEvent: () => initialState, + setEventShouldPreview: (state, action) => { + state.shouldPreview = action.payload; + } + } +}); + +export const { + setEvent, + resetEvent, + setEventShouldPreview, + setEventTags, + setEventId +} = eventSlice.actions; +export default eventSlice.reducer; diff --git a/frontend/mobile/src/store/store.ts b/frontend/mobile/src/store/store.ts index c505d2ff9..21e502edc 100644 --- a/frontend/mobile/src/store/store.ts +++ b/frontend/mobile/src/store/store.ts @@ -16,10 +16,14 @@ import persistReducer from 'redux-persist/es/persistReducer'; import autoMergeLevel2 from 'redux-persist/es/stateReconciler/autoMergeLevel2'; import { thunk } from 'redux-thunk'; +import clubReducer from '@/src/store/slices/clubSlice'; +import eventReducer from '@/src/store/slices/eventSlice'; import userReducer from '@/src/store/slices/userSlice'; const rootReducer = combineReducers({ user: userReducer, + event: eventReducer, + club: clubReducer, [baseApi.reducerPath]: baseApi.reducer }); @@ -30,7 +34,7 @@ const persistanceConfig = { storage: AsyncStorage, stateReconciler: autoMergeLevel2, blacklist: [baseApi.reducerPath], - whitelist: ['user'] + whitelist: ['user', 'event', 'club'] }; const persistedReducer = persistReducer( diff --git a/frontend/mobile/yarn.lock b/frontend/mobile/yarn.lock index 26a578108..07999f6a7 100644 --- a/frontend/mobile/yarn.lock +++ b/frontend/mobile/yarn.lock @@ -1402,10 +1402,10 @@ humps "^2.0.1" prop-types "^15.7.2" -"@generatesac/lib@0.0.170": - version "0.0.170" - resolved "https://registry.yarnpkg.com/@generatesac/lib/-/lib-0.0.170.tgz#2401ac4216d43dc555f142633aba16d1141ca356" - integrity sha512-oPkqZTudQd7tGEQUpAXQrhD/cngIUV595q069d7SMQZ3zZleQqGXirvk1TOjdoEzNug1gN21yS238stq6CL4Nw== +"@generatesac/lib@0.0.171": + version "0.0.171" + resolved "https://registry.yarnpkg.com/@generatesac/lib/-/lib-0.0.171.tgz#e5ea2c800dc319e4938616e0033de6d1b2fb3ee9" + integrity sha512-/9RN79Oa5b8+bJsWaMD+8A+fyyaVdzc1KVt+9QVnHog8OaQivx6fAFu+lV2X/j25+KbKsEd/rCSWX5u6DfMe3Q== dependencies: "@reduxjs/toolkit" "^2.2.3" react "^18.2.0"