forked from cybrespace/pinafore
		
	
		
			
				
	
	
		
			128 lines
		
	
	
		
			No EOL
		
	
	
		
			4.3 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			128 lines
		
	
	
		
			No EOL
		
	
	
		
			4.3 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
| <div class="compose-autosuggest {{shown ? 'shown' : ''}}"
 | |
|      aria-hidden="true" >
 | |
|   <ComposeAutosuggestionList
 | |
|     items="{{searchResults}}"
 | |
|     on:click="onUserSelected(event)"
 | |
|     :selected
 | |
|   />
 | |
| </div>
 | |
| <style>
 | |
|   .compose-autosuggest {
 | |
|     opacity: 0;
 | |
|     pointer-events: none;
 | |
|     transition: opacity 0.1s linear;
 | |
|     min-width: 400px;
 | |
|     max-width: calc(100vw - 20px);
 | |
|   }
 | |
|   .compose-autosuggest.shown {
 | |
|     pointer-events: auto;
 | |
|     opacity: 1
 | |
|   }
 | |
| 
 | |
|   @media (max-width: 479px) {
 | |
|     .compose-autosuggest {
 | |
|       /* hack: move this over to the left on mobile so it's easier to see */
 | |
|       transform: translateX(-58px); /* avatar size 48px + 10px padding */
 | |
|       width: calc(100vw - 20px);
 | |
|     }
 | |
|   }
 | |
| 
 | |
| </style>
 | |
| <script>
 | |
|   import { store } from '../../_store/store'
 | |
|   import { database } from '../../_database/database'
 | |
|   import { insertUsername } from '../../_actions/compose'
 | |
|   import { scheduleIdleTask } from '../../_utils/scheduleIdleTask'
 | |
|   import { once } from '../../_utils/once'
 | |
|   import ComposeAutosuggestionList from './ComposeAutosuggestionList.html'
 | |
| 
 | |
|   const SEARCH_RESULTS_LIMIT = 4
 | |
|   const DATABASE_SEARCH_RESULTS_LIMIT = 30
 | |
|   const MIN_PREFIX_LENGTH = 1
 | |
|   const SEARCH_REGEX = new RegExp(`(?:\\s|^)(@\\S{${MIN_PREFIX_LENGTH},})$`)
 | |
| 
 | |
|   export default {
 | |
|     oncreate() {
 | |
|       // perf improves for input responsiveness
 | |
|       this.observe('composeSelectionStart', () => {
 | |
|         scheduleIdleTask(() => {
 | |
|           this.set({composeSelectionStartDeferred: this.get('composeSelectionStart')})
 | |
|         })
 | |
|       })
 | |
|       this.observe('composeFocused', (composeFocused) => {
 | |
|         let updateFocusedState = () => {
 | |
|           scheduleIdleTask(() => {
 | |
|             this.set({composeFocusedDeferred: this.get('composeFocused')})
 | |
|           })
 | |
|         }
 | |
| 
 | |
|         // TODO: hack so that when the user clicks the button, and the textarea blurs,
 | |
|         // we don't immediately hide the dropdown which would cause the click to get lost
 | |
|         if (composeFocused) {
 | |
|           updateFocusedState()
 | |
|         } else {
 | |
|           Promise.race([
 | |
|             new Promise(resolve => setTimeout(resolve, 200)),
 | |
|             new Promise(resolve => this.once('userSelected', resolve))
 | |
|           ]).then(updateFocusedState)
 | |
|         }
 | |
|       })
 | |
|       this.observe('searchText', async searchText => {
 | |
|         if (!searchText) {
 | |
|           return
 | |
|         }
 | |
|         let results = await this.search(searchText)
 | |
|         this.store.set({
 | |
|           composeAutosuggestionSelected: 0,
 | |
|           composeAutosuggestionSearchText: searchText,
 | |
|           composeAutosuggestionSearchResults: results
 | |
|         })
 | |
|       })
 | |
|       this.observe('shown', shown => {
 | |
|         this.store.set({composeAutosuggestionShown: shown})
 | |
|       })
 | |
|     },
 | |
|     methods: {
 | |
|       once: once,
 | |
|       onUserSelected(account) {
 | |
|         this.fire('userSelected')
 | |
|         let realm = this.get('realm')
 | |
|         let selectionStart = this.store.get('composeSelectionStart')
 | |
|         let searchText = this.store.get('composeAutosuggestionSearchText')
 | |
|         let startIndex = selectionStart - searchText.length
 | |
|         let endIndex = selectionStart
 | |
|         /* no await */ insertUsername(realm, account.acct, startIndex, endIndex)
 | |
|       },
 | |
|       async search(searchText) {
 | |
|         let currentInstance = this.store.get('currentInstance')
 | |
|         let results = await database.searchAccountsByUsername(
 | |
|           currentInstance, searchText.substring(1), DATABASE_SEARCH_RESULTS_LIMIT)
 | |
|         return results.slice(0, SEARCH_RESULTS_LIMIT)
 | |
|       }
 | |
|     },
 | |
|     computed: {
 | |
|       composeSelectionStart: ($composeSelectionStart) => $composeSelectionStart,
 | |
|       composeFocused: ($composeFocused) => $composeFocused,
 | |
|       searchResults: ($composeAutosuggestionSearchResults) => $composeAutosuggestionSearchResults || [],
 | |
|       selected: ($composeAutosuggestionSelected) => $composeAutosuggestionSelected || 0,
 | |
|       searchText: (text, composeSelectionStartDeferred) => {
 | |
|         let selectionStart = composeSelectionStartDeferred || 0
 | |
|         if (!text || selectionStart < MIN_PREFIX_LENGTH) {
 | |
|           return
 | |
|         }
 | |
| 
 | |
|         let match = text.substring(0, selectionStart).match(SEARCH_REGEX)
 | |
|         return match && match[1]
 | |
|       },
 | |
|       shown: (composeFocusedDeferred, searchText, searchResults) => {
 | |
|         return !!(composeFocusedDeferred &&
 | |
|           searchText &&
 | |
|           searchResults.length)
 | |
|       }
 | |
|     },
 | |
|     store: () => store,
 | |
|     components: {
 | |
|       ComposeAutosuggestionList
 | |
|     }
 | |
|   }
 | |
| </script> |