import { Component, linkEvent } from 'inferno'; import { Subscription } from 'rxjs'; import { UserOperation, CommentView, SortType, GetReplies, GetRepliesResponse, GetUserMentions, GetUserMentionsResponse, UserMentionResponse, CommentResponse, PrivateMessageView, GetPrivateMessages, PrivateMessagesResponse, PrivateMessageResponse, SiteView, UserMentionView, } from 'lemmy-js-client'; import { WebSocketService, UserService } from '../services'; import { wsJsonToRes, fetchLimit, toast, editCommentRes, saveCommentRes, createCommentLikeRes, commentsToFlatNodes, setupTippy, setIsoData, wsSubscribe, isBrowser, wsUserOp, wsClient, authField, } from '../utils'; import { CommentNodes } from './comment-nodes'; import { PrivateMessage } from './private-message'; import { HtmlTags } from './html-tags'; import { SortSelect } from './sort-select'; import { i18n } from '../i18next'; import { InitialFetchRequest } from 'shared/interfaces'; enum UnreadOrAll { Unread, All, } enum MessageType { All, Replies, Mentions, Messages, } enum ReplyEnum { Reply, Mention, Message, } type ReplyType = { id: number; type_: ReplyEnum; view: CommentView | PrivateMessageView | UserMentionView; published: string; }; interface InboxState { unreadOrAll: UnreadOrAll; messageType: MessageType; replies: CommentView[]; mentions: UserMentionView[]; messages: PrivateMessageView[]; sort: SortType; page: number; site_view: SiteView; loading: boolean; } export class Inbox extends Component { private isoData = setIsoData(this.context); private subscription: Subscription; private emptyState: InboxState = { unreadOrAll: UnreadOrAll.Unread, messageType: MessageType.All, replies: [], mentions: [], messages: [], sort: SortType.New, page: 1, site_view: this.isoData.site_res.site_view, loading: true, }; constructor(props: any, context: any) { super(props, context); this.state = this.emptyState; this.handleSortChange = this.handleSortChange.bind(this); if (!UserService.Instance.user && isBrowser()) { toast(i18n.t('not_logged_in'), 'danger'); this.context.router.history.push(`/login`); } this.parseMessage = this.parseMessage.bind(this); this.subscription = wsSubscribe(this.parseMessage); // Only fetch the data if coming from another route if (this.isoData.path == this.context.router.route.match.url) { this.state.replies = this.isoData.routeData[0].replies || []; this.state.mentions = this.isoData.routeData[1].mentions || []; this.state.messages = this.isoData.routeData[2].messages || []; this.state.loading = false; } else { this.refetch(); } } componentWillUnmount() { if (isBrowser()) { this.subscription.unsubscribe(); } } get documentTitle(): string { return `@${UserService.Instance.user.name} ${i18n.t('inbox')} - ${ this.state.site_view.site.name }`; } render() { return (
{this.state.loading ? (
) : (
{i18n.t('inbox')} #
{this.state.replies.length + this.state.mentions.length + this.state.messages.length > 0 && this.state.unreadOrAll == UnreadOrAll.Unread && (
  • {i18n.t('mark_all_as_read')}
)} {this.selects()} {this.state.messageType == MessageType.All && this.all()} {this.state.messageType == MessageType.Replies && this.replies()} {this.state.messageType == MessageType.Mentions && this.mentions()} {this.state.messageType == MessageType.Messages && this.messages()} {this.paginator()}
)}
); } unreadOrAllRadios() { return (
); } messageTypeRadios() { return (
); } selects() { return (
{this.unreadOrAllRadios()} {this.messageTypeRadios()}
); } combined(): ReplyType[] { let id = 0; let replies: ReplyType[] = this.state.replies.map(r => ({ id: id++, type_: ReplyEnum.Reply, view: r, published: r.comment.published, })); let mentions: ReplyType[] = this.state.mentions.map(r => ({ id: id++, type_: ReplyEnum.Mention, view: r, published: r.comment.published, })); let messages: ReplyType[] = this.state.messages.map(r => ({ id: id++, type_: ReplyEnum.Message, view: r, published: r.private_message.published, })); return [...replies, ...mentions, ...messages].sort((a, b) => b.published.localeCompare(a.published) ); } renderReplyType(i: ReplyType) { switch (i.type_) { case ReplyEnum.Reply: return ( ); case ReplyEnum.Mention: return ( ); case ReplyEnum.Message: return ( ); default: return
; } } all() { return
{this.combined().map(i => this.renderReplyType(i))}
; } replies() { return (
); } mentions() { return (
{this.state.mentions.map(umv => ( ))}
); } messages() { return (
{this.state.messages.map(pmv => ( ))}
); } paginator() { return (
{this.state.page > 1 && ( )} {this.unreadCount() > 0 && ( )}
); } nextPage(i: Inbox) { i.state.page++; i.setState(i.state); i.refetch(); } prevPage(i: Inbox) { i.state.page--; i.setState(i.state); i.refetch(); } handleUnreadOrAllChange(i: Inbox, event: any) { i.state.unreadOrAll = Number(event.target.value); i.state.page = 1; i.setState(i.state); i.refetch(); } handleMessageTypeChange(i: Inbox, event: any) { i.state.messageType = Number(event.target.value); i.state.page = 1; i.setState(i.state); i.refetch(); } static fetchInitialData(req: InitialFetchRequest): Promise[] { let promises: Promise[] = []; // It can be /u/me, or /username/1 let repliesForm: GetReplies = { sort: SortType.New, unread_only: true, page: 1, limit: fetchLimit, auth: req.auth, }; promises.push(req.client.getReplies(repliesForm)); let userMentionsForm: GetUserMentions = { sort: SortType.New, unread_only: true, page: 1, limit: fetchLimit, auth: req.auth, }; promises.push(req.client.getUserMentions(userMentionsForm)); let privateMessagesForm: GetPrivateMessages = { unread_only: true, page: 1, limit: fetchLimit, auth: req.auth, }; promises.push(req.client.getPrivateMessages(privateMessagesForm)); return promises; } refetch() { let repliesForm: GetReplies = { sort: this.state.sort, unread_only: this.state.unreadOrAll == UnreadOrAll.Unread, page: this.state.page, limit: fetchLimit, auth: authField(), }; WebSocketService.Instance.send(wsClient.getReplies(repliesForm)); let userMentionsForm: GetUserMentions = { sort: this.state.sort, unread_only: this.state.unreadOrAll == UnreadOrAll.Unread, page: this.state.page, limit: fetchLimit, auth: authField(), }; WebSocketService.Instance.send(wsClient.getUserMentions(userMentionsForm)); let privateMessagesForm: GetPrivateMessages = { unread_only: this.state.unreadOrAll == UnreadOrAll.Unread, page: this.state.page, limit: fetchLimit, auth: authField(), }; WebSocketService.Instance.send( wsClient.getPrivateMessages(privateMessagesForm) ); } handleSortChange(val: SortType) { this.state.sort = val; this.state.page = 1; this.setState(this.state); this.refetch(); } markAllAsRead(i: Inbox) { WebSocketService.Instance.send( wsClient.markAllAsRead({ auth: authField(), }) ); i.state.replies = []; i.state.mentions = []; i.state.messages = []; i.sendUnreadCount(); window.scrollTo(0, 0); i.setState(i.state); } parseMessage(msg: any) { let op = wsUserOp(msg); if (msg.error) { toast(i18n.t(msg.error), 'danger'); return; } else if (msg.reconnect) { this.refetch(); } else if (op == UserOperation.GetReplies) { let data = wsJsonToRes(msg).data; this.state.replies = data.replies; this.state.loading = false; this.sendUnreadCount(); window.scrollTo(0, 0); this.setState(this.state); setupTippy(); } else if (op == UserOperation.GetUserMentions) { let data = wsJsonToRes(msg).data; this.state.mentions = data.mentions; this.sendUnreadCount(); window.scrollTo(0, 0); this.setState(this.state); setupTippy(); } else if (op == UserOperation.GetPrivateMessages) { let data = wsJsonToRes(msg).data; this.state.messages = data.private_messages; this.sendUnreadCount(); window.scrollTo(0, 0); this.setState(this.state); setupTippy(); } else if (op == UserOperation.EditPrivateMessage) { let data = wsJsonToRes(msg).data; let found: PrivateMessageView = this.state.messages.find( m => m.private_message.id === data.private_message_view.private_message.id ); if (found) { found.private_message.content = data.private_message_view.private_message.content; found.private_message.updated = data.private_message_view.private_message.updated; } this.setState(this.state); } else if (op == UserOperation.DeletePrivateMessage) { let data = wsJsonToRes(msg).data; let found: PrivateMessageView = this.state.messages.find( m => m.private_message.id === data.private_message_view.private_message.id ); if (found) { found.private_message.deleted = data.private_message_view.private_message.deleted; found.private_message.updated = data.private_message_view.private_message.updated; } this.setState(this.state); } else if (op == UserOperation.MarkPrivateMessageAsRead) { let data = wsJsonToRes(msg).data; let found: PrivateMessageView = this.state.messages.find( m => m.private_message.id === data.private_message_view.private_message.id ); if (found) { found.private_message.updated = data.private_message_view.private_message.updated; // If youre in the unread view, just remove it from the list if ( this.state.unreadOrAll == UnreadOrAll.Unread && data.private_message_view.private_message.read ) { this.state.messages = this.state.messages.filter( r => r.private_message.id !== data.private_message_view.private_message.id ); } else { let found = this.state.messages.find( c => c.private_message.id == data.private_message_view.private_message.id ); found.private_message.read = data.private_message_view.private_message.read; } } this.sendUnreadCount(); this.setState(this.state); } else if (op == UserOperation.MarkAllAsRead) { // Moved to be instant } else if ( op == UserOperation.EditComment || op == UserOperation.DeleteComment || op == UserOperation.RemoveComment ) { let data = wsJsonToRes(msg).data; editCommentRes(data.comment_view, this.state.replies); this.setState(this.state); } else if (op == UserOperation.MarkCommentAsRead) { let data = wsJsonToRes(msg).data; // If youre in the unread view, just remove it from the list if ( this.state.unreadOrAll == UnreadOrAll.Unread && data.comment_view.comment.read ) { this.state.replies = this.state.replies.filter( r => r.comment.id !== data.comment_view.comment.id ); } else { let found = this.state.replies.find( c => c.comment.id == data.comment_view.comment.id ); found.comment.read = data.comment_view.comment.read; } this.sendUnreadCount(); this.setState(this.state); setupTippy(); } else if (op == UserOperation.MarkUserMentionAsRead) { let data = wsJsonToRes(msg).data; // TODO this might not be correct, it might need to use the comment id let found = this.state.mentions.find( c => c.user_mention.id == data.user_mention_view.user_mention.id ); found.comment.content = data.user_mention_view.comment.content; found.comment.updated = data.user_mention_view.comment.updated; found.comment.removed = data.user_mention_view.comment.removed; found.comment.deleted = data.user_mention_view.comment.deleted; found.counts.upvotes = data.user_mention_view.counts.upvotes; found.counts.downvotes = data.user_mention_view.counts.downvotes; found.counts.score = data.user_mention_view.counts.score; // If youre in the unread view, just remove it from the list if ( this.state.unreadOrAll == UnreadOrAll.Unread && data.user_mention_view.user_mention.read ) { this.state.mentions = this.state.mentions.filter( r => r.user_mention.id !== data.user_mention_view.user_mention.id ); } else { let found = this.state.mentions.find( c => c.user_mention.id == data.user_mention_view.user_mention.id ); // TODO test to make sure these mentions are getting marked as read found.user_mention.read = data.user_mention_view.user_mention.read; } this.sendUnreadCount(); this.setState(this.state); } else if (op == UserOperation.CreateComment) { let data = wsJsonToRes(msg).data; if (data.recipient_ids.includes(UserService.Instance.user.id)) { this.state.replies.unshift(data.comment_view); this.setState(this.state); } else if (data.comment_view.creator.id == UserService.Instance.user.id) { // TODO this seems wrong, you should be using form_id toast(i18n.t('reply_sent')); } } else if (op == UserOperation.CreatePrivateMessage) { let data = wsJsonToRes(msg).data; if ( data.private_message_view.recipient.id == UserService.Instance.user.id ) { this.state.messages.unshift(data.private_message_view); this.setState(this.state); } } else if (op == UserOperation.SaveComment) { let data = wsJsonToRes(msg).data; saveCommentRes(data.comment_view, this.state.replies); this.setState(this.state); setupTippy(); } else if (op == UserOperation.CreateCommentLike) { let data = wsJsonToRes(msg).data; createCommentLikeRes(data.comment_view, this.state.replies); this.setState(this.state); } } sendUnreadCount() { UserService.Instance.unreadCountSub.next(this.unreadCount()); } unreadCount(): number { return ( this.state.replies.filter(r => !r.comment.read).length + this.state.mentions.filter(r => !r.user_mention.read).length + this.state.messages.filter( r => UserService.Instance.user && !r.private_message.read && // TODO also seems very strang and wrong r.creator.id !== UserService.Instance.user.id ).length ); } }