|
|
@ -1,9 +1,8 @@
|
|
|
|
import _, { omit } from 'lodash';
|
|
|
|
import _, { omit } from 'lodash';
|
|
|
|
|
|
|
|
|
|
|
|
import { Constants } from '../../session';
|
|
|
|
import { Constants } from '../../session';
|
|
|
|
import { createAsyncThunk } from '@reduxjs/toolkit';
|
|
|
|
import { createAsyncThunk, createSlice, PayloadAction } from '@reduxjs/toolkit';
|
|
|
|
import { getConversationController } from '../../session/conversations';
|
|
|
|
import { getConversationController } from '../../session/conversations';
|
|
|
|
import { MessageModel } from '../../models/message';
|
|
|
|
|
|
|
|
import { getMessagesByConversation } from '../../data/data';
|
|
|
|
import { getMessagesByConversation } from '../../data/data';
|
|
|
|
import {
|
|
|
|
import {
|
|
|
|
ConversationNotificationSettingType,
|
|
|
|
ConversationNotificationSettingType,
|
|
|
@ -231,6 +230,9 @@ export type ConversationsStateType = {
|
|
|
|
conversationLookup: ConversationLookupType;
|
|
|
|
conversationLookup: ConversationLookupType;
|
|
|
|
selectedConversation?: string;
|
|
|
|
selectedConversation?: string;
|
|
|
|
messages: Array<SortedMessageModelProps>;
|
|
|
|
messages: Array<SortedMessageModelProps>;
|
|
|
|
|
|
|
|
messageDetailProps: MessagePropsDetails | undefined;
|
|
|
|
|
|
|
|
showRightPanel: boolean;
|
|
|
|
|
|
|
|
selectedMessageIds: Array<string>;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
async function getMessages(
|
|
|
|
async function getMessages(
|
|
|
@ -338,246 +340,15 @@ export const fetchMessagesForConversation = createAsyncThunk(
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
// Actions
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
type ConversationAddedActionType = {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_ADDED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
data: ReduxConversationType;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
type ConversationChangedActionType = {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_CHANGED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
data: ReduxConversationType;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
type ConversationRemovedActionType = {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_REMOVED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type RemoveAllConversationsActionType = {
|
|
|
|
|
|
|
|
type: 'CONVERSATIONS_REMOVE_ALL';
|
|
|
|
|
|
|
|
payload: null;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type MessageExpiredActionType = {
|
|
|
|
|
|
|
|
type: 'MESSAGE_EXPIRED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type MessageChangedActionType = {
|
|
|
|
|
|
|
|
type: 'MESSAGE_CHANGED';
|
|
|
|
|
|
|
|
payload: MessageModelProps;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type MessagesChangedActionType = {
|
|
|
|
|
|
|
|
type: 'MESSAGES_CHANGED';
|
|
|
|
|
|
|
|
payload: Array<MessageModelProps>;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type MessageAddedActionType = {
|
|
|
|
|
|
|
|
type: 'MESSAGE_ADDED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messageModelProps: MessageModelProps;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type MessageDeletedActionType = {
|
|
|
|
|
|
|
|
type: 'MESSAGE_DELETED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type ConversationResetActionType = {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_RESET';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
export type SelectedConversationChangedActionType = {
|
|
|
|
|
|
|
|
type: 'SELECTED_CONVERSATION_CHANGED';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
messageId?: string;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export type FetchMessagesForConversationType = {
|
|
|
|
|
|
|
|
type: 'messages/fetchByConversationKey/fulfilled';
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messages: Array<MessageModelProps>;
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export type ConversationActionType =
|
|
|
|
|
|
|
|
| ConversationAddedActionType
|
|
|
|
|
|
|
|
| ConversationChangedActionType
|
|
|
|
|
|
|
|
| ConversationRemovedActionType
|
|
|
|
|
|
|
|
| ConversationResetActionType
|
|
|
|
|
|
|
|
| RemoveAllConversationsActionType
|
|
|
|
|
|
|
|
| MessageExpiredActionType
|
|
|
|
|
|
|
|
| MessageAddedActionType
|
|
|
|
|
|
|
|
| MessageDeletedActionType
|
|
|
|
|
|
|
|
| MessageChangedActionType
|
|
|
|
|
|
|
|
| MessagesChangedActionType
|
|
|
|
|
|
|
|
| SelectedConversationChangedActionType
|
|
|
|
|
|
|
|
| SelectedConversationChangedActionType
|
|
|
|
|
|
|
|
| FetchMessagesForConversationType;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Action Creators
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export const actions = {
|
|
|
|
|
|
|
|
conversationAdded,
|
|
|
|
|
|
|
|
conversationChanged,
|
|
|
|
|
|
|
|
conversationRemoved,
|
|
|
|
|
|
|
|
removeAllConversations,
|
|
|
|
|
|
|
|
messageExpired,
|
|
|
|
|
|
|
|
messageAdded,
|
|
|
|
|
|
|
|
messageDeleted,
|
|
|
|
|
|
|
|
conversationReset,
|
|
|
|
|
|
|
|
messageChanged,
|
|
|
|
|
|
|
|
messagesChanged,
|
|
|
|
|
|
|
|
fetchMessagesForConversation,
|
|
|
|
|
|
|
|
openConversationExternal,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function conversationAdded(id: string, data: ReduxConversationType): ConversationAddedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_ADDED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id,
|
|
|
|
|
|
|
|
data,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function conversationChanged(
|
|
|
|
|
|
|
|
id: string,
|
|
|
|
|
|
|
|
data: ReduxConversationType
|
|
|
|
|
|
|
|
): ConversationChangedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_CHANGED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id,
|
|
|
|
|
|
|
|
data,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function conversationRemoved(id: string): ConversationRemovedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_REMOVED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function removeAllConversations(): RemoveAllConversationsActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'CONVERSATIONS_REMOVE_ALL',
|
|
|
|
|
|
|
|
payload: null,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function messageExpired({
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
messageId,
|
|
|
|
|
|
|
|
}: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
}): MessageExpiredActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'MESSAGE_EXPIRED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
messageId,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function messageChanged(messageModelProps: MessageModelProps): MessageChangedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'MESSAGE_CHANGED',
|
|
|
|
|
|
|
|
payload: messageModelProps,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function messagesChanged(messageModelsProps: Array<MessageModelProps>): MessagesChangedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'MESSAGES_CHANGED',
|
|
|
|
|
|
|
|
payload: messageModelsProps,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function messageAdded({
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
messageModelProps,
|
|
|
|
|
|
|
|
}: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messageModelProps: MessageModelProps;
|
|
|
|
|
|
|
|
}): MessageAddedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'MESSAGE_ADDED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
messageModelProps,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function messageDeleted({
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
messageId,
|
|
|
|
|
|
|
|
}: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
}): MessageDeletedActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'MESSAGE_DELETED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
messageId,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function conversationReset({
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
}: {
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
}): ConversationResetActionType {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'CONVERSATION_RESET',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
conversationKey,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export function openConversationExternal(
|
|
|
|
|
|
|
|
id: string,
|
|
|
|
|
|
|
|
messageId?: string
|
|
|
|
|
|
|
|
): SelectedConversationChangedActionType {
|
|
|
|
|
|
|
|
window?.log?.info(`openConversationExternal with convoId: ${id}; messageId: ${messageId}`);
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
type: 'SELECTED_CONVERSATION_CHANGED',
|
|
|
|
|
|
|
|
payload: {
|
|
|
|
|
|
|
|
id,
|
|
|
|
|
|
|
|
messageId,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Reducer
|
|
|
|
// Reducer
|
|
|
|
|
|
|
|
|
|
|
|
function getEmptyState(): ConversationsStateType {
|
|
|
|
function getEmptyState(): ConversationsStateType {
|
|
|
|
return {
|
|
|
|
return {
|
|
|
|
conversationLookup: {},
|
|
|
|
conversationLookup: {},
|
|
|
|
messages: [],
|
|
|
|
messages: [],
|
|
|
|
|
|
|
|
messageDetailProps: undefined,
|
|
|
|
|
|
|
|
showRightPanel: false,
|
|
|
|
|
|
|
|
selectedMessageIds: [],
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -607,7 +378,13 @@ function sortMessages(
|
|
|
|
return messagesSorted;
|
|
|
|
return messagesSorted;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function handleMessageAdded(state: ConversationsStateType, action: MessageAddedActionType) {
|
|
|
|
function handleMessageAdded(
|
|
|
|
|
|
|
|
state: ConversationsStateType,
|
|
|
|
|
|
|
|
action: PayloadAction<{
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
messageModelProps: MessageModelProps;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
const { messages } = state;
|
|
|
|
const { messages } = state;
|
|
|
|
const { conversationKey, messageModelProps: addedMessageProps } = action.payload;
|
|
|
|
const { conversationKey, messageModelProps: addedMessageProps } = action.payload;
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
@ -631,9 +408,7 @@ function handleMessageAdded(state: ConversationsStateType, action: MessageAddedA
|
|
|
|
return state;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function handleMessageChanged(state: ConversationsStateType, action: MessageChangedActionType) {
|
|
|
|
function handleMessageChanged(state: ConversationsStateType, payload: MessageModelProps) {
|
|
|
|
const { payload } = action;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const messageInStoreIndex = state?.messages?.findIndex(
|
|
|
|
const messageInStoreIndex = state?.messages?.findIndex(
|
|
|
|
m => m.propsForMessage.id === payload.propsForMessage.id
|
|
|
|
m => m.propsForMessage.id === payload.propsForMessage.id
|
|
|
|
);
|
|
|
|
);
|
|
|
@ -661,15 +436,10 @@ function handleMessageChanged(state: ConversationsStateType, action: MessageChan
|
|
|
|
return state;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function handleMessagesChanged(state: ConversationsStateType, action: MessagesChangedActionType) {
|
|
|
|
function handleMessagesChanged(state: ConversationsStateType, payload: Array<MessageModelProps>) {
|
|
|
|
const { payload } = action;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
payload.forEach(element => {
|
|
|
|
payload.forEach(element => {
|
|
|
|
// tslint:disable-next-line: no-parameter-reassignment
|
|
|
|
// tslint:disable-next-line: no-parameter-reassignment
|
|
|
|
state = handleMessageChanged(state, {
|
|
|
|
state = handleMessageChanged(state, element);
|
|
|
|
payload: element,
|
|
|
|
|
|
|
|
type: 'MESSAGE_CHANGED',
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
return state;
|
|
|
|
return state;
|
|
|
@ -677,7 +447,10 @@ function handleMessagesChanged(state: ConversationsStateType, action: MessagesCh
|
|
|
|
|
|
|
|
|
|
|
|
function handleMessageExpiredOrDeleted(
|
|
|
|
function handleMessageExpiredOrDeleted(
|
|
|
|
state: ConversationsStateType,
|
|
|
|
state: ConversationsStateType,
|
|
|
|
action: MessageDeletedActionType | MessageExpiredActionType
|
|
|
|
action: PayloadAction<{
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
}>
|
|
|
|
) {
|
|
|
|
) {
|
|
|
|
const { conversationKey, messageId } = action.payload;
|
|
|
|
const { conversationKey, messageId } = action.payload;
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
@ -709,11 +482,8 @@ function handleMessageExpiredOrDeleted(
|
|
|
|
return state;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function handleConversationReset(
|
|
|
|
function handleConversationReset(state: ConversationsStateType, action: PayloadAction<string>) {
|
|
|
|
state: ConversationsStateType,
|
|
|
|
const conversationKey = action.payload;
|
|
|
|
action: ConversationResetActionType
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
const { conversationKey } = action.payload;
|
|
|
|
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
|
// just empty the list of messages
|
|
|
|
// just empty the list of messages
|
|
|
|
return {
|
|
|
|
return {
|
|
|
@ -724,110 +494,220 @@ function handleConversationReset(
|
|
|
|
return state;
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// tslint:disable: cyclomatic-complexity
|
|
|
|
const conversationsSlice = createSlice({
|
|
|
|
// tslint:disable: max-func-body-length
|
|
|
|
name: 'conversations',
|
|
|
|
export function reducer(
|
|
|
|
initialState: getEmptyState(),
|
|
|
|
state: ConversationsStateType = getEmptyState(),
|
|
|
|
reducers: {
|
|
|
|
action: ConversationActionType
|
|
|
|
showMessageDetailsView(
|
|
|
|
): ConversationsStateType {
|
|
|
|
state: ConversationsStateType,
|
|
|
|
if (action.type === 'CONVERSATION_ADDED') {
|
|
|
|
action: PayloadAction<MessagePropsDetails>
|
|
|
|
const { payload } = action;
|
|
|
|
) {
|
|
|
|
const { id, data } = payload;
|
|
|
|
// force the right panel to be hidden when showing message detail view
|
|
|
|
const { conversationLookup } = state;
|
|
|
|
return { ...state, messageDetailProps: action.payload, showRightPanel: false };
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
closeMessageDetailsView(state: ConversationsStateType) {
|
|
|
|
...state,
|
|
|
|
return { ...state, messageDetailProps: undefined };
|
|
|
|
conversationLookup: {
|
|
|
|
},
|
|
|
|
...conversationLookup,
|
|
|
|
|
|
|
|
[id]: data,
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (action.type === 'CONVERSATION_CHANGED') {
|
|
|
|
|
|
|
|
const { payload } = action;
|
|
|
|
|
|
|
|
const { id, data } = payload;
|
|
|
|
|
|
|
|
const { conversationLookup, selectedConversation } = state;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const existing = conversationLookup[id];
|
|
|
|
|
|
|
|
// In the change case we only modify the lookup if we already had that conversation
|
|
|
|
|
|
|
|
if (!existing) {
|
|
|
|
|
|
|
|
return state;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return {
|
|
|
|
openRightPanel(state: ConversationsStateType) {
|
|
|
|
...state,
|
|
|
|
return { ...state, showRightPanel: true };
|
|
|
|
selectedConversation,
|
|
|
|
},
|
|
|
|
conversationLookup: {
|
|
|
|
closeRightPanel(state: ConversationsStateType) {
|
|
|
|
...conversationLookup,
|
|
|
|
return { ...state, showRightPanel: false };
|
|
|
|
[id]: data,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
addMessageIdToSelection(state: ConversationsStateType, action: PayloadAction<string>) {
|
|
|
|
};
|
|
|
|
if (state.selectedMessageIds.some(id => id === action.payload)) {
|
|
|
|
}
|
|
|
|
return state;
|
|
|
|
if (action.type === 'CONVERSATION_REMOVED') {
|
|
|
|
}
|
|
|
|
const { payload } = action;
|
|
|
|
return { ...state, selectedMessageIds: [...state.selectedMessageIds, action.payload] };
|
|
|
|
const { id } = payload;
|
|
|
|
},
|
|
|
|
const { conversationLookup, selectedConversation } = state;
|
|
|
|
removeMessageIdFromSelection(state: ConversationsStateType, action: PayloadAction<string>) {
|
|
|
|
return {
|
|
|
|
const index = state.selectedMessageIds.findIndex(id => id === action.payload);
|
|
|
|
...state,
|
|
|
|
|
|
|
|
conversationLookup: omit(conversationLookup, [id]),
|
|
|
|
if (index === -1) {
|
|
|
|
selectedConversation: selectedConversation === id ? undefined : selectedConversation,
|
|
|
|
return state;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return { ...state, selectedMessageIds: state.selectedMessageIds.splice(index, 1) };
|
|
|
|
if (action.type === 'CONVERSATIONS_REMOVE_ALL') {
|
|
|
|
},
|
|
|
|
return getEmptyState();
|
|
|
|
toggleSelectedMessageId(state: ConversationsStateType, action: PayloadAction<string>) {
|
|
|
|
}
|
|
|
|
const index = state.selectedMessageIds.findIndex(id => id === action.payload);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (index === -1) {
|
|
|
|
|
|
|
|
return { ...state, selectedMessageIds: [...state.selectedMessageIds, action.payload] };
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return { ...state, selectedMessageIds: state.selectedMessageIds.splice(index, 1) };
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
resetSelectedMessageIds(state: ConversationsStateType) {
|
|
|
|
|
|
|
|
return { ...state, selectedMessageIds: [] };
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
if (action.type === 'SELECTED_CONVERSATION_CHANGED') {
|
|
|
|
conversationAdded(
|
|
|
|
const { payload } = action;
|
|
|
|
state: ConversationsStateType,
|
|
|
|
const { id } = payload;
|
|
|
|
action: PayloadAction<{
|
|
|
|
const oldSelectedConversation = state.selectedConversation;
|
|
|
|
id: string;
|
|
|
|
const newSelectedConversation = id;
|
|
|
|
data: ReduxConversationType;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
const { conversationLookup } = state;
|
|
|
|
|
|
|
|
|
|
|
|
if (newSelectedConversation !== oldSelectedConversation) {
|
|
|
|
|
|
|
|
// empty the message list
|
|
|
|
|
|
|
|
return {
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
...state,
|
|
|
|
messages: [],
|
|
|
|
conversationLookup: {
|
|
|
|
selectedConversation: id,
|
|
|
|
...conversationLookup,
|
|
|
|
|
|
|
|
[action.payload.id]: action.payload.data,
|
|
|
|
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
},
|
|
|
|
return {
|
|
|
|
conversationChanged(
|
|
|
|
...state,
|
|
|
|
state: ConversationsStateType,
|
|
|
|
selectedConversation: id,
|
|
|
|
action: PayloadAction<{
|
|
|
|
};
|
|
|
|
id: string;
|
|
|
|
}
|
|
|
|
data: ReduxConversationType;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
const { payload } = action;
|
|
|
|
|
|
|
|
const { id, data } = payload;
|
|
|
|
|
|
|
|
const { conversationLookup, selectedConversation } = state;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const existing = conversationLookup[id];
|
|
|
|
|
|
|
|
// In the change case we only modify the lookup if we already had that conversation
|
|
|
|
|
|
|
|
if (!existing) {
|
|
|
|
|
|
|
|
return state;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// this is called once the messages are loaded from the db for the currently selected conversation
|
|
|
|
|
|
|
|
if (action.type === fetchMessagesForConversation.fulfilled.type) {
|
|
|
|
|
|
|
|
const { messagesProps, conversationKey } = action.payload as FetchedMessageResults;
|
|
|
|
|
|
|
|
// double check that this update is for the shown convo
|
|
|
|
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
|
|
|
|
|
return {
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
...state,
|
|
|
|
messages: messagesProps,
|
|
|
|
selectedConversation,
|
|
|
|
|
|
|
|
conversationLookup: {
|
|
|
|
|
|
|
|
...conversationLookup,
|
|
|
|
|
|
|
|
[id]: data,
|
|
|
|
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
|
|
|
},
|
|
|
|
return state;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (action.type === 'MESSAGE_CHANGED') {
|
|
|
|
conversationRemoved(
|
|
|
|
return handleMessageChanged(state, action);
|
|
|
|
state: ConversationsStateType,
|
|
|
|
}
|
|
|
|
action: PayloadAction<{
|
|
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
const { payload } = action;
|
|
|
|
|
|
|
|
const { id } = payload;
|
|
|
|
|
|
|
|
const { conversationLookup, selectedConversation } = state;
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
...state,
|
|
|
|
|
|
|
|
conversationLookup: omit(conversationLookup, [id]),
|
|
|
|
|
|
|
|
selectedConversation: selectedConversation === id ? undefined : selectedConversation,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
if (action.type === 'MESSAGES_CHANGED') {
|
|
|
|
removeAllConversations() {
|
|
|
|
return handleMessagesChanged(state, action);
|
|
|
|
return getEmptyState();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
if (action.type === 'MESSAGE_ADDED') {
|
|
|
|
messageAdded(
|
|
|
|
return handleMessageAdded(state, action);
|
|
|
|
state: ConversationsStateType,
|
|
|
|
}
|
|
|
|
action: PayloadAction<{
|
|
|
|
if (action.type === 'MESSAGE_EXPIRED' || action.type === 'MESSAGE_DELETED') {
|
|
|
|
conversationKey: string;
|
|
|
|
return handleMessageExpiredOrDeleted(state, action);
|
|
|
|
messageModelProps: MessageModelProps;
|
|
|
|
}
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
return handleMessageAdded(state, action);
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
if (action.type === 'CONVERSATION_RESET') {
|
|
|
|
messageChanged(state: ConversationsStateType, action: PayloadAction<MessageModelProps>) {
|
|
|
|
return handleConversationReset(state, action);
|
|
|
|
return handleMessageChanged(state, action.payload);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
messagesChanged(
|
|
|
|
|
|
|
|
state: ConversationsStateType,
|
|
|
|
|
|
|
|
action: PayloadAction<Array<MessageModelProps>>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
return handleMessagesChanged(state, action.payload);
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
return state;
|
|
|
|
messageExpired(
|
|
|
|
}
|
|
|
|
state: ConversationsStateType,
|
|
|
|
|
|
|
|
action: PayloadAction<{
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
return handleMessageExpiredOrDeleted(state, action);
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
messageDeleted(
|
|
|
|
|
|
|
|
state: ConversationsStateType,
|
|
|
|
|
|
|
|
action: PayloadAction<{
|
|
|
|
|
|
|
|
messageId: string;
|
|
|
|
|
|
|
|
conversationKey: string;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
return handleMessageExpiredOrDeleted(state, action);
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
conversationReset(state: ConversationsStateType, action: PayloadAction<string>) {
|
|
|
|
|
|
|
|
return handleConversationReset(state, action);
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
openConversationExternal(
|
|
|
|
|
|
|
|
state: ConversationsStateType,
|
|
|
|
|
|
|
|
action: PayloadAction<{
|
|
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
|
|
messageId?: string;
|
|
|
|
|
|
|
|
}>
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
if (state.selectedConversation === action.payload.id) {
|
|
|
|
|
|
|
|
return state;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
state.showRightPanel = false;
|
|
|
|
|
|
|
|
state.messageDetailProps = undefined;
|
|
|
|
|
|
|
|
state.selectedMessageIds = [];
|
|
|
|
|
|
|
|
state.selectedConversation = action.payload.id;
|
|
|
|
|
|
|
|
state.messages = [];
|
|
|
|
|
|
|
|
return state;
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
extraReducers: (builder: any) => {
|
|
|
|
|
|
|
|
// Add reducers for additional action types here, and handle loading state as needed
|
|
|
|
|
|
|
|
builder.addCase(
|
|
|
|
|
|
|
|
fetchMessagesForConversation.fulfilled,
|
|
|
|
|
|
|
|
(state: ConversationsStateType, action: any) => {
|
|
|
|
|
|
|
|
// this is called once the messages are loaded from the db for the currently selected conversation
|
|
|
|
|
|
|
|
const { messagesProps, conversationKey } = action.payload as FetchedMessageResults;
|
|
|
|
|
|
|
|
// double check that this update is for the shown convo
|
|
|
|
|
|
|
|
if (conversationKey === state.selectedConversation) {
|
|
|
|
|
|
|
|
return {
|
|
|
|
|
|
|
|
...state,
|
|
|
|
|
|
|
|
messages: messagesProps,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return state;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// destructures
|
|
|
|
|
|
|
|
export const { actions, reducer } = conversationsSlice;
|
|
|
|
|
|
|
|
export const {
|
|
|
|
|
|
|
|
// conversation and messages list
|
|
|
|
|
|
|
|
conversationAdded,
|
|
|
|
|
|
|
|
conversationChanged,
|
|
|
|
|
|
|
|
conversationRemoved,
|
|
|
|
|
|
|
|
removeAllConversations,
|
|
|
|
|
|
|
|
messageExpired,
|
|
|
|
|
|
|
|
messageAdded,
|
|
|
|
|
|
|
|
messageDeleted,
|
|
|
|
|
|
|
|
conversationReset,
|
|
|
|
|
|
|
|
messageChanged,
|
|
|
|
|
|
|
|
messagesChanged,
|
|
|
|
|
|
|
|
openConversationExternal,
|
|
|
|
|
|
|
|
// layout stuff
|
|
|
|
|
|
|
|
showMessageDetailsView,
|
|
|
|
|
|
|
|
closeMessageDetailsView,
|
|
|
|
|
|
|
|
openRightPanel,
|
|
|
|
|
|
|
|
closeRightPanel,
|
|
|
|
|
|
|
|
addMessageIdToSelection,
|
|
|
|
|
|
|
|
resetSelectedMessageIds,
|
|
|
|
|
|
|
|
} = actions;
|
|
|
|