mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-04-06 03:07:04 +02:00
Refactor ResultPanel/ResultItem with MVVM
This commit is contained in:
378
Wox/ViewModel/ResultPanelViewModel.cs
Normal file
378
Wox/ViewModel/ResultPanelViewModel.cs
Normal file
@@ -0,0 +1,378 @@
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Collections.ObjectModel;
|
||||
using System.Collections.Specialized;
|
||||
using System.ComponentModel;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
using System.Threading.Tasks;
|
||||
using System.Windows;
|
||||
using Wox.Core.UserSettings;
|
||||
using Wox.Plugin;
|
||||
using Wox.Storage;
|
||||
|
||||
namespace Wox.ViewModel
|
||||
{
|
||||
public class ResultPanelViewModel : BaseViewModel
|
||||
{
|
||||
#region Private Fields
|
||||
|
||||
private ResultItemViewModel _selectedResult;
|
||||
private ResultCollection _results;
|
||||
private bool _isVisible;
|
||||
private Thickness _margin;
|
||||
|
||||
private readonly object _resultsUpdateLock = new object();
|
||||
|
||||
#endregion
|
||||
|
||||
#region Constructor
|
||||
|
||||
public ResultPanelViewModel()
|
||||
{
|
||||
this._results = new ResultCollection(
|
||||
|
||||
(o, e)=> {
|
||||
|
||||
if(null != ResultOpenedInPanel)
|
||||
{
|
||||
this.ResultOpenedInPanel(this, new ResultOpenedInPanelEventArgs(o as ResultItemViewModel, e.HideWindow));
|
||||
}
|
||||
},
|
||||
|
||||
(o, e) => {
|
||||
|
||||
if(null != ResultActionPanelOpenedInPanel)
|
||||
{
|
||||
this.ResultActionPanelOpenedInPanel(this, new ResultActionPanelOpenedInPanelEventArgs(o as ResultItemViewModel));
|
||||
}
|
||||
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region ViewModel Properties
|
||||
|
||||
public int MaxHeight
|
||||
{
|
||||
get
|
||||
{
|
||||
return UserSettingStorage.Instance.MaxResultsToShow * 50;
|
||||
}
|
||||
}
|
||||
|
||||
public ResultCollection Results
|
||||
{
|
||||
get
|
||||
{
|
||||
return this._results;
|
||||
}
|
||||
}
|
||||
|
||||
public ResultItemViewModel SelectedResult
|
||||
{
|
||||
get
|
||||
{
|
||||
return this._selectedResult;
|
||||
}
|
||||
set
|
||||
{
|
||||
this._selectedResult = value;
|
||||
OnPropertyChanged("SelectedResult");
|
||||
}
|
||||
}
|
||||
|
||||
public Thickness Margin
|
||||
{
|
||||
get
|
||||
{
|
||||
return this._margin;
|
||||
}
|
||||
set
|
||||
{
|
||||
this._margin = value;
|
||||
OnPropertyChanged("Margin");
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Private Methods
|
||||
|
||||
private bool IsTopMostResult(Result result)
|
||||
{
|
||||
return TopMostRecordStorage.Instance.IsTopMost(result);
|
||||
}
|
||||
|
||||
private int InsertIndexOf(int newScore, IList<ResultItemViewModel> 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
|
||||
|
||||
public void Clear()
|
||||
{
|
||||
this._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<ResultItemViewModel>();
|
||||
newRawResults.ForEach((re) => { newResults.Add(new ResultItemViewModel(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(this._results.Count > 0)
|
||||
{
|
||||
this.Margin = new Thickness { Top = 8 };
|
||||
this.SelectedResult = this._results[0];
|
||||
}
|
||||
else
|
||||
{
|
||||
this.Margin = new Thickness { Top = 0 };
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
#endregion
|
||||
|
||||
public event EventHandler<ResultOpenedInPanelEventArgs> ResultOpenedInPanel;
|
||||
|
||||
public event EventHandler<ResultActionPanelOpenedInPanelEventArgs> ResultActionPanelOpenedInPanel;
|
||||
|
||||
public class ResultCollection : ObservableCollection<ResultItemViewModel>
|
||||
// todo implement custom moveItem,removeItem,insertItem for better performance
|
||||
{
|
||||
|
||||
private EventHandler<ResultOpenedEventArgs> _resultOpenedHandler;
|
||||
private EventHandler _resultActionPanelOpenedHandler;
|
||||
|
||||
public ResultCollection(EventHandler<ResultOpenedEventArgs> resultOpenedHandler,
|
||||
EventHandler resultActionPanelOpenedHandler)
|
||||
{
|
||||
this._resultOpenedHandler = resultOpenedHandler;
|
||||
this._resultActionPanelOpenedHandler = resultActionPanelOpenedHandler;
|
||||
}
|
||||
|
||||
protected override void OnCollectionChanged(NotifyCollectionChangedEventArgs e)
|
||||
{
|
||||
base.OnCollectionChanged(e);
|
||||
|
||||
if(e.Action == NotifyCollectionChangedAction.Add)
|
||||
{
|
||||
foreach(var item in e.NewItems)
|
||||
{
|
||||
var resultVM = item as ResultItemViewModel;
|
||||
resultVM.ResultOpened += this._resultOpenedHandler;
|
||||
resultVM.ResultActionPanelOpened += this._resultActionPanelOpenedHandler;
|
||||
}
|
||||
}
|
||||
|
||||
if(e.Action == NotifyCollectionChangedAction.Remove)
|
||||
{
|
||||
foreach (var item in e.OldItems)
|
||||
{
|
||||
var resultVM = item as ResultItemViewModel;
|
||||
resultVM.ResultOpened -= this._resultOpenedHandler;
|
||||
resultVM.ResultActionPanelOpened -= this._resultActionPanelOpenedHandler;
|
||||
}
|
||||
}
|
||||
|
||||
if(e.Action == NotifyCollectionChangedAction.Replace)
|
||||
{
|
||||
foreach (var item in e.NewItems)
|
||||
{
|
||||
var resultVM = item as ResultItemViewModel;
|
||||
resultVM.ResultOpened += this._resultOpenedHandler;
|
||||
resultVM.ResultActionPanelOpened += this._resultActionPanelOpenedHandler;
|
||||
}
|
||||
|
||||
foreach (var item in e.OldItems)
|
||||
{
|
||||
var resultVM = item as ResultItemViewModel;
|
||||
resultVM.ResultOpened -= this._resultOpenedHandler;
|
||||
resultVM.ResultActionPanelOpened -= this._resultActionPanelOpenedHandler;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void RemoveAll(Predicate<ResultItemViewModel> predicate)
|
||||
{
|
||||
CheckReentrancy();
|
||||
|
||||
List<ResultItemViewModel> 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<ResultItemViewModel> newItems)
|
||||
{
|
||||
int newCount = newItems.Count;
|
||||
int oldCount = Items.Count;
|
||||
int location = newCount > oldCount ? oldCount : newCount;
|
||||
for (int i = 0; i < location; i++)
|
||||
{
|
||||
ResultItemViewModel oldItem = Items[i];
|
||||
ResultItemViewModel newItem = newItems[i];
|
||||
if (!oldItem.Equals(newItem))
|
||||
{
|
||||
this[i] = newItem;
|
||||
}
|
||||
else if (oldItem.RawResult.Score != newItem.RawResult.Score)
|
||||
{
|
||||
this[i].RawResult.Score = newItem.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);
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
public class ResultOpenedInPanelEventArgs : EventArgs
|
||||
{
|
||||
|
||||
public bool HideWindow
|
||||
{
|
||||
get;
|
||||
private set;
|
||||
}
|
||||
|
||||
public ResultItemViewModel Result
|
||||
{
|
||||
get;
|
||||
private set;
|
||||
}
|
||||
|
||||
public ResultOpenedInPanelEventArgs(ResultItemViewModel result, bool hideWindow)
|
||||
{
|
||||
this.HideWindow = hideWindow;
|
||||
this.Result = result;
|
||||
}
|
||||
}
|
||||
|
||||
public class ResultActionPanelOpenedInPanelEventArgs : EventArgs
|
||||
{
|
||||
|
||||
public ResultItemViewModel Result
|
||||
{
|
||||
get;
|
||||
private set;
|
||||
}
|
||||
|
||||
public ResultActionPanelOpenedInPanelEventArgs(ResultItemViewModel result)
|
||||
{
|
||||
this.Result = result;
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user