PowerToys/Wox/ViewModel/ResultsViewModel.cs

317 lines
9.5 KiB
C#
Raw Normal View History

using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Collections.Specialized;
using System.ComponentModel;
using System.Linq;
using System.Windows;
using Wox.Core.UserSettings;
using Wox.Plugin;
using Wox.Storage;
namespace Wox.ViewModel
{
public class ResultsViewModel : BaseViewModel
{
#region Private Fields
private ResultViewModel _selectedResult;
public ResultCollection Results { get; } = new ResultCollection();
private Thickness _margin;
private readonly object _resultsUpdateLock = new object();
#endregion
#region ViewModel Properties
public int MaxHeight => UserSettingStorage.Instance.MaxResultsToShow * 50;
public ResultViewModel SelectedResult
{
get
{
return _selectedResult;
}
set
{
if (value != null)
{
if (_selectedResult != null)
{
_selectedResult.IsSelected = false;
}
_selectedResult = value;
if (_selectedResult != null)
{
_selectedResult.IsSelected = true;
}
}
OnPropertyChanged();
}
}
public Thickness Margin
{
get
{
return _margin;
}
set
{
_margin = value;
OnPropertyChanged();
}
}
#endregion
#region Private Methods
private bool IsTopMostResult(Result result)
{
return TopMostRecordStorage.Instance.IsTopMost(result);
}
private int InsertIndexOf(int newScore, IList<ResultViewModel> list)
{
int index = 0;
for (; index < list.Count; index++)
{
var result = list[index];
if (newScore > result.RawResult.Score)
{
break;
}
}
return index;
}
#endregion
#region Public Methods
2016-02-12 14:21:12 +08:00
public void SelectResult(int index)
{
if (index <= Results.Count - 1)
2016-02-12 14:21:12 +08:00
{
SelectedResult = Results[index];
2016-02-12 14:21:12 +08:00
}
}
public void SelectNextResult()
{
if (SelectedResult != null)
2016-02-12 14:21:12 +08:00
{
var index = Results.IndexOf(SelectedResult);
if (index == Results.Count - 1)
2016-02-12 14:21:12 +08:00
{
index = -1;
}
SelectedResult = Results.ElementAt(index + 1);
2016-02-12 14:21:12 +08:00
}
}
public void SelectPrevResult()
{
if (SelectedResult != null)
2016-02-12 14:21:12 +08:00
{
var index = Results.IndexOf(SelectedResult);
2016-02-12 14:21:12 +08:00
if (index == 0)
{
index = Results.Count;
2016-02-12 14:21:12 +08:00
}
SelectedResult = Results.ElementAt(index - 1);
2016-02-12 14:21:12 +08:00
}
}
public void SelectNextPage()
{
var index = 0;
if (SelectedResult != null)
2016-02-12 14:21:12 +08:00
{
index = Results.IndexOf(SelectedResult);
2016-02-12 14:21:12 +08:00
}
index += 5;
if (index > Results.Count - 1)
2016-02-12 14:21:12 +08:00
{
index = Results.Count - 1;
2016-02-12 14:21:12 +08:00
}
SelectedResult = Results.ElementAt(index);
2016-02-12 14:21:12 +08:00
}
public void SelectPrevPage()
{
var index = 0;
if (SelectedResult != null)
2016-02-12 14:21:12 +08:00
{
index = Results.IndexOf(SelectedResult);
2016-02-12 14:21:12 +08:00
}
index -= 5;
if (index < 0)
{
index = 0;
}
SelectedResult = Results.ElementAt(index);
2016-02-12 14:21:12 +08:00
}
public void Clear()
{
Results.Clear();
}
public void RemoveResultsExcept(PluginMetadata metadata)
{
lock (_resultsUpdateLock)
{
Results.RemoveAll(r => r.RawResult.PluginID != metadata.ID);
}
}
public void RemoveResultsFor(PluginMetadata metadata)
{
lock (_resultsUpdateLock)
{
Results.RemoveAll(r => r.RawResult.PluginID == metadata.ID);
}
}
public void AddResults(List<Result> newRawResults, string resultId)
{
lock (_resultsUpdateLock)
{
var newResults = new List<ResultViewModel>();
newRawResults.ForEach((re) => { newResults.Add(new ResultViewModel(re)); });
// todo use async to do new result calculation
var resultsCopy = Results.ToList();
var oldResults = resultsCopy.Where(r => r.RawResult.PluginID == resultId).ToList();
// intersection of A (old results) and B (new newResults)
var intersection = oldResults.Intersect(newResults).ToList();
// remove result of relative complement of B in A
foreach (var result in oldResults.Except(intersection))
{
resultsCopy.Remove(result);
}
// update scores
foreach (var result in newResults)
{
if (IsTopMostResult(result.RawResult))
{
result.RawResult.Score = int.MaxValue;
}
}
// update index for result in intersection of A and B
foreach (var commonResult in intersection)
{
int oldIndex = resultsCopy.IndexOf(commonResult);
int oldScore = resultsCopy[oldIndex].RawResult.Score;
int newScore = newResults[newResults.IndexOf(commonResult)].RawResult.Score;
if (newScore != oldScore)
{
var oldResult = resultsCopy[oldIndex];
oldResult.RawResult.Score = newScore;
resultsCopy.RemoveAt(oldIndex);
int newIndex = InsertIndexOf(newScore, resultsCopy);
resultsCopy.Insert(newIndex, oldResult);
}
}
// insert result in relative complement of A in B
foreach (var result in newResults.Except(intersection))
{
int newIndex = InsertIndexOf(result.RawResult.Score, resultsCopy);
resultsCopy.Insert(newIndex, result);
}
// update UI in one run, so it can avoid UI flickering
Results.Update(resultsCopy);
if (Results.Count > 0)
{
Margin = new Thickness { Top = 8 };
SelectedResult = Results[0];
}
else
{
Margin = new Thickness { Top = 0 };
}
}
}
#endregion
public class ResultCollection : ObservableCollection<ResultViewModel>
{
public void RemoveAll(Predicate<ResultViewModel> predicate)
{
CheckReentrancy();
List<ResultViewModel> itemsToRemove = Items.Where(x => predicate(x)).ToList();
if (itemsToRemove.Count > 0)
{
itemsToRemove.ForEach(item => { Items.Remove(item); });
OnPropertyChanged(new PropertyChangedEventArgs("Count"));
OnPropertyChanged(new PropertyChangedEventArgs("Item[]"));
// fuck ms
// http://blogs.msdn.com/b/nathannesbit/archive/2009/04/20/addrange-and-observablecollection.aspx
// http://geekswithblogs.net/NewThingsILearned/archive/2008/01/16/listcollectionviewcollectionview-doesnt-support-notifycollectionchanged-with-multiple-items.aspx
// PS: don't use Reset for other data updates, it will cause UI flickering
OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
}
}
public void Update(List<ResultViewModel> newItems)
{
int newCount = newItems.Count;
int oldCount = Items.Count;
int location = newCount > oldCount ? oldCount : newCount;
for (int i = 0; i < location; i++)
{
ResultViewModel oldResult = Items[i];
ResultViewModel newResult = newItems[i];
if (!oldResult.Equals(newResult))
{
this[i] = newResult;
}
else if (oldResult.RawResult.Score != newResult.RawResult.Score)
{
this[i].RawResult.Score = newResult.RawResult.Score;
}
}
if (newCount > oldCount)
{
for (int i = oldCount; i < newCount; i++)
{
Add(newItems[i]);
}
}
else
{
int removeIndex = newCount;
for (int i = newCount; i < oldCount; i++)
{
RemoveAt(removeIndex);
}
}
}
}
}
}