From b746fbf7d2b59b4b0b91a7a5d896c5e8d8d37b5d Mon Sep 17 00:00:00 2001 From: emkartal1 Date: Mon, 25 Dec 2023 16:28:37 +0100 Subject: [PATCH] Update chat screen and add a StubMessageService :white_check_mark: --- README.md | 2 +- src/Api/src/controllers/._userController.ts | Bin 0 -> 4096 bytes src/Api/src/controllers/userController.ts | 2 +- src/Api/src/database/UserSchema.ts | 2 +- src/FLAD/components/FriendComponent.tsx | 34 +++- src/FLAD/components/PaginatorComponent.tsx | 2 +- src/FLAD/components/SimilarMusicComponent.tsx | 2 +- src/FLAD/models/Conversation.ts | 47 +++++ src/FLAD/models/Message.ts | 55 ++++++ src/FLAD/navigation/HomeNavigation.tsx | 11 +- src/FLAD/navigation/MessagingNavigation.tsx | 52 ++++- src/FLAD/package.json | 5 +- src/FLAD/redux/actions/appActions.ts | 17 ++ src/FLAD/redux/reducers/appReducer.tsx | 9 + src/FLAD/redux/thunk/appThunk.tsx | 1 + src/FLAD/redux/thunk/chatThunk.tsx | 42 ++++ src/FLAD/redux/types/chatTypes.tsx | 4 + src/FLAD/screens/ChatScreen.tsx | 186 ++++++++++++++++-- src/FLAD/screens/ConversationScreen.tsx | 81 ++++++-- src/FLAD/screens/FavoriteScreen.tsx | 2 +- src/FLAD/screens/SpotScreen.tsx | 6 +- .../messages/interfaces/IMessageService.ts | 8 + .../messages/stub/StubMessageService.ts | 57 ++++++ .../services/musics/spotify/SpotifyService.ts | 4 +- 24 files changed, 577 insertions(+), 54 deletions(-) create mode 100644 src/Api/src/controllers/._userController.ts create mode 100644 src/FLAD/models/Conversation.ts create mode 100644 src/FLAD/models/Message.ts create mode 100644 src/FLAD/redux/thunk/chatThunk.tsx create mode 100644 src/FLAD/redux/types/chatTypes.tsx create mode 100644 src/FLAD/services/messages/interfaces/IMessageService.ts create mode 100644 src/FLAD/services/messages/stub/StubMessageService.ts diff --git a/README.md b/README.md index c8e7d0e..b97f7b7 100644 --- a/README.md +++ b/README.md @@ -116,7 +116,7 @@ Une fois sur la page, saisissez votre nom, votre adresse e-mail, et votre mot de Pour accéder aux détails d'une musique, maintenez votre doigt appuyé sur un Spot ou rendez-vous sur la page des favoris. Vous pourrez écouter la musique :arrow_forward:, obtenir des informations sur l'artiste et la chanson, découvrir des musiques similaires, et même l'ajouter à votre playlist Spotify ou la partager.
-`Dans la page **settings** ⚙️, vous avez accès à toutes vos informations ```Spotify```, que vous pouvez modifier à votre guise. Toutefois, ces modifications ne seront prises en compte que dans notre application. Vous pouvez également choisir le mode sombre (dark mode) dans les paramètres pour une expérience de navigation plus confortable. +Dans la page **settings** ⚙️, vous avez accès à toutes vos informations ```Spotify```, que vous pouvez modifier à votre guise. Toutefois, ces modifications ne seront prises en compte que dans notre application. Vous pouvez également choisir le mode sombre (dark mode) dans les paramètres pour une expérience de navigation plus confortable.
### Voici un petit récapitulatif diff --git a/src/Api/src/controllers/._userController.ts b/src/Api/src/controllers/._userController.ts new file mode 100644 index 0000000000000000000000000000000000000000..88aa1e667760fc6853f61c2229eecd9eef004f5b GIT binary patch literal 4096 zcmZQz6=P>$Vqox1Ojhs@R)|o50+1L3ClDJkFz{^v(m+1nBL)UWIUt(=a103v`v3o@ zC`1Rt0-$mMG%bukK2%&PIX_n~v7jI)RWB#8xTLf=H6Kf7%s{i3$kztVg G{~rJtx-9wt literal 0 HcmV?d00001 diff --git a/src/Api/src/controllers/userController.ts b/src/Api/src/controllers/userController.ts index 92956fe..7e3866c 100644 --- a/src/Api/src/controllers/userController.ts +++ b/src/Api/src/controllers/userController.ts @@ -1,4 +1,4 @@ -import { Router, Request, Response, NextFunction } from 'express'; +import { Router, Request, Response } from 'express'; import IController from './interfaces/IController'; import User from '../models/User'; import UserService from '../services/UserService'; diff --git a/src/Api/src/database/UserSchema.ts b/src/Api/src/database/UserSchema.ts index 51ae320..1d90764 100644 --- a/src/Api/src/database/UserSchema.ts +++ b/src/Api/src/database/UserSchema.ts @@ -38,7 +38,7 @@ const userSchema = new Schema({ userId: String, date: Date }], - default: [] + default: [] } }, { timestamps: true } diff --git a/src/FLAD/components/FriendComponent.tsx b/src/FLAD/components/FriendComponent.tsx index ec0d73c..5126844 100644 --- a/src/FLAD/components/FriendComponent.tsx +++ b/src/FLAD/components/FriendComponent.tsx @@ -4,11 +4,12 @@ import { useSelector } from 'react-redux'; import { colorsDark } from '../constants/colorsDark'; import { colorsLight } from '../constants/colorsLight'; import normalize from './Normalize'; +import Message from '../models/Message'; type FriendProps = { image: string; name: string; - lastMessage: string; + lastMessage: Message; } export default function Friend(props: FriendProps) { @@ -67,14 +68,41 @@ export default function Friend(props: FriendProps) { } }) + const getTimeDifferenceString = (date: Date): string => { + const now = new Date(); + const differenceInSeconds = Math.floor((now.getTime() - date.getTime()) / 1000); + + const intervals = { + an: 31536000, + mois: 2592000, + sem: 604800, + jour: 86400, + heure: 3600, + min: 60, + }; + + for (const [intervalName, seconds] of Object.entries(intervals)) { + const intervalCount = Math.floor(differenceInSeconds / seconds); + if (intervalCount > 0) { + if (intervalName === 'mois' || intervalName === 'min') { + return `il y a ${intervalCount} ${intervalName}`; + } else { + return `il y a ${intervalCount} ${intervalName}${intervalCount !== 1 ? 's' : ''}`; + } + } + } + + return 'À l’instant'; + }; + return ( {props.name} - {props.lastMessage} - · 1sem + {props.lastMessage.content} + · {getTimeDifferenceString(props.lastMessage.date)} diff --git a/src/FLAD/components/PaginatorComponent.tsx b/src/FLAD/components/PaginatorComponent.tsx index 0f3c830..66bd7f0 100644 --- a/src/FLAD/components/PaginatorComponent.tsx +++ b/src/FLAD/components/PaginatorComponent.tsx @@ -8,7 +8,7 @@ export default function Paginator({ data, scrollX }) { return ( - {data.map((_ : any, i : any) => { + { data.map((_ : any, i : any) => { const inputRange = [(i - 1) * width, i * width, (i + 1) * width]; const dotWidth = scrollX.interpolate({ diff --git a/src/FLAD/components/SimilarMusicComponent.tsx b/src/FLAD/components/SimilarMusicComponent.tsx index a0da245..dd42465 100644 --- a/src/FLAD/components/SimilarMusicComponent.tsx +++ b/src/FLAD/components/SimilarMusicComponent.tsx @@ -3,7 +3,7 @@ import Music from '../models/Music'; import normalize from './Normalize'; export interface RenderCellProps { - music : Music; + music: Music; } export const SimilarMusic = (props: RenderCellProps) => { return ( diff --git a/src/FLAD/models/Conversation.ts b/src/FLAD/models/Conversation.ts new file mode 100644 index 0000000..a972af9 --- /dev/null +++ b/src/FLAD/models/Conversation.ts @@ -0,0 +1,47 @@ +import Message from "./Message"; + +export default class Conversation { + private _id: string; + private _name: string; + private _image: string; + private _lastMessage: Message; + + constructor(id: string, name: string, image: string, lastMessage: Message) { + this._id = id; + this._name = name; + this._image = image; + this._lastMessage = lastMessage; + } + + get id(): string { + return this._id; + } + + set id(value: string) { + this._id = value; + } + + get name(): string { + return this._name; + } + + set name(value: string) { + this._name = value; + } + + get image(): string { + return this._image; + } + + set image(value: string) { + this._image = value; + } + + get lastMessage(): Message { + return this._lastMessage; + } + + set lastMessage(value: Message) { + this._lastMessage = value; + } +} \ No newline at end of file diff --git a/src/FLAD/models/Message.ts b/src/FLAD/models/Message.ts new file mode 100644 index 0000000..4e8b6e5 --- /dev/null +++ b/src/FLAD/models/Message.ts @@ -0,0 +1,55 @@ +export default class Message { + private _id: string; + private _content: string; + private _sender: string; + private _date: Date; + private _audio: string; + + constructor(id: string, content: string, sender: string, date: Date, audio: string = '') { + this._id = id; + this._content = content; + this._sender = sender; + this._date = date; + this._audio = audio; + } + + get id(): string { + return this._id; + } + + set id(value: string) { + this._id = value; + } + + get content(): string { + return this._content; + } + + set content(value: string) { + this._content = value; + } + + get sender(): string { + return this._sender; + } + + set sender(value: string) { + this._sender = value; + } + + get date(): Date { + return this._date; + } + + set date(value: Date) { + this._date = value; + } + + get audio(): string { + return this._audio; + } + + set audio(value: string) { + this._audio = value; + } +} \ No newline at end of file diff --git a/src/FLAD/navigation/HomeNavigation.tsx b/src/FLAD/navigation/HomeNavigation.tsx index 8094b39..7765a1d 100644 --- a/src/FLAD/navigation/HomeNavigation.tsx +++ b/src/FLAD/navigation/HomeNavigation.tsx @@ -2,7 +2,7 @@ import React, { useEffect, useState } from 'react'; import { View, Alert, Platform } from 'react-native'; import { faUser, faEnvelope, faHeart, faMusic } from "@fortawesome/free-solid-svg-icons" import { createBottomTabNavigator } from '@react-navigation/bottom-tabs'; -import { NavigationContainer } from '@react-navigation/native'; +import { NavigationContainer, getFocusedRouteNameFromRoute } from '@react-navigation/native'; import FavoriteNavigation from './FavoriteNavigation'; import SettingNavigation from './SettingNavigation'; import { FontAwesomeIcon } from "@fortawesome/react-native-fontawesome"; @@ -156,10 +156,15 @@ export default function HomeNavigation() { tabBarIcon: ({ color }) => , }} /> ({ headerShown: false, + tabBarStyle: { + display: getFocusedRouteNameFromRoute(route) !== "Chat" ? "flex" : "none", + position: "absolute", + borderTopColor: isDark ? 'rgba(255, 255, 255, 0.25)' : 'rgba(50, 50, 50, 0.07)', + }, tabBarIcon: ({ color }) => , - }} /> + })} /> state.userReducer.dark); + const style = isDark ? colorsDark : colorsLight; const Stack = createStackNavigator(); + + const styles = StyleSheet.create({ + headerContainer: { + flexDirection: 'row', + alignItems: 'center' + }, + headerImage: { + width: 30, + height: 30, + borderRadius: 20, + marginRight: 8 + }, + headerText: { + color: style.Text, + fontSize: 16, + fontWeight: 'bold' + } + }); + return ( - + ({ + headerShown: true, + headerBackTitleVisible: false, + headerStyle: { + backgroundColor: style.Card + }, + headerTitle: () => ( + + + + {/* @ts-ignore */} + {route.params.username} + + + ), + headerTitleStyle: { + color: style.Text + }, + })} /> ) diff --git a/src/FLAD/package.json b/src/FLAD/package.json index e42515b..0460782 100644 --- a/src/FLAD/package.json +++ b/src/FLAD/package.json @@ -25,6 +25,7 @@ "expo-image-picker": "~14.0.2", "expo-linear-gradient": "~12.0.1", "expo-location": "~15.0.1", + "expo-random": "^13.0.0", "expo-secure-store": "~12.0.0", "expo-splash-screen": "~0.17.5", "react": "18.1.0", @@ -37,8 +38,9 @@ "react-native-safe-area-context": "4.4.1", "react-native-screens": "~3.18.0", "react-native-svg": "13.4.0", - "react-navigation-shared-element": "^3.1.3", + "react-native-vector-icons": "^10.0.3", "react-native-web": "~0.18.9", + "react-navigation-shared-element": "^3.1.3", "react-redux": "^8.0.5", "redux": "^4.2.1" }, @@ -46,6 +48,7 @@ "@babel/core": "^7.12.9", "@types/react": "~18.0.14", "@types/react-native": "~0.70.8", + "@types/react-native-vector-icons": "^6.4.18", "typescript": "^4.6.3" }, "private": true diff --git a/src/FLAD/redux/actions/appActions.ts b/src/FLAD/redux/actions/appActions.ts index a55b3ed..241d176 100644 --- a/src/FLAD/redux/actions/appActions.ts +++ b/src/FLAD/redux/actions/appActions.ts @@ -1,5 +1,8 @@ +import Conversation from "../../models/Conversation"; +import Message from "../../models/Message"; import Music from "../../models/Music"; import { Spot } from "../../models/Spot"; +import { chatTypes } from "../types/chatTypes"; import { favoritesTypes } from "../types/favoritesTypes"; import { spotifyTypes } from "../types/spotifyTypes"; @@ -28,4 +31,18 @@ export const resetNbAddedFavoriteMusic = () => { return { type: favoritesTypes.RESET_NB_ADDED_FAVORITE_MUSIC }; +} + +export const setConversations = (conversations: Conversation[]) => { + return { + type: chatTypes.FETCH_CONVERSATIONS, + payload: conversations, + }; +} + +export const setMessages = (messages: Message[]) => { + return { + type: chatTypes.FETCH_MESSAGES, + payload: messages, + }; } \ No newline at end of file diff --git a/src/FLAD/redux/reducers/appReducer.tsx b/src/FLAD/redux/reducers/appReducer.tsx index 1c0944f..7fbdfb2 100644 --- a/src/FLAD/redux/reducers/appReducer.tsx +++ b/src/FLAD/redux/reducers/appReducer.tsx @@ -1,4 +1,7 @@ +import Conversation from "../../models/Conversation"; +import Message from "../../models/Message"; import { Spot } from "../../models/Spot"; +import { chatTypes } from "../types/chatTypes"; import { favoritesTypes } from "../types/favoritesTypes"; import { spotifyTypes } from "../types/spotifyTypes"; import { spotTypes } from "../types/spotTypes"; @@ -10,6 +13,8 @@ const initialState = { userCurrentMusic: null, nbAddedFavoriteMusic: 0, oldSpot: [] as string[], + conversations: [] as Conversation[], + messages: [] as Message[] } const appReducer = (state = initialState, action: any) => { @@ -23,6 +28,10 @@ const appReducer = (state = initialState, action: any) => { }; case favoritesTypes.RESET_NB_ADDED_FAVORITE_MUSIC: return { ...state, nbAddedFavoriteMusic: 0 }; + case chatTypes.FETCH_CONVERSATIONS: + return { ...state, conversations: action.payload }; + case chatTypes.FETCH_MESSAGES: + return { ...state, messages: action.payload }; case spotTypes.FETCH_SPOT: const uniqueSpots = action.payload.filter((spot: Spot) => { const spotKey = `${spot.user}_${spot.music.id}`; diff --git a/src/FLAD/redux/thunk/appThunk.tsx b/src/FLAD/redux/thunk/appThunk.tsx index 76c3c95..2933b71 100644 --- a/src/FLAD/redux/thunk/appThunk.tsx +++ b/src/FLAD/redux/thunk/appThunk.tsx @@ -28,6 +28,7 @@ export const getUserCurrentMusic = () => { const music = await MusicServiceProvider.musicService.getMusicById(idTrack); dispatch(setUserCurrentMusic(music)) } catch (error: any) { + console.log(error); switch (error.response.status) { case 401: dispatch(logout); diff --git a/src/FLAD/redux/thunk/chatThunk.tsx b/src/FLAD/redux/thunk/chatThunk.tsx new file mode 100644 index 0000000..f3d847e --- /dev/null +++ b/src/FLAD/redux/thunk/chatThunk.tsx @@ -0,0 +1,42 @@ +import Message from "../../models/Message"; +import IMessageService from "../../services/messages/interfaces/IMessageService" +import StubMessageService from "../../services/messages/stub/StubMessageService"; +import { setConversations, setMessages } from "../actions/appActions"; + +const chatService: IMessageService = new StubMessageService(); + +export const getConversations = () => { + //@ts-ignore + return async dispatch => { + try { + const conversations = await chatService.getConversations(); + dispatch(setConversations(conversations)); + } catch (error: any) { + + } + } +} + +export const getMessages = (id: string) => { + //@ts-ignore + return async dispatch => { + try { + const messages = await chatService.getMessagesWithIdConversation(id); + dispatch(setMessages(messages)); + } catch (error: any) { + + } + } +} + +export const sendMessage = (id: string, message: Message) => { + //@ts-ignore + return async dispatch => { + try { + await chatService.sendMessage(id, message); + dispatch(getMessages(id)); + } catch (error: any) { + + } + } +} \ No newline at end of file diff --git a/src/FLAD/redux/types/chatTypes.tsx b/src/FLAD/redux/types/chatTypes.tsx new file mode 100644 index 0000000..9c832e5 --- /dev/null +++ b/src/FLAD/redux/types/chatTypes.tsx @@ -0,0 +1,4 @@ +export const chatTypes = { + FETCH_CONVERSATIONS: 'FETCH_CONVERSATIONS', + FETCH_MESSAGES: 'FETCH_MESSAGES', +} \ No newline at end of file diff --git a/src/FLAD/screens/ChatScreen.tsx b/src/FLAD/screens/ChatScreen.tsx index 07ecbb9..bc56b9c 100644 --- a/src/FLAD/screens/ChatScreen.tsx +++ b/src/FLAD/screens/ChatScreen.tsx @@ -1,22 +1,178 @@ -import { useNavigation } from "@react-navigation/native"; -import React, { useEffect } from "react"; -import { GiftedChat } from "react-native-gifted-chat"; +import React, { useCallback, useEffect, useState } from "react"; +import { Bubble, GiftedChat, IMessage, InputToolbar, Send } from "react-native-gifted-chat"; +import { faFileImage, faMicrophone } from "@fortawesome/free-solid-svg-icons" +import { colorsDark } from "../constants/colorsDark"; +import { colorsLight } from "../constants/colorsLight"; +import { useDispatch, useSelector } from 'react-redux'; +import { SafeAreaView, TouchableOpacity, View } from "react-native"; +import Icon from 'react-native-vector-icons/Ionicons'; +import FontAwesome from 'react-native-vector-icons/FontAwesome'; +import Message from "../models/Message"; +import { getMessages, sendMessage } from "../redux/thunk/chatThunk"; +import { FontAwesomeIcon } from "@fortawesome/react-native-fontawesome"; -export default function Chat() { +//@ts-ignore +export default function Chat({ route }) { + const item: string = route.params.conversation; + const [messages, setMessages] = useState(); + //@ts-ignore + const conversations: Message[] = useSelector(state => state.appReducer.messages); - const navigation = useNavigation(); + // @ts-ignore + const isDark = useSelector(state => state.userReducer.dark); + const style = isDark ? colorsDark : colorsLight; + + const dispatch = useDispatch(); useEffect(() => { - navigation.getParent()?.setOptions({ - tabBarStyle: { - display: "none" - } - }); - return () => navigation.getParent()?.setOptions({ - tabBarStyle: undefined + // @ts-ignore + dispatch(getMessages(item)); + }, []); + + useEffect(() => { + const mappedMessages = conversations.map((msg: Message) => ({ + _id: msg.id, + text: msg.content, + createdAt: msg.date, + user: { + _id: msg.sender, + name: msg.sender, + avatar: 'https://picsum.photos/536/354', + }, + audio: msg.audio + })); + + mappedMessages.reverse(); + + mappedMessages.push({ + _id: "0", + text: 'Vous avez matché !!!', + system: true, }); - }, [navigation]); + + setMessages(mappedMessages); + }, [conversations]); + + const onSend = useCallback((messages: any = []) => { + + const newMessage = new Message( + "-1", + messages[0].text, + "User1", + new Date() + ); + + // @ts-ignore + dispatch(sendMessage(item, newMessage)); + }, []) + + const renderBubble = (props: any) => { + return ( + + ); + }; + + const renderInputToolbar = (props: any) => { + return ( + + ); + }; + + function handleImageIconPress(): void { + console.log("Image"); + } + + function handleMicrophoneIconPress(): void { + console.log("Audio"); + } + + const renderActions = (props: any) => { + return ( + + + + + + + + + + ); + }; + + + const renderSend = (props: any) => { + return ( + + + + + + ); + }; + + const scrollToBottomComponent = () => { + return ( + + ) + } + return ( - - ) + + onSend(messages)} + user={{ + _id: "User1", + }} + listViewProps={{ + style: { + backgroundColor: style.body + }, + }} + // @ts-ignore + textInputStyle={{ + backgroundColor: style.Line, + borderRadius: 20, + paddingHorizontal: 15, + paddingTop: 10, + marginTop: 5, + color: style.Text + }} + renderActions={renderActions} + maxInputLength={255} + renderBubble={renderBubble} + renderInputToolbar={renderInputToolbar} + renderSend={renderSend} + scrollToBottom + scrollToBottomComponent={scrollToBottomComponent} + placeholder="Chat" + /> + + ); } \ No newline at end of file diff --git a/src/FLAD/screens/ConversationScreen.tsx b/src/FLAD/screens/ConversationScreen.tsx index 5990ae7..cd1da8d 100644 --- a/src/FLAD/screens/ConversationScreen.tsx +++ b/src/FLAD/screens/ConversationScreen.tsx @@ -1,25 +1,39 @@ import { useNavigation } from "@react-navigation/native"; -import { StyleSheet, Text, View, FlatList, TouchableOpacity } from "react-native"; -import { useSelector } from "react-redux"; +import { StyleSheet, Text, View, FlatList, TouchableOpacity, RefreshControl } from "react-native"; +import { useSelector, useDispatch } from "react-redux"; import { colorsDark } from '../constants/colorsDark'; import { colorsLight } from '../constants/colorsLight'; import Friend from "../components/FriendComponent"; import normalize from '../components/Normalize'; import { useSafeAreaInsets } from "react-native-safe-area-context"; +import { useCallback, useEffect, useState } from "react"; +import { getConversations } from "../redux/thunk/chatThunk"; export default function ConversationScreen() { + //@ts-ignore + const friends = useSelector(state => state.appReducer.conversations); // @ts-ignore const isDark = useSelector(state => state.userReducer.dark); + const [refreshing, setRefreshing] = useState(false); - const navigation = useNavigation(); + const dispatch = useDispatch(); + + useEffect(() => { + // @ts-ignore + dispatch(getConversations()); + }, []); + + const handleRefresh = () => { + setRefreshing(true); + //@ts-ignore + dispatch(getConversations()); + setTimeout(() => { + setRefreshing(false); + }, 700); + }; - const friends = [ - { id: 1, name: "Lucas", lastMessage: "J'en ai marre de provot", source: "https://i1.sndcdn.com/artworks-ncJnbnDbNOFd-0-t500x500.jpg" }, - { id: 2, name: "Louison", lastMessage: "Tu vien piscine ?", source: "https://i1.sndcdn.com/artworks-ncJnbnDbNOFd-0-t500x500.jpg" }, - { id: 3, name: "Dave", lastMessage: "Ok c noté !", source: "https://img.lemde.fr/2019/04/05/0/0/960/960/664/0/75/0/18299d3_tUvp2AZPH_jnsIL2ypVFGUro.jpg" }, - { id: 4, name: "Valentin", lastMessage: "Haha react native c incroyable !!!", source: "https://i1.sndcdn.com/artworks-ncJnbnDbNOFd-0-t500x500.jpg" }, - ]; + const navigation = useNavigation(); const style = isDark ? colorsDark : colorsLight; @@ -44,6 +58,18 @@ export default function ConversationScreen() { fontSize: normalize(20), color: '#787878', marginBottom: 5 + }, + body: { + alignItems: 'center', + justifyContent: 'center', + flex: 1, + marginHorizontal: "7%" + }, + text: { + color: style.Text, + fontSize: normalize(18), + opacity: 0.8, + textAlign: 'center' } }) @@ -53,18 +79,33 @@ export default function ConversationScreen() { Messages Retrouvez ici les discussions - item.id.toString()} - renderItem={({ item }) => ( - + + Pas de conversations pour le moment. 🥲{'\n'} + Va liker des musiques pour créer des conversations avec des gens dans le monde ! 🔥🎆 + + + ) : ( + b.lastMessage.date - a.lastMessage.date)} + keyExtractor={(item) => item.id.toString()} + renderItem={({ item }) => ( // @ts-ignore - onPress={() => navigation.navigate('Chat')}> - - - )} - /> + navigation.navigate('Chat', { username: item.name, image: item.image, conversation: item.id })}> + + + )} + refreshControl={ + + } + /> + )} ) } \ No newline at end of file diff --git a/src/FLAD/screens/FavoriteScreen.tsx b/src/FLAD/screens/FavoriteScreen.tsx index dd4ee64..dcc61a0 100644 --- a/src/FLAD/screens/FavoriteScreen.tsx +++ b/src/FLAD/screens/FavoriteScreen.tsx @@ -136,7 +136,7 @@ export default function FavoriteScreen() { Retrouvez ici vos musiques favorites state.appReducer.spot) - // @ts-ignore - const isDark = useSelector(state => state.userReducer.dark); - const style = isDark ? colorsDark : colorsLight; + // @ts-ignore + const isDark = useSelector(state => state.userReducer.dark); + const style = isDark ? colorsDark : colorsLight; const [cards, setCards] = useState(spotReducer); const [currentCard, setcurrentCard] = useState(cards[cards.length - 1]); diff --git a/src/FLAD/services/messages/interfaces/IMessageService.ts b/src/FLAD/services/messages/interfaces/IMessageService.ts new file mode 100644 index 0000000..4445711 --- /dev/null +++ b/src/FLAD/services/messages/interfaces/IMessageService.ts @@ -0,0 +1,8 @@ +import Conversation from "../../../models/Conversation"; +import Message from "../../../models/Message"; + +export default interface IMessageService { + getConversations(): Promise; + getMessagesWithIdConversation(id: string): Promise; + sendMessage(id: string, mes: Message): Promise; +} \ No newline at end of file diff --git a/src/FLAD/services/messages/stub/StubMessageService.ts b/src/FLAD/services/messages/stub/StubMessageService.ts new file mode 100644 index 0000000..17854c6 --- /dev/null +++ b/src/FLAD/services/messages/stub/StubMessageService.ts @@ -0,0 +1,57 @@ +import Conversation from "../../../models/Conversation"; +import Message from "../../../models/Message"; +import IMessageService from "../interfaces/IMessageService"; + +export default class StubMessageService implements IMessageService { + + getRandomDate(): Date { + const startDate = new Date(2022, 0, 1); // Jan 1, 2022 + const endDate = new Date(); // Current date + const randomDate = new Date(startDate.getTime() + Math.random() * (endDate.getTime() - startDate.getTime())); + return randomDate; + } + + private messages: { id: string, message: Message }[] = [ + { id: "1", message: new Message("1", "Hello", "User1", this.getRandomDate(), "https://p.scdn.co/mp3-preview/d2cdc7726fcdc244849620b427a0c37461bb3586?cid=eb2aab666a43490f82eef0bb064d363f") }, + { id: "2", message: new Message("2", "Hi there!", "User2", new Date()) }, + { id: "3", message: new Message("3", "Greetings", "User3", this.getRandomDate()) }, + { id: "1", message: new Message("4", "La chiennete", "User0", this.getRandomDate()) }, + { id: "2", message: new Message("5", "Grr paw", "User2", new Date()) }, + { id: "3", message: new Message("6", "Greetings", "User3", this.getRandomDate()) }, + ]; + + private conversations: Conversation[] = [ + new Conversation("1", "Imri", "https://i1.sndcdn.com/artworks-ncJnbnDbNOFd-0-t500x500.jpg", this.messages[0].message), + new Conversation("2", "Dave", "", this.messages[1].message), + new Conversation("3", "Benzema", "", this.messages[2].message), + ]; + + getConversations(): Promise { + return Promise.resolve(this.conversations); + } + + getMessagesWithIdConversation(id: string): Promise { + return Promise.resolve(this.messages.filter(msg => msg.id === id) + .map(msg => msg.message)); + + } + + sendMessage(id: string, mes: Message): Promise { + return new Promise((resolve, reject) => { + + const conversation = this.conversations.find(conv => conv.id === id); + + if (!conversation) { + reject(new Error(`Conversation with id ${id} not found.`)); + return; + } + + this.messages.push({ id, message: mes }); + + conversation.lastMessage = mes; + + resolve(); + }); + } + +} \ No newline at end of file diff --git a/src/FLAD/services/musics/spotify/SpotifyService.ts b/src/FLAD/services/musics/spotify/SpotifyService.ts index 1d59404..65ffbf6 100644 --- a/src/FLAD/services/musics/spotify/SpotifyService.ts +++ b/src/FLAD/services/musics/spotify/SpotifyService.ts @@ -53,9 +53,11 @@ export default class SpotifyService implements IMusicService { }, }); - if (response.data.item === undefined) { + if (response.data.item === undefined || response.data.item === null) { return null; } + + return response.data.item.id }