mirror of
https://github.com/Foundry376/Mailspring.git
synced 2024-11-11 18:32:20 +08:00
313fdc2fd0
If that method is fired immediately twice in a row, it's possible for this._set to not re-instantiate causing the results to be blow away
195 lines
5.3 KiB
JavaScript
195 lines
5.3 KiB
JavaScript
import _ from 'underscore'
|
|
import {
|
|
Actions,
|
|
NylasAPI,
|
|
Thread,
|
|
DatabaseStore,
|
|
ComponentRegistry,
|
|
FocusedContentStore,
|
|
MutableQuerySubscription,
|
|
} from 'nylas-exports'
|
|
import SearchActions from './search-actions'
|
|
|
|
const {LongConnectionStatus} = NylasAPI
|
|
|
|
|
|
class SearchQuerySubscription extends MutableQuerySubscription {
|
|
|
|
constructor(searchQuery, accountIds) {
|
|
super(null, {emitResultSet: true})
|
|
this._searchQuery = searchQuery
|
|
this._accountIds = accountIds
|
|
|
|
this.resetData()
|
|
|
|
this._connections = []
|
|
this._unsubscribers = [
|
|
FocusedContentStore.listen(::this.onFocusedContentChanged),
|
|
]
|
|
this._extDisposables = []
|
|
|
|
_.defer(() => this.performSearch())
|
|
}
|
|
|
|
replaceRange = () => {
|
|
// TODO
|
|
}
|
|
|
|
resetData() {
|
|
this._searchStartedAt = null
|
|
this._resultsReceivedAt = null
|
|
this._firstThreadSelectedAt = null
|
|
this._lastFocusedThread = null
|
|
this._focusedThreadCount = 0
|
|
}
|
|
|
|
performSearch() {
|
|
this._searchStartedAt = Date.now()
|
|
|
|
this.performLocalSearch()
|
|
this.performRemoteSearch()
|
|
this.performExtensionSearch()
|
|
}
|
|
|
|
performLocalSearch() {
|
|
let dbQuery = DatabaseStore.findAll(Thread).distinct()
|
|
if (this._accountIds.length === 1) {
|
|
dbQuery = dbQuery.where({accountId: this._accountIds[0]})
|
|
}
|
|
dbQuery = dbQuery
|
|
.search(this._searchQuery)
|
|
.order(Thread.attributes.lastMessageReceivedTimestamp.descending())
|
|
.limit(100)
|
|
|
|
dbQuery.then((results) => {
|
|
if (results.length > 0) {
|
|
this.replaceQuery(dbQuery)
|
|
}
|
|
})
|
|
}
|
|
|
|
_addThreadIdsToSearch(ids = []) {
|
|
const currentResults = this._set && this._set.ids().length > 0;
|
|
let searchIds = ids;
|
|
if (currentResults) {
|
|
const currentResultIds = this._set.ids()
|
|
searchIds = _.uniq(currentResultIds.concat(ids))
|
|
}
|
|
const dbQuery = (
|
|
DatabaseStore.findAll(Thread)
|
|
.where({id: searchIds})
|
|
.order(Thread.attributes.lastMessageReceivedTimestamp.descending())
|
|
)
|
|
this.replaceQuery(dbQuery)
|
|
}
|
|
|
|
performRemoteSearch() {
|
|
const accountsSearched = new Set()
|
|
let resultIds = []
|
|
|
|
const allAccountsSearched = () => accountsSearched.size === this._accountIds.length
|
|
const resultsReturned = () => {
|
|
// Don't emit a "result" until we have at least one thread to display.
|
|
// Otherwise it will show "No Results Found"
|
|
if (resultIds.length > 0 || allAccountsSearched()) {
|
|
this._addThreadIdsToSearch(resultIds)
|
|
}
|
|
}
|
|
|
|
this._connections = this._accountIds.map((accountId) => {
|
|
return NylasAPI.startLongConnection({
|
|
accountId,
|
|
path: `/threads/search/streaming?q=${encodeURIComponent(this._searchQuery)}`,
|
|
onResults: (results) => {
|
|
if (!this._resultsReceivedAt) {
|
|
this._resultsReceivedAt = Date.now()
|
|
}
|
|
const threads = results[0]
|
|
resultIds = resultIds.concat(_.pluck(threads, 'id'))
|
|
resultsReturned()
|
|
},
|
|
onStatusChanged: (status) => {
|
|
const hasClosed = [
|
|
LongConnectionStatus.Closed,
|
|
LongConnectionStatus.Ended,
|
|
].includes(status)
|
|
|
|
if (hasClosed) {
|
|
accountsSearched.add(accountId)
|
|
if (allAccountsSearched()) {
|
|
SearchActions.searchCompleted()
|
|
}
|
|
}
|
|
},
|
|
})
|
|
})
|
|
}
|
|
|
|
performExtensionSearch() {
|
|
const searchExtensions = ComponentRegistry.findComponentsMatching({
|
|
role: "SearchBarResults",
|
|
})
|
|
|
|
this._extDisposables = searchExtensions.map((ext) => {
|
|
return ext.observeThreadIdsForQuery(this._searchQuery)
|
|
.subscribe((ids = []) => {
|
|
const allIds = _.compact(_.flatten(ids))
|
|
if (allIds.length === 0) return;
|
|
this._addThreadIdsToSearch(allIds)
|
|
})
|
|
})
|
|
}
|
|
|
|
onFocusedContentChanged() {
|
|
const thread = FocusedContentStore.focused('thread')
|
|
const shouldRecordChange = (
|
|
thread &&
|
|
(this._lastFocusedThread || {}).id !== thread.id
|
|
)
|
|
if (shouldRecordChange) {
|
|
if (this._focusedThreadCount === 0) {
|
|
this._firstThreadSelectedAt = Date.now()
|
|
}
|
|
this._focusedThreadCount += 1
|
|
this._lastFocusedThread = thread
|
|
}
|
|
}
|
|
|
|
reportSearchMetrics() {
|
|
if (!this._searchStartedAt) {
|
|
return;
|
|
}
|
|
|
|
let timeToFirstServerResults = null;
|
|
let timeToFirstThreadSelected = null;
|
|
const timeInsideSearch = Math.round((Date.now() - this._searchStartedAt) / 1000)
|
|
const numItems = this._focusedThreadCount
|
|
const didSelectAnyThreads = numItems > 0
|
|
|
|
if (this._firstThreadSelectedAt) {
|
|
timeToFirstThreadSelected = Math.round((this._firstThreadSelectedAt - this._searchStartedAt) / 1000)
|
|
}
|
|
if (this._resultsReceivedAt) {
|
|
timeToFirstServerResults = Math.round((this._resultsReceivedAt - this._searchStartedAt) / 1000)
|
|
}
|
|
|
|
const data = {
|
|
numItems,
|
|
timeInsideSearch,
|
|
didSelectAnyThreads,
|
|
timeToFirstServerResults,
|
|
timeToFirstThreadSelected,
|
|
}
|
|
Actions.recordUserEvent("Search Performed", data)
|
|
this.resetData()
|
|
}
|
|
|
|
onLastCallbackRemoved() {
|
|
this.reportSearchMetrics();
|
|
this._connections.forEach((conn) => conn.end())
|
|
this._unsubscribers.forEach((unsub) => unsub())
|
|
this._extDisposables.forEach((disposable) => disposable.dispose())
|
|
}
|
|
}
|
|
|
|
export default SearchQuerySubscription
|