The code powering m.abunchtell.com https://m.abunchtell.com
Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.
 
 
 
 

198 linhas
5.5 KiB

  1. import AutosuggestAccountContainer from '../features/compose/containers/autosuggest_account_container';
  2. import ImmutablePropTypes from 'react-immutable-proptypes';
  3. import { isRtl } from '../rtl';
  4. const textAtCursorMatchesToken = (str, caretPosition) => {
  5. let word;
  6. let left = str.slice(0, caretPosition).search(/\S+$/);
  7. let right = str.slice(caretPosition).search(/\s/);
  8. if (right < 0) {
  9. word = str.slice(left);
  10. } else {
  11. word = str.slice(left, right + caretPosition);
  12. }
  13. if (!word || word.trim().length < 2 || word[0] !== '@') {
  14. return [null, null];
  15. }
  16. word = word.trim().toLowerCase().slice(1);
  17. if (word.length > 0) {
  18. return [left + 1, word];
  19. } else {
  20. return [null, null];
  21. }
  22. };
  23. const AutosuggestTextarea = React.createClass({
  24. propTypes: {
  25. value: React.PropTypes.string,
  26. suggestions: ImmutablePropTypes.list,
  27. disabled: React.PropTypes.bool,
  28. placeholder: React.PropTypes.string,
  29. onSuggestionSelected: React.PropTypes.func.isRequired,
  30. onSuggestionsClearRequested: React.PropTypes.func.isRequired,
  31. onSuggestionsFetchRequested: React.PropTypes.func.isRequired,
  32. onChange: React.PropTypes.func.isRequired,
  33. onKeyUp: React.PropTypes.func,
  34. onKeyDown: React.PropTypes.func,
  35. onPaste: React.PropTypes.func.isRequired,
  36. },
  37. getInitialState () {
  38. return {
  39. suggestionsHidden: false,
  40. selectedSuggestion: 0,
  41. lastToken: null,
  42. tokenStart: 0
  43. };
  44. },
  45. onChange (e) {
  46. const [ tokenStart, token ] = textAtCursorMatchesToken(e.target.value, e.target.selectionStart);
  47. if (token !== null && this.state.lastToken !== token) {
  48. this.setState({ lastToken: token, selectedSuggestion: 0, tokenStart });
  49. this.props.onSuggestionsFetchRequested(token);
  50. } else if (token === null) {
  51. this.setState({ lastToken: null });
  52. this.props.onSuggestionsClearRequested();
  53. }
  54. this.props.onChange(e);
  55. },
  56. onKeyDown (e) {
  57. const { suggestions, disabled } = this.props;
  58. const { selectedSuggestion, suggestionsHidden } = this.state;
  59. if (disabled) {
  60. e.preventDefault();
  61. return;
  62. }
  63. switch(e.key) {
  64. case 'Escape':
  65. if (!suggestionsHidden) {
  66. e.preventDefault();
  67. this.setState({ suggestionsHidden: true });
  68. }
  69. break;
  70. case 'ArrowDown':
  71. if (suggestions.size > 0 && !suggestionsHidden) {
  72. e.preventDefault();
  73. this.setState({ selectedSuggestion: Math.min(selectedSuggestion + 1, suggestions.size - 1) });
  74. }
  75. break;
  76. case 'ArrowUp':
  77. if (suggestions.size > 0 && !suggestionsHidden) {
  78. e.preventDefault();
  79. this.setState({ selectedSuggestion: Math.max(selectedSuggestion - 1, 0) });
  80. }
  81. break;
  82. case 'Enter':
  83. case 'Tab':
  84. // Select suggestion
  85. if (this.state.lastToken !== null && suggestions.size > 0 && !suggestionsHidden) {
  86. e.preventDefault();
  87. e.stopPropagation();
  88. this.props.onSuggestionSelected(this.state.tokenStart, this.state.lastToken, suggestions.get(selectedSuggestion));
  89. }
  90. break;
  91. }
  92. if (e.defaultPrevented || !this.props.onKeyDown) {
  93. return;
  94. }
  95. this.props.onKeyDown(e);
  96. },
  97. onBlur () {
  98. // If we hide the suggestions immediately, then this will prevent the
  99. // onClick for the suggestions themselves from firing.
  100. // Setting a short window for that to take place before hiding the
  101. // suggestions ensures that can't happen.
  102. setTimeout(() => {
  103. this.setState({ suggestionsHidden: true });
  104. }, 100);
  105. },
  106. onSuggestionClick (suggestion, e) {
  107. e.preventDefault();
  108. this.props.onSuggestionSelected(this.state.tokenStart, this.state.lastToken, suggestion);
  109. this.textarea.focus();
  110. },
  111. componentWillReceiveProps (nextProps) {
  112. if (nextProps.suggestions !== this.props.suggestions && nextProps.suggestions.size > 0 && this.state.suggestionsHidden) {
  113. this.setState({ suggestionsHidden: false });
  114. }
  115. },
  116. setTextarea (c) {
  117. this.textarea = c;
  118. },
  119. onPaste (e) {
  120. if (e.clipboardData && e.clipboardData.files.length === 1) {
  121. this.props.onPaste(e.clipboardData.files)
  122. e.preventDefault();
  123. }
  124. },
  125. render () {
  126. const { value, suggestions, disabled, placeholder, onKeyUp } = this.props;
  127. const { suggestionsHidden, selectedSuggestion } = this.state;
  128. const className = 'autosuggest-textarea__textarea';
  129. const style = { direction: 'ltr' };
  130. if (isRtl(value)) {
  131. style.direction = 'rtl';
  132. }
  133. return (
  134. <div className='autosuggest-textarea'>
  135. <textarea
  136. ref={this.setTextarea}
  137. className={className}
  138. disabled={disabled}
  139. placeholder={placeholder}
  140. autoFocus={true}
  141. value={value}
  142. onChange={this.onChange}
  143. onKeyDown={this.onKeyDown}
  144. onKeyUp={onKeyUp}
  145. onBlur={this.onBlur}
  146. onPaste={this.onPaste}
  147. style={style}
  148. />
  149. <div style={{ display: (suggestions.size > 0 && !suggestionsHidden) ? 'block' : 'none' }} className='autosuggest-textarea__suggestions'>
  150. {suggestions.map((suggestion, i) => (
  151. <div
  152. role='button'
  153. tabIndex='0'
  154. key={suggestion}
  155. className={`autosuggest-textarea__suggestions__item ${i === selectedSuggestion ? 'selected' : ''}`}
  156. onClick={this.onSuggestionClick.bind(this, suggestion)}>
  157. <AutosuggestAccountContainer id={suggestion} />
  158. </div>
  159. ))}
  160. </div>
  161. </div>
  162. );
  163. }
  164. });
  165. export default AutosuggestTextarea;