Files
mattermost/web/react/components/create_post.jsx

280 lines
10 KiB
React
Raw Normal View History

2015-06-14 23:53:32 -08:00
// Copyright (c) 2015 Spinpunch, Inc. All Rights Reserved.
// See License.txt for license information.
var AppDispatcher = require('../dispatcher/app_dispatcher.jsx');
var client = require('../utils/client.jsx');
var AsyncClient = require('../utils/async_client.jsx');
var ChannelStore = require('../stores/channel_store.jsx');
var PostStore = require('../stores/post_store.jsx');
var UserStore = require('../stores/user_store.jsx');
var SocketStore = require('../stores/socket_store.jsx');
var MsgTyping = require('./msg_typing.jsx');
var Textbox = require('./textbox.jsx');
var FileUpload = require('./file_upload.jsx');
var FilePreview = require('./file_preview.jsx');
var utils = require('../utils/utils.jsx');
var Constants = require('../utils/constants.jsx');
var ActionTypes = Constants.ActionTypes;
module.exports = React.createClass({
lastTime: 0,
handleSubmit: function(e) {
e.preventDefault();
if (this.state.uploadsInProgress.length > 0 || this.state.submitting) {
return;
}
2015-06-14 23:53:32 -08:00
var post = {};
post.filenames = [];
post.message = this.state.messageText;
2015-06-18 17:39:12 -04:00
if (post.message.trim().length === 0 && this.state.previews.length === 0) {
2015-06-14 23:53:32 -08:00
return;
}
if (post.message.length > Constants.CHARACTER_LIMIT) {
this.setState({postError: 'Post length must be less than ' + Constants.CHARACTER_LIMIT + ' characters.'});
2015-06-14 23:53:32 -08:00
return;
}
this.setState({submitting: true, serverError: null});
2015-06-14 23:53:32 -08:00
if (post.message.indexOf('/') === 0) {
2015-06-14 23:53:32 -08:00
client.executeCommand(
this.state.channelId,
2015-06-14 23:53:32 -08:00
post.message,
false,
function(data) {
PostStore.storeDraft(data.channel_id, null);
this.setState({messageText: '', submitting: false, postError: null, previews: [], serverError: null});
2015-06-14 23:53:32 -08:00
if (data.goto_location.length > 0) {
window.location.href = data.goto_location;
}
}.bind(this),
function(err) {
var state = {};
state.serverError = err.message;
2015-06-14 23:53:32 -08:00
state.submitting = false;
this.setState(state);
}.bind(this)
);
} else {
post.channel_id = this.state.channelId;
2015-06-14 23:53:32 -08:00
post.filenames = this.state.previews;
client.createPost(post, ChannelStore.getCurrent(),
function(data) {
PostStore.storeDraft(data.channel_id, null);
this.setState({messageText: '', submitting: false, postError: null, previews: [], serverError: null});
2015-06-14 23:53:32 -08:00
this.resizePostHolder();
AsyncClient.getPosts(true);
var channel = ChannelStore.get(this.state.channelId);
var member = ChannelStore.getMember(this.state.channelId);
2015-06-14 23:53:32 -08:00
member.msg_count = channel.total_msg_count;
member.last_viewed_at = Date.now();
2015-06-14 23:53:32 -08:00
ChannelStore.setChannelMember(member);
}.bind(this),
function(err) {
var state = {};
state.serverError = err.message;
2015-06-14 23:53:32 -08:00
state.submitting = false;
this.setState(state);
}.bind(this)
);
}
$('.post-list-holder-by-time').perfectScrollbar('update');
2015-06-14 23:53:32 -08:00
},
componentDidUpdate: function() {
this.resizePostHolder();
},
postMsgKeyPress: function(e) {
if (e.which === 13 && !e.shiftKey && !e.altKey) {
2015-06-14 23:53:32 -08:00
e.preventDefault();
this.refs.textbox.getDOMNode().blur();
this.handleSubmit(e);
}
var t = Date.now();
2015-06-14 23:53:32 -08:00
if ((t - this.lastTime) > 5000) {
SocketStore.sendMessage({channelId: this.state.channelId, action: 'typing', props: {'parent_id': ''}, state: {}});
2015-06-14 23:53:32 -08:00
this.lastTime = t;
}
},
handleUserInput: function(messageText) {
this.resizePostHolder();
this.setState({messageText: messageText});
2015-06-14 23:53:32 -08:00
var draft = PostStore.getCurrentDraft();
draft['message'] = messageText;
PostStore.storeCurrentDraft(draft);
},
resizePostHolder: function() {
var height = $(window).height() - $(this.refs.topDiv.getDOMNode()).height() - $('#error_bar').outerHeight() - 50;
$('.post-list-holder-by-time').css('height', height + 'px');
2015-06-14 23:53:32 -08:00
$(window).trigger('resize');
},
handleUploadStart: function(clientIds, channelId) {
var draft = PostStore.getDraft(channelId);
2015-06-14 23:53:32 -08:00
draft['uploadsInProgress'] = draft['uploadsInProgress'].concat(clientIds);
PostStore.storeDraft(channelId, draft);
2015-06-14 23:53:32 -08:00
this.setState({uploadsInProgress: draft['uploadsInProgress']});
},
handleFileUploadComplete: function(filenames, clientIds, channelId) {
var draft = PostStore.getDraft(channelId);
2015-06-14 23:53:32 -08:00
// remove each finished file from uploads
for (var i = 0; i < clientIds.length; i++) {
var index = draft['uploadsInProgress'].indexOf(clientIds[i]);
if (index !== -1) {
draft['uploadsInProgress'].splice(index, 1);
}
2015-06-14 23:53:32 -08:00
}
draft['previews'] = draft['previews'].concat(filenames);
PostStore.storeDraft(channelId, draft);
this.setState({uploadsInProgress: draft['uploadsInProgress'], previews: draft['previews']});
2015-06-14 23:53:32 -08:00
},
handleUploadError: function(err, clientId) {
var draft = PostStore.getDraft(this.state.channelId);
var index = draft['uploadsInProgress'].indexOf(clientId);
if (index !== -1) {
draft['uploadsInProgress'].splice(index, 1);
}
PostStore.storeDraft(this.state.channelId, draft);
this.setState({uploadsInProgress: draft['uploadsInProgress'], serverError: err});
2015-06-14 23:53:32 -08:00
},
removePreview: function(id) {
2015-06-14 23:53:32 -08:00
var previews = this.state.previews;
var uploadsInProgress = this.state.uploadsInProgress;
// id can either be the path of an uploaded file or the client id of an in progress upload
var index = previews.indexOf(id);
if (index !== -1) {
previews.splice(index, 1);
} else {
index = uploadsInProgress.indexOf(id);
if (index !== -1) {
uploadsInProgress.splice(index, 1);
this.refs.fileUpload.cancelUpload(id);
2015-06-14 23:53:32 -08:00
}
}
2015-06-14 23:53:32 -08:00
var draft = PostStore.getCurrentDraft();
draft['previews'] = previews;
draft['uploadsInProgress'] = uploadsInProgress;
2015-06-14 23:53:32 -08:00
PostStore.storeCurrentDraft(draft);
this.setState({previews: previews, uploadsInProgress: uploadsInProgress});
2015-06-14 23:53:32 -08:00
},
componentDidMount: function() {
ChannelStore.addChangeListener(this._onChange);
this.resizePostHolder();
},
componentWillUnmount: function() {
ChannelStore.removeChangeListener(this._onChange);
},
_onChange: function() {
var channelId = ChannelStore.getCurrentId();
if (this.state.channelId !== channelId) {
2015-06-14 23:53:32 -08:00
var draft = PostStore.getCurrentDraft();
this.setState({
channelId: channelId, messageText: draft['message'], initialText: draft['message'], submitting: false,
serverError: null, postError: null, previews: draft['previews'], uploadsInProgress: draft['uploadsInProgress']
});
2015-06-14 23:53:32 -08:00
}
},
getInitialState: function() {
PostStore.clearDraftUploads();
var draft = PostStore.getCurrentDraft();
return {
channelId: ChannelStore.getCurrentId(), messageText: draft['message'], uploadsInProgress: draft['uploadsInProgress'],
previews: draft['previews'], submitting: false, initialText: draft['message']
};
2015-06-14 23:53:32 -08:00
},
getFileCount: function(channelId) {
if (channelId === this.state.channelId) {
return this.state.previews.length + this.state.uploadsInProgress.length;
} else {
var draft = PostStore.getDraft(channelId);
return draft['previews'].length + draft['uploadsInProgress'].length;
2015-06-14 23:53:32 -08:00
}
},
render: function() {
var serverError = null;
if (this.state.serverError) {
serverError = (
<div className='has-error'>
<label className='control-label'>{this.state.serverError}</label>
</div>
);
}
2015-06-14 23:53:32 -08:00
var postError = null;
if (this.state.postError) {
postError = <label className='control-label'>{this.state.postError}</label>;
}
var preview = null;
if (this.state.previews.length > 0 || this.state.uploadsInProgress.length > 0) {
2015-06-14 23:53:32 -08:00
preview = (
<FilePreview
files={this.state.previews}
onRemove={this.removePreview}
uploadsInProgress={this.state.uploadsInProgress} />
);
}
var postFooterClassName = 'post-create-footer';
if (postError) {
postFooterClassName += ' has-error';
}
2015-06-14 23:53:32 -08:00
return (
<form id='create_post' ref='topDiv' role='form' onSubmit={this.handleSubmit}>
<div className='post-create'>
<div className='post-create-body'>
2015-06-14 23:53:32 -08:00
<Textbox
onUserInput={this.handleUserInput}
onKeyPress={this.postMsgKeyPress}
messageText={this.state.messageText}
createMessage='Write a message...'
channelId={this.state.channelId}
id='post_textbox'
ref='textbox' />
2015-06-14 23:53:32 -08:00
<FileUpload
ref='fileUpload'
getFileCount={this.getFileCount}
onUploadStart={this.handleUploadStart}
onFileUpload={this.handleFileUploadComplete}
onUploadError={this.handleUploadError}
postType='post'
channelId='' />
2015-06-14 23:53:32 -08:00
</div>
<div className={postFooterClassName}>
{postError}
{serverError}
{preview}
<MsgTyping channelId={this.state.channelId} parentId=''/>
2015-06-14 23:53:32 -08:00
</div>
</div>
</form>
);
}
});