Files
mattermost-mobile/app/actions/websocket/posts.ts
Elias Nahum ac2a3a143a MM-24895 Load team member and roles in the WebSocket event (#4603)
* Load team member and roles in the WebSocket event

* Split WebSocket actions and events into multiple files
2020-07-23 10:19:54 -07:00

209 lines
7.9 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import {
fetchMyChannel,
fetchMyChannelMember,
makeDirectChannelVisibleIfNecessary,
makeGroupMessageVisibleIfNecessary,
markChannelAsUnread,
} from '@actions/helpers/channels';
import {markAsViewedAndReadBatch} from '@actions/views/channel';
import {getPostsAdditionalDataBatch, getPostThread} from '@actions/views/post';
import {WebsocketEvents} from '@constants';
import {ChannelTypes} from '@mm-redux/action_types';
import {getUnreadPostData, postDeleted, receivedNewPost, receivedPost} from '@mm-redux/actions/posts';
import {General} from '@mm-redux/constants';
import {
getChannel,
getCurrentChannelId,
getMyChannelMember as selectMyChannelMember,
isManuallyUnread,
} from '@mm-redux/selectors/entities/channels';
import {getCurrentUserId} from '@mm-redux/selectors/entities/users';
import {getPost as selectPost} from '@mm-redux/selectors/entities/posts';
import {getUserIdFromChannelName} from '@mm-redux/utils/channel_utils';
import EventEmitter from '@mm-redux/utils/event_emitter';
import {isFromWebhook, isSystemMessage, shouldIgnorePost} from '@mm-redux/utils/post_utils';
import {ActionResult, DispatchFunc, GenericAction, GetStateFunc, batchActions} from '@mm-redux/types/actions';
import {WebSocketMessage} from '@mm-redux/types/websocket';
export function handleNewPostEvent(msg: WebSocketMessage) {
return async (dispatch: DispatchFunc, getState: GetStateFunc): Promise<ActionResult> => {
const state = getState();
const currentChannelId = getCurrentChannelId(state);
const currentUserId = getCurrentUserId(state);
const data = JSON.parse(msg.data.post);
const post = {
...data,
ownPost: data.user_id === currentUserId,
};
const actions: Array<GenericAction> = [];
const exists = selectPost(state, post.pending_post_id);
if (!exists) {
if (getCurrentChannelId(state) === post.channel_id) {
EventEmitter.emit(WebsocketEvents.INCREASE_POST_VISIBILITY_BY_ONE);
}
const myChannel = getChannel(state, post.channel_id);
if (!myChannel) {
const channel = await fetchMyChannel(post.channel_id);
if (channel.data) {
actions.push({
type: ChannelTypes.RECEIVED_CHANNEL,
data: channel.data,
});
}
}
const myChannelMember = selectMyChannelMember(state, post.channel_id);
if (!myChannelMember) {
const member = await fetchMyChannelMember(post.channel_id);
if (member.data) {
actions.push({
type: ChannelTypes.RECEIVED_MY_CHANNEL_MEMBER,
data: member.data,
});
}
}
actions.push(receivedNewPost(post));
// If we don't have the thread for this post, fetch it from the server
// and include the actions in the batch
if (post.root_id) {
const rootPost = selectPost(state, post.root_id);
if (!rootPost) {
const thread: any = await dispatch(getPostThread(post.root_id, true));
if (thread.data?.length) {
actions.push(...thread.data);
}
}
}
if (post.channel_id === currentChannelId) {
const id = post.channel_id + post.root_id;
const {typing} = state.entities;
if (typing[id]) {
actions.push({
type: WebsocketEvents.STOP_TYPING,
data: {
id,
userId: post.user_id,
now: Date.now(),
},
});
}
}
// Fetch and batch additional post data
const additional: any = await dispatch(getPostsAdditionalDataBatch([post]));
if (additional.data.length) {
actions.push(...additional.data);
}
if (msg.data.channel_type === General.DM_CHANNEL) {
const otherUserId = getUserIdFromChannelName(currentUserId, msg.data.channel_name);
const dmAction = makeDirectChannelVisibleIfNecessary(state, otherUserId);
if (dmAction) {
actions.push(dmAction);
}
} else if (msg.data.channel_type === General.GM_CHANNEL) {
const gmActions = await makeGroupMessageVisibleIfNecessary(state, post.channel_id);
if (gmActions) {
actions.push(...gmActions);
}
}
if (!shouldIgnorePost(post)) {
let markAsRead = false;
let markAsReadOnServer = false;
if (!isManuallyUnread(state, post.channel_id)) {
if (
post.user_id === getCurrentUserId(state) &&
!isSystemMessage(post) &&
!isFromWebhook(post)
) {
markAsRead = true;
markAsReadOnServer = false;
} else if (post.channel_id === currentChannelId) {
markAsRead = true;
markAsReadOnServer = true;
}
}
if (markAsRead) {
const readActions = markAsViewedAndReadBatch(state, post.channel_id, undefined, markAsReadOnServer);
actions.push(...readActions);
} else {
const unreadActions = markChannelAsUnread(state, msg.data.team_id, post.channel_id, msg.data.mentions);
actions.push(...unreadActions);
}
}
dispatch(batchActions(actions, 'BATCH_WS_NEW_POST'));
}
return {data: true};
};
}
export function handlePostEdited(msg: WebSocketMessage) {
return async (dispatch: DispatchFunc, getState: GetStateFunc): Promise<ActionResult> => {
const state = getState();
const currentUserId = getCurrentUserId(state);
const data = JSON.parse(msg.data.post);
const post = {
...data,
ownPost: data.user_id === currentUserId,
};
const actions = [receivedPost(post)];
const additional: any = await dispatch(getPostsAdditionalDataBatch([post]));
if (additional.data.length) {
actions.push(...additional.data);
}
dispatch(batchActions(actions, 'BATCH_WS_POST_EDITED'));
return {data: true};
};
}
export function handlePostDeleted(msg: WebSocketMessage): GenericAction {
const data = JSON.parse(msg.data.post);
return postDeleted(data);
}
export function handlePostUnread(msg: WebSocketMessage) {
return (dispatch: DispatchFunc, getState: GetStateFunc): ActionResult => {
const state = getState();
const manual = isManuallyUnread(state, msg.broadcast.channel_id);
if (!manual) {
const member = selectMyChannelMember(state, msg.broadcast.channel_id);
const delta = member ? member.msg_count - msg.data.msg_count : msg.data.msg_count;
const info = {
...msg.data,
user_id: msg.broadcast.user_id,
team_id: msg.broadcast.team_id,
channel_id: msg.broadcast.channel_id,
deltaMsgs: delta,
};
const data = getUnreadPostData(info, state);
dispatch({
type: ChannelTypes.POST_UNREAD_SUCCESS,
data,
});
return {data};
}
return {data: null};
};
}