mirror of
https://github.com/Flow-Launcher/Flow.Launcher.git
synced 2026-03-11 08:54:32 +00:00
316 lines
10 KiB
C#
316 lines
10 KiB
C#
using Flow.Launcher.Infrastructure.UserSettings;
|
|
using Flow.Launcher.Plugin;
|
|
using System.Collections.Generic;
|
|
using System.Collections.Specialized;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using System.Windows;
|
|
using System.Windows.Controls;
|
|
using System.Windows.Data;
|
|
using System.Windows.Documents;
|
|
using System.Windows.Input;
|
|
using JetBrains.Annotations;
|
|
|
|
namespace Flow.Launcher.ViewModel
|
|
{
|
|
public class ResultsViewModel : BaseModel
|
|
{
|
|
#region Private Fields
|
|
|
|
public ResultCollection Results { get; }
|
|
|
|
private readonly object _collectionLock = new object();
|
|
private readonly Settings _settings;
|
|
private int MaxResults => _settings?.MaxResultsToShow ?? 6;
|
|
|
|
public ResultsViewModel()
|
|
{
|
|
Results = new ResultCollection();
|
|
BindingOperations.EnableCollectionSynchronization(Results, _collectionLock);
|
|
}
|
|
public ResultsViewModel(Settings settings) : this()
|
|
{
|
|
_settings = settings;
|
|
_settings.PropertyChanged += (s, e) =>
|
|
{
|
|
if (e.PropertyName == nameof(_settings.MaxResultsToShow))
|
|
{
|
|
OnPropertyChanged(nameof(MaxHeight));
|
|
}
|
|
};
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Properties
|
|
|
|
public int MaxHeight => MaxResults * 52;
|
|
|
|
public int SelectedIndex { get; set; }
|
|
|
|
public ResultViewModel SelectedItem { get; set; }
|
|
public Thickness Margin { get; set; }
|
|
public Visibility Visbility { get; set; } = Visibility.Collapsed;
|
|
|
|
public ICommand RightClickResultCommand { get; init; }
|
|
public ICommand LeftClickResultCommand { get; init; }
|
|
|
|
#endregion
|
|
|
|
#region Private Methods
|
|
|
|
private int InsertIndexOf(int newScore, IList<ResultViewModel> list)
|
|
{
|
|
int index = 0;
|
|
for (; index < list.Count; index++)
|
|
{
|
|
var result = list[index];
|
|
if (newScore > result.Result.Score)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
return index;
|
|
}
|
|
|
|
private int NewIndex(int i)
|
|
{
|
|
var n = Results.Count;
|
|
if (n > 0)
|
|
{
|
|
i = (n + i) % n;
|
|
return i;
|
|
}
|
|
else
|
|
{
|
|
// SelectedIndex returns -1 if selection is empty.
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
|
|
#endregion
|
|
|
|
#region Public Methods
|
|
|
|
public void SelectNextResult()
|
|
{
|
|
SelectedIndex = NewIndex(SelectedIndex + 1);
|
|
}
|
|
|
|
public void SelectPrevResult()
|
|
{
|
|
SelectedIndex = NewIndex(SelectedIndex - 1);
|
|
}
|
|
|
|
public void SelectNextPage()
|
|
{
|
|
SelectedIndex = NewIndex(SelectedIndex + MaxResults);
|
|
}
|
|
|
|
public void SelectPrevPage()
|
|
{
|
|
SelectedIndex = NewIndex(SelectedIndex - MaxResults);
|
|
}
|
|
|
|
public void SelectFirstResult()
|
|
{
|
|
SelectedIndex = NewIndex(0);
|
|
}
|
|
|
|
public void Clear()
|
|
{
|
|
lock (_collectionLock)
|
|
Results.RemoveAll();
|
|
}
|
|
|
|
public void KeepResultsFor(PluginMetadata metadata)
|
|
{
|
|
lock (_collectionLock)
|
|
Results.Update(Results.Where(r => r.Result.PluginID == metadata.ID).ToList());
|
|
}
|
|
|
|
public void KeepResultsExcept(PluginMetadata metadata)
|
|
{
|
|
lock (_collectionLock)
|
|
Results.Update(Results.Where(r => r.Result.PluginID != metadata.ID).ToList());
|
|
}
|
|
|
|
/// <summary>
|
|
/// To avoid deadlock, this method should not called from main thread
|
|
/// </summary>
|
|
public void AddResults(List<Result> newRawResults, string resultId)
|
|
{
|
|
var newResults = NewResults(newRawResults, resultId);
|
|
|
|
UpdateResults(newResults);
|
|
}
|
|
/// <summary>
|
|
/// To avoid deadlock, this method should not called from main thread
|
|
/// </summary>
|
|
public void AddResults(IEnumerable<ResultsForUpdate> resultsForUpdates, CancellationToken token)
|
|
{
|
|
var newResults = NewResults(resultsForUpdates);
|
|
|
|
if (token.IsCancellationRequested)
|
|
return;
|
|
|
|
UpdateResults(newResults, token);
|
|
}
|
|
|
|
private void UpdateResults(List<ResultViewModel> newResults, CancellationToken token = default)
|
|
{
|
|
lock (_collectionLock)
|
|
{
|
|
// update UI in one run, so it can avoid UI flickering
|
|
Results.Update(newResults, token);
|
|
if (Results.Any())
|
|
SelectedItem = Results[0];
|
|
}
|
|
|
|
switch (Visbility)
|
|
{
|
|
case Visibility.Collapsed when Results.Count > 0:
|
|
SelectedIndex = 0;
|
|
Visbility = Visibility.Visible;
|
|
break;
|
|
case Visibility.Visible when Results.Count == 0:
|
|
Visbility = Visibility.Collapsed;
|
|
break;
|
|
}
|
|
}
|
|
|
|
private List<ResultViewModel> NewResults(List<Result> newRawResults, string resultId)
|
|
{
|
|
if (newRawResults.Count == 0)
|
|
return Results;
|
|
|
|
|
|
var newResults = newRawResults.Select(r => new ResultViewModel(r, _settings));
|
|
|
|
return Results.Where(r => r.Result.PluginID != resultId)
|
|
.Concat(newResults)
|
|
.OrderByDescending(r => r.Result.Score)
|
|
.ToList();
|
|
}
|
|
|
|
private List<ResultViewModel> NewResults(IEnumerable<ResultsForUpdate> resultsForUpdates)
|
|
{
|
|
if (!resultsForUpdates.Any())
|
|
return Results;
|
|
|
|
return Results.Where(r => r != null && !resultsForUpdates.Any(u => u.ID == r.Result.PluginID))
|
|
.Concat(resultsForUpdates.SelectMany(u => u.Results, (u, r) => new ResultViewModel(r, _settings)))
|
|
.OrderByDescending(rv => rv.Result.Score)
|
|
.ToList();
|
|
}
|
|
#endregion
|
|
|
|
#region FormattedText Dependency Property
|
|
public static readonly DependencyProperty FormattedTextProperty = DependencyProperty.RegisterAttached(
|
|
"FormattedText",
|
|
typeof(Inline),
|
|
typeof(ResultsViewModel),
|
|
new PropertyMetadata(null, FormattedTextPropertyChanged));
|
|
|
|
public static void SetFormattedText(DependencyObject textBlock, IList<int> value)
|
|
{
|
|
textBlock.SetValue(FormattedTextProperty, value);
|
|
}
|
|
|
|
public static Inline GetFormattedText(DependencyObject textBlock)
|
|
{
|
|
return (Inline)textBlock.GetValue(FormattedTextProperty);
|
|
}
|
|
|
|
private static void FormattedTextPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
|
|
{
|
|
var textBlock = d as TextBlock;
|
|
if (textBlock == null) return;
|
|
|
|
var inline = (Inline)e.NewValue;
|
|
|
|
textBlock.Inlines.Clear();
|
|
if (inline == null) return;
|
|
|
|
textBlock.Inlines.Add(inline);
|
|
}
|
|
#endregion
|
|
|
|
public class ResultCollection : List<ResultViewModel>, INotifyCollectionChanged
|
|
{
|
|
private long editTime = 0;
|
|
|
|
private CancellationToken _token;
|
|
|
|
public event NotifyCollectionChangedEventHandler CollectionChanged;
|
|
|
|
|
|
protected void OnCollectionChanged(NotifyCollectionChangedEventArgs e)
|
|
{
|
|
CollectionChanged?.Invoke(this, e);
|
|
}
|
|
|
|
public void BulkAddAll(List<ResultViewModel> resultViews)
|
|
{
|
|
AddRange(resultViews);
|
|
|
|
// can return because the list will be cleared next time updated, which include a reset event
|
|
if (_token.IsCancellationRequested)
|
|
return;
|
|
|
|
// manually update event
|
|
// wpf use directx / double buffered already, so just reset all won't cause ui flickering
|
|
OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
|
|
}
|
|
private void AddAll(List<ResultViewModel> Items)
|
|
{
|
|
for (int i = 0; i < Items.Count; i++)
|
|
{
|
|
var item = Items[i];
|
|
if (_token.IsCancellationRequested)
|
|
return;
|
|
Add(item);
|
|
OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add, item, i));
|
|
}
|
|
}
|
|
public void RemoveAll(int Capacity = 512)
|
|
{
|
|
Clear();
|
|
if (this.Capacity > 8000 && Capacity < this.Capacity)
|
|
this.Capacity = Capacity;
|
|
|
|
OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update the results collection with new results, try to keep identical results
|
|
/// </summary>
|
|
/// <param name="newItems"></param>
|
|
public void Update(List<ResultViewModel> newItems, CancellationToken token = default)
|
|
{
|
|
_token = token;
|
|
if (Count == 0 && newItems.Count == 0 || _token.IsCancellationRequested)
|
|
return;
|
|
|
|
if (editTime < 10 || newItems.Count < 30)
|
|
{
|
|
if (Count != 0) RemoveAll(newItems.Count);
|
|
AddAll(newItems);
|
|
editTime++;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
BulkAddAll(newItems);
|
|
if (Capacity > 8000 && newItems.Count < 3000)
|
|
{
|
|
Capacity = newItems.Count;
|
|
}
|
|
editTime++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|