|  |  |  | @ -17,9 +17,13 @@@@ -17,9 +17,13 @@ | 
			
		
	
		
			
				
					|  |  |  |  | // DEALINGS IN THE SOFTWARE.
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | using System; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Collections.Concurrent; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Collections.Generic; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Collections.ObjectModel; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Collections.Specialized; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Diagnostics; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Threading; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Threading.Tasks; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Windows; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Windows.Controls; | 
			
		
	
		
			
				
					|  |  |  |  | using System.Windows.Input; | 
			
		
	
	
		
			
				
					|  |  |  | @ -36,8 +40,18 @@ namespace ICSharpCode.ILSpy@@ -36,8 +40,18 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 	/// </summary>
 | 
			
		
	
		
			
				
					|  |  |  |  | 	public partial class SearchPane : UserControl, IPane | 
			
		
	
		
			
				
					|  |  |  |  | 	{ | 
			
		
	
		
			
				
					|  |  |  |  | 		const int MAX_RESULTS = 1000; | 
			
		
	
		
			
				
					|  |  |  |  | 		const int MAX_REFRESH_TIME_MS = 10; // More means quicker forward of data, less means better responsibility
 | 
			
		
	
		
			
				
					|  |  |  |  | 		static SearchPane instance; | 
			
		
	
		
			
				
					|  |  |  |  | 		RunningSearch currentSearch; | 
			
		
	
		
			
				
					|  |  |  |  | 		bool runSearchOnNextShow; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		public static readonly DependencyProperty ResultsProperty = | 
			
		
	
		
			
				
					|  |  |  |  | 			DependencyProperty.Register("Results", typeof(ObservableCollection<SearchResult>), typeof(SearchPane),  | 
			
		
	
		
			
				
					|  |  |  |  | 				new PropertyMetadata(new ObservableCollection<SearchResult>())); | 
			
		
	
		
			
				
					|  |  |  |  | 		public ObservableCollection<SearchResult> Results { | 
			
		
	
		
			
				
					|  |  |  |  | 			get { return (ObservableCollection<SearchResult>)GetValue(ResultsProperty); } | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		public static SearchPane Instance { | 
			
		
	
		
			
				
					|  |  |  |  | 			get { | 
			
		
	
	
		
			
				
					|  |  |  | @ -61,14 +75,14 @@ namespace ICSharpCode.ILSpy@@ -61,14 +75,14 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 			searchModeComboBox.Items.Add(new { Image = Images.Event, Name = "Event" }); | 
			
		
	
		
			
				
					|  |  |  |  | 			searchModeComboBox.Items.Add(new { Image = Images.Literal, Name = "Constant" }); | 
			
		
	
		
			
				
					|  |  |  |  | 			searchModeComboBox.Items.Add(new { Image = Images.Library, Name = "Metadata Token" }); | 
			
		
	
		
			
				
					|  |  |  |  | 			searchModeComboBox.SelectedIndex = (int)MainWindow.Instance.SessionSettings.SelectedSearchMode; | 
			
		
	
		
			
				
					|  |  |  |  | 			searchModeComboBox.SelectionChanged += (sender, e) => MainWindow.Instance.SessionSettings.SelectedSearchMode = (SearchMode)searchModeComboBox.SelectedIndex; | 
			
		
	
		
			
				
					|  |  |  |  | 			ContextMenuProvider.Add(listBox); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			ContextMenuProvider.Add(listBox); | 
			
		
	
		
			
				
					|  |  |  |  | 			MainWindow.Instance.CurrentAssemblyListChanged += MainWindow_Instance_CurrentAssemblyListChanged; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 			CompositionTarget.Rendering += UpdateResults; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		bool runSearchOnNextShow; | 
			
		
	
		
			
				
					|  |  |  |  | 			// This starts empty search right away, so do at the end (we're still in ctor)
 | 
			
		
	
		
			
				
					|  |  |  |  | 			searchModeComboBox.SelectedIndex = (int)MainWindow.Instance.SessionSettings.SelectedSearchMode; | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		void MainWindow_Instance_CurrentAssemblyListChanged(object sender, NotifyCollectionChangedEventArgs e) | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
	
		
			
				
					|  |  |  | @ -114,26 +128,10 @@ namespace ICSharpCode.ILSpy@@ -114,26 +128,10 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		void SearchModeComboBox_SelectionChanged(object sender, SelectionChangedEventArgs e) | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			MainWindow.Instance.SessionSettings.SelectedSearchMode = (SearchMode)searchModeComboBox.SelectedIndex; | 
			
		
	
		
			
				
					|  |  |  |  | 			StartSearch(this.SearchTerm); | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		void StartSearch(string searchTerm) | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			if (currentSearch != null) { | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch.Cancel(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			if (string.IsNullOrEmpty(searchTerm)) { | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch = null; | 
			
		
	
		
			
				
					|  |  |  |  | 				listBox.ItemsSource = null; | 
			
		
	
		
			
				
					|  |  |  |  | 			} else { | 
			
		
	
		
			
				
					|  |  |  |  | 				MainWindow mainWindow = MainWindow.Instance; | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch = new RunningSearch(mainWindow.CurrentAssemblyList.GetAssemblies(), searchTerm, | 
			
		
	
		
			
				
					|  |  |  |  | 					(SearchMode)searchModeComboBox.SelectedIndex, mainWindow.CurrentLanguage); | 
			
		
	
		
			
				
					|  |  |  |  | 				listBox.ItemsSource = currentSearch.Results; | 
			
		
	
		
			
				
					|  |  |  |  | 				new Thread(currentSearch.Run).Start(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		 | 
			
		
	
		
			
				
					|  |  |  |  | 		void IPane.Closed() | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			this.SearchTerm = string.Empty; | 
			
		
	
	
		
			
				
					|  |  |  | @ -150,14 +148,10 @@ namespace ICSharpCode.ILSpy@@ -150,14 +148,10 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 			if (e.Key == Key.Return) { | 
			
		
	
		
			
				
					|  |  |  |  | 				e.Handled = true; | 
			
		
	
		
			
				
					|  |  |  |  | 				JumpToSelectedItem(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		 | 
			
		
	
		
			
				
					|  |  |  |  | 		void JumpToSelectedItem() | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			SearchResult result = listBox.SelectedItem as SearchResult; | 
			
		
	
		
			
				
					|  |  |  |  | 			if (result != null) { | 
			
		
	
		
			
				
					|  |  |  |  | 				MainWindow.Instance.JumpToReference(result.Member); | 
			
		
	
		
			
				
					|  |  |  |  | 			} else if(e.Key == Key.Up && listBox.SelectedIndex == 0) { | 
			
		
	
		
			
				
					|  |  |  |  | 				e.Handled = true; | 
			
		
	
		
			
				
					|  |  |  |  | 				listBox.SelectedIndex = -1; | 
			
		
	
		
			
				
					|  |  |  |  | 				searchBox.Focus(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -185,26 +179,94 @@ namespace ICSharpCode.ILSpy@@ -185,26 +179,94 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		void UpdateResults(object sender, EventArgs e) | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			if (currentSearch == null) | 
			
		
	
		
			
				
					|  |  |  |  | 				return; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			var timer = Stopwatch.StartNew(); | 
			
		
	
		
			
				
					|  |  |  |  | 			int resultsAdded = 0; | 
			
		
	
		
			
				
					|  |  |  |  | 			while (Results.Count < MAX_RESULTS && timer.ElapsedMilliseconds < MAX_REFRESH_TIME_MS && currentSearch.resultQueue.TryTake(out var result)) { | 
			
		
	
		
			
				
					|  |  |  |  | 				InsertResult(Results, result); | 
			
		
	
		
			
				
					|  |  |  |  | 				++resultsAdded; | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			if (resultsAdded > 0 && Results.Count == MAX_RESULTS) { | 
			
		
	
		
			
				
					|  |  |  |  | 				Results.Add(new SearchResult { Name = "Search aborted, more than 1000 results found." }); | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch.Cancel(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		async void StartSearch(string searchTerm) | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			if (currentSearch != null) { | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch.Cancel(); | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch = null; | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			Results.Clear(); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			RunningSearch startedSearch = null; | 
			
		
	
		
			
				
					|  |  |  |  | 			if (!string.IsNullOrEmpty(searchTerm)) { | 
			
		
	
		
			
				
					|  |  |  |  | 				MainWindow mainWindow = MainWindow.Instance; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 				searchProgressBar.IsIndeterminate = true; | 
			
		
	
		
			
				
					|  |  |  |  | 				startedSearch = new RunningSearch(mainWindow.CurrentAssemblyList.GetAssemblies(), searchTerm, | 
			
		
	
		
			
				
					|  |  |  |  | 					(SearchMode)searchModeComboBox.SelectedIndex, mainWindow.CurrentLanguage); | 
			
		
	
		
			
				
					|  |  |  |  | 				currentSearch = startedSearch; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 				await startedSearch.Run(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			if (currentSearch == startedSearch) { //are we still running the same search
 | 
			
		
	
		
			
				
					|  |  |  |  | 				searchProgressBar.IsIndeterminate = false; | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		void InsertResult(IList<SearchResult> results, SearchResult result) | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			if (results.Count == 0) { | 
			
		
	
		
			
				
					|  |  |  |  | 				results.Add(result); | 
			
		
	
		
			
				
					|  |  |  |  | 			} else if (Options.DisplaySettingsPanel.CurrentDisplaySettings.SortResults) { | 
			
		
	
		
			
				
					|  |  |  |  | 				// Keep results collection sorted by "Fitness" by inserting result into correct place
 | 
			
		
	
		
			
				
					|  |  |  |  | 				// Inserts in the beginning shifts all elements, but there can be no more than 1000 items.
 | 
			
		
	
		
			
				
					|  |  |  |  | 				for (int i = 0; i < results.Count; i++) { | 
			
		
	
		
			
				
					|  |  |  |  | 					if (results[i].Fitness < result.Fitness) { | 
			
		
	
		
			
				
					|  |  |  |  | 						results.Insert(i, result); | 
			
		
	
		
			
				
					|  |  |  |  | 						return; | 
			
		
	
		
			
				
					|  |  |  |  | 					} | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				results.Insert(results.Count - 1, result); | 
			
		
	
		
			
				
					|  |  |  |  | 			} else { | 
			
		
	
		
			
				
					|  |  |  |  | 				// Original Code
 | 
			
		
	
		
			
				
					|  |  |  |  | 				int index = results.BinarySearch(result, 0, results.Count - 1, SearchResult.Comparer); | 
			
		
	
		
			
				
					|  |  |  |  | 				results.Insert(index < 0 ? ~index : index, result); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 		void JumpToSelectedItem() | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			if (listBox.SelectedItem is SearchResult result) { | 
			
		
	
		
			
				
					|  |  |  |  | 				MainWindow.Instance.JumpToReference(result.Member); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 		 | 
			
		
	
		
			
				
					|  |  |  |  | 		sealed class RunningSearch | 
			
		
	
		
			
				
					|  |  |  |  | 		{ | 
			
		
	
		
			
				
					|  |  |  |  | 			readonly Dispatcher dispatcher; | 
			
		
	
		
			
				
					|  |  |  |  | 			readonly CancellationTokenSource cts = new CancellationTokenSource(); | 
			
		
	
		
			
				
					|  |  |  |  | 			readonly LoadedAssembly[] assemblies; | 
			
		
	
		
			
				
					|  |  |  |  | 			readonly string[] searchTerm; | 
			
		
	
		
			
				
					|  |  |  |  | 			readonly SearchMode searchMode; | 
			
		
	
		
			
				
					|  |  |  |  | 			readonly Language language; | 
			
		
	
		
			
				
					|  |  |  |  | 			public readonly ObservableCollection<SearchResult> Results = new ObservableCollection<SearchResult>(); | 
			
		
	
		
			
				
					|  |  |  |  | 			int resultCount; | 
			
		
	
		
			
				
					|  |  |  |  | 			public readonly IProducerConsumerCollection<SearchResult> resultQueue = new ConcurrentQueue<SearchResult>();  | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			public RunningSearch(LoadedAssembly[] assemblies, string searchTerm, SearchMode searchMode, Language language) | 
			
		
	
		
			
				
					|  |  |  |  | 			{ | 
			
		
	
		
			
				
					|  |  |  |  | 				this.dispatcher = Dispatcher.CurrentDispatcher; | 
			
		
	
		
			
				
					|  |  |  |  | 				this.assemblies = assemblies; | 
			
		
	
		
			
				
					|  |  |  |  | 				this.searchTerm = NativeMethods.CommandLineToArgumentArray(searchTerm); | 
			
		
	
		
			
				
					|  |  |  |  | 				this.language = language; | 
			
		
	
		
			
				
					|  |  |  |  | 				this.searchMode = searchMode; | 
			
		
	
		
			
				
					|  |  |  |  | 				 | 
			
		
	
		
			
				
					|  |  |  |  | 				this.Results.Add(new SearchResult { Name = "Searching..." }); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			 | 
			
		
	
		
			
				
					|  |  |  |  | 			public void Cancel() | 
			
		
	
	
		
			
				
					|  |  |  | @ -212,114 +274,75 @@ namespace ICSharpCode.ILSpy@@ -212,114 +274,75 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 				cts.Cancel(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			 | 
			
		
	
		
			
				
					|  |  |  |  | 			public void Run() | 
			
		
	
		
			
				
					|  |  |  |  | 			public async Task Run() | 
			
		
	
		
			
				
					|  |  |  |  | 			{ | 
			
		
	
		
			
				
					|  |  |  |  | 				await Task.Factory.StartNew(() => { | 
			
		
	
		
			
				
					|  |  |  |  | 					var searcher = GetSearchStrategy(); | 
			
		
	
		
			
				
					|  |  |  |  | 					try { | 
			
		
	
		
			
				
					|  |  |  |  | 					var searcher = GetSearchStrategy(searchMode, searchTerm); | 
			
		
	
		
			
				
					|  |  |  |  | 					// TODO : parallelize
 | 
			
		
	
		
			
				
					|  |  |  |  | 						foreach (var loadedAssembly in assemblies) { | 
			
		
	
		
			
				
					|  |  |  |  | 							var module = loadedAssembly.GetPEFileOrNull(); | 
			
		
	
		
			
				
					|  |  |  |  | 							if (module == null) | 
			
		
	
		
			
				
					|  |  |  |  | 								continue; | 
			
		
	
		
			
				
					|  |  |  |  | 						CancellationToken cancellationToken = cts.Token; | 
			
		
	
		
			
				
					|  |  |  |  | 						searcher.Search(module); | 
			
		
	
		
			
				
					|  |  |  |  | 							searcher.Search(module, cts.Token); | 
			
		
	
		
			
				
					|  |  |  |  | 						} | 
			
		
	
		
			
				
					|  |  |  |  | 					} catch (OperationCanceledException) { | 
			
		
	
		
			
				
					|  |  |  |  | 						// ignore cancellation
 | 
			
		
	
		
			
				
					|  |  |  |  | 					} | 
			
		
	
		
			
				
					|  |  |  |  | 				// remove the 'Searching...' entry
 | 
			
		
	
		
			
				
					|  |  |  |  | 				dispatcher.BeginInvoke( | 
			
		
	
		
			
				
					|  |  |  |  | 					DispatcherPriority.Normal, | 
			
		
	
		
			
				
					|  |  |  |  | 					new Action(delegate { this.Results.RemoveAt(this.Results.Count - 1); })); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 			 | 
			
		
	
		
			
				
					|  |  |  |  | 			void AddResult(SearchResult result) | 
			
		
	
		
			
				
					|  |  |  |  | 			{ | 
			
		
	
		
			
				
					|  |  |  |  | 				if (++resultCount == 1000) { | 
			
		
	
		
			
				
					|  |  |  |  | 					result = new SearchResult { Name = "Search aborted, more than 1000 results found." }; | 
			
		
	
		
			
				
					|  |  |  |  | 					cts.Cancel(); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				dispatcher.BeginInvoke( | 
			
		
	
		
			
				
					|  |  |  |  | 					DispatcherPriority.Normal, | 
			
		
	
		
			
				
					|  |  |  |  | 					new Action(delegate { InsertResult(this.Results, result); })); | 
			
		
	
		
			
				
					|  |  |  |  | 				cts.Token.ThrowIfCancellationRequested(); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			void InsertResult(ObservableCollection<SearchResult> results, SearchResult result) | 
			
		
	
		
			
				
					|  |  |  |  | 			{ | 
			
		
	
		
			
				
					|  |  |  |  | 				if (Options.DisplaySettingsPanel.CurrentDisplaySettings.SortResults) | 
			
		
	
		
			
				
					|  |  |  |  | 				{ | 
			
		
	
		
			
				
					|  |  |  |  | 					// Keep results collection sorted by "Fitness" by inserting result into correct place
 | 
			
		
	
		
			
				
					|  |  |  |  | 					// Inserts in the beginning shifts all elements, but there can be no more than 1000 items.
 | 
			
		
	
		
			
				
					|  |  |  |  | 					for (int i = 0; i < results.Count; i++) | 
			
		
	
		
			
				
					|  |  |  |  | 					{ | 
			
		
	
		
			
				
					|  |  |  |  | 						if (results[i].Fitness < result.Fitness) | 
			
		
	
		
			
				
					|  |  |  |  | 						{ | 
			
		
	
		
			
				
					|  |  |  |  | 							results.Insert(i, result); | 
			
		
	
		
			
				
					|  |  |  |  | 							return; | 
			
		
	
		
			
				
					|  |  |  |  | 						} | 
			
		
	
		
			
				
					|  |  |  |  | 					} | 
			
		
	
		
			
				
					|  |  |  |  | 					results.Insert(results.Count - 1, result); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				else | 
			
		
	
		
			
				
					|  |  |  |  | 				{ | 
			
		
	
		
			
				
					|  |  |  |  | 					// Original Code
 | 
			
		
	
		
			
				
					|  |  |  |  | 					int index = results.BinarySearch(result, 0, results.Count - 1, SearchResult.Comparer); | 
			
		
	
		
			
				
					|  |  |  |  | 					results.Insert(index < 0 ? ~index : index, result); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 				}, cts.Token, TaskCreationOptions.LongRunning, TaskScheduler.Current).ConfigureAwait(false); | 
			
		
	
		
			
				
					|  |  |  |  | 			} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 			AbstractSearchStrategy GetSearchStrategy(SearchMode mode, string[] terms) | 
			
		
	
		
			
				
					|  |  |  |  | 			AbstractSearchStrategy GetSearchStrategy() | 
			
		
	
		
			
				
					|  |  |  |  | 			{ | 
			
		
	
		
			
				
					|  |  |  |  | 				if (terms.Length == 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("tm:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(3)); | 
			
		
	
		
			
				
					|  |  |  |  | 				if (searchTerm.Length == 1) { | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("tm:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(3), resultQueue); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("t:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(2), MemberSearchKind.Type); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("t:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(2), resultQueue, MemberSearchKind.Type); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("m:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(2), MemberSearchKind.Member); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("m:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(2), resultQueue, MemberSearchKind.Member); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("md:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(3), MemberSearchKind.Method); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("md:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(3), resultQueue, MemberSearchKind.Method); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("f:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(2), MemberSearchKind.Field); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("f:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(2), resultQueue, MemberSearchKind.Field); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("p:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(2), MemberSearchKind.Property); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("p:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(2), resultQueue, MemberSearchKind.Property); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("e:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms[0].Substring(2), MemberSearchKind.Event); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("e:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, searchTerm[0].Substring(2), resultQueue, MemberSearchKind.Event); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("c:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new LiteralSearchStrategy(language, AddResult, terms[0].Substring(2)); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("c:", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new LiteralSearchStrategy(language, resultQueue, searchTerm[0].Substring(2)); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 					if (terms[0].StartsWith("@", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MetadataTokenSearchStrategy(language, AddResult, terms[0].Substring(1)); | 
			
		
	
		
			
				
					|  |  |  |  | 					if (searchTerm[0].StartsWith("@", StringComparison.Ordinal)) | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MetadataTokenSearchStrategy(language, resultQueue, searchTerm[0].Substring(1)); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 				switch (mode) | 
			
		
	
		
			
				
					|  |  |  |  | 				switch (searchMode) | 
			
		
	
		
			
				
					|  |  |  |  | 				{ | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.TypeAndMember: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Type: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms, MemberSearchKind.Type); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm, MemberSearchKind.Type); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Member: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms, MemberSearchKind.Member); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm, MemberSearchKind.Member); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Literal: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new LiteralSearchStrategy(language, AddResult, terms); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new LiteralSearchStrategy(language, resultQueue, searchTerm); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Method: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms, MemberSearchKind.Method); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm, MemberSearchKind.Method); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Field: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms, MemberSearchKind.Field); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm, MemberSearchKind.Field); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Property: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms, MemberSearchKind.Property); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm, MemberSearchKind.Property); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Event: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, AddResult, terms, MemberSearchKind.Event); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MemberSearchStrategy(language, resultQueue, searchTerm, MemberSearchKind.Event); | 
			
		
	
		
			
				
					|  |  |  |  | 					case SearchMode.Token: | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MetadataTokenSearchStrategy(language, AddResult, terms); | 
			
		
	
		
			
				
					|  |  |  |  | 						return new MetadataTokenSearchStrategy(language, resultQueue, searchTerm); | 
			
		
	
		
			
				
					|  |  |  |  | 				} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 				return null; | 
			
		
	
	
		
			
				
					|  |  |  | @ -327,7 +350,7 @@ namespace ICSharpCode.ILSpy@@ -327,7 +350,7 @@ namespace ICSharpCode.ILSpy | 
			
		
	
		
			
				
					|  |  |  |  | 		} | 
			
		
	
		
			
				
					|  |  |  |  | 	} | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | 	sealed class SearchResult : IMemberTreeNode | 
			
		
	
		
			
				
					|  |  |  |  | 	public sealed class SearchResult : IMemberTreeNode | 
			
		
	
		
			
				
					|  |  |  |  | 	{ | 
			
		
	
		
			
				
					|  |  |  |  | 		public static readonly System.Collections.Generic.IComparer<SearchResult> Comparer = new SearchResultComparer(); | 
			
		
	
		
			
				
					|  |  |  |  | 		 | 
			
		
	
	
		
			
				
					|  |  |  | 
 |