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
イベントをサブスクライブする必要があります。コンポーネントにチャットメッセージを提供するコードを次に示します。
重要: チャットメッセージリスナーがメッセージの状態を更新するときに、多くの再レンダリングが発生する可能性があるため、パフォーマンス上の理由から、ChatMessageContext
と ChatRoomProvider
は分離されています。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 />
</>
);
};