The code powering m.abunchtell.com https://m.abunchtell.com
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

154 line
4.4 KiB

  1. import AutosuggestAccountContainer from '../features/compose/containers/autosuggest_account_container';
  2. import ImmutablePropTypes from 'react-immutable-proptypes';
  3. const textAtCursorMatchesToken = (str, caretPosition) => {
  4. let word;
  5. let left = str.slice(0, caretPosition).search(/\S+$/);
  6. let right = str.slice(caretPosition).search(/\s/);
  7. if (right < 0) {
  8. word = str.slice(left);
  9. } else {
  10. word = str.slice(left, right + caretPosition);
  11. }
  12. if (!word || word.trim().length < 2 || word[0] !== '@') {
  13. return [null, null];
  14. }
  15. word = word.trim().toLowerCase().slice(1);
  16. if (word.length > 0) {
  17. return [left + 1, word];
  18. } else {
  19. return [null, null];
  20. }
  21. };
  22. const AutosuggestTextarea = React.createClass({
  23. propTypes: {
  24. value: React.PropTypes.string,
  25. suggestions: ImmutablePropTypes.list,
  26. disabled: React.PropTypes.bool,
  27. placeholder: React.PropTypes.string,
  28. onSuggestionSelected: React.PropTypes.func.isRequired,
  29. onSuggestionsClearRequested: React.PropTypes.func.isRequired,
  30. onSuggestionsFetchRequested: React.PropTypes.func.isRequired,
  31. onChange: React.PropTypes.func.isRequired
  32. },
  33. getInitialState () {
  34. return {
  35. suggestionsHidden: false,
  36. selectedSuggestion: 0,
  37. lastToken: null,
  38. tokenStart: 0
  39. };
  40. },
  41. onChange (e) {
  42. const [ tokenStart, token ] = textAtCursorMatchesToken(e.target.value, e.target.selectionStart);
  43. if (token != null && this.state.lastToken !== token) {
  44. this.setState({ lastToken: token, selectedSuggestion: 0, tokenStart });
  45. this.props.onSuggestionsFetchRequested(token);
  46. } else if (token === null && this.state.lastToken != null) {
  47. this.setState({ lastToken: null });
  48. this.props.onSuggestionsClearRequested();
  49. }
  50. this.props.onChange(e);
  51. },
  52. onKeyDown (e) {
  53. const { suggestions, disabled } = this.props;
  54. const { selectedSuggestion, suggestionsHidden } = this.state;
  55. if (disabled) {
  56. e.preventDefault();
  57. return;
  58. }
  59. switch(e.key) {
  60. case 'Escape':
  61. if (!suggestionsHidden) {
  62. e.preventDefault();
  63. this.setState({ suggestionsHidden: true });
  64. }
  65. break;
  66. case 'ArrowDown':
  67. if (suggestions.size > 0 && !suggestionsHidden) {
  68. e.preventDefault();
  69. this.setState({ selectedSuggestion: Math.min(selectedSuggestion + 1, suggestions.size - 1) });
  70. }
  71. break;
  72. case 'ArrowUp':
  73. if (suggestions.size > 0 && !suggestionsHidden) {
  74. e.preventDefault();
  75. this.setState({ selectedSuggestion: Math.max(selectedSuggestion - 1, 0) });
  76. }
  77. break;
  78. case 'Enter':
  79. case 'Tab':
  80. // Select suggestion
  81. if (this.state.lastToken != null && suggestions.size > 0 && !suggestionsHidden) {
  82. e.preventDefault();
  83. e.stopPropagation();
  84. this.props.onSuggestionSelected(this.state.tokenStart, this.state.lastToken, suggestions.get(selectedSuggestion));
  85. }
  86. break;
  87. }
  88. },
  89. onSuggestionClick (suggestion, e) {
  90. e.preventDefault();
  91. this.props.onSuggestionSelected(this.state.tokenStart, this.state.lastToken, suggestion);
  92. },
  93. componentWillReceiveProps (nextProps) {
  94. if (nextProps.suggestions !== this.props.suggestions && nextProps.suggestions.size > 0 && this.state.suggestionsHidden) {
  95. this.setState({ suggestionsHidden: false });
  96. }
  97. },
  98. setTextarea (c) {
  99. this.textarea = c;
  100. },
  101. render () {
  102. const { value, suggestions, disabled, placeholder } = this.props;
  103. const { suggestionsHidden, selectedSuggestion } = this.state;
  104. return (
  105. <div className='autosuggest-textarea'>
  106. <textarea
  107. ref={this.setTextarea}
  108. className='autosuggest-textarea__textarea'
  109. disabled={disabled}
  110. placeholder={placeholder}
  111. value={value}
  112. onChange={this.onChange}
  113. onKeyDown={this.onKeyDown}
  114. />
  115. <div style={{ display: (suggestions.size > 0 && !suggestionsHidden) ? 'block' : 'none' }} className='autosuggest-textarea__suggestions'>
  116. {suggestions.map((suggestion, i) => (
  117. <div key={suggestion} className={`autosuggest-textarea__suggestions__item ${i === selectedSuggestion ? 'selected' : ''}`} onClick={this.onSuggestionClick.bind(this, suggestion)}>
  118. <AutosuggestAccountContainer id={suggestion} />
  119. </div>
  120. ))}
  121. </div>
  122. </div>
  123. );
  124. }
  125. });
  126. export default AutosuggestTextarea;