mirror of
https://github.com/microsoft/PowerToys.git
synced 2025-01-18 22:43:31 +08:00
323 lines
11 KiB
C#
323 lines
11 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Runtime.Remoting.Contexts;
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using System.Windows.Input;
|
|
using System.Windows.Media;
|
|
using Wox.Core.UserSettings;
|
|
using Wox.Helper;
|
|
using Wox.Plugin;
|
|
using Wox.Storage;
|
|
|
|
namespace Wox
|
|
{
|
|
[Synchronization]
|
|
public partial class ResultPanel : UserControl
|
|
{
|
|
public event Action<Result> LeftMouseClickEvent;
|
|
public event Action<Result> RightMouseClickEvent;
|
|
public event Action<Result, IDataObject, DragEventArgs> ItemDropEvent;
|
|
private readonly ListBoxItems _results;
|
|
private readonly object _resultsUpdateLock = new object();
|
|
|
|
protected virtual void OnRightMouseClick(Result result)
|
|
{
|
|
Action<Result> handler = RightMouseClickEvent;
|
|
if (handler != null) handler(result);
|
|
}
|
|
|
|
protected virtual void OnLeftMouseClick(Result result)
|
|
{
|
|
Action<Result> handler = LeftMouseClickEvent;
|
|
if (handler != null) handler(result);
|
|
}
|
|
|
|
|
|
public int MaxResultsToShow { get { return UserSettingStorage.Instance.MaxResultsToShow * 50; } }
|
|
|
|
internal void RemoveResultsFor(PluginMetadata metadata)
|
|
{
|
|
lock (_resultsUpdateLock)
|
|
{
|
|
_results.RemoveAll(r => r.PluginID == metadata.ID);
|
|
}
|
|
}
|
|
|
|
internal void RemoveResultsExcept(PluginMetadata metadata)
|
|
{
|
|
lock (_resultsUpdateLock)
|
|
{
|
|
_results.RemoveAll(r => r.PluginID != metadata.ID);
|
|
}
|
|
}
|
|
|
|
public void AddResults(List<Result> newResults, string resultId)
|
|
{
|
|
lock (_resultsUpdateLock)
|
|
{
|
|
// todo use async to do new result calculation
|
|
var resultsCopy = _results.ToList();
|
|
var oldResults = resultsCopy.Where(r => r.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))
|
|
{
|
|
result.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].Score;
|
|
int newScore = newResults[newResults.IndexOf(commonResult)].Score;
|
|
if (newScore != oldScore)
|
|
{
|
|
var oldResult = resultsCopy[oldIndex];
|
|
oldResult.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.Score, resultsCopy);
|
|
resultsCopy.Insert(newIndex, result);
|
|
}
|
|
|
|
// update UI in one run, so it can avoid UI flickering
|
|
_results.Update(resultsCopy);
|
|
|
|
lbResults.Margin = lbResults.Items.Count > 0 ? new Thickness { Top = 8 } : new Thickness { Top = 0 };
|
|
SelectFirst();
|
|
}
|
|
}
|
|
|
|
private bool IsTopMostResult(Result result)
|
|
{
|
|
return TopMostRecordStorage.Instance.IsTopMost(result);
|
|
}
|
|
|
|
private int InsertIndexOf(int newScore, IList<Result> list)
|
|
{
|
|
int index = 0;
|
|
for (; index < list.Count; index++)
|
|
{
|
|
var result = list[index];
|
|
if (newScore > result.Score)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
return index;
|
|
}
|
|
|
|
public void SelectNext()
|
|
{
|
|
int index = lbResults.SelectedIndex;
|
|
if (index == lbResults.Items.Count - 1)
|
|
{
|
|
index = -1;
|
|
}
|
|
Select(index + 1);
|
|
}
|
|
|
|
public void SelectPrev()
|
|
{
|
|
int index = lbResults.SelectedIndex;
|
|
if (index == 0)
|
|
{
|
|
index = lbResults.Items.Count;
|
|
}
|
|
Select(index - 1);
|
|
}
|
|
|
|
private void SelectFirst()
|
|
{
|
|
Select(0);
|
|
}
|
|
|
|
private void Select(int index)
|
|
{
|
|
if (index >= 0 && index < lbResults.Items.Count)
|
|
{
|
|
lbResults.SelectedItem = lbResults.Items.GetItemAt(index);
|
|
}
|
|
}
|
|
|
|
public List<Result> GetVisibleResults()
|
|
{
|
|
List<Result> visibleElements = new List<Result>();
|
|
VirtualizingStackPanel virtualizingStackPanel = GetInnerStackPanel(lbResults);
|
|
for (int i = (int)virtualizingStackPanel.VerticalOffset; i <= virtualizingStackPanel.VerticalOffset + virtualizingStackPanel.ViewportHeight; i++)
|
|
{
|
|
ListBoxItem item = lbResults.ItemContainerGenerator.ContainerFromIndex(i) as ListBoxItem;
|
|
if (item != null)
|
|
{
|
|
visibleElements.Add(item.DataContext as Result);
|
|
}
|
|
}
|
|
return visibleElements;
|
|
}
|
|
|
|
private void UpdateItemNumber()
|
|
{
|
|
//VirtualizingStackPanel virtualizingStackPanel = GetInnerStackPanel(lbResults);
|
|
//int index = 0;
|
|
//for (int i = (int)virtualizingStackPanel.VerticalOffset; i <= virtualizingStackPanel.VerticalOffset + virtualizingStackPanel.ViewportHeight; i++)
|
|
//{
|
|
// index++;
|
|
// ListBoxItem item = lbResults.ItemContainerGenerator.ContainerFromIndex(i) as ListBoxItem;
|
|
// if (item != null)
|
|
// {
|
|
// ContentPresenter myContentPresenter = FindVisualChild<ContentPresenter>(item);
|
|
// if (myContentPresenter != null)
|
|
// {
|
|
// DataTemplate dataTemplate = myContentPresenter.ContentTemplate;
|
|
// TextBlock tbItemNumber = (TextBlock)dataTemplate.FindName("tbItemNumber", myContentPresenter);
|
|
// tbItemNumber.Text = index.ToString();
|
|
// }
|
|
// }
|
|
//}
|
|
}
|
|
|
|
private childItem FindVisualChild<childItem>(DependencyObject obj) where childItem : DependencyObject
|
|
{
|
|
for (int i = 0; i < VisualTreeHelper.GetChildrenCount(obj); i++)
|
|
{
|
|
DependencyObject child = VisualTreeHelper.GetChild(obj, i);
|
|
if (child != null && child is childItem)
|
|
return (childItem)child;
|
|
else
|
|
{
|
|
childItem childOfChild = FindVisualChild<childItem>(child);
|
|
if (childOfChild != null)
|
|
return childOfChild;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private VirtualizingStackPanel GetInnerStackPanel(FrameworkElement element)
|
|
{
|
|
for (int i = 0; i < VisualTreeHelper.GetChildrenCount(element); i++)
|
|
{
|
|
var child = VisualTreeHelper.GetChild(element, i) as FrameworkElement;
|
|
|
|
if (child == null) continue;
|
|
|
|
if (child is VirtualizingStackPanel) return child as VirtualizingStackPanel;
|
|
|
|
var panel = GetInnerStackPanel(child);
|
|
|
|
if (panel != null)
|
|
return panel;
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
public Result GetActiveResult()
|
|
{
|
|
int index = lbResults.SelectedIndex;
|
|
if (index < 0) return null;
|
|
|
|
return lbResults.Items[index] as Result;
|
|
}
|
|
|
|
public ResultPanel()
|
|
{
|
|
InitializeComponent();
|
|
_results = new ListBoxItems();
|
|
lbResults.ItemsSource = _results;
|
|
}
|
|
|
|
public void Clear()
|
|
{
|
|
lock (_resultsUpdateLock)
|
|
{
|
|
_results.Clear();
|
|
lbResults.Margin = new Thickness { Top = 0 };
|
|
}
|
|
}
|
|
|
|
private void lbResults_SelectionChanged(object sender, SelectionChangedEventArgs e)
|
|
{
|
|
if (e.AddedItems.Count > 0 && e.AddedItems[0] != null)
|
|
{
|
|
lbResults.ScrollIntoView(e.AddedItems[0]);
|
|
//Dispatcher.DelayInvoke("UpdateItemNumber", () =>
|
|
//{
|
|
//UpdateItemNumber();
|
|
//}, TimeSpan.FromMilliseconds(3));
|
|
}
|
|
}
|
|
|
|
private void LbResults_OnPreviewMouseDown(object sender, MouseButtonEventArgs e)
|
|
{
|
|
var item = ItemsControl.ContainerFromElement(lbResults, e.OriginalSource as DependencyObject) as ListBoxItem;
|
|
if (item != null && e.ChangedButton == MouseButton.Left)
|
|
{
|
|
OnLeftMouseClick(item.DataContext as Result);
|
|
}
|
|
if (item != null && e.ChangedButton == MouseButton.Right)
|
|
{
|
|
OnRightMouseClick(item.DataContext as Result);
|
|
}
|
|
}
|
|
|
|
public void SelectNextPage()
|
|
{
|
|
int index = lbResults.SelectedIndex;
|
|
index += 5;
|
|
if (index >= lbResults.Items.Count)
|
|
{
|
|
index = lbResults.Items.Count - 1;
|
|
}
|
|
Select(index);
|
|
}
|
|
|
|
public void SelectPrevPage()
|
|
{
|
|
int index = lbResults.SelectedIndex;
|
|
index -= 5;
|
|
if (index < 0)
|
|
{
|
|
index = 0;
|
|
}
|
|
Select(index);
|
|
}
|
|
|
|
private void ListBoxItem_OnDrop(object sender, DragEventArgs e)
|
|
{
|
|
var item = ItemsControl.ContainerFromElement(lbResults, e.OriginalSource as DependencyObject) as ListBoxItem;
|
|
if (item != null)
|
|
{
|
|
OnItemDropEvent(item.DataContext as Result, e.Data, e);
|
|
}
|
|
}
|
|
|
|
protected virtual void OnItemDropEvent(Result obj, IDataObject data, DragEventArgs e)
|
|
{
|
|
var handler = ItemDropEvent;
|
|
if (handler != null) handler(obj, data, e);
|
|
}
|
|
}
|
|
} |