mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
PLT-3101 Added message history (#3205)
* Added message history * Minor logical changes * Fixed indexes resetting * Fixed double messages * Fixed resetting main history when RHS opened
This commit is contained in:
committed by
Harrison Healey
parent
96e8fc165f
commit
2c42294bbc
@@ -11,6 +11,7 @@ import UserStore from 'stores/user_store.jsx';
|
||||
import PostDeletedModal from './post_deleted_modal.jsx';
|
||||
import PostStore from 'stores/post_store.jsx';
|
||||
import PreferenceStore from 'stores/preference_store.jsx';
|
||||
import MessageHistoryStore from 'stores/message_history_store.jsx';
|
||||
import Textbox from './textbox.jsx';
|
||||
import MsgTyping from './msg_typing.jsx';
|
||||
import FileUpload from './file_upload.jsx';
|
||||
@@ -68,11 +69,11 @@ class CreateComment extends React.Component {
|
||||
this.hidePostDeletedModal = this.hidePostDeletedModal.bind(this);
|
||||
|
||||
PostStore.clearCommentDraftUploads();
|
||||
MessageHistoryStore.resetHistoryIndex('comment');
|
||||
|
||||
const draft = PostStore.getCommentDraft(this.props.rootId);
|
||||
this.state = {
|
||||
messageText: draft.message,
|
||||
lastMessage: '',
|
||||
uploadsInProgress: draft.uploadsInProgress,
|
||||
previews: draft.previews,
|
||||
submitting: false,
|
||||
@@ -80,18 +81,22 @@ class CreateComment extends React.Component {
|
||||
showPostDeletedModal: false
|
||||
};
|
||||
}
|
||||
|
||||
componentDidMount() {
|
||||
PreferenceStore.addChangeListener(this.onPreferenceChange);
|
||||
this.focusTextbox();
|
||||
}
|
||||
|
||||
componentWillUnmount() {
|
||||
PreferenceStore.removeChangeListener(this.onPreferenceChange);
|
||||
}
|
||||
|
||||
onPreferenceChange() {
|
||||
this.setState({
|
||||
ctrlSend: PreferenceStore.getBool(Constants.Preferences.CATEGORY_ADVANCED_SETTINGS, 'send_on_ctrl_enter')
|
||||
});
|
||||
}
|
||||
|
||||
componentDidUpdate(prevProps, prevState) {
|
||||
if (prevState.uploadsInProgress < this.state.uploadsInProgress) {
|
||||
$('.post-right__scroll').scrollTop($('.post-right__scroll')[0].scrollHeight);
|
||||
@@ -101,6 +106,7 @@ class CreateComment extends React.Component {
|
||||
this.focusTextbox();
|
||||
}
|
||||
}
|
||||
|
||||
handleSubmit(e) {
|
||||
e.preventDefault();
|
||||
|
||||
@@ -125,6 +131,8 @@ class CreateComment extends React.Component {
|
||||
return;
|
||||
}
|
||||
|
||||
MessageHistoryStore.storeMessageInHistory(this.state.messageText);
|
||||
|
||||
const userId = UserStore.getCurrentId();
|
||||
|
||||
post.channel_id = this.props.channelId;
|
||||
@@ -173,13 +181,13 @@ class CreateComment extends React.Component {
|
||||
|
||||
this.setState({
|
||||
messageText: '',
|
||||
lastMessage: this.state.messageText,
|
||||
submitting: false,
|
||||
postError: null,
|
||||
previews: [],
|
||||
serverError: null
|
||||
});
|
||||
}
|
||||
|
||||
commentMsgKeyPress(e) {
|
||||
if (this.state.ctrlSend && e.ctrlKey || !this.state.ctrlSend) {
|
||||
if (e.which === KeyCodes.ENTER && !e.shiftKey && !e.altKey) {
|
||||
@@ -191,6 +199,7 @@ class CreateComment extends React.Component {
|
||||
|
||||
GlobalActions.emitLocalUserTypingEvent(this.props.channelId, this.props.rootId);
|
||||
}
|
||||
|
||||
handleUserInput(messageText) {
|
||||
const draft = PostStore.getCommentDraft(this.props.rootId);
|
||||
draft.message = messageText;
|
||||
@@ -199,6 +208,7 @@ class CreateComment extends React.Component {
|
||||
$('.post-right__scroll').parent().scrollTop($('.post-right__scroll')[0].scrollHeight);
|
||||
this.setState({messageText: messageText});
|
||||
}
|
||||
|
||||
handleKeyDown(e) {
|
||||
if (this.state.ctrlSend && e.keyCode === KeyCodes.ENTER && e.ctrlKey === true) {
|
||||
this.commentMsgKeyPress(e);
|
||||
@@ -224,22 +234,21 @@ class CreateComment extends React.Component {
|
||||
});
|
||||
}
|
||||
|
||||
if ((e.ctrlKey || e.metaKey) && !e.altKey && !e.shiftKey && e.keyCode === KeyCodes.UP) {
|
||||
const lastPost = PostStore.getCurrentUsersLatestPost(this.props.channelId, this.props.rootId);
|
||||
if (!lastPost || this.state.messageText !== '') {
|
||||
return;
|
||||
if ((e.ctrlKey || e.metaKey) && !e.altKey && !e.shiftKey && (e.keyCode === Constants.KeyCodes.UP || e.keyCode === Constants.KeyCodes.DOWN)) {
|
||||
const lastMessage = MessageHistoryStore.nextMessageInHistory(e.keyCode, this.state.messageText, 'comment');
|
||||
if (lastMessage !== null) {
|
||||
e.preventDefault();
|
||||
this.setState({
|
||||
messageText: lastMessage
|
||||
});
|
||||
}
|
||||
e.preventDefault();
|
||||
let message = lastPost.message;
|
||||
if (this.state.lastMessage !== '') {
|
||||
message = this.state.lastMessage;
|
||||
}
|
||||
this.setState({messageText: message});
|
||||
}
|
||||
}
|
||||
|
||||
handleUploadClick() {
|
||||
this.focusTextbox();
|
||||
}
|
||||
|
||||
handleUploadStart(clientIds) {
|
||||
const draft = PostStore.getCommentDraft(this.props.rootId);
|
||||
|
||||
@@ -252,6 +261,7 @@ class CreateComment extends React.Component {
|
||||
// but this also resets the focus after a drag and drop
|
||||
this.focusTextbox();
|
||||
}
|
||||
|
||||
handleFileUploadComplete(filenames, clientIds) {
|
||||
const draft = PostStore.getCommentDraft(this.props.rootId);
|
||||
|
||||
@@ -269,6 +279,7 @@ class CreateComment extends React.Component {
|
||||
|
||||
this.setState({uploadsInProgress: draft.uploadsInProgress, previews: draft.previews});
|
||||
}
|
||||
|
||||
handleUploadError(err, clientId) {
|
||||
if (clientId === -1) {
|
||||
this.setState({serverError: err});
|
||||
@@ -285,6 +296,7 @@ class CreateComment extends React.Component {
|
||||
this.setState({uploadsInProgress: draft.uploadsInProgress, serverError: err});
|
||||
}
|
||||
}
|
||||
|
||||
removePreview(id) {
|
||||
const previews = this.state.previews;
|
||||
const uploadsInProgress = this.state.uploadsInProgress;
|
||||
@@ -309,30 +321,36 @@ class CreateComment extends React.Component {
|
||||
|
||||
this.setState({previews: previews, uploadsInProgress: uploadsInProgress});
|
||||
}
|
||||
|
||||
componentWillReceiveProps(newProps) {
|
||||
if (newProps.rootId !== this.props.rootId) {
|
||||
const draft = PostStore.getCommentDraft(newProps.rootId);
|
||||
this.setState({messageText: draft.message, uploadsInProgress: draft.uploadsInProgress, previews: draft.previews});
|
||||
}
|
||||
}
|
||||
|
||||
getFileCount() {
|
||||
return this.state.previews.length + this.state.uploadsInProgress.length;
|
||||
}
|
||||
|
||||
focusTextbox() {
|
||||
if (!Utils.isMobile()) {
|
||||
this.refs.textbox.focus();
|
||||
}
|
||||
}
|
||||
|
||||
showPostDeletedModal() {
|
||||
this.setState({
|
||||
showPostDeletedModal: true
|
||||
});
|
||||
}
|
||||
|
||||
hidePostDeletedModal() {
|
||||
this.setState({
|
||||
showPostDeletedModal: false
|
||||
});
|
||||
}
|
||||
|
||||
render() {
|
||||
let serverError = null;
|
||||
if (this.state.serverError) {
|
||||
|
||||
@@ -17,6 +17,7 @@ import * as ChannelActions from 'actions/channel_actions.jsx';
|
||||
|
||||
import ChannelStore from 'stores/channel_store.jsx';
|
||||
import PostStore from 'stores/post_store.jsx';
|
||||
import MessageHistoryStore from 'stores/message_history_store.jsx';
|
||||
import UserStore from 'stores/user_store.jsx';
|
||||
import PreferenceStore from 'stores/preference_store.jsx';
|
||||
|
||||
@@ -79,7 +80,6 @@ class CreatePost extends React.Component {
|
||||
this.state = {
|
||||
channelId: ChannelStore.getCurrentId(),
|
||||
messageText: draft.messageText,
|
||||
lastMessage: '',
|
||||
uploadsInProgress: draft.uploadsInProgress,
|
||||
previews: draft.previews,
|
||||
submitting: false,
|
||||
@@ -90,6 +90,7 @@ class CreatePost extends React.Component {
|
||||
showPostDeletedModal: false
|
||||
};
|
||||
}
|
||||
|
||||
getCurrentDraft() {
|
||||
const draft = PostStore.getCurrentDraft();
|
||||
const safeDraft = {previews: [], messageText: '', uploadsInProgress: []};
|
||||
@@ -108,6 +109,7 @@ class CreatePost extends React.Component {
|
||||
|
||||
return safeDraft;
|
||||
}
|
||||
|
||||
handleSubmit(e) {
|
||||
e.preventDefault();
|
||||
|
||||
@@ -128,8 +130,10 @@ class CreatePost extends React.Component {
|
||||
return;
|
||||
}
|
||||
|
||||
MessageHistoryStore.storeMessageInHistory(this.state.messageText);
|
||||
|
||||
this.setState({submitting: true, serverError: null});
|
||||
this.setState({lastMessage: this.state.messageText});
|
||||
|
||||
if (post.message.indexOf('/') === 0) {
|
||||
ChannelActions.executeCommand(
|
||||
this.state.channelId,
|
||||
@@ -158,6 +162,7 @@ class CreatePost extends React.Component {
|
||||
this.sendMessage(post);
|
||||
}
|
||||
}
|
||||
|
||||
sendMessage(post) {
|
||||
post.channel_id = this.state.channelId;
|
||||
post.filenames = this.state.previews;
|
||||
@@ -193,11 +198,13 @@ class CreatePost extends React.Component {
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
focusTextbox() {
|
||||
if (!Utils.isMobile()) {
|
||||
this.refs.textbox.focus();
|
||||
}
|
||||
}
|
||||
|
||||
postMsgKeyPress(e) {
|
||||
if (this.state.ctrlSend && e.ctrlKey || !this.state.ctrlSend) {
|
||||
if (e.which === KeyCodes.ENTER && !e.shiftKey && !e.altKey) {
|
||||
@@ -209,6 +216,7 @@ class CreatePost extends React.Component {
|
||||
|
||||
GlobalActions.emitLocalUserTypingEvent(this.state.channelId, '');
|
||||
}
|
||||
|
||||
handleUserInput(messageText) {
|
||||
this.setState({messageText});
|
||||
|
||||
@@ -216,9 +224,11 @@ class CreatePost extends React.Component {
|
||||
draft.message = messageText;
|
||||
PostStore.storeCurrentDraft(draft);
|
||||
}
|
||||
|
||||
handleUploadClick() {
|
||||
this.focusTextbox();
|
||||
}
|
||||
|
||||
handleUploadStart(clientIds, channelId) {
|
||||
const draft = PostStore.getDraft(channelId);
|
||||
|
||||
@@ -231,6 +241,7 @@ class CreatePost extends React.Component {
|
||||
// but this also resets the focus after a drag and drop
|
||||
this.focusTextbox();
|
||||
}
|
||||
|
||||
handleFileUploadComplete(filenames, clientIds, channelId) {
|
||||
const draft = PostStore.getDraft(channelId);
|
||||
|
||||
@@ -250,6 +261,7 @@ class CreatePost extends React.Component {
|
||||
this.setState({uploadsInProgress: draft.uploadsInProgress, previews: draft.previews});
|
||||
}
|
||||
}
|
||||
|
||||
handleUploadError(err, clientId, channelId) {
|
||||
let message = err;
|
||||
if (message && typeof message !== 'string') {
|
||||
@@ -274,6 +286,7 @@ class CreatePost extends React.Component {
|
||||
|
||||
this.setState({serverError: message});
|
||||
}
|
||||
|
||||
removePreview(id) {
|
||||
const previews = Object.assign([], this.state.previews);
|
||||
const uploadsInProgress = this.state.uploadsInProgress;
|
||||
@@ -298,6 +311,7 @@ class CreatePost extends React.Component {
|
||||
|
||||
this.setState({previews, uploadsInProgress});
|
||||
}
|
||||
|
||||
componentWillMount() {
|
||||
const tutorialStep = PreferenceStore.getInt(Preferences.TUTORIAL_STEP, UserStore.getCurrentId(), 999);
|
||||
|
||||
@@ -308,6 +322,7 @@ class CreatePost extends React.Component {
|
||||
showTutorialTip: tutorialStep === TutorialSteps.POST_POPOVER
|
||||
});
|
||||
}
|
||||
|
||||
componentDidMount() {
|
||||
ChannelStore.addChangeListener(this.onChange);
|
||||
PreferenceStore.addChangeListener(this.onPreferenceChange);
|
||||
@@ -315,11 +330,13 @@ class CreatePost extends React.Component {
|
||||
this.focusTextbox();
|
||||
document.addEventListener('keydown', this.showShortcuts);
|
||||
}
|
||||
|
||||
componentDidUpdate(prevProps, prevState) {
|
||||
if (prevState.channelId !== this.state.channelId) {
|
||||
this.focusTextbox();
|
||||
}
|
||||
}
|
||||
|
||||
componentWillUnmount() {
|
||||
ChannelStore.removeChangeListener(this.onChange);
|
||||
PreferenceStore.removeChangeListener(this.onPreferenceChange);
|
||||
@@ -342,6 +359,7 @@ class CreatePost extends React.Component {
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
onChange() {
|
||||
const channelId = ChannelStore.getCurrentId();
|
||||
if (this.state.channelId !== channelId) {
|
||||
@@ -350,6 +368,7 @@ class CreatePost extends React.Component {
|
||||
this.setState({channelId, messageText: draft.messageText, initialText: draft.messageText, submitting: false, serverError: null, postError: null, previews: draft.previews, uploadsInProgress: draft.uploadsInProgress});
|
||||
}
|
||||
}
|
||||
|
||||
onPreferenceChange() {
|
||||
const tutorialStep = PreferenceStore.getInt(Preferences.TUTORIAL_STEP, UserStore.getCurrentId(), 999);
|
||||
this.setState({
|
||||
@@ -358,6 +377,7 @@ class CreatePost extends React.Component {
|
||||
centerTextbox: PreferenceStore.get(Preferences.CATEGORY_DISPLAY_SETTINGS, Preferences.CHANNEL_DISPLAY_MODE, Preferences.CHANNEL_DISPLAY_MODE_DEFAULT) === Preferences.CHANNEL_DISPLAY_MODE_CENTERED
|
||||
});
|
||||
}
|
||||
|
||||
getFileCount(channelId) {
|
||||
if (channelId === this.state.channelId) {
|
||||
return this.state.previews.length + this.state.uploadsInProgress.length;
|
||||
@@ -366,6 +386,7 @@ class CreatePost extends React.Component {
|
||||
const draft = PostStore.getDraft(channelId);
|
||||
return draft.previews.length + draft.uploadsInProgress.length;
|
||||
}
|
||||
|
||||
handleKeyDown(e) {
|
||||
if (this.state.ctrlSend && e.keyCode === KeyCodes.ENTER && e.ctrlKey === true) {
|
||||
this.postMsgKeyPress(e);
|
||||
@@ -394,30 +415,29 @@ class CreatePost extends React.Component {
|
||||
});
|
||||
}
|
||||
|
||||
if ((e.ctrlKey || e.metaKey) && !e.altKey && !e.shiftKey && e.keyCode === KeyCodes.UP) {
|
||||
const channelId = ChannelStore.getCurrentId();
|
||||
const lastPost = PostStore.getCurrentUsersLatestPost(channelId);
|
||||
if (!lastPost || this.state.messageText !== '') {
|
||||
return;
|
||||
if ((e.ctrlKey || e.metaKey) && !e.altKey && !e.shiftKey && (e.keyCode === Constants.KeyCodes.UP || e.keyCode === Constants.KeyCodes.DOWN)) {
|
||||
const lastMessage = MessageHistoryStore.nextMessageInHistory(e.keyCode, this.state.messageText, 'post');
|
||||
if (lastMessage !== null) {
|
||||
e.preventDefault();
|
||||
this.setState({
|
||||
messageText: lastMessage
|
||||
});
|
||||
}
|
||||
e.preventDefault();
|
||||
let message = lastPost.message;
|
||||
if (this.state.lastMessage !== '') {
|
||||
message = this.state.lastMessage;
|
||||
}
|
||||
this.setState({messageText: message});
|
||||
}
|
||||
}
|
||||
|
||||
showPostDeletedModal() {
|
||||
this.setState({
|
||||
showPostDeletedModal: true
|
||||
});
|
||||
}
|
||||
|
||||
hidePostDeletedModal() {
|
||||
this.setState({
|
||||
showPostDeletedModal: false
|
||||
});
|
||||
}
|
||||
|
||||
createTutorialTip() {
|
||||
const screens = [];
|
||||
|
||||
@@ -438,6 +458,7 @@ class CreatePost extends React.Component {
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
render() {
|
||||
let serverError = null;
|
||||
if (this.state.serverError) {
|
||||
|
||||
@@ -9,6 +9,7 @@ import * as GlobalActions from 'actions/global_actions.jsx';
|
||||
import Textbox from './textbox.jsx';
|
||||
import BrowserStore from 'stores/browser_store.jsx';
|
||||
import PostStore from 'stores/post_store.jsx';
|
||||
import MessageHistoryStore from 'stores/message_history_store.jsx';
|
||||
import PreferenceStore from 'stores/preference_store.jsx';
|
||||
|
||||
import Constants from 'utils/constants.jsx';
|
||||
@@ -49,6 +50,8 @@ class EditPostModal extends React.Component {
|
||||
return;
|
||||
}
|
||||
|
||||
MessageHistoryStore.storeMessageInHistory(updatedPost.message);
|
||||
|
||||
if (updatedPost.message.length === 0) {
|
||||
var tempState = this.state;
|
||||
Reflect.deleteProperty(tempState, 'editText');
|
||||
|
||||
79
webapp/stores/message_history_store.jsx
Normal file
79
webapp/stores/message_history_store.jsx
Normal file
@@ -0,0 +1,79 @@
|
||||
// Copyright (c) 2016 Mattermost, Inc. All Rights Reserved.
|
||||
// See License.txt for license information.
|
||||
|
||||
import Constants from 'utils/constants.jsx';
|
||||
|
||||
const TYPE_POST = 'post';
|
||||
const TYPE_COMMENT = 'comment';
|
||||
|
||||
class MessageHistoryStoreClass {
|
||||
constructor() {
|
||||
this.messageHistory = [];
|
||||
this.index = [];
|
||||
this.index[TYPE_POST] = 0;
|
||||
this.index[TYPE_COMMENT] = 0;
|
||||
}
|
||||
|
||||
getMessageInHistory(type) {
|
||||
if (this.index[type] >= this.messageHistory.length) {
|
||||
return '';
|
||||
} else if (this.index[type] < 0) {
|
||||
return null;
|
||||
}
|
||||
|
||||
return this.messageHistory[this.index[type]];
|
||||
}
|
||||
|
||||
getHistoryLength() {
|
||||
if (this.messageHistory === null) {
|
||||
return 0;
|
||||
}
|
||||
return this.messageHistory.length;
|
||||
}
|
||||
|
||||
storeMessageInHistory(message) {
|
||||
this.messageHistory.push(message);
|
||||
this.resetAllHistoryIndex();
|
||||
if (this.messageHistory.length > Constants.MAX_PREV_MSGS) {
|
||||
this.messageHistory = this.messageHistory.slice(1, Constants.MAX_PREV_MSGS + 1);
|
||||
}
|
||||
}
|
||||
|
||||
storeMessageInHistoryByIndex(index, message) {
|
||||
this.messageHistory[index] = message;
|
||||
}
|
||||
|
||||
resetAllHistoryIndex() {
|
||||
this.index[TYPE_POST] = this.messageHistory.length;
|
||||
this.index[TYPE_COMMENT] = this.messageHistory.length;
|
||||
}
|
||||
|
||||
resetHistoryIndex(type) {
|
||||
this.index[type] = this.messageHistory.length;
|
||||
}
|
||||
|
||||
nextMessageInHistory(keyCode, messageText, type) {
|
||||
if (messageText !== '' && messageText !== this.getMessageInHistory(type)) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (keyCode === Constants.KeyCodes.UP) {
|
||||
this.index[type]--;
|
||||
} else if (keyCode === Constants.KeyCodes.DOWN) {
|
||||
this.index[type]++;
|
||||
}
|
||||
|
||||
if (this.index[type] < 0) {
|
||||
this.index[type] = 0;
|
||||
return null;
|
||||
} else if (this.index[type] >= this.getHistoryLength()) {
|
||||
this.index[type] = this.getHistoryLength();
|
||||
}
|
||||
|
||||
return this.getMessageInHistory(type);
|
||||
}
|
||||
}
|
||||
|
||||
var MessageHistoryStore = new MessageHistoryStoreClass();
|
||||
|
||||
export default MessageHistoryStore;
|
||||
@@ -752,6 +752,7 @@ export default {
|
||||
MHPNS: 'https://push.mattermost.com',
|
||||
MTPNS: 'http://push-test.mattermost.com',
|
||||
BOT_NAME: 'BOT',
|
||||
MAX_PREV_MSGS: 100,
|
||||
POST_COLLAPSE_TIMEOUT: 1000 * 60 * 5, // five minutes
|
||||
LICENSE_EXPIRY_NOTIFICATION: 1000 * 60 * 60 * 24 * 15, // 15 days
|
||||
LICENSE_GRACE_PERIOD: 1000 * 60 * 60 * 24 * 15 // 15 days
|
||||
|
||||
@@ -2,7 +2,6 @@
|
||||
// See License.txt for license information.
|
||||
|
||||
import Client from 'utils/web_client.jsx';
|
||||
|
||||
import Constants from 'utils/constants.jsx';
|
||||
|
||||
export function isSystemMessage(post) {
|
||||
|
||||
Reference in New Issue
Block a user