IVS Chat Client Messaging SDK: React と React Native のベストプラクティス - Amazon IVS

IVS Chat Client Messaging SDK: React と React Native のベストプラクティス

このドキュメントは、React と React Native のために Amazon IVS Chat Messaging SDK を使用する際の最も重要なプラクティスについて記載しています。この情報により、React アプリ内で一般的なチャット機能を構築できるようになり、IVS Chat Messaging SDK のより高度な部分を深く掘り下げるために必要な背景知識が得られるはずです。

チャットルームイニシャライザーフックの作成

ChatRoom クラスには、接続状態を管理し、メッセージの受信やメッセージの削除などのイベントをリッスンするためのコアチャットメソッドとリスナーが含まれています。ここでは、チャットインスタンスをフックに適切に格納する方法を示します。

実装

TypeScript
// useChatRoom.ts import React from 'react'; import { ChatRoom, ChatRoomConfig } from 'amazon-ivs-chat-messaging'; export const useChatRoom = (config: ChatRoomConfig) => { const [room] = React.useState(() => new ChatRoom(config)); return { room }; };
JavaScript
import React from 'react'; import { ChatRoom } from 'amazon-ivs-chat-messaging'; export const useChatRoom = (config) => { const [room] = React.useState(() => new ChatRoom(config)); return { room }; };

注: 設定パラメータをその場で更新できないため、setState フックからの dispatch メソッドは使用しません。SDK はインスタンスを一度作成します。また、トークンプロバイダーを更新することはできません。

重要: ChatRoom イニシャライザーフックを 1 回使用して、新しいチャットルームインスタンスを初期化します。

TypeScript/JavaScript:

// ... const MyChatScreen = () => { const userId = 'Mike'; const { room } = useChatRoom({ regionOrUrl: SOCKET_URL, tokenProvider: () => tokenProvider(ROOM_ID, ['SEND_MESSAGE']), }); const handleConnect = () => { room.connect(); }; // ... }; // ...

接続状態のリスニング

オプションで、チャットルームフックで接続状態の更新をサブスクライブできます。

実装

TypeScript
// useChatRoom.ts import React from 'react'; import { ChatRoom, ChatRoomConfig, ConnectionState } from 'amazon-ivs-chat-messaging'; export const useChatRoom = (config: ChatRoomConfig) => { const [room] = useState(() => new ChatRoom(config)); const [state, setState] = React.useState<ConnectionState>('disconnected'); React.useEffect(() => { const unsubscribeOnConnecting = room.addListener('connecting', () => { setState('connecting'); }); const unsubscribeOnConnected = room.addListener('connect', () => { setState('connected'); }); const unsubscribeOnDisconnected = room.addListener('disconnect', () => { setState('disconnected'); }); return () => { unsubscribeOnConnecting(); unsubscribeOnConnected(); unsubscribeOnDisconnected(); }; }, []); return { room, state }; };
JavaScript
// useChatRoom.js import React from 'react'; import { ChatRoom } from 'amazon-ivs-chat-messaging'; export const useChatRoom = (config) => { const [room] = useState(() => new ChatRoom(config)); const [state, setState] = React.useState('disconnected'); React.useEffect(() => { const unsubscribeOnConnecting = room.addListener('connecting', () => { setState('connecting'); }); const unsubscribeOnConnected = room.addListener('connect', () => { setState('connected'); }); const unsubscribeOnDisconnected = room.addListener('disconnect', () => { setState('disconnected'); }); return () => { unsubscribeOnConnecting(); unsubscribeOnConnected(); unsubscribeOnDisconnected(); }; }, []); return { room, state }; };

ChatRoom インスタンスプロバイダー

(Prop Drilling (バケツリレー) を避けるために) 他のコンポーネントでフックを使用するには、React context を使用してチャットルームプロバイダーを作成できます。

実装

TypeScript
// ChatRoomContext.tsx import React from 'react'; import { ChatRoom } from 'amazon-ivs-chat-messaging'; const ChatRoomContext = React.createContext<ChatRoom | undefined>(undefined); export const useChatRoomContext = () => { const context = React.useContext(ChatRoomContext); if (context === undefined) { throw new Error('useChatRoomContext must be within ChatRoomProvider'); } return context; }; export const ChatRoomProvider = ChatRoomContext.Provider;
JavaScript
// ChatRoomContext.jsx import React from 'react'; import { ChatRoom } from 'amazon-ivs-chat-messaging'; const ChatRoomContext = React.createContext(undefined); export const useChatRoomContext = () => { const context = React.useContext(ChatRoomContext); if (context === undefined) { throw new Error('useChatRoomContext must be within ChatRoomProvider'); } return context; }; export const ChatRoomProvider = ChatRoomContext.Provider;

ChatRoomProvider を作成したら、インスタンスを useChatRoomContext で使用できます。

重要: 接続をリッスンしている場合に不要な再レンダリングを避けるために、チャット画面と中間の他のコンポーネントの間で context へのアクセスが必要な場合にのみ、プロバイダーをルートレベルに配置してください。それ以外の場合は、チャット画面のできるだけ近くにプロバイダーを配置してください。

TypeScript/JavaScript:

// AppContainer const AppContainer = () => { const { room } = useChatRoom({ regionOrUrl: SOCKET_URL, tokenProvider: () => tokenProvider(ROOM_ID, ['SEND_MESSAGE']), }); return ( <ChatRoomProvider value={room}> <MyChatScreen /> </ChatRoomProvider> ); }; // MyChatScreen const MyChatScreen = () => { const room = useChatRoomContext(); const handleConnect = () => { room.connect(); }; // ... }; // ...

メッセージリスナーの作成

すべての着信メッセージについての最新の状態を知るには、message および deleteMessage イベントをサブスクライブする必要があります。コンポーネントにチャットメッセージを提供するコードを次に示します。

重要: チャットメッセージリスナーがメッセージの状態を更新するときに、多くの再レンダリングが発生する可能性があるため、パフォーマンス上の理由から、ChatMessageContextChatRoomProvider は分離されています。ChatMessageProvider を使用するコンポーネントで ChatMessageContext を適用することを忘れないでください。

実装

TypeScript
// ChatMessagesContext.tsx import React from 'react'; import { ChatMessage } from 'amazon-ivs-chat-messaging'; import { useChatRoomContext } from './ChatRoomContext'; const ChatMessagesContext = React.createContext<ChatMessage[] | undefined>(undefined); export const useChatMessagesContext = () => { const context = React.useContext(ChatMessagesContext); if (context === undefined) { throw new Error('useChatMessagesContext must be within ChatMessagesProvider); } return context; }; export const ChatMessagesProvider = ({ children }: { children: React.ReactNode }) => { const room = useChatRoomContext(); const [messages, setMessages] = React.useState<ChatMessage[]>([]); React.useEffect(() => { const unsubscribeOnMessageReceived = room.addListener('message', (message) => { setMessages((msgs) => [message, ...msgs]); }); const unsubscribeOnMessageDeleted = room.addListener('messageDelete', (deleteEvent) => { setMessages((prev) => prev.filter((message) => message.id !== deleteEvent.messageId)); }); return () => { unsubscribeOnMessageDeleted(); unsubscribeOnMessageReceived(); }; }, [room]); return <ChatMessagesContext.Provider value={messages}>{children}</ChatMessagesContext.Provider>; };
JavaScript
// ChatMessagesContext.jsx import React from 'react'; import { useChatRoomContext } from './ChatRoomContext'; const ChatMessagesContext = React.createContext(undefined); export const useChatMessagesContext = () => { const context = React.useContext(ChatMessagesContext); if (context === undefined) { throw new Error('useChatMessagesContext must be within ChatMessagesProvider); } return context; }; export const ChatMessagesProvider = ({ children }) => { const room = useChatRoomContext(); const [messages, setMessages] = React.useState([]); React.useEffect(() => { const unsubscribeOnMessageReceived = room.addListener('message', (message) => { setMessages((msgs) => [message, ...msgs]); }); const unsubscribeOnMessageDeleted = room.addListener('messageDelete', (deleteEvent) => { setMessages((prev) => prev.filter((message) => message.id !== deleteEvent.messageId)); }); return () => { unsubscribeOnMessageDeleted(); unsubscribeOnMessageReceived(); }; }, [room]); return <ChatMessagesContext.Provider value={messages}>{children}</ChatMessagesContext.Provider>; };

React の例

重要: メッセージコンテナを ChatMessagesProvider で必ずラップしてください。Message 行は、メッセージの内容を表示するコンポーネントの例です。

TypeScript/JavaScript:

// your message list component... import React from 'react'; import { useChatMessagesContext } from './ChatMessagesContext'; const MessageListContainer = () => { const messages = useChatMessagesContext(); return ( <React.Fragment> {messages.map((message) => ( <MessageRow message={message} /> ))} </React.Fragment> ); };

React Native の例

デフォルトでは、ChatMessage には id が含まれており、これは各行の FlatList の React キーとして自動的に使用されます。したがって、keyExtractor を渡す必要はありません。

TypeScript
// MessageListContainer.tsx import React from 'react'; import { ListRenderItemInfo, FlatList } from 'react-native'; import { ChatMessage } from 'amazon-ivs-chat-messaging'; import { useChatMessagesContext } from './ChatMessagesContext'; const MessageListContainer = () => { const messages = useChatMessagesContext(); const renderItem = useCallback(({ item }: ListRenderItemInfo<ChatMessage>) => <MessageRow />, []); return <FlatList data={messages} renderItem={renderItem} />; };
JavaScript
// MessageListContainer.jsx import React from 'react'; import { FlatList } from 'react-native'; import { useChatMessagesContext } from './ChatMessagesContext'; const MessageListContainer = () => { const messages = useChatMessagesContext(); const renderItem = useCallback(({ item }) => <MessageRow />, []); return <FlatList data={messages} renderItem={renderItem} />; };

アプリ内の複数のチャットルームインスタンス

アプリで複数のチャットルームを同時に使用する場合は、チャットごとに各プロバイダーを作成し、チャットプロバイダーで使用することをお勧めします。この例では、ヘルプボットとカスタマーヘルプチャットを作成しています。両方のためにプロバイダーを作成します。

TypeScript
// SupportChatProvider.tsx import React from 'react'; import { SUPPORT_ROOM_ID, SOCKET_URL } from '../../config'; import { tokenProvider } from '../tokenProvider'; import { ChatRoomProvider } from './ChatRoomContext'; import { useChatRoom } from './useChatRoom'; export const SupportChatProvider = ({ children }: { children: React.ReactNode }) => { const { room } = useChatRoom({ regionOrUrl: SOCKET_URL, tokenProvider: () => tokenProvider(SUPPORT_ROOM_ID, ['SEND_MESSAGE']), }); return <ChatRoomProvider value={room}>{children}</ChatRoomProvider>; }; // SalesChatProvider.tsx import React from 'react'; import { SALES_ROOM_ID, SOCKET_URL } from '../../config'; import { tokenProvider } from '../tokenProvider'; import { ChatRoomProvider } from './ChatRoomContext'; import { useChatRoom } from './useChatRoom'; export const SalesChatProvider = ({ children }: { children: React.ReactNode }) => { const { room } = useChatRoom({ regionOrUrl: SOCKET_URL, tokenProvider: () => tokenProvider(SALES_ROOM_ID, ['SEND_MESSAGE']), }); return <ChatRoomProvider value={room}>{children}</ChatRoomProvider>; };
JavaScript
// SupportChatProvider.jsx import React from 'react'; import { SUPPORT_ROOM_ID, SOCKET_URL } from '../../config'; import { tokenProvider } from '../tokenProvider'; import { ChatRoomProvider } from './ChatRoomContext'; import { useChatRoom } from './useChatRoom'; export const SupportChatProvider = ({ children }) => { const { room } = useChatRoom({ regionOrUrl: SOCKET_URL, tokenProvider: () => tokenProvider(SUPPORT_ROOM_ID, ['SEND_MESSAGE']), }); return <ChatRoomProvider value={room}>{children}</ChatRoomProvider>; }; // SalesChatProvider.jsx import React from 'react'; import { SALES_ROOM_ID, SOCKET_URL } from '../../config'; import { tokenProvider } from '../tokenProvider'; import { ChatRoomProvider } from './ChatRoomContext'; import { useChatRoom } from './useChatRoom'; export const SalesChatProvider = ({ children }) => { const { room } = useChatRoom({ regionOrUrl: SOCKET_URL, tokenProvider: () => tokenProvider(SALES_ROOM_ID, ['SEND_MESSAGE']), }); return <ChatRoomProvider value={room}>{children}</ChatRoomProvider>; };

React の例

同じ ChatRoomProvider を使用する別のチャットプロバイダーを使用できるようになりました。後で、各画面/ビュー内で同じ useChatRoomContext を再利用できます。

TypeScript/JavaScript:

// App.tsx / App.jsx const App = () => { return ( <Routes> <Route element={ <SupportChatProvider> <SupportChatScreen /> </SupportChatProvider> } /> <Route element={ <SalesChatProvider> <SalesChatScreen /> </SalesChatProvider> } /> </Routes> ); };

React Native の例

TypeScript/JavaScript:

// App.tsx / App.jsx const App = () => { return ( <Stack.Navigator> <Stack.Screen name="SupportChat"> <SupportChatProvider> <SupportChatScreen /> </SupportChatProvider> </Stack.Screen> <Stack.Screen name="SalesChat"> <SalesChatProvider> <SalesChatScreen /> </SalesChatProvider> </Stack.Screen> </Stack.Navigator> ); };

TypeScript/JavaScript:

// SupportChatScreen.tsx / SupportChatScreen.jsx // ... const SupportChatScreen = () => { const room = useChatRoomContext(); const handleConnect = () => { room.connect(); }; return ( <> <Button title="Connect" onPress={handleConnect} /> <MessageListContainer /> </> ); }; // SalesChatScreen.tsx / SalesChatScreen.jsx // ... const SalesChatScreen = () => { const room = useChatRoomContext(); const handleConnect = () => { room.connect(); }; return ( <> <Button title="Connect" onPress={handleConnect} /> <MessageListContainer /> </> ); };