faefd8ec8f
* Reorder javascript English locale file * Reorder translation string in order of the locale key. * Add javascript English locale missing language keys * Search all javascript language keys by command: `grep -REho '<FormattedMessage .*\/>' ./app/assets/javascripts/.` * Add all the missing language keys and their values to `en.jsx`. * Add javascript English locale missing language keys (2) * Find all `defineMessages` calls with this command: `grep -Rl 'defineMessages({.*' ./app/assets/javascripts/.` * Open all these files. Find the language key (`id`) in these statements. * Add all the missing language keys and their values to `en.jsx`. * Remove javascript English locale obsoleted language keys * Find all language keys that no longer exists in the source code and remove them. The removed keys include: * "compose_form.private" * "compose_form.unlisted" * "getting_started.about_addressing" * "getting_started.about_shortcuts" * "notification.mention" * "search.account" * "search.hashtag" * "tabs_bar.mentions" * "tabs_bar.public" * Javascript English locale file add note * Add notes to contributors about the English translation files. Hope that will make translation process smoother. * Update javascript locale defaultValue in code * Update the defaultValue in code according to the relevant translation in English locale file.
211 lines
8.1 KiB
JavaScript
211 lines
8.1 KiB
JavaScript
import CharacterCounter from './character_counter';
|
|
import Button from '../../../components/button';
|
|
import PureRenderMixin from 'react-addons-pure-render-mixin';
|
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
|
import ReplyIndicatorContainer from '../containers/reply_indicator_container';
|
|
import AutosuggestTextarea from '../../../components/autosuggest_textarea';
|
|
import { debounce } from 'react-decoration';
|
|
import UploadButtonContainer from '../containers/upload_button_container';
|
|
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
|
import Toggle from 'react-toggle';
|
|
import Collapsable from '../../../components/collapsable';
|
|
import SpoilerButtonContainer from '../containers/spoiler_button_container';
|
|
import PrivacyDropdownContainer from '../containers/privacy_dropdown_container';
|
|
import SensitiveButtonContainer from '../containers/sensitive_button_container';
|
|
import EmojiPickerDropdown from './emoji_picker_dropdown';
|
|
import UploadFormContainer from '../containers/upload_form_container';
|
|
import TextIconButton from './text_icon_button';
|
|
|
|
const messages = defineMessages({
|
|
placeholder: { id: 'compose_form.placeholder', defaultMessage: 'What is on your mind?' },
|
|
spoiler_placeholder: { id: 'compose_form.spoiler_placeholder', defaultMessage: 'Content warning' },
|
|
publish: { id: 'compose_form.publish', defaultMessage: 'Toot' }
|
|
});
|
|
|
|
const ComposeForm = React.createClass({
|
|
|
|
propTypes: {
|
|
intl: React.PropTypes.object.isRequired,
|
|
text: React.PropTypes.string.isRequired,
|
|
suggestion_token: React.PropTypes.string,
|
|
suggestions: ImmutablePropTypes.list,
|
|
spoiler: React.PropTypes.bool,
|
|
privacy: React.PropTypes.string,
|
|
spoiler_text: React.PropTypes.string,
|
|
focusDate: React.PropTypes.instanceOf(Date),
|
|
preselectDate: React.PropTypes.instanceOf(Date),
|
|
is_submitting: React.PropTypes.bool,
|
|
is_uploading: React.PropTypes.bool,
|
|
me: React.PropTypes.number,
|
|
needsPrivacyWarning: React.PropTypes.bool,
|
|
mentionedDomains: React.PropTypes.array.isRequired,
|
|
onChange: React.PropTypes.func.isRequired,
|
|
onSubmit: React.PropTypes.func.isRequired,
|
|
onClearSuggestions: React.PropTypes.func.isRequired,
|
|
onFetchSuggestions: React.PropTypes.func.isRequired,
|
|
onSuggestionSelected: React.PropTypes.func.isRequired,
|
|
onChangeSpoilerText: React.PropTypes.func.isRequired,
|
|
onPaste: React.PropTypes.func.isRequired,
|
|
onPickEmoji: React.PropTypes.func.isRequired
|
|
},
|
|
|
|
mixins: [PureRenderMixin],
|
|
|
|
handleChange (e) {
|
|
this.props.onChange(e.target.value);
|
|
},
|
|
|
|
handleKeyDown (e) {
|
|
if (e.keyCode === 13 && (e.ctrlKey || e.metaKey)) {
|
|
this.props.onSubmit();
|
|
}
|
|
},
|
|
|
|
handleSubmit () {
|
|
this.props.onSubmit();
|
|
},
|
|
|
|
onSuggestionsClearRequested () {
|
|
this.props.onClearSuggestions();
|
|
},
|
|
|
|
@debounce(500)
|
|
onSuggestionsFetchRequested (token) {
|
|
this.props.onFetchSuggestions(token);
|
|
},
|
|
|
|
onSuggestionSelected (tokenStart, token, value) {
|
|
this._restoreCaret = null;
|
|
this.props.onSuggestionSelected(tokenStart, token, value);
|
|
},
|
|
|
|
handleChangeSpoilerText (e) {
|
|
this.props.onChangeSpoilerText(e.target.value);
|
|
},
|
|
|
|
componentWillReceiveProps (nextProps) {
|
|
// If this is the update where we've finished uploading,
|
|
// save the last caret position so we can restore it below!
|
|
if (!nextProps.is_uploading && this.props.is_uploading) {
|
|
this._restoreCaret = this.autosuggestTextarea.textarea.selectionStart;
|
|
}
|
|
},
|
|
|
|
componentDidUpdate (prevProps) {
|
|
// This statement does several things:
|
|
// - If we're beginning a reply, and,
|
|
// - Replying to zero or one users, places the cursor at the end of the textbox.
|
|
// - Replying to more than one user, selects any usernames past the first;
|
|
// this provides a convenient shortcut to drop everyone else from the conversation.
|
|
// - If we've just finished uploading an image, and have a saved caret position,
|
|
// restores the cursor to that position after the text changes!
|
|
if (this.props.focusDate !== prevProps.focusDate || (prevProps.is_uploading && !this.props.is_uploading && typeof this._restoreCaret === 'number')) {
|
|
let selectionEnd, selectionStart;
|
|
|
|
if (this.props.preselectDate !== prevProps.preselectDate) {
|
|
selectionEnd = this.props.text.length;
|
|
selectionStart = this.props.text.search(/\s/) + 1;
|
|
} else if (typeof this._restoreCaret === 'number') {
|
|
selectionStart = this._restoreCaret;
|
|
selectionEnd = this._restoreCaret;
|
|
} else {
|
|
selectionEnd = this.props.text.length;
|
|
selectionStart = selectionEnd;
|
|
}
|
|
|
|
this.autosuggestTextarea.textarea.setSelectionRange(selectionStart, selectionEnd);
|
|
this.autosuggestTextarea.textarea.focus();
|
|
}
|
|
},
|
|
|
|
setAutosuggestTextarea (c) {
|
|
this.autosuggestTextarea = c;
|
|
},
|
|
|
|
handleEmojiPick (data) {
|
|
const position = this.autosuggestTextarea.textarea.selectionStart;
|
|
this._restoreCaret = position + data.shortname.length + 1;
|
|
this.props.onPickEmoji(position, data);
|
|
},
|
|
|
|
render () {
|
|
const { intl, needsPrivacyWarning, mentionedDomains, onPaste } = this.props;
|
|
const disabled = this.props.is_submitting;
|
|
|
|
let publishText = '';
|
|
let privacyWarning = '';
|
|
let reply_to_other = false;
|
|
|
|
if (needsPrivacyWarning) {
|
|
privacyWarning = (
|
|
<div className='compose-form__warning'>
|
|
<FormattedMessage
|
|
id='compose_form.privacy_disclaimer'
|
|
defaultMessage='Your private status will be delivered to mentioned users on {domains}. Do you trust {domainsCount, plural, one {that server} other {those servers}} to not leak your status?'
|
|
values={{ domains: <strong>{mentionedDomains.join(', ')}</strong>, domainsCount: mentionedDomains.length }}
|
|
/>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (this.props.privacy === 'private' || this.props.privacy === 'direct') {
|
|
publishText = <span><i className='fa fa-lock' /> {intl.formatMessage(messages.publish)}</span>;
|
|
} else {
|
|
publishText = intl.formatMessage(messages.publish) + (this.props.privacy !== 'unlisted' ? '!' : '');
|
|
}
|
|
|
|
return (
|
|
<div style={{ padding: '10px' }}>
|
|
<Collapsable isVisible={this.props.spoiler} fullHeight={50}>
|
|
<div className="spoiler-input">
|
|
<input placeholder={intl.formatMessage(messages.spoiler_placeholder)} value={this.props.spoiler_text} onChange={this.handleChangeSpoilerText} onKeyDown={this.handleKeyDown} type="text" className="spoiler-input__input" />
|
|
</div>
|
|
</Collapsable>
|
|
|
|
{privacyWarning}
|
|
|
|
<ReplyIndicatorContainer />
|
|
|
|
<div style={{ position: 'relative' }}>
|
|
<AutosuggestTextarea
|
|
ref={this.setAutosuggestTextarea}
|
|
placeholder={intl.formatMessage(messages.placeholder)}
|
|
disabled={disabled}
|
|
value={this.props.text}
|
|
onChange={this.handleChange}
|
|
suggestions={this.props.suggestions}
|
|
onKeyDown={this.handleKeyDown}
|
|
onSuggestionsFetchRequested={this.onSuggestionsFetchRequested}
|
|
onSuggestionsClearRequested={this.onSuggestionsClearRequested}
|
|
onSuggestionSelected={this.onSuggestionSelected}
|
|
onPaste={onPaste}
|
|
/>
|
|
|
|
<EmojiPickerDropdown onPickEmoji={this.handleEmojiPick} />
|
|
</div>
|
|
|
|
<div className='compose-form__modifiers'>
|
|
<UploadFormContainer />
|
|
</div>
|
|
|
|
<div style={{ display: 'flex', justifyContent: 'space-between' }}>
|
|
<div className='compose-form__buttons'>
|
|
<UploadButtonContainer />
|
|
<PrivacyDropdownContainer />
|
|
<SensitiveButtonContainer />
|
|
<SpoilerButtonContainer />
|
|
</div>
|
|
|
|
<div style={{ display: 'flex' }}>
|
|
<div style={{ paddingTop: '10px', marginRight: '16px', lineHeight: '36px' }}><CharacterCounter max={500} text={[this.props.spoiler_text, this.props.text].join('')} /></div>
|
|
<div style={{ paddingTop: '10px' }}><Button text={publishText} onClick={this.handleSubmit} disabled={disabled} /></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
});
|
|
|
|
export default injectIntl(ComposeForm);
|